#include "types.h" #include "VGA.h" #include "kmalloc.h" #include "i386.h" #include "i8253.h" #include "Keyboard.h" #include "Process.h" #include "system.h" #include "Disk.h" #include "PIC.h" #include "StdLib.h" #include "Syscall.h" #include "CMOS.h" #include "IDEDiskDevice.h" #include #include #include #include #include #include #include #include #include "MemoryManager.h" #include #include "Console.h" #include "ProcFileSystem.h" #include "RTC.h" #include "VirtualConsole.h" #include "Scheduler.h" #define TEST_VFS #define KSYMS #define SPAWN_MULTIPLE_SHELLS //#define STRESS_TEST_SPAWNING //#define TEST_ELF_LOADER system_t system; VirtualConsole* tty0; VirtualConsole* tty1; VirtualConsole* tty2; VirtualConsole* tty3; Keyboard* keyboard; static byte parseHexDigit(char nibble) { if (nibble >= '0' && nibble <= '9') return nibble - '0'; ASSERT(nibble >= 'a' && nibble <= 'f'); return 10 + (nibble - 'a'); } #ifdef KSYMS static Vector* s_ksyms; static bool s_ksyms_ready; Vector& ksyms() { return *s_ksyms; } volatile bool ksyms_ready() { return s_ksyms_ready; } const KSym* ksymbolicate(dword address) { if (address < ksyms().first().address || address > ksyms().last().address) return nullptr; for (unsigned i = 0; i < ksyms().size(); ++i) { if (address < ksyms()[i + 1].address) return &ksyms()[i]; } return nullptr; } static void loadKsyms(const ByteBuffer& buffer) { // FIXME: It's gross that this vector grows dynamically rather than being sized-to-fit. // We're wasting that eternal kmalloc memory. s_ksyms = new Vector; auto* bufptr = (const char*)buffer.pointer(); auto* startOfName = bufptr; dword address = 0; while (bufptr < buffer.endPointer()) { for (unsigned i = 0; i < 8; ++i) address = (address << 4) | parseHexDigit(*(bufptr++)); bufptr += 3; startOfName = bufptr; while (*(++bufptr)) { if (*bufptr == '\n') { break; } } // FIXME: The Strings here should be eternally allocated too. ksyms().append({ address, String(startOfName, bufptr - startOfName) }); ++bufptr; } s_ksyms_ready = true; } void dump_backtrace(bool use_ksyms) { if (!current) { HANG; return; } extern volatile bool ksyms_ready(); if (use_ksyms && !ksyms_ready()) { HANG; return; } struct RecognizedSymbol { dword address; const KSym* ksym; }; Vector recognizedSymbols; if (use_ksyms) { for (dword* stackPtr = (dword*)&use_ksyms; current->isValidAddressForKernel(LinearAddress((dword)stackPtr)); stackPtr = (dword*)*stackPtr) { dword retaddr = stackPtr[1]; if (auto* ksym = ksymbolicate(retaddr)) recognizedSymbols.append({ retaddr, ksym }); } } else{ for (dword* stackPtr = (dword*)&use_ksyms; current->isValidAddressForKernel(LinearAddress((dword)stackPtr)); stackPtr = (dword*)*stackPtr) { dword retaddr = stackPtr[1]; kprintf("%x (next: %x)\n", retaddr, stackPtr ? (dword*)*stackPtr : 0); } return; } size_t bytesNeeded = 0; for (auto& symbol : recognizedSymbols) { bytesNeeded += symbol.ksym->name.length() + 8 + 16; } for (auto& symbol : recognizedSymbols) { unsigned offset = symbol.address - symbol.ksym->address; dbgprintf("%p %s +%u\n", symbol.address, symbol.ksym->name.characters(), offset); } } #endif static void undertaker_main() NORETURN; static void undertaker_main() { for (;;) { Process::doHouseKeeping(); sleep(300); } } static void spawn_stress() NORETURN; static void spawn_stress() { dword lastAlloc = sum_alloc; for (unsigned i = 0; i < 10000; ++i) { int error; Process::create_user_process("/bin/id", (uid_t)100, (gid_t)100, (pid_t)0, error, Vector(), Vector(), tty0); // kprintf("malloc stats: alloc:%u free:%u page_aligned:%u eternal:%u\n", sum_alloc, sum_free, kmalloc_page_aligned, kmalloc_sum_eternal); // kprintf("delta:%u\n", sum_alloc - lastAlloc); lastAlloc = sum_alloc; sleep(60); } for (;;) { asm volatile("hlt"); } } static void init_stage2() NORETURN; static void init_stage2() { Syscall::initialize(); Disk::initialize(); #ifdef TEST_VFS auto vfs = make(); auto dev_zero = make(); vfs->registerCharacterDevice(*dev_zero); auto dev_null = make(); vfs->registerCharacterDevice(*dev_null); auto dev_full = make(); vfs->registerCharacterDevice(*dev_full); auto dev_random = make(); vfs->registerCharacterDevice(*dev_random); vfs->registerCharacterDevice(*keyboard); vfs->registerCharacterDevice(*tty0); vfs->registerCharacterDevice(*tty1); vfs->registerCharacterDevice(*tty2); vfs->registerCharacterDevice(*tty3); auto dev_hd0 = IDEDiskDevice::create(); auto e2fs = Ext2FileSystem::create(dev_hd0.copyRef()); e2fs->initialize(); vfs->mountRoot(e2fs.copyRef()); #ifdef KSYMS { int error; auto descriptor = vfs->open("/kernel.map", error); if (!descriptor) { kprintf("Failed to open /kernel.map\n"); } else { auto buffer = descriptor->readEntireFile(); ASSERT(buffer); loadKsyms(buffer); } } #endif vfs->mount(ProcFileSystem::the(), "/proc"); #endif #ifdef TEST_ELF_LOADER { auto testExecutable = vfs->open("/bin/id"); ASSERT(testExecutable); auto testExecutableData = testExecutable->readEntireFile(); ASSERT(testExecutableData); ExecSpace space; space.loadELF(move(testExecutableData)); auto* elf_entry = space.symbol_ptr("_start"); ASSERT(elf_entry); typedef int (*MainFunctionPtr)(void); kprintf("elf_entry: %p\n", elf_entry); int rc = reinterpret_cast(elf_entry)(); kprintf("it returned %d\n", rc); } #endif int error; auto* sh0 = Process::create_user_process("/bin/sh", (uid_t)100, (gid_t)100, (pid_t)0, error, Vector(), Vector(), tty0); #ifdef SPAWN_MULTIPLE_SHELLS auto* sh1 = Process::create_user_process("/bin/sh", (uid_t)100, (gid_t)100, (pid_t)0, error, Vector(), Vector(), tty1); auto* sh2 = Process::create_user_process("/bin/sh", (uid_t)100, (gid_t)100, (pid_t)0, error, Vector(), Vector(), tty2); auto* sh3 = Process::create_user_process("/bin/sh", (uid_t)100, (gid_t)100, (pid_t)0, error, Vector(), Vector(), tty3); #endif #ifdef STRESS_TEST_SPAWNING Process::createKernelProcess(spawn_stress, "spawn_stress"); #endif for (;;) { //sleep(3600 * TICKS_PER_SECOND); asm("hlt"); } } void init() { cli(); #ifdef KSYMS s_ksyms = nullptr; s_ksyms_ready = false; #endif kmalloc_init(); vga_init(); auto console = make(); RTC::initialize(); PIC::initialize(); gdt_init(); idt_init(); keyboard = new Keyboard; VirtualConsole::initialize(); tty0 = new VirtualConsole(0, VirtualConsole::AdoptCurrentVGABuffer); tty1 = new VirtualConsole(1); tty2 = new VirtualConsole(2); tty3 = new VirtualConsole(3); VirtualConsole::switch_to(0); kprintf("Starting Serenity Operating System...\n"); MemoryManager::initialize(); VirtualFileSystem::initializeGlobals(); StringImpl::initializeGlobals(); PIT::initialize(); memset(&system, 0, sizeof(system)); WORD base_memory = (CMOS::read(0x16) << 8) | CMOS::read(0x15); WORD ext_memory = (CMOS::read(0x18) << 8) | CMOS::read(0x17); kprintf("%u kB base memory\n", base_memory); kprintf("%u kB extended memory\n", ext_memory); auto procfs = ProcFileSystem::create(); procfs->initialize(); Process::initialize(); Process::create_kernel_process(undertaker_main, "undertaker"); Process::create_kernel_process(init_stage2, "init"); Scheduler::pick_next(); sti(); // This now becomes the idle process :^) for (;;) { asm("hlt"); } } void log_try_lock(const char* where) { kprintf("[%u] >>> locking... (%s)\n", current->pid(), where); } void log_locked(const char* where) { kprintf("[%u] >>> locked() in %s\n", current->pid(), where); } void log_unlocked(const char* where) { kprintf("[%u] <<< unlocked()\n", current->pid(), where); }