#include <lunaix/device.h>
-#include <lunaix/fs.h>
+#include <lunaix/fs/api.h>
#include <lunaix/fs/devfs.h>
#include <lunaix/spike.h>
{
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;
}
{
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;
}
{
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;
}
{
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;
}
}
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);
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;
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);