#include #include #include #include #include #include #include #include #include #include #include #include u32 pid = 0; struct process *root; struct process *current_proc = NULL; struct regs hold_root; extern u32 stack_hold; void scheduler(struct regs *regs) { cli(); memcpy(¤t_proc->registers, regs, sizeof(struct regs)); timer_handler(regs); current_proc = current_proc->next; if (current_proc == NULL) { current_proc = root; } debug("Max pid: %d", pid); debug("Task switch to %s with pid %d", current_proc->name, current_proc->pid); while (current_proc->state == PROC_ASLEEP) { current_proc = current_proc->next; if (current_proc == NULL) { warn("No next process!"); current_proc = root; } } memcpy(regs, ¤t_proc->registers, sizeof(struct regs)); paging_switch_directory(current_proc->cr3); sti(); } void process_init(struct process *proc) { log("Initializing process %d", pid + 1); cli(); root = proc; root->pid = pid++; root->next = NULL; root->thread = PROC_ROOT; // Unkillable root->state = PROC_RUNNING; current_proc = root; irq_install_handler(0, scheduler); userspace_enter(proc); } void process_kill(u32 pid) { debug("Killing process %d", pid); struct process *proc = process_from_pid(pid); if (proc == PID_NOT_FOUND) panic("Can't kill unknown PID"); // flush(proc->stdout); // flush(proc->stderr); proc->state = PROC_ASLEEP; if (proc->parent != NULL) { //warn("Child had parent"); //process_wake(proc->parent->pid); } } u32 process_spawn(struct process *process) { debug("Spawning process %d", process->pid); process->next = root->next; root->next = process; process->state = PROC_RUNNING; process->parent = current_proc; //process_suspend(current_proc->pid); return process->pid; } u32 process_wait_gid(u32 gid, u32 *status) { struct process *i = root; while (i != NULL) { if (i->gid == gid) if (i->state == PROC_ASLEEP) { *status = i->registers.ebx; return i->pid; } i = i->next; } return WAIT_OKAY; } u32 process_wait_pid(u32 pid, u32 *status) { struct process *i = current_proc->next; while (i != NULL) { if (i->pid == pid) { if (i->state == PROC_ASLEEP) { *status = i->registers.ebx; return i->pid; } else { return WAIT_OKAY; } } i = i->next; } return WAIT_ERROR; } void process_suspend(u32 pid) { debug("Suspending process %d", pid); struct process *proc = process_from_pid(pid); if (proc == PID_NOT_FOUND) { warn("Couldn't find PID for suspension"); return; } proc->state = PROC_ASLEEP; } void process_wake(u32 pid) { debug("Waking process %d", pid); struct process *proc = process_from_pid(pid); if (proc == PID_NOT_FOUND) return; proc->state = PROC_RUNNING; } u32 process_child(struct process *child, u32 pid) { debug("Spawning child process %d", pid); process_suspend(pid); struct process *parent = process_from_pid(pid); if (parent == PID_NOT_FOUND) panic("Child process spawned without parent"); child->parent = parent; return process_spawn(child); } u32 process_fork(u32 pid) { warn("Fork is not implemented"); // With struct regs *regs /*struct page_directory *dir = paging_copy_user_directory(current_proc->cr3); struct process *proc = process_make_new(); proc->cr3 = dir; memcpy(&proc->registers, regs, sizeof(struct regs)); proc->registers.eax = proc->pid; proc->pid = current_proc->pid; process_spawn(proc);*/ return 0; //pid++; } struct process *process_from_pid(u32 pid) { struct process *proc = root; while (proc != NULL) { if (proc->pid == pid) return proc; proc = proc->next; } return PID_NOT_FOUND; } struct process *process_make_new() { debug("Making new process %d", pid); struct process *proc = (struct process *)kmalloc_a(sizeof(struct process)); proc->registers.cs = 0x1B; proc->registers.ds = 0x23; proc->registers.ss = 0x23; proc->cr3 = paging_make_directory(); proc->brk = 0x50000000; for (int i = 0; i < 1024; i++) proc->cr3->tables[i] = paging_root_directory->tables[i]; proc->pid = pid++; return proc; } u32 kexec(char *path) { debug("Starting kernel process %s", path); struct process *proc = elf_load(path); if (proc == NULL) return -1; // TODO: Add stdin etc support (?) proc->stdin = NULL; proc->stdout = NULL; proc->stderr = NULL; process_init(proc); return 0; } u32 uexec(char *path) { debug("Starting user process %s", path); process_suspend(current_proc->pid); struct process *proc = elf_load(path); if (proc == NULL) return -1; proc->stdin = NULL; proc->stdout = NULL; proc->stderr = NULL; proc->parent = current_proc; proc->gid = current_proc->pid; process_spawn(proc); return 0; }