aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorIan Moffett <ian@osmora.org>2024-04-10 11:24:46 -0400
committerIan Moffett <ian@osmora.org>2024-04-10 11:24:46 -0400
commit190f059455b7057c99cb3b24ca63d5ff0872a562 (patch)
tree5ec53b33cf6a5ad98f532b76152791f31e293d3c
parent0296ec6874f69034488f0c90d8d54059f32aa3c0 (diff)
kernel: fs: Add devfs
Signed-off-by: Ian Moffett <ian@osmora.org>
-rw-r--r--sys/fs/devfs.c208
-rw-r--r--sys/include/fs/devfs.h41
-rw-r--r--sys/include/sys/device.h1
-rw-r--r--sys/include/sys/vnode.h2
-rw-r--r--sys/kern/vfs_init.c3
5 files changed, 255 insertions, 0 deletions
diff --git a/sys/fs/devfs.c b/sys/fs/devfs.c
new file mode 100644
index 0000000..0438b14
--- /dev/null
+++ b/sys/fs/devfs.c
@@ -0,0 +1,208 @@
+/*
+ * Copyright (c) 2023-2024 Ian Marco Moffett and the Osmora Team.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright notice,
+ * this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Hyra nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include <fs/devfs.h>
+#include <sys/vfs.h>
+#include <sys/mount.h>
+#include <sys/spinlock.h>
+#include <sys/queue.h>
+#include <sys/vnode.h>
+#include <sys/errno.h>
+#include <sys/cdefs.h>
+#include <vm/dynalloc.h>
+#include <string.h>
+
+struct device_node {
+ struct spinlock lock;
+ char *name;
+ uint8_t is_block : 1;
+ dev_t major, minor;
+ TAILQ_ENTRY(device_node) link;
+};
+
+static TAILQ_HEAD(, device_node) nodes;
+static bool nodelist_init = false;
+
+static struct device_node *
+node_from_name(const char *name)
+{
+ struct device_node *n;
+
+ TAILQ_FOREACH(n, &nodes, link) {
+ if (strcmp(n->name, name) == 0) {
+ return n;
+ }
+ }
+
+ return NULL;
+}
+
+static int
+blkdev_read(struct device *dev, struct device_node *node, struct sio_txn *sio)
+{
+ char *buf;
+ struct sio_txn dev_txn = {0};
+ size_t buf_size = dev->blocksize * sio->len;
+
+ if (dev->blocksize == 0 || sio->len == 0) {
+ /* Sizes can't be zero! */
+ return -EIO;
+ }
+
+ spinlock_acquire(&node->lock);
+ buf = dynalloc_memalign(buf_size, 0x1000);
+ if (buf == NULL) {
+ spinlock_release(&node->lock);
+ return -ENOMEM;
+ }
+
+ dev_txn.len = __DIV_ROUNDUP(sio->len, 512);
+ dev_txn.buf = buf;
+ dev_txn.offset = sio->offset;
+ dev->read(dev, &dev_txn);
+ spinlock_release(&node->lock);
+
+ for (size_t i = 0; i < sio->len; ++i) {
+ ((uint8_t *)sio->buf)[i] = buf[i];
+ }
+
+ dynfree(buf);
+ return sio->len;
+}
+
+
+static int
+vop_vget(struct vnode *parent, const char *name, struct vnode **vp)
+{
+ struct device_node *dev;
+ struct vnode *vnode;
+ int status, vtype;
+
+ if (!nodelist_init) {
+ return -EIO;
+ }
+
+ if ((dev = node_from_name(name)) == NULL) {
+ return -ENOENT;
+ }
+
+ vtype = dev->is_block ? VBLK : VCHR;
+ if ((status = vfs_alloc_vnode(&vnode, NULL, vtype)) != 0) {
+ return status;
+ }
+
+ vnode->parent = parent;
+ vnode->data = dev;
+ vnode->vops = &g_devfs_vops;
+ *vp = vnode;
+ return 0;
+}
+
+static int
+vop_read(struct vnode *vp, struct sio_txn *sio)
+{
+ struct device_node *node;
+ struct device *dev;
+
+ if (vp == NULL) {
+ return -EIO;
+ }
+
+ node = vp->data;
+ dev = device_fetch(node->major, node->minor);
+ return blkdev_read(dev, node, sio);
+}
+
+static int
+devfs_init(struct fs_info *info)
+{
+ TAILQ_INIT(&nodes);
+ nodelist_init = true;
+ return 0;
+}
+
+static int
+devfs_make_devicenode(const char *name, struct device_node **node_out)
+{
+ size_t name_len = 0;
+ const char *p = name;
+ struct device_node *node;
+
+ /*
+ * Only one filename, no paths.
+ *
+ * TODO: Do something better here... */
+ for (; *p; ++p, ++name_len) {
+ if (*p == '/')
+ return -EINVAL;
+ }
+
+ /* Ensure this filename has valid chars */
+ if (!vfs_is_valid_path(name)) {
+ return -EINVAL;
+ }
+
+ node = dynalloc(sizeof(struct device_node));
+ if (node == NULL)
+ return -ENOMEM;
+
+ node->name = dynalloc(sizeof(char) * name_len);
+ if (node->name == NULL)
+ return -ENOMEM;
+
+ memcpy(node->name, name, name_len + 1);
+ *node_out = node;
+ return 0;
+}
+
+int
+devfs_add_blkdev(const char *name, const struct device *blkdev)
+{
+ struct device_node *node;
+ int status;
+
+ if ((status = devfs_make_devicenode(name, &node)) != 0) {
+ return status;
+ }
+
+ node->major = blkdev->major;
+ node->minor = blkdev->minor;
+ node->is_block = 1;
+ TAILQ_INSERT_HEAD(&nodes, node, link);
+ return 0;
+}
+
+struct vfsops g_devfs_ops = {
+ .init = devfs_init
+};
+
+struct vops g_devfs_vops = {
+ .vget = vop_vget,
+ .read = vop_read
+};
diff --git a/sys/include/fs/devfs.h b/sys/include/fs/devfs.h
new file mode 100644
index 0000000..1fa4ada
--- /dev/null
+++ b/sys/include/fs/devfs.h
@@ -0,0 +1,41 @@
+/*
+ * Copyright (c) 2023-2024 Ian Marco Moffett and the Osmora Team.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright notice,
+ * this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Hyra nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#ifndef _FS_DEVFS_H_
+#define _FS_DEVFS_H_
+
+#include <sys/vnode.h>
+#include <sys/device.h>
+
+extern struct vfsops g_devfs_ops;
+extern struct vops g_devfs_vops;
+
+int devfs_add_blkdev(const char *name, const struct device *blkdev);
+
+#endif
diff --git a/sys/include/sys/device.h b/sys/include/sys/device.h
index 0dd1744..43b3db7 100644
--- a/sys/include/sys/device.h
+++ b/sys/include/sys/device.h
@@ -38,6 +38,7 @@
struct device {
dev_t major, minor;
+ size_t blocksize;
int(*write)(struct device *dev, struct sio_txn *sio);
int(*read)(struct device *dev, struct sio_txn *sio);
TAILQ_ENTRY(device) link;
diff --git a/sys/include/sys/vnode.h b/sys/include/sys/vnode.h
index ac66044..ea537f6 100644
--- a/sys/include/sys/vnode.h
+++ b/sys/include/sys/vnode.h
@@ -65,6 +65,8 @@ struct vnode {
*/
#define VREG 0x01 /* Regular file */
#define VDIR 0x02 /* Directory */
+#define VCHR 0x03 /* Character device */
+#define VBLK 0x04 /* Block device */
#if defined(_KERNEL)
int vfs_alloc_vnode(struct vnode **vnode, struct mount *mp, int type);
diff --git a/sys/kern/vfs_init.c b/sys/kern/vfs_init.c
index 9039e6a..8b0b119 100644
--- a/sys/kern/vfs_init.c
+++ b/sys/kern/vfs_init.c
@@ -33,6 +33,7 @@
#include <sys/types.h>
#include <sys/vnode.h>
#include <fs/initramfs.h>
+#include <fs/devfs.h>
#include <assert.h>
#include <string.h>
@@ -41,9 +42,11 @@ __KERNEL_META("$Hyra$: vfs.c, Ian Marco Moffett, "
"Hyra Virtual File System");
#define INITRAMFS_ID 0
+#define DEVFS_ID 1
static struct fs_info filesystems[] = {
[INITRAMFS_ID] = { "initramfs", &g_initramfs_ops, NULL},
+ [DEVFS_ID] = { "dev", &g_devfs_ops, &g_devfs_vops }
};
struct vnode *g_root_vnode = NULL;