123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122 |
- #include <AK/Assertions.h>
- #include <LibCore/CFile.h>
- #include <errno.h>
- #include <fcntl.h>
- #include <sched.h>
- #include <signal.h>
- #include <stdio.h>
- #include <string.h>
- #include <sys/types.h>
- #include <sys/wait.h>
- #include <unistd.h>
- void sigchld_handler(int)
- {
- int status = 0;
- pid_t pid = waitpid(-1, &status, WNOHANG);
- if (pid)
- dbg() << "reaped pid " << pid;
- }
- void start_process(const String& program, const Vector<String>& 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<char*>(program.characters());
- for (int i = 0; i < arguments.size() && i < 254; i++)
- progv[i + 1] = const_cast<char*>(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");
- // Drop privileges.
- setgid(100);
- setuid(100);
- signal(SIGCHLD, sigchld_handler);
- 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);
- }
- }
|