X-Git-Url: https://scm.lunaixsky.com/lunaix-os.git/blobdiff_plain/21ffd87b228fa0bd0c25a09b710c6aeac8b81281..96e23fa3c6eabf8a6efebac24b740c5d4a2a1050:/lunaix-os/kernel/fs/mount.c diff --git a/lunaix-os/kernel/fs/mount.c b/lunaix-os/kernel/fs/mount.c index 3d4b0be..486d695 100644 --- a/lunaix-os/kernel/fs/mount.c +++ b/lunaix-os/kernel/fs/mount.c @@ -1,3 +1,4 @@ +#include #include #include #include @@ -89,13 +90,27 @@ mnt_chillax(struct v_mount* mnt) } int -vfs_mount(const char* target, const char* fs_name, struct device* device) +vfs_mount_root(const char* fs_name, struct device* device) +{ + int errno = 0; + if (vfs_sysroot->mnt && (errno = vfs_unmount_at(vfs_sysroot))) { + return errno; + } + return vfs_mount_at(fs_name, device, vfs_sysroot, 0); +} + +int +vfs_mount(const char* target, + const char* fs_name, + struct device* device, + int options) { int errno; struct v_dnode* mnt; - if (!(errno = vfs_walk(__current->cwd, target, &mnt, NULL, 0))) { - errno = vfs_mount_at(fs_name, device, mnt); + if (!(errno = + vfs_walk(__current->cwd, target, &mnt, NULL, VFS_WALK_MKPARENT))) { + errno = vfs_mount_at(fs_name, device, mnt, options); } return errno; @@ -117,7 +132,8 @@ vfs_unmount(const char* target) int vfs_mount_at(const char* fs_name, struct device* device, - struct v_dnode* mnt_point) + struct v_dnode* mnt_point, + int options) { if (mnt_point->inode && !(mnt_point->inode->itype & VFS_IFDIR)) { return ENOTDIR; @@ -134,17 +150,25 @@ vfs_mount_at(const char* fs_name, int errno = 0; if (!(errno = fs->mount(sb, mnt_point))) { + mnt_point->super_block = sb; sb->fs = fs; sb->root = mnt_point; - mnt_point->super_block = sb; if (!(mnt_point->mnt = vfs_create_mount(parent_mnt, mnt_point))) { errno = ENOMEM; - vfs_sb_free(sb); + goto cleanup; } + + mnt_point->mnt->flags = options; + } else { + goto cleanup; } return errno; + +cleanup: + vfs_sb_free(sb); + return errno; } int @@ -170,3 +194,59 @@ vfs_unmount_at(struct v_dnode* mnt_point) return errno; } + +int +vfs_check_writable(struct v_dnode* dnode) +{ + if ((dnode->mnt->flags & MNT_RO)) { + return EROFS; + } + return 0; +} + +__DEFINE_LXSYSCALL4(int, + mount, + const char*, + source, + const char*, + target, + const char*, + fstype, + int, + options) +{ + struct v_dnode *dev, *mnt; + int errno = 0; + + if ((errno = vfs_walk(__current->cwd, source, &dev, NULL, 0))) { + goto done; + } + + if ((errno = vfs_walk(__current->cwd, target, &mnt, NULL, 0))) { + goto done; + } + + if (mnt->ref_count > 1) { + errno = EBUSY; + goto done; + } + + // By our convention. + // XXX could we do better? + struct device* device = (struct device*)dev->inode->data; + + if (!(dev->inode->itype & VFS_IFVOLDEV) || !device) { + errno = ENOTDEV; + goto done; + } + + errno = vfs_mount_at(fstype, device, mnt, options); + +done: + return DO_STATUS(errno); +} + +__DEFINE_LXSYSCALL1(int, unmount, const char*, target) +{ + return vfs_unmount(target); +} \ No newline at end of file