/** * @brief Some platform-specific abstractions. * * Things like initial entry point, utility functions we need * in BIOS but don't already have, BIOS trampoline management, * and so on. * * @copyright * This file is part of ToaruOS and is released under the terms * of the NCSA / University of Illinois License - see LICENSE.md * Copyright (C) 2018-2021 K. Lange */ extern int kmain(); #ifdef EFI_PLATFORM #include #include EFI_HANDLE ImageHandleIn; extern int init_graphics(); EFI_STATUS EFIAPI efi_main (EFI_HANDLE ImageHandle, EFI_SYSTEM_TABLE *SystemTable) { InitializeLib(ImageHandle, SystemTable); ST = SystemTable; ImageHandleIn = ImageHandle; init_graphics(); return kmain(); } #else #include #include #include "iso9660.h" #include "util.h" #include "text.h" extern char _bss_start[]; extern char _bss_end[]; void * memcpy(void * restrict dest, const void * restrict src, long n) { asm volatile("cld; rep movsb" : "=c"((int){0}) : "D"(dest), "S"(src), "c"(n) : "flags", "memory"); return dest; } void * memset(void * dest, int c, long n) { asm volatile("cld; rep stosb" : "=c"((int){0}) : "D"(dest), "a"(c), "c"(n) : "flags", "memory"); return dest; } extern void init_graphics(void); extern void do_bios_call(uint32_t function, uint32_t arg1); extern uint32_t vbe_cont_info_mode_off; extern uint16_t vbe_info_pitch; extern uint16_t vbe_info_width; extern uint16_t vbe_info_height; extern uint8_t vbe_info_bpp; extern uint16_t vbe_info; void text_reset(void) { /* Hide the cursor */ outportb(0x3D4, 14); outportb(0x3D5, 0xFF); outportb(0x3D4, 15); outportb(0x3D5, 0xFF); /* iirc this disables blink? */ inportb(0x3DA); outportb(0x3C0, 0x30); char b = inportb(0x3C1); b &= ~8; outportb(0x3c0, b); } extern int in_graphics_mode; int bios_text_mode(void) { do_bios_call(3, 3); vbe_info_width = 0; in_graphics_mode = 0; text_reset(); } int last_video_mode = -1; void bios_set_video(int mode) { last_video_mode = mode; do_bios_call(2, mode); do_bios_call(3, mode | 0x4000); init_graphics(); } int bios_video_mode(void) { int best_match = 0; int match_score = 0; #define MATCH(w,h,s) if (match_score < s && vbe_info_width == w && vbe_info_height == h) { best_match = *x; match_score = s; } uint32_t vbe_addr = ((vbe_cont_info_mode_off & 0xFFFF0000) >> 12) + (vbe_cont_info_mode_off & 0xFFFF); for (uint16_t * x = (uint16_t*)vbe_addr; *x != 0xFFFF; x++) { /* Query mode info */ do_bios_call(2, *x); if (!(vbe_info & (1 << 7))) continue; if (vbe_info_bpp < 24) continue; if (vbe_info_bpp == 32) { if (match_score < 9) { best_match = *x; match_score = 9; } MATCH(1024,768,10); MATCH(1280,720,50); MATCH(1280,800,60); MATCH(1920,1080,75); MATCH(1440,900,100); } else if (vbe_info_bpp == 24) { if (!match_score) { best_match = *x; match_score = 1; } MATCH(1024,768,3); MATCH(1280,720,4); MATCH(1280,800,5); MATCH(1920,1080,6); MATCH(1440,900,7); } } if (best_match) { bios_set_video(best_match); } else { vbe_info_width = 0; } } void bios_toggle_mode(void) { if (in_graphics_mode) { bios_text_mode(); } else if (last_video_mode != -1) { bios_set_video(last_video_mode); } } int bios_main(void) { /* Zero BSS */ memset(&_bss_start,0,(uintptr_t)&_bss_end-(uintptr_t)&_bss_start); text_reset(); bios_video_mode(); return kmain(); } extern volatile uint16_t dap_sectors; extern volatile uint32_t dap_buffer; extern volatile uint32_t dap_lba_low; extern volatile uint32_t dap_lba_high; extern volatile uint16_t drive_params_bps; extern uint8_t disk_space[]; int bios_call(char * into, uint32_t sector) { dap_sectors = 2048 / drive_params_bps; dap_buffer = (uint32_t)disk_space; dap_lba_low = sector * dap_sectors; dap_lba_high = 0; do_bios_call(1,0); memcpy(into, disk_space, 2048); } iso_9660_volume_descriptor_t * root = NULL; iso_9660_directory_entry_t * dir_entry = NULL; static char * dir_entries = NULL; int navigate(char * name) { dir_entry = (iso_9660_directory_entry_t*)&root->root; dir_entries = (char*)(DATA_LOAD_BASE + dir_entry->extent_start_LSB * ISO_SECTOR_SIZE); bios_call(dir_entries, dir_entry->extent_start_LSB); long offset = 0; while (1) { iso_9660_directory_entry_t * dir = (iso_9660_directory_entry_t *)(dir_entries + offset); if (dir->length == 0) { if (offset < dir_entry->extent_length_LSB) { offset += 1; goto try_again; } break; } if (!(dir->flags & FLAG_HIDDEN)) { char file_name[dir->name_len + 1]; memcpy(file_name, dir->name, dir->name_len); file_name[dir->name_len] = 0; char * s = strchr(file_name,';'); if (s) { *s = '\0'; } if (!strcmp(file_name, name)) { dir_entry = dir; return 1; } } offset += dir->length; try_again: if ((long)(offset) > dir_entry->extent_length_LSB) break; } return 0; } #endif