#include #include #include #include #include #include #include #include #include //#define SPAWN_MULTIPLE_VIRTUAL_CONSOLES void start_process(const String& program, const Vector& arguments, int prio, const char* tty = nullptr) { pid_t pid = 0; while (true) { dbg() << "Forking for " << program << "..."; int pid = fork(); if (pid < 0) { dbg() << "Fork " << program << " failed! " << strerror(errno); continue; } else if (pid > 0) { // parent... dbg() << "Process " << program << " hopefully started with priority " << prio << "..."; return; } break; } while (true) { dbg() << "Executing for " << program << "... at prio " << prio; struct sched_param p; p.sched_priority = prio; int ret = sched_setparam(pid, &p); ASSERT(ret == 0); if (tty != nullptr) { close(0); ASSERT(open(tty, O_RDWR) == 0); dup2(0, 1); dup2(0, 2); } char* progv[256]; progv[0] = const_cast(program.characters()); for (int i = 0; i < arguments.size() && i < 254; i++) progv[i+1] = const_cast(arguments[i].characters()); progv[arguments.size() + 1] = nullptr; ret = execv(progv[0], progv); if (ret < 0) { dbg() << "Exec " << progv[0] << " failed! " << strerror(errno); continue; } break; } } static void check_for_test_mode() { auto f = CFile::construct("/proc/cmdline"); if (!f->open(CIODevice::ReadOnly)) { dbg() << "Failed to read command line: " << f->error_string(); ASSERT(false); } const String cmd = String::copy(f->read_all()); dbg() << "Read command line: " << cmd; if (cmd.matches("*testmode=1*")) { // Eventually, we should run a test binary and wait for it to finish // before shutting down. But this is good enough for now. dbg() << "Waiting for testmode shutdown..."; sleep(5); dbg() << "Shutting down due to testmode..."; if (fork() == 0) { execl("/bin/shutdown", "/bin/shutdown", "-n", nullptr); ASSERT_NOT_REACHED(); } } else { dbg() << "Continuing normally"; } } int main(int, char**) { int lowest_prio = sched_get_priority_min(SCHED_OTHER); int highest_prio = sched_get_priority_max(SCHED_OTHER); // Mount the filesystems. start_process("/bin/mount", { "-a" }, highest_prio); wait(nullptr); // NOTE: We don't start anything on tty0 since that's the "active" TTY while WindowServer is up. start_process("/bin/TTYServer", { "tty1" }, highest_prio, "/dev/tty1"); #ifdef SPAWN_MULTIPLE_VIRTUAL_CONSOLES start_process("/bin/TTYServer", { "tty2" }, highest_prio, "/dev/tty2"); start_process("/bin/TTYServer", { "tty3" }, highest_prio, "/dev/tty3"); #endif // Drop privileges. setgid(100); setuid(100); start_process("/bin/LookupServer", {}, lowest_prio); start_process("/bin/WindowServer", {}, highest_prio); start_process("/bin/AudioServer", {}, highest_prio); start_process("/bin/Taskbar", {}, highest_prio); start_process("/bin/Terminal", {}, highest_prio - 1); start_process("/bin/Launcher", {}, highest_prio); // This won't return if we're in test mode. check_for_test_mode(); while (1) { sleep(3600); } }