X-Git-Url: https://scm.lunaixsky.com/lunaix-os.git/blobdiff_plain/2236410f4582ab45ae8c384dd6eeeef5d10aab15..270869139db617e29a35bb9ded41087bb702f9ac:/lunaix-os/kernel/device/devfs.c diff --git a/lunaix-os/kernel/device/devfs.c b/lunaix-os/kernel/device/devfs.c index f586add..28f923a 100644 --- a/lunaix-os/kernel/device/devfs.c +++ b/lunaix-os/kernel/device/devfs.c @@ -1,5 +1,5 @@ #include -#include +#include #include #include @@ -13,9 +13,9 @@ devfs_read(struct v_inode* inode, void* buffer, size_t len, size_t fpos) { assert(inode->data); - struct device* dev = (struct device*)inode->data; + struct device* dev = resolve_device(inode->data); - if (!dev->ops.read) { + if (!dev || !dev->ops.read) { return ENOTSUP; } @@ -27,9 +27,9 @@ devfs_write(struct v_inode* inode, void* buffer, size_t len, size_t fpos) { assert(inode->data); - struct device* dev = (struct device*)inode->data; + struct device* dev = resolve_device(inode->data); - if (!dev->ops.write) { + if (!dev || !dev->ops.write) { return ENOTSUP; } @@ -41,9 +41,9 @@ devfs_read_page(struct v_inode* inode, void* buffer, size_t fpos) { assert(inode->data); - struct device* dev = (struct device*)inode->data; + struct device* dev = resolve_device(inode->data); - if (!dev->ops.read_page) { + if (!dev || !dev->ops.read_page) { return ENOTSUP; } @@ -55,9 +55,9 @@ devfs_write_page(struct v_inode* inode, void* buffer, size_t fpos) { assert(inode->data); - struct device* dev = (struct device*)inode->data; + struct device* dev = resolve_device(inode->data); - if (!dev->ops.read_page) { + if (!dev || !dev->ops.read_page) { return ENOTSUP; } @@ -65,36 +65,36 @@ devfs_write_page(struct v_inode* inode, void* buffer, size_t fpos) } int -devfs_get_itype(struct device* dev) +devfs_get_itype(struct device_meta* dm) { - int itype = VFS_IFFILE; + int itype = VFS_IFDEV; + + if (valid_device_subtype_ref(dm, DEV_CAT)) { + return VFS_IFDIR; + } + + struct device* dev = resolve_device(dm); int dev_if = dev->dev_type & DEV_MSKIF; - if (dev_if == DEV_IFCAT) { - itype = VFS_IFDIR; - } else if (dev_if == DEV_IFVOL) { + + if (dev_if == DEV_IFVOL) { itype |= VFS_IFVOLDEV; - } else if (dev_if == DEV_IFSEQ) { - itype |= VFS_IFSEQDEV; - } else { - itype |= VFS_IFDEV; } + + // otherwise, the mapping is considered to be generic seq dev. return itype; } int -devfs_get_dtype(struct device* dev) +devfs_get_dtype(struct device_meta* dev) { - switch (dev->dev_type & DEV_MSKIF) { - case DEV_IFCAT: - return DT_DIR; - - default: - return DT_FILE; + if (valid_device_subtype_ref(dev, DEV_CAT)) { + return DT_DIR; } + return DT_FILE; } int -devfs_mknod(struct v_dnode* dnode, struct device* dev) +devfs_mknod(struct v_dnode* dnode, struct device_meta* dev) { assert(dev); @@ -118,22 +118,44 @@ devfs_mknod(struct v_dnode* dnode, struct device* dev) int devfs_dirlookup(struct v_inode* this, struct v_dnode* dnode) { - struct device* dev = - device_getbyhname((struct device*)this->data, &dnode->name); + void* data = this->data; + struct device_meta* rootdev = resolve_device_meta(data); + + if (data && !rootdev) { + return ENOTDIR; + } + + struct device_meta* dev = + device_getbyhname(rootdev, &dnode->name); + if (!dev) { return ENOENT; } + return devfs_mknod(dnode, dev); } int devfs_readdir(struct v_file* file, struct dir_context* dctx) { - struct device* dev = - device_getbyoffset((struct device*)(file->inode->data), dctx->index); + void* data = file->inode->data; + struct device_meta* rootdev = resolve_device_meta(data); + + if (data && !rootdev) { + return ENOTDIR; + } + + if (fsapi_handle_pseudo_dirent(file, dctx)) { + return 1; + } + + struct device_meta* dev = + device_getbyoffset(rootdev, file->f_pos - 2); + if (!dev) { return 0; } + dctx->read_complete_callback( dctx, dev->name.value, dev->name.len, devfs_get_dtype(dev)); return 1; @@ -175,10 +197,11 @@ devfs_unmount(struct v_superblock* vsb) void devfs_init() { - struct filesystem* fs = fsm_new_fs("devfs", 5); - fsm_register(fs); - fs->mount = devfs_mount; - fs->unmount = devfs_unmount; + struct filesystem* fs; + fs = fsapi_fs_declare("devfs", FSTYPE_PSEUDO); + + fsapi_fs_set_mntops(fs, devfs_mount, devfs_unmount); + fsapi_fs_finalise(fs); } EXPORT_FILE_SYSTEM(devfs, devfs_init);