diff options
author | Marvin Borner | 2021-02-14 17:07:29 +0100 |
---|---|---|
committer | Marvin Borner | 2021-02-14 17:07:29 +0100 |
commit | 1287f9dfe987f0456e4fb0741385d5f0278ef53b (patch) | |
tree | 64d3da484a4d186a725779f20d339432d51f441a /kernel/features | |
parent | bc4e62f629a392e1a4cf204665e91c57f4e619b2 (diff) |
Message waiting and more!
Diffstat (limited to 'kernel/features')
-rw-r--r-- | kernel/features/fs.c | 36 | ||||
-rw-r--r-- | kernel/features/proc.c | 98 | ||||
-rw-r--r-- | kernel/features/syscall.c | 6 |
3 files changed, 105 insertions, 35 deletions
diff --git a/kernel/features/fs.c b/kernel/features/fs.c index 16a12f7..1744b51 100644 --- a/kernel/features/fs.c +++ b/kernel/features/fs.c @@ -120,10 +120,11 @@ s32 vfs_mount(struct device *dev, const char *path) s32 vfs_read(const char *path, void *buf, u32 offset, u32 count) { + /* printf("%s READ: %s\n", proc_current()->name, path); */ if (!count) return 0; - if (offset > count) + if (offset > count || !buf) return -1; while (*path == ' ') @@ -144,10 +145,11 @@ s32 vfs_read(const char *path, void *buf, u32 offset, u32 count) s32 vfs_write(const char *path, void *buf, u32 offset, u32 count) { + /* printf("%s WRITE: %s\n", proc_current()->name, path); */ if (!count) return 0; - if (offset > count) + if (offset > count || !buf) return -1; while (*path == ' ') @@ -171,6 +173,9 @@ s32 vfs_stat(const char *path, struct stat *buf) while (*path == ' ') path++; + if (!buf) + return -1; + struct mount_info *m = vfs_find_mount_info(path); assert(m && m->dev && m->dev->vfs && m->dev->vfs->stat); @@ -181,6 +186,27 @@ s32 vfs_stat(const char *path, struct stat *buf) return m->dev->vfs->stat(path, buf, m->dev); } +s32 vfs_wait(const char *path, s32 (*func)()) +{ + while (*path == ' ') + path++; + + struct mount_info *m = vfs_find_mount_info(path); + assert(m && m->dev && m->dev->vfs); + + // Default wait + if (!m->dev->vfs->wait) { + proc_wait_for(vfs_find_dev(path)->id, PROC_WAIT_DEV, func); + return 1; + } + + u32 len = strlen(m->path); + if (len > 1) + path += len; + + return m->dev->vfs->wait(path, func, m->dev); +} + s32 vfs_poll(const char **files) { if (!files) @@ -191,7 +217,7 @@ s32 vfs_poll(const char **files) return p - files; for (const char **p = files; *p && **p; p++) - proc_wait_for(vfs_find_dev(*p)->id, PROC_WAIT_DEV, vfs_poll); + vfs_wait(*p, vfs_poll); return PROC_MAX_WAIT_IDS + 1; } @@ -280,12 +306,12 @@ void device_install(void) { devices = list_new(); - struct vfs *vfs = malloc(sizeof(*vfs)); + struct vfs *vfs = zalloc(sizeof(*vfs)); vfs->type = VFS_DEVFS; vfs->read = devfs_read; vfs->perm = devfs_perm; vfs->ready = devfs_ready; - struct device *dev = malloc(sizeof(*dev)); + struct device *dev = zalloc(sizeof(*dev)); dev->name = "dev"; dev->type = DEV_CHAR; dev->vfs = vfs; diff --git a/kernel/features/proc.c b/kernel/features/proc.c index 9c7fcdc..fdf509c 100644 --- a/kernel/features/proc.c +++ b/kernel/features/proc.c @@ -164,30 +164,27 @@ void proc_enable_waiting(u32 id, enum proc_wait_type type) struct proc *p = iterator->data; struct proc_wait *w = &p->wait; - if (!p || !w || w->id_cnt == 0 || w->type != type) { + if (!p || !w || w->id_cnt == 0) { iterator = iterator->next; continue; } - u8 dispatched = 0; - current = list_first_data(proc_list, p); - for (u32 i = 0; i < PROC_MAX_WAIT_IDS; i++) { - if (w->ids[i] == id) { + assert(w->id_cnt < PROC_MAX_WAIT_IDS); + for (u32 i = 0; i < w->id_cnt; i++) { + if (w->ids[i].magic == PROC_WAIT_MAGIC && w->ids[i].id == id && + w->ids[i].type == type) { struct regs *r = &p->regs; - if (w->func) - r->eax = (u32)w->func((char *)r->ebx, (void *)r->ecx, - r->edx, r->esi); - w->ids[i] = 0; + if (w->ids[i].func) + r->eax = (u32)w->ids[i].func((char *)r->ebx, (void *)r->ecx, + r->edx, r->esi); + memset(&w->ids[i], 0, sizeof(w->ids[i])); + p->wait.id_cnt--; p->state = PROC_RUNNING; - dispatched = 1; break; } } - if (dispatched) - memset(&p->wait, 0, sizeof(p->wait)); - iterator = iterator->next; } @@ -197,25 +194,45 @@ void proc_enable_waiting(u32 id, enum proc_wait_type type) void proc_wait_for(u32 id, enum proc_wait_type type, s32 (*func)()) { + u8 already_exists = 0; struct proc *p = proc_current(); - if (p->wait.id_cnt > 0) { - p->wait.ids[p->wait.id_cnt++] = id; - assert(func == p->wait.func && type == p->wait.type); - } else { - p->wait.type = type; - p->wait.id_cnt = 1; - p->wait.ids[0] = id; - p->wait.func = func; + // 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 == func); + 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 = func; + p->wait.id_cnt++; +end: p->state = PROC_SLEEPING; } struct proc *proc_make(void) { - struct proc *proc = malloc(sizeof(*proc)); - memset(proc, 0, sizeof(*proc)); + struct proc *proc = zalloc(sizeof(*proc)); proc->pid = current_pid++; proc->super = 0; proc->messages = stack_new(); @@ -274,7 +291,7 @@ s32 procfs_write(const char *path, void *buf, u32 offset, u32 count, struct devi path++; if (!memcmp(path, "msg", 4)) { stack_push_bot(p->messages, buf); // TODO: Use offset and count - proc_enable_waiting(dev->id, PROC_WAIT_DEV); // TODO: Better wakeup solution + proc_enable_waiting(pid, PROC_WAIT_MSG); return count; } else if (!memcmp(path, "io/", 3)) { path += 3; @@ -324,10 +341,11 @@ s32 procfs_read(const char *path, void *buf, u32 offset, u32 count, struct devic return count; } else if (!memcmp(path, "msg", 4)) { if (stack_empty(p->messages)) { - return 0; + return 0; // This shouldn't happen } else { u8 *msg = stack_pop(p->messages); - printf("Pop: %s\n", msg); + if (!msg) + return -1; memcpy(buf, msg + offset, count); return count; } @@ -346,6 +364,29 @@ s32 procfs_read(const char *path, void *buf, u32 offset, u32 count, struct devic return -1; } +s32 procfs_wait(const char *path, s32 (*func)(), struct device *dev) +{ + u32 pid = 0; + procfs_parse_path(&path, &pid); + + if (pid) { + struct proc *p = proc_from_pid(pid); + if (!p || path[0] != '/') + return -1; + + path++; + if (!memcmp(path, "msg", 4)) { + proc_wait_for(pid, PROC_WAIT_MSG, func); + return 1; + } else { + proc_wait_for(dev->id, PROC_WAIT_DEV, func); + return 1; + } + } + + return -1; +} + u8 procfs_perm(const char *path, enum vfs_perm perm, struct device *dev) { (void)path; @@ -398,14 +439,15 @@ void proc_init(void) proc_list = list_new(); // Procfs - struct vfs *vfs = malloc(sizeof(*vfs)); + struct vfs *vfs = zalloc(sizeof(*vfs)); vfs->type = VFS_PROCFS; vfs->read = procfs_read; vfs->write = procfs_write; + vfs->wait = procfs_wait; vfs->perm = procfs_perm; vfs->ready = procfs_ready; vfs->data = NULL; - struct device *dev = malloc(sizeof(*dev)); + struct device *dev = zalloc(sizeof(*dev)); dev->name = "proc"; dev->type = DEV_CHAR; dev->vfs = vfs; diff --git a/kernel/features/syscall.c b/kernel/features/syscall.c index 80c3e9e..9f05471 100644 --- a/kernel/features/syscall.c +++ b/kernel/features/syscall.c @@ -40,8 +40,10 @@ void syscall_handler(struct regs *r) if (vfs_ready((char *)r->ebx)) { r->eax = (u32)vfs_read((char *)r->ebx, (void *)r->ecx, r->edx, r->esi); } else { - proc_wait_for(vfs_find_dev((char *)r->ebx)->id, PROC_WAIT_DEV, vfs_read); - proc_yield(r); + if (vfs_wait((char *)r->ebx, vfs_read) < 0) + r->eax = -1; + else + proc_yield(r); } break; } |