diff options
Diffstat (limited to 'kernel/features/fs.c')
-rw-r--r-- | kernel/features/fs.c | 189 |
1 files changed, 112 insertions, 77 deletions
diff --git a/kernel/features/fs.c b/kernel/features/fs.c index c692aac..56d253a 100644 --- a/kernel/features/fs.c +++ b/kernel/features/fs.c @@ -2,6 +2,7 @@ #include <assert.h> #include <def.h> +#include <errno.h> #include <fs.h> #include <ide.h> #include <mem.h> @@ -107,9 +108,14 @@ static void vfs_list_mounts() s32 vfs_mount(struct device *dev, const char *path) { - // TODO: Check if already mounted - if (!dev || !dev->id || vfs_mounted(dev, path)) - return -1; + if (!path || !memory_valid(path)) + return -EFAULT; + + if (!dev || !memory_valid(dev) || !dev->id) + return -EFAULT; + + if (vfs_mounted(dev, path)) + return -EBUSY; char *fixed = vfs_normalize_path(path); @@ -124,25 +130,31 @@ 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() ? proc_current()->name : "Unknown", path); */ - if (!count) - return 0; + if (!path || !memory_valid(path)) + return -EFAULT; - if (offset > count || !buf) - return -1; + if (!buf || !memory_valid(buf)) + return -EFAULT; - while (*path == ' ') - path++; + if (offset > count) + return -EINVAL; struct mount_info *m = vfs_find_mount_info(path); - if (!(m && m->dev && m->dev->vfs && m->dev->vfs->read && m->dev->vfs->perm)) - return -1; + if (!m || !m->dev || !m->dev->vfs) + return -ENOENT; + + if (!m->dev->vfs->read || !m->dev->vfs->perm) + return -EINVAL; u32 len = strlen(m->path); if (len > 1) path += len; if (!m->dev->vfs->perm(path, VFS_READ, m->dev) && !proc_super()) - return -1; + return -EACCES; + + if (!count) + return 0; return m->dev->vfs->read(path, buf, offset, count, m->dev); } @@ -150,83 +162,95 @@ 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() ? proc_current()->name : "Unknown", path); */ - if (!count) - return 0; + if (!path || !memory_valid(path)) + return -EFAULT; - if (offset > count || !buf) - return -1; + if (!buf || !memory_valid(buf)) + return -EFAULT; - while (*path == ' ') - path++; + if (offset > count) + return -EINVAL; struct mount_info *m = vfs_find_mount_info(path); - if (!(m && m->dev && m->dev->vfs && m->dev->vfs->write && m->dev->vfs->perm)) - return -1; + if (!m || !m->dev || !m->dev->vfs) + return -ENOENT; + + if (!m->dev->vfs->write || !m->dev->vfs->perm) + return -EINVAL; u32 len = strlen(m->path); if (len > 1) path += len; if (!m->dev->vfs->perm(path, VFS_WRITE, m->dev) && !proc_super()) - return -1; + return -EACCES; + + if (!count) + return 0; return m->dev->vfs->write(path, buf, offset, count, m->dev); } s32 vfs_ioctl(const char *path, u32 request, void *arg1, void *arg2, void *arg3) { - while (*path == ' ') - path++; + if (!path || !memory_valid(path)) + return -EFAULT; struct mount_info *m = vfs_find_mount_info(path); - if (!(m && m->dev && m->dev->vfs && m->dev->vfs->ioctl && m->dev->vfs->perm)) - return -1; + if (!m || !m->dev || !m->dev->vfs) + return -ENOENT; + + if (!m->dev->vfs->ioctl || !m->dev->vfs->perm) + return -EINVAL; u32 len = strlen(m->path); if (len > 1) path += len; if (!m->dev->vfs->perm(path, VFS_WRITE, m->dev) && !proc_super()) - return -1; + return -EACCES; return m->dev->vfs->ioctl(path, request, arg1, arg2, arg3, m->dev); } s32 vfs_stat(const char *path, struct stat *buf) { - while (*path == ' ') - path++; + if (!path || !memory_valid(path)) + return -EFAULT; - if (!buf) - return -1; + if (!buf || !memory_valid(buf)) + return -EFAULT; struct mount_info *m = vfs_find_mount_info(path); - if (!(m && m->dev && m->dev->vfs && m->dev->vfs->stat && m->dev->vfs->perm)) - return -1; + if (!m || !m->dev || !m->dev->vfs) + return -ENOENT; + + if (!m->dev->vfs->stat || !m->dev->vfs->perm) + return -EINVAL; u32 len = strlen(m->path); if (len > 1) path += len; - if (!m->dev->vfs->perm(path, VFS_WRITE, m->dev) && !proc_super()) - return -1; + if (!m->dev->vfs->perm(path, VFS_READ, m->dev) && !proc_super()) + return -EACCES; return m->dev->vfs->stat(path, buf, m->dev); } s32 vfs_wait(const char *path, u32 func_ptr) { - while (*path == ' ') - path++; + if (!path || !func_ptr || !memory_valid(path)) + return -EFAULT; struct mount_info *m = vfs_find_mount_info(path); - if (!(m && m->dev && m->dev->vfs)) - return -1; + if (!m || !m->dev || !m->dev->vfs) + return -ENOENT; // Default wait if (!m->dev->vfs->wait) { proc_wait_for(vfs_find_dev(path)->id, PROC_WAIT_DEV, func_ptr); - return 1; + return 0; } u32 len = strlen(m->path); @@ -238,26 +262,34 @@ s32 vfs_wait(const char *path, u32 func_ptr) s32 vfs_poll(const char **files) { - if (!files) - return -1; + if (!files || !memory_valid(files)) + return -EFAULT; - for (const char **p = files; memory_user_valid((u32)*p) && *p && **p; p++) - if (vfs_ready(*p)) + for (const char **p = files; *p && memory_valid(*p) && **p; p++) { + s32 ready = vfs_ready(*p); + if (ready == 1) return p - files; + else if (ready < 0) + return ready; + } - for (const char **p = files; memory_user_valid((u32)*p) && *p && **p; p++) + for (const char **p = files; *p && memory_valid(*p) && **p; p++) vfs_wait(*p, (u32)vfs_poll); return PROC_MAX_WAIT_IDS + 1; } -u8 vfs_ready(const char *path) +s32 vfs_ready(const char *path) { - while (*path == ' ') - path++; + if (!path || !memory_valid(path)) + return -EFAULT; struct mount_info *m = vfs_find_mount_info(path); - assert(m && m->dev && m->dev->vfs && m->dev->vfs->ready); + if (!m || !m->dev || !m->dev->vfs) + return -ENOENT; + + if (!m->dev->vfs->ready) + return -EINVAL; u32 len = strlen(m->path); if (len > 1) @@ -308,8 +340,10 @@ struct device *device_get_by_name(const char *name) static s32 devfs_read(const char *path, void *buf, u32 offset, u32 count, struct device *dev) { struct device *target = device_get_by_name(path + 1); - if (!target || !target->read) - return -1; + if (!target) + return -ENOENT; + if (!target->read) + return -EINVAL; return target->read(buf, offset, count, dev); } @@ -317,26 +351,30 @@ static s32 devfs_ioctl(const char *path, u32 request, void *arg1, void *arg2, vo struct device *dev) { struct device *target = device_get_by_name(path + 1); - if (!target || !target->ioctl) - return -1; + if (!target) + return -ENOENT; + if (!target->ioctl) + return -EINVAL; return target->ioctl(request, arg1, arg2, arg3, dev); } -static u8 devfs_perm(const char *path, enum vfs_perm perm, struct device *dev) +static s32 devfs_perm(const char *path, enum vfs_perm perm, struct device *dev) { - (void)path; - (void)perm; - (void)dev; + UNUSED(path); + UNUSED(perm); + UNUSED(dev); return 1; } -static u8 devfs_ready(const char *path, struct device *dev) +static s32 devfs_ready(const char *path, struct device *dev) { - (void)dev; + UNUSED(dev); struct device *target = device_get_by_name(path + 1); - if (!target || !target->ready) - return 0; + if (!target) + return -ENOENT; + if (!target->ready) + return -EINVAL; return target->ready(); } @@ -367,7 +405,7 @@ void device_install(void) // TODO: Remove malloc from buffer_read (attempt in #56cd63f199) static void *buffer_read(u32 block, struct device *dev) { - void *buf = malloc(BLOCK_SIZE); + void *buf = zalloc(BLOCK_SIZE); dev->read(buf, block * SECTOR_COUNT, SECTOR_COUNT, dev); return buf; } @@ -448,15 +486,15 @@ static u32 read_indirect(u32 indirect, u32 block_num, struct device *dev) static s32 read_inode(struct ext2_inode *in, void *buf, u32 offset, u32 count, struct device *dev) { // TODO: Support read offset - (void)offset; + UNUSED(offset); if (!in || !buf) - return -1; + return -EINVAL; u32 num_blocks = in->blocks / (BLOCK_SIZE / SECTOR_SIZE); if (!num_blocks) - return -1; + return -EINVAL; // TODO: memcpy block chunks until count is copied while (BLOCK_SIZE * num_blocks > count) @@ -538,7 +576,7 @@ static struct ext2_inode *find_inode_by_path(const char *path, struct ext2_inode struct device *dev) { if (path[0] != '/') - return 0; + return NULL; char *path_cp = strdup(path); char *init = path_cp; // For freeing @@ -560,7 +598,7 @@ static struct ext2_inode *find_inode_by_path(const char *path, struct ext2_inode if (current_inode == 0) { free(init); - return 0; + return NULL; } path_cp += i + 1; @@ -569,7 +607,7 @@ static struct ext2_inode *find_inode_by_path(const char *path, struct ext2_inode u32 inode = find_inode(path_cp, current_inode, dev); free(init); if ((signed)inode <= 0) - return 0; + return NULL; return get_inode(inode, in_buf, dev); } @@ -580,17 +618,14 @@ s32 ext2_read(const char *path, void *buf, u32 offset, u32 count, struct device if (find_inode_by_path(path, &in, dev) == &in) return read_inode(&in, buf, offset, count, dev); else - return -1; + return -ENOENT; } s32 ext2_stat(const char *path, struct stat *buf, struct device *dev) { - if (!buf) - return -1; - struct ext2_inode in = { 0 }; if (find_inode_by_path(path, &in, dev) != &in) - return -1; + return -ENOENT; u32 num_blocks = in.blocks / (BLOCK_SIZE / SECTOR_SIZE); u32 sz = BLOCK_SIZE * num_blocks; @@ -601,11 +636,11 @@ s32 ext2_stat(const char *path, struct stat *buf, struct device *dev) return 0; } -u8 ext2_perm(const char *path, enum vfs_perm perm, struct device *dev) +s32 ext2_perm(const char *path, enum vfs_perm perm, struct device *dev) { struct ext2_inode in = { 0 }; if (find_inode_by_path(path, &in, dev) != &in) - return 0; + return -ENOENT; switch (perm) { case VFS_EXEC: @@ -619,9 +654,9 @@ u8 ext2_perm(const char *path, enum vfs_perm perm, struct device *dev) } } -u8 ext2_ready(const char *path, struct device *dev) +s32 ext2_ready(const char *path, struct device *dev) { - (void)path; - (void)dev; + UNUSED(path); + UNUSED(dev); return 1; } |