diff options
Diffstat (limited to 'kernel/features/proc.c')
-rw-r--r-- | kernel/features/proc.c | 294 |
1 files changed, 154 insertions, 140 deletions
diff --git a/kernel/features/proc.c b/kernel/features/proc.c index fde49bd..a17a655 100644 --- a/kernel/features/proc.c +++ b/kernel/features/proc.c @@ -14,72 +14,58 @@ #include <str.h> #include <timer.h> -u32 current_pid = 0; -u32 quantum = 0; -struct proc *priority_proc = NULL; -struct list *proc_list = NULL; -struct node *idle_proc = NULL; -struct node *current = NULL; - -// TODO: Use less memcpy and only copy relevant registers (rewrite for efficiency argh) +#define PROC(node) ((struct proc *)node->data) + +static u8 locked = 0; +static u32 current_pid = 0; +static struct node *idle_proc = NULL; +static struct node *current = NULL; + +static struct list *proc_list_running = NULL; +static struct list *proc_list_blocked = NULL; +static struct list *proc_list_idle = NULL; + +// TODO: Use less memcpy and only copy relevant registers // TODO: 20 priority queues (https://www.kernel.org/doc/html/latest/scheduler/sched-nice-design.html) -// TODO: Optimize scheduler HOT FLATTEN void scheduler(struct regs *regs) { - if (quantum == 0) { - quantum = PROC_QUANTUM; + spinlock(&locked); + + PROC(current)->ticks++; + + if (PROC(current)->quantum.cnt >= PROC(current)->quantum.val) { + PROC(current)->quantum.cnt = 0; } else { - quantum--; + PROC(current)->quantum.cnt++; + locked = 0; return; } - assert(proc_list->head); - - if (current) - memcpy(&((struct proc *)current->data)->regs, regs, sizeof(struct regs)); + memcpy(&PROC(current)->regs, regs, sizeof(*regs)); - if (priority_proc && priority_proc->state == PROC_RUNNING) { - current = list_first_data(proc_list, priority_proc); - priority_proc = NULL; - assert(current); - } else if (current && current->next && - ((struct proc *)current->next->data)->state == PROC_RUNNING) { + if (current->next) { current = current->next; - } else if (((struct proc *)proc_list->head->data)->state == PROC_RUNNING) { - current = proc_list->head; + } else if (proc_list_running->head) { + current = proc_list_running->head; } else { current = idle_proc; - struct node *iterator = proc_list->head; - while (iterator) { - if (((struct proc *)iterator->data)->state == PROC_RUNNING) { - current = iterator; - break; - } - iterator = iterator->next; - } } - struct proc *p = current->data; - memory_switch_dir(p->page_dir); - memcpy(regs, &p->regs, sizeof(*regs)); - - if (current == idle_proc) - quantum = 0; + memory_switch_dir(PROC(current)->page_dir); + memcpy(regs, &PROC(current)->regs, sizeof(*regs)); - /* printf("{%d}", ((struct proc *)current->data)->pid); */ + locked = 0; } void proc_print(void) { - struct node *node = proc_list->head; + struct node *node = proc_list_running->head; + struct proc *proc = NULL; printf("--- PROCESSES ---\n"); - struct proc *proc = NULL; while (node && (proc = node->data)) { - printf("Process %d: %s [%s] [entry: %x; stack: %x]\n", proc->pid, proc->name, - proc->state == PROC_RUNNING ? "RUNNING" : "SLEEPING", - virtual_to_physical(proc->page_dir, proc->entry), - virtual_to_physical(proc->page_dir, proc->regs.ebp)); + printf("Process %d: %s [%s]\n", proc->pid, proc->name, + proc->state == PROC_RUNNING ? "RUNNING" : "SLEEPING"); node = node->next; } printf("\n"); @@ -103,37 +89,60 @@ u8 proc_super(void) struct proc *proc_from_pid(u32 pid) { - struct node *iterator = proc_list->head; - while (iterator != NULL) { + struct node *iterator = NULL; + + iterator = proc_list_blocked->head; + while (iterator) { if (((struct proc *)iterator->data)->pid == pid) return iterator->data; iterator = iterator->next; } + + iterator = proc_list_running->head; + while (iterator) { + if (((struct proc *)iterator->data)->pid == pid) + return iterator->data; + iterator = iterator->next; + } + return NULL; } -void proc_clear_quantum(void) +void proc_set_quantum(struct proc *proc, u32 value) { - quantum = 0; + proc->quantum.val = value; } -void proc_exit(struct proc *proc, s32 status) +void proc_reset_quantum(struct proc *proc) { - u8 found = 0; - struct node *iterator = proc_list->head; - while (iterator) { - if (iterator->data == proc) { - found = 1; - list_remove(proc_list, iterator); - break; - } - iterator = iterator->next; + proc->quantum.cnt = proc->quantum.val; +} + +void proc_state(struct proc *proc, enum proc_state state) +{ + if (state == PROC_RUNNING && !list_first_data(proc_list_running, proc)) { + assert(list_remove(proc_list_blocked, list_first_data(proc_list_blocked, proc))); + assert(list_add(proc_list_running, proc)); + } else if (state == PROC_BLOCKED && !list_first_data(proc_list_blocked, proc)) { + assert(list_remove(proc_list_running, list_first_data(proc_list_running, proc))); + assert(list_add(proc_list_blocked, proc)); } + // else: Nothing to do! +} - assert(found); +void proc_exit(struct proc *proc, struct regs *r, s32 status) +{ + struct node *running = list_first_data(proc_list_running, proc); + if (!running || !list_remove(proc_list_running, running)) { + struct node *blocked = list_first_data(proc_list_blocked, proc); + assert(blocked && list_remove(proc_list_blocked, blocked)); + // Idle procs can't be killed -> assertion failure. + } - if (memcmp(proc, current->data, sizeof(*proc)) == 0) - current = NULL; + if (current->data == proc) { + current = idle_proc; + memcpy(r, &PROC(idle_proc)->regs, sizeof(*r)); + } printf("Process %s (%d) exited with status %d (%s)\n", proc->name[0] ? proc->name : "UNKNOWN", proc->pid, status, @@ -145,40 +154,77 @@ void proc_exit(struct proc *proc, s32 status) free(proc); - proc_clear_quantum(); // TODO: Add quantum to each process struct? - - // The caller has to yield itself + if (current->data == proc) + proc_yield(r); } void proc_yield(struct regs *r) { - proc_clear_quantum(); + proc_reset_quantum(PROC(current)); scheduler(r); } -void proc_enable_waiting(u32 id, enum proc_wait_type type) +void proc_block(u32 id, enum proc_block_type type, u32 func_ptr) +{ + u8 already_exists = 0; + struct proc *p = proc_current(); + + // Check if already exists + for (u32 i = 0; i < p->block.id_cnt; i++) { + if (p->block.ids[i].id == id && p->block.ids[i].type == type) { + assert(p->block.ids[i].func_ptr == func_ptr); + already_exists = 1; + } + } + + if (already_exists) + goto end; + + assert(p->block.id_cnt + 1 < PROC_MAX_BLOCK_IDS); + + // Find slot + struct proc_block_identifier *slot = NULL; + for (u32 i = 0; i < PROC_MAX_BLOCK_IDS; i++) { + if (p->block.ids[i].magic != PROC_BLOCK_MAGIC) { + slot = &p->block.ids[i]; + break; + } + } + assert(slot); + + slot->magic = PROC_BLOCK_MAGIC; + slot->id = id; + slot->type = type; + slot->func_ptr = func_ptr; + p->block.id_cnt++; + +end: + proc_state(p, PROC_BLOCKED); +} + +void proc_unblock(u32 id, enum proc_block_type type) { struct page_dir *dir_bak; memory_backup_dir(&dir_bak); - struct proc *proc_bak = proc_current(); + struct node *proc_bak = current; if (!proc_bak) return; - struct node *iterator = proc_list->head; + struct node *iterator = proc_list_blocked->head; while (iterator) { struct proc *p = iterator->data; - struct proc_wait *w = &p->wait; + struct proc_block *w = &p->block; if (!p || !w || w->id_cnt == 0) { iterator = iterator->next; continue; } - current = list_first_data(proc_list, p); - assert(w->id_cnt < PROC_MAX_WAIT_IDS); + current = list_first_data(proc_list_blocked, p); + assert(w->id_cnt < PROC_MAX_BLOCK_IDS); for (u32 i = 0; i < w->id_cnt; i++) { - if (w->ids[i].magic == PROC_WAIT_MAGIC && w->ids[i].id == id && + if (w->ids[i].magic == PROC_BLOCK_MAGIC && w->ids[i].id == id && w->ids[i].type == type) { struct regs *r = &p->regs; u32 (*func)(u32, u32, u32, u32) = @@ -189,8 +235,8 @@ void proc_enable_waiting(u32 id, enum proc_wait_type type) memory_switch_dir(dir_bak); } memset(&w->ids[i], 0, sizeof(w->ids[i])); - p->wait.id_cnt--; - p->state = PROC_RUNNING; + p->block.id_cnt--; + proc_state(p, PROC_RUNNING); break; } } @@ -198,46 +244,8 @@ void proc_enable_waiting(u32 id, enum proc_wait_type type) iterator = iterator->next; } - if (current->data != proc_bak) - current = list_first_data(proc_list, proc_bak); -} - -void proc_wait_for(u32 id, enum proc_wait_type type, u32 func_ptr) -{ - u8 already_exists = 0; - struct proc *p = proc_current(); - - // Check if already exists - for (u32 i = 0; i < p->wait.id_cnt; i++) { - if (p->wait.ids[i].id == id && p->wait.ids[i].type == type) { - assert(p->wait.ids[i].func_ptr == func_ptr); - already_exists = 1; - } - } - - if (already_exists) - goto end; - - assert(p->wait.id_cnt + 1 < PROC_MAX_WAIT_IDS); - - // Find slot - struct proc_wait_identifier *slot = NULL; - for (u32 i = 0; i < PROC_MAX_WAIT_IDS; i++) { - if (p->wait.ids[i].magic != PROC_WAIT_MAGIC) { - slot = &p->wait.ids[i]; - break; - } - } - assert(slot != NULL); - - slot->magic = PROC_WAIT_MAGIC; - slot->id = id; - slot->type = type; - slot->func_ptr = func_ptr; - p->wait.id_cnt++; - -end: - p->state = PROC_SLEEPING; + if (current != proc_bak) + current = proc_bak; } struct proc *proc_make(enum proc_priv priv) @@ -249,6 +257,8 @@ struct proc *proc_make(enum proc_priv priv) proc->memory = list_new(); proc->state = PROC_RUNNING; proc->page_dir = virtual_create_dir(); + proc->quantum.val = PROC_QUANTUM; + proc->quantum.cnt = 0; // Init regs u8 is_kernel = priv == PROC_PRIV_KERNEL; @@ -262,8 +272,7 @@ struct proc *proc_make(enum proc_priv priv) proc->regs.cs = code; proc->regs.eflags = EFLAGS_ALWAYS | EFLAGS_INTERRUPTS; - if (current) - list_add(proc_list, proc); + list_add(proc_list_running, proc); return proc; } @@ -339,7 +348,7 @@ static res procfs_write(const char *path, void *buf, u32 offset, u32 count, stru msg->data = msg_data; msg->size = count; stack_push_bot(p->messages, msg); // TODO: Use offset - proc_enable_waiting(pid, PROC_WAIT_MSG); + proc_unblock(pid, PROC_BLOCK_MSG); return count; } else if (!memcmp(path, "io/", 3)) { path += 3; @@ -355,7 +364,7 @@ static res procfs_write(const char *path, void *buf, u32 offset, u32 count, stru assert(stream->offset_write + count < STREAM_MAX_SIZE); // TODO: Resize memcpy((char *)(stream->data + stream->offset_write), buf, count); stream->offset_write += count; - proc_enable_waiting(dev->id, PROC_WAIT_DEV); + proc_unblock(dev->id, PROC_BLOCK_DEV); return count; } } @@ -414,7 +423,7 @@ static res procfs_read(const char *path, void *buf, u32 offset, u32 count, struc return -ENOENT; } -static res procfs_wait(const char *path, u32 func_ptr, struct device *dev) +static res procfs_block(const char *path, u32 func_ptr, struct device *dev) { u32 pid = 0; procfs_parse_path(&path, &pid); @@ -426,10 +435,10 @@ static res procfs_wait(const char *path, u32 func_ptr, struct device *dev) path++; if (!memcmp(path, "msg", 4)) { - proc_wait_for(pid, PROC_WAIT_MSG, func_ptr); + proc_block(pid, PROC_BLOCK_MSG, func_ptr); return EOK; } else { - proc_wait_for(dev->id, PROC_WAIT_DEV, func_ptr); + proc_block(dev->id, PROC_BLOCK_DEV, func_ptr); return EOK; } } @@ -479,21 +488,23 @@ static res procfs_ready(const char *path, struct device *dev) extern void proc_jump_userspace(void); u32 _esp, _eip; -void proc_init(void) +NORETURN void proc_init(void) { - if (proc_list) - return; + if (proc_list_running) + panic("Already initialized processes!"); cli(); scheduler_enable(); - proc_list = list_new(); + proc_list_running = list_new(); + proc_list_blocked = list_new(); + proc_list_idle = list_new(); // Procfs struct vfs *vfs = zalloc(sizeof(*vfs)); vfs->type = VFS_PROCFS; vfs->read = procfs_read; vfs->write = procfs_write; - vfs->wait = procfs_wait; + vfs->block = procfs_block; vfs->perm = procfs_perm; vfs->ready = procfs_ready; vfs->data = NULL; @@ -502,29 +513,32 @@ void proc_init(void) dev->type = DEV_CHAR; dev->vfs = vfs; device_add(dev); - assert(vfs_mount(dev, "/proc/") == 0); + assert(vfs_mount(dev, "/proc/") == EOK); // Idle proc struct proc *kernel_proc = proc_make(PROC_PRIV_KERNEL); - assert(elf_load("/bin/idle", kernel_proc) == 0); - kernel_proc->state = PROC_SLEEPING; - idle_proc = list_add(proc_list, kernel_proc); + assert(elf_load("/bin/idle", kernel_proc) == EOK); + kernel_proc->state = PROC_BLOCKED; + kernel_proc->quantum.val = 0; + kernel_proc->quantum.cnt = 0; + idle_proc = list_add(proc_list_idle, kernel_proc); + list_remove(proc_list_running, list_first_data(proc_list_running, kernel_proc)); // Init proc (root) - struct node *new = list_add(proc_list, proc_make(PROC_PRIV_ROOT)); - assert(elf_load("/bin/init", new->data) == 0); - current = new; - proc_stack_push(new->data, 0); + struct proc *init = proc_make(PROC_PRIV_ROOT); + assert(elf_load("/bin/init", init) == EOK); + proc_stack_push(init, 0); + current = list_first_data(proc_list_running, init); - _eip = ((struct proc *)new->data)->regs.eip; - _esp = ((struct proc *)new->data)->regs.useresp; + _eip = init->regs.eip; + _esp = init->regs.useresp; - memory_switch_dir(((struct proc *)new->data)->page_dir); + memory_switch_dir(init->page_dir); printf("Jumping to userspace!\n"); + // You're waiting for a train. A train that will take you far away... proc_jump_userspace(); - while (1) { - }; + panic("Returned from limbo!\n"); } |