#include #include #include #include #include #include #include #include #include #include #include #define SYSCALL_RETURN(val) do { \ assert(process_current->state == PS_RUNNING); \ return regs_savereturn(&process_current->regs, val); \ } while (0) _Noreturn void _syscall_exit(int ret) { process_kill(process_current, ret); process_switch_any(); } int _syscall_await(void) { bool has_children = false; process_transition(process_current, PS_WAITS4CHILDDEATH); for (struct process *iter = process_current->child; iter; iter = iter->sibling) { if (iter->noreap) continue; has_children = true; if (iter->state == PS_DEAD) // TODO this path crashes SYSCALL_RETURN(process_try2collect(iter)); } if (!has_children) { process_transition(process_current, PS_RUNNING); SYSCALL_RETURN(~0); // TODO errno } return -1; // dummy } int _syscall_fork(int flags, handle_t __user *fs_front) { struct process *child; handle_t front; if ((flags & FORK_NEWFS) && fs_front) { /* we'll need to return a handle, check if that's possible */ front = process_find_free_handle(process_current, 1); if (front < 0) SYSCALL_RETURN(-1); } child = process_fork(process_current, flags); regs_savereturn(&child->regs, 0); if ((flags & FORK_NEWFS) && fs_front) { struct vfs_backend *backend = kmalloc(sizeof *backend); process_current->handles[front] = handle_init(HANDLE_FS_FRONT); backend->heap = true; backend->is_user = true; backend->potential_handlers = 1; backend->refcount = 2; // child + handle backend->user.handler = NULL; backend->queue = NULL; child->controlled = backend; process_current->handles[front]->backend = backend; if (fs_front) { /* failure ignored. if you pass an invalid pointer to this function, * you just don't receive the handle. you'll probably segfault * trying to access it anyways */ virt_cpy_to(process_current->pages, fs_front, &front, sizeof front); } } SYSCALL_RETURN(1); } handle_t _syscall_open(const char __user *path, int len, int flags) { struct vfs_mount *mount; char *path_buf = NULL; if (PATH_MAX < len) SYSCALL_RETURN(-1); if (process_find_free_handle(process_current, 0) < 0) SYSCALL_RETURN(-1); path_buf = kmalloc(len); if (!path_buf) goto fail; if (!virt_cpy_from(process_current->pages, path_buf, path, len)) goto fail; len = path_simplify(path_buf, path_buf, len); if (len < 0) goto fail; mount = vfs_mount_resolve(process_current->mount, path_buf, len); if (!mount) goto fail; if (mount->prefix_len > 0) { // strip prefix len -= mount->prefix_len; // i can't just adjust path_buf, because it needs to be passed to free() // later on memcpy(path_buf, path_buf + mount->prefix_len, len); } vfsreq_create((struct vfs_request) { .type = VFSOP_OPEN, .input = { .kern = true, .buf_kern = path_buf, .len = len, }, .caller = process_current, .backend = mount->backend, .flags = flags, }); return -1; // dummy fail: kfree(path_buf); SYSCALL_RETURN(-1); } int _syscall_mount(handle_t hid, const char __user *path, int len) { struct vfs_mount *mount = NULL; struct vfs_backend *backend = NULL; char *path_buf = NULL; if (PATH_MAX < len) SYSCALL_RETURN(-1); path_buf = kmalloc(len); if (!path_buf) goto fail; if (!virt_cpy_from(process_current->pages, path_buf, path, len)) goto fail; len = path_simplify(path_buf, path_buf, len); if (len < 0) goto fail; // remove trailing slash // mounting something under `/this` and `/this/` should be equalivent if (path_buf[len - 1] == '/') { if (len == 0) goto fail; len--; } if (hid >= 0) { // mounting a real backend? struct handle *handle = process_handle_get(process_current, hid, HANDLE_FS_FRONT); if (!handle) goto fail; backend = handle->backend; backend->refcount++; } // otherwise backend == NULL // append to mount list // TODO move to kernel/vfs/mount.c mount = kmalloc(sizeof *mount); mount->prev = process_current->mount; mount->prefix = path_buf; mount->prefix_owned = true; mount->prefix_len = len; mount->backend = backend; mount->refs = 1; process_current->mount = mount; kmalloc_sanity(mount); kmalloc_sanity(mount->prefix); SYSCALL_RETURN(0); fail: kfree(path_buf); kfree(mount); SYSCALL_RETURN(-1); } int _syscall_read(handle_t handle_num, void __user *buf, size_t len, int offset) { struct handle *h; // TODO get rid of the type argument in process_handle_get if ((h = process_handle_get(process_current, handle_num, HANDLE_FILE))) { vfsreq_create((struct vfs_request) { .type = VFSOP_READ, .output = { .buf = (userptr_t) buf, .len = len, }, .id = h->file_id, .offset = offset, .caller = process_current, .backend = h->backend, }); } else if ((h = process_handle_get(process_current, handle_num, HANDLE_PIPE))) { pipe_joinqueue(h, true, process_current, buf, len); pipe_trytransfer(h); } else { SYSCALL_RETURN(-1); } return -1; // dummy } int _syscall_write(handle_t handle_num, const void __user *buf, size_t len, int offset) { struct handle *h; if ((h = process_handle_get(process_current, handle_num, HANDLE_FILE))) { vfsreq_create((struct vfs_request) { .type = VFSOP_WRITE, .input = { .buf = (userptr_t) buf, .len = len, }, .id = h->file_id, .offset = offset, .caller = process_current, .backend = h->backend, }); } else if ((h = process_handle_get(process_current, handle_num, HANDLE_PIPE))) { pipe_joinqueue(h, false, process_current, (void __user *)buf, len); pipe_trytransfer(h); } else { SYSCALL_RETURN(-1); } return -1; // dummy } int _syscall_close(handle_t hid) { if (hid < 0 || hid >= HANDLE_MAX) return -1; struct handle **h = &process_current->handles[hid]; if (!*h) SYSCALL_RETURN(-1); handle_close(*h); *h = NULL; SYSCALL_RETURN(0); } int _syscall_fs_wait(char __user *buf, int max_len, struct fs_wait_response __user *res) { struct vfs_backend *backend = process_current->controlled; if (!backend) SYSCALL_RETURN(-1); process_transition(process_current, PS_WAITS4REQUEST); assert(!backend->user.handler); // TODO allow multiple processes to wait on the same backend backend->user.handler = process_current; /* checking the validity of those pointers here would make * vfs_backend_accept simpler. TODO? */ process_current->awaited_req.buf = buf; process_current->awaited_req.max_len = max_len; process_current->awaited_req.res = res; vfs_backend_tryaccept(backend); // sets return value return -1; // dummy } int _syscall_fs_respond(char __user *buf, int ret, int flags) { struct vfs_request *req = process_current->handled_req; if (!req) SYSCALL_RETURN(-1); if (req->output.len > 0 && ret > 0) { // if this vfsop outputs data and ret is positive, it's the length of the buffer // TODO document ret = min(ret, capped_cast32(req->output.len)); if (!virt_cpy(req->caller->pages, req->output.buf, process_current->pages, buf, ret)) { // how should this error even be handled? TODO } } process_current->handled_req = NULL; vfsreq_finish(req, ret, flags, process_current); SYSCALL_RETURN(0); } void __user *_syscall_memflag(void __user *addr, size_t len, int flags) { struct pagedir *pages = process_current->pages; void *phys; addr = (userptr_t)((int __force)addr & ~PAGE_MASK); // align to page boundary if (flags & MEMFLAG_FINDFREE) { addr = pagedir_findfree(pages, addr, len); if (!(flags & MEMFLAG_PRESENT)) goto ret; } for (userptr_t iter = addr; iter < addr + len; iter += PAGE_SIZE) { if (pagedir_iskern(pages, iter)) { // TODO reflect failure in return value continue; } phys = pagedir_virt2phys(pages, iter, false, false); if (!(flags & MEMFLAG_PRESENT)) { if (phys) page_free(pagedir_unmap(pages, iter), 1); continue; } if (!phys) { phys = page_alloc(1); memset(phys, 0, PAGE_SIZE); // TODO somehow test this pagedir_map(pages, iter, phys, true, true); } } ret: // the macro is too stupid to handle returning pointers assert(process_current->state == PS_RUNNING); // TODO move to regs_savereturn regs_savereturn(&process_current->regs, (uintptr_t)addr); return addr; } handle_t _syscall_pipe(int flags) { if (flags) return -1; handle_t h = process_find_free_handle(process_current, 0); if (h < 0) return -1; process_current->handles[h] = handle_init(HANDLE_PIPE); assert(process_current->handles[h]->pipe.reader == NULL); assert(process_current->handles[h]->pipe.writer == NULL); SYSCALL_RETURN(h); } void _syscall_debug_klog(const void __user *buf, size_t len) { (void)buf; (void)len; // static char kbuf[256]; // if (len >= sizeof(kbuf)) len = sizeof(kbuf) - 1; // virt_cpy_from(process_current->pages, kbuf, buf, len); // kbuf[len] = '\0'; // kprintf("[klog] %x\t%s\n", process_current->id, kbuf); } int _syscall(int num, int a, int b, int c, int d) { switch (num) { case _SYSCALL_EXIT: _syscall_exit(a); // _syscall_exit doesn't exit case _SYSCALL_AWAIT: _syscall_await(); break; case _SYSCALL_FORK: _syscall_fork(a, (userptr_t)b); break; case _SYSCALL_OPEN: _syscall_open((userptr_t)a, b, c); break; case _SYSCALL_MOUNT: _syscall_mount(a, (userptr_t)b, c); break; case _SYSCALL_READ: _syscall_read(a, (userptr_t)b, c, d); break; case _SYSCALL_WRITE: _syscall_write(a, (userptr_t)b, c, d); break; case _SYSCALL_CLOSE: _syscall_close(a); break; case _SYSCALL_FS_WAIT: _syscall_fs_wait((userptr_t)a, b, (userptr_t)c); break; case _SYSCALL_FS_RESPOND: _syscall_fs_respond((userptr_t)a, b, c); break; case _SYSCALL_MEMFLAG: _syscall_memflag((userptr_t)a, b, c); break; case _SYSCALL_PIPE: _syscall_pipe(a); break; case _SYSCALL_DEBUG_KLOG: _syscall_debug_klog((userptr_t)a, b); break; default: regs_savereturn(&process_current->regs, -1); break; } /* return value is unused. execution continues in sysenter_stage2 */ return -1; }