aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--apps/wm/wm.c54
-rw-r--r--kernel/Makefile1
-rw-r--r--kernel/drivers/fb.c12
-rw-r--r--kernel/drivers/ide.c9
-rw-r--r--kernel/drivers/keyboard.c9
-rw-r--r--kernel/drivers/mouse.c8
-rw-r--r--kernel/drivers/rtc.c6
-rw-r--r--kernel/features/fs.c207
-rw-r--r--kernel/features/io.c67
-rw-r--r--kernel/features/load.c2
-rw-r--r--kernel/features/mm.c2
-rw-r--r--kernel/features/proc.c160
-rw-r--r--kernel/features/syscall.c44
-rw-r--r--kernel/inc/fs.h52
-rw-r--r--kernel/inc/io.h30
-rw-r--r--kernel/inc/proc.h19
-rw-r--r--kernel/main.c3
-rw-r--r--libs/libc/crt/crt0.c2
-rw-r--r--libs/libc/inc/rand.h (renamed from libs/libc/inc/random.h)0
-rw-r--r--libs/libc/inc/sys.h9
-rw-r--r--libs/libc/random.c2
-rw-r--r--libs/libc/sys.c5
22 files changed, 225 insertions, 478 deletions
diff --git a/apps/wm/wm.c b/apps/wm/wm.c
index 0b3689f..7ed67af 100644
--- a/apps/wm/wm.c
+++ b/apps/wm/wm.c
@@ -11,7 +11,7 @@
#include <libgui/vesa.h>
#include <libtxt/keymap.h>
#include <list.h>
-#include <random.h>
+#include <rand.h>
struct client {
u32 pid;
@@ -519,33 +519,33 @@ int main(int argc, char **argv)
gfx_load_wallpaper(&cursor->ctx, "/res/cursor.png");
window_redraw(wallpaper);
- u8 msg[1024] = { 0 };
- struct event_keyboard event_keyboard = { 0 };
- struct event_mouse event_mouse = { 0 };
- const char *listeners[] = { "/dev/kbd", "/dev/mouse", "/proc/self/msg", NULL };
+ /* u8 msg[1024] = { 0 }; */
+ /* struct event_keyboard event_keyboard = { 0 }; */
+ /* struct event_mouse event_mouse = { 0 }; */
+ /* const char *listeners[] = { "/dev/kbd", "/dev/mouse", "/proc/self/msg", NULL }; */
while (1) {
- int poll_ret = 0;
- if ((poll_ret = poll(listeners)) >= 0) {
- if (poll_ret == 0) {
- if (read(listeners[poll_ret], &event_keyboard, 0,
- sizeof(event_keyboard)) > 0) {
- handle_event_keyboard(&event_keyboard);
- continue;
- }
- } else if (poll_ret == 1) {
- if (read(listeners[poll_ret], &event_mouse, 0,
- sizeof(event_mouse)) > 0) {
- handle_event_mouse(&event_mouse);
- continue;
- }
- } else if (poll_ret == 2) {
- if (msg_receive(msg, sizeof(msg)) > 0) {
- handle_message(msg);
- continue;
- }
- }
- }
- panic("Poll/read error: %s\n", strerror(errno));
+ /* int poll_ret = 0; */
+ /* if (1) { */
+ /* if (poll_ret == 0) { */
+ /* if (read(listeners[poll_ret], &event_keyboard, 0, */
+ /* sizeof(event_keyboard)) > 0) { */
+ /* handle_event_keyboard(&event_keyboard); */
+ /* continue; */
+ /* } */
+ /* } else if (poll_ret == 1) { */
+ /* if (read(listeners[poll_ret], &event_mouse, 0, */
+ /* sizeof(event_mouse)) > 0) { */
+ /* handle_event_mouse(&event_mouse); */
+ /* continue; */
+ /* } */
+ /* } else if (poll_ret == 2) { */
+ /* if (msg_receive(msg, sizeof(msg)) > 0) { */
+ /* handle_message(msg); */
+ /* continue; */
+ /* } */
+ /* } */
+ /* } */
+ /* panic("Poll/read error: %s\n", strerror(errno)); */
}
return 1;
diff --git a/kernel/Makefile b/kernel/Makefile
index e28d2d8..81f19b7 100644
--- a/kernel/Makefile
+++ b/kernel/Makefile
@@ -14,6 +14,7 @@ COBJS = main.o \
drivers/timer.o \
features/mm.o \
features/fs.o \
+ features/io.o \
features/load.o \
features/proc.o \
features/proc_asm.o \
diff --git a/kernel/drivers/fb.c b/kernel/drivers/fb.c
index 78b3984..7f05014 100644
--- a/kernel/drivers/fb.c
+++ b/kernel/drivers/fb.c
@@ -26,7 +26,7 @@ PROTECTED static u32 dev_id = 0;
PROTECTED static struct vid_info *info = NULL;
static u32 fb_owner = 0;
-static res fb_ioctl(u32 request, void *arg1, void *arg2, void *arg3, struct device *dev)
+static res fb_ioctl(u32 request, void *arg1, void *arg2, void *arg3, struct vfs_dev *dev)
{
UNUSED(arg2);
UNUSED(arg3);
@@ -56,11 +56,6 @@ static res fb_ioctl(u32 request, void *arg1, void *arg2, void *arg3, struct devi
}
}
-static res fb_ready(void)
-{
- return 1;
-}
-
void fb_map_buffer(struct page_dir *dir, struct vid_info *boot)
{
struct vbe_basic *vbe = (struct vbe_basic *)boot->vbe;
@@ -72,11 +67,10 @@ CLEAR void fb_install(struct vid_info *boot)
{
info = boot;
- struct device *dev = zalloc(sizeof(*dev));
+ struct vfs_dev *dev = zalloc(sizeof(*dev));
dev->name = strdup("fb");
dev->type = DEV_CHAR;
dev->ioctl = fb_ioctl;
- dev->ready = fb_ready;
- device_add(dev);
+ /* device_add(dev); */
dev_id = dev->id;
}
diff --git a/kernel/drivers/ide.c b/kernel/drivers/ide.c
index 408630d..68e797f 100644
--- a/kernel/drivers/ide.c
+++ b/kernel/drivers/ide.c
@@ -85,7 +85,7 @@ static void ide_poll(u16 io)
} while (!(status & ATA_SR_DRQ));
}
-static u8 ata_read_one(u8 *buf, u32 lba, struct device *dev)
+static u8 ata_read_one(u8 *buf, u32 lba, struct vfs_dev *dev)
{
u8 drive = ((struct ata_data *)dev->data)->drive;
u16 io = (drive & ATA_PRIMARY << 1) == ATA_PRIMARY ? ATA_PRIMARY_IO : ATA_SECONDARY_IO;
@@ -108,7 +108,7 @@ static u8 ata_read_one(u8 *buf, u32 lba, struct device *dev)
return 1;
}
-static res ata_read(void *buf, u32 lba, u32 sector_count, struct device *dev)
+static res ata_read(void *buf, u32 lba, u32 sector_count, struct vfs_dev *dev)
{
u8 *b = buf; // I love bytes, yk
for (u32 i = 0; i < sector_count; i++) {
@@ -128,7 +128,7 @@ CLEAR static void ata_probe(void)
if (!ide_find(bus, drive))
continue;
- struct device *dev = zalloc(sizeof(*dev));
+ struct vfs_dev *dev = zalloc(sizeof(*dev));
struct ata_data *data = malloc(sizeof(*data));
data->drive = (bus << 1) | drive;
@@ -139,7 +139,7 @@ CLEAR static void ata_probe(void)
dev->name = str;
dev->type = DEV_BLOCK;
dev->read = ata_read;
- device_add(dev);
+ vfs_add_dev(dev);
if (vfs_mounted(dev, "/"))
continue;
@@ -149,7 +149,6 @@ CLEAR static void ata_probe(void)
vfs->read = ext2_read;
vfs->stat = ext2_stat;
vfs->perm = ext2_perm;
- vfs->ready = ext2_ready;
dev->vfs = vfs;
dev->data = data;
vfs_mount(dev, "/");
diff --git a/kernel/drivers/keyboard.c b/kernel/drivers/keyboard.c
index 95b8a21..6b4b0fa 100644
--- a/kernel/drivers/keyboard.c
+++ b/kernel/drivers/keyboard.c
@@ -44,8 +44,6 @@ static void keyboard_handler(struct regs *r)
state = 0;
merged = 0;
-
- proc_unblock(dev_id, PROC_BLOCK_DEV);
}
/*static void keyboard_acknowledge(void)
@@ -61,7 +59,7 @@ static void keyboard_rate(void)
outb(0x60, 0x0); // Rate{00000} Delay{00} 0
}*/
-static res keyboard_read(void *buf, u32 offset, u32 count, struct device *dev)
+static res keyboard_read(void *buf, u32 offset, u32 count, struct vfs_dev *dev)
{
UNUSED(dev);
if (stack_empty(queue))
@@ -89,11 +87,10 @@ CLEAR void keyboard_install(void)
irq_install_handler(1, keyboard_handler);
queue = stack_new();
- struct device *dev = zalloc(sizeof(*dev));
+ struct vfs_dev *dev = zalloc(sizeof(*dev));
dev->name = strdup("kbd");
dev->type = DEV_CHAR;
dev->read = keyboard_read;
- dev->ready = keyboard_ready;
- device_add(dev);
+ /* device_add(dev); */
dev_id = dev->id;
}
diff --git a/kernel/drivers/mouse.c b/kernel/drivers/mouse.c
index 0440057..3a78bc8 100644
--- a/kernel/drivers/mouse.c
+++ b/kernel/drivers/mouse.c
@@ -46,7 +46,6 @@ static void mouse_handler(struct regs *r)
event->but3 = (mouse_byte[0] >> 2) & 1;
stack_push_bot(queue, event);
mouse_cycle = 0;
- proc_unblock(dev_id, PROC_BLOCK_DEV);
break;
default:
break;
@@ -88,7 +87,7 @@ static res mouse_ready(void)
return !stack_empty(queue);
}
-static res mouse_read(void *buf, u32 offset, u32 count, struct device *dev)
+static res mouse_read(void *buf, u32 offset, u32 count, struct vfs_dev *dev)
{
(void)dev;
if (stack_empty(queue))
@@ -186,11 +185,10 @@ CLEAR void mouse_install(void)
irq_install_handler(12, mouse_handler);
queue = stack_new();
- struct device *dev = zalloc(sizeof(*dev));
+ struct vfs_dev *dev = zalloc(sizeof(*dev));
dev->name = strdup("mouse");
dev->type = DEV_CHAR;
dev->read = mouse_read;
- dev->ready = mouse_ready;
- device_add(dev);
+ /* device_add(dev); */
dev_id = dev->id;
}
diff --git a/kernel/drivers/rtc.c b/kernel/drivers/rtc.c
index a82864f..a814fa9 100644
--- a/kernel/drivers/rtc.c
+++ b/kernel/drivers/rtc.c
@@ -70,7 +70,7 @@ u32 rtc_stamp(void)
rtc.year * 360 * 24 * 365;
}
-static res rtc_dev_read(void *buf, u32 offset, u32 count, struct device *dev)
+static res rtc_dev_read(void *buf, u32 offset, u32 count, struct vfs_dev *dev)
{
UNUSED(offset);
UNUSED(dev);
@@ -83,9 +83,9 @@ static res rtc_dev_read(void *buf, u32 offset, u32 count, struct device *dev)
CLEAR void rtc_install(void)
{
- struct device *dev = zalloc(sizeof(*dev));
+ struct vfs_dev *dev = zalloc(sizeof(*dev));
dev->name = strdup("rtc");
dev->type = DEV_CHAR;
dev->read = rtc_dev_read;
- device_add(dev);
+ /* device_add(dev); */
}
diff --git a/kernel/features/fs.c b/kernel/features/fs.c
index 82d00b6..c3d5402 100644
--- a/kernel/features/fs.c
+++ b/kernel/features/fs.c
@@ -10,7 +10,7 @@
#include <mem.h>
#include <mm.h>
#include <print.h>
-#include <random.h>
+#include <rand.h>
#include <str.h>
/**
@@ -28,7 +28,7 @@ static char *vfs_normalize_path(const char *path)
return fixed;
}
-u8 vfs_mounted(struct device *dev, const char *path)
+u8 vfs_mounted(struct vfs_dev *dev, const char *path)
{
struct node *iterator = mount_points->head;
while (iterator) {
@@ -77,7 +77,12 @@ static struct mount_info *vfs_find_mount_info(const char *path)
return ret;
}
-struct device *vfs_find_dev(const char *path)
+CLEAR void vfs_add_dev(struct vfs_dev *dev)
+{
+ dev->id = rand() + 1;
+}
+
+struct vfs_dev *vfs_find_dev(const char *path)
{
stac();
if (path[0] != '/') {
@@ -86,8 +91,6 @@ struct device *vfs_find_dev(const char *path)
}
clac();
struct mount_info *m = vfs_find_mount_info(path);
- if (m->dev->vfs->type == VFS_DEVFS) // TODO: ?
- return device_get_by_name(path + strlen(m->path) + 1);
return m && m->dev ? m->dev : NULL;
}
@@ -118,7 +121,7 @@ static void vfs_list_mounts()
}
}*/
-res vfs_mount(struct device *dev, const char *path)
+res vfs_mount(struct vfs_dev *dev, const char *path)
{
if (!memory_readable(path))
return -EFAULT;
@@ -242,187 +245,24 @@ res vfs_stat(const char *path, struct stat *buf)
return m->dev->vfs->stat(path, buf, m->dev);
}
-res vfs_block(const char *path, u32 func_ptr)
-{
- if (!func_ptr || !memory_readable(path))
- return -EFAULT;
-
- struct mount_info *m = vfs_find_mount_info(path);
- if (!m || !m->dev || !m->dev->vfs)
- return -ENOENT;
-
- // Default block
- if (!m->dev->vfs->block) {
- proc_block(vfs_find_dev(path)->id, PROC_BLOCK_DEV, func_ptr);
- return EOK;
- }
-
- u32 len = strlen(m->path);
- if (len > 1)
- path += len;
-
- return m->dev->vfs->block(path, func_ptr, m->dev);
-}
-
-// TODO: Reduce stac clac?
-// TODO: Fix page fault when called too often/fast
-res vfs_poll(const char **files)
-{
- if (!memory_readable(files))
- return -EFAULT;
-
- stac();
- for (const char **p = files; *p && memory_readable(*p) && **p; p++) {
- res ready = vfs_ready(*p);
- clac();
- if (ready == 1)
- return p - files;
- else if (ready < 0)
- return ready;
- stac();
- }
-
- for (const char **p = files; *p && memory_readable(*p) && **p; p++) {
- vfs_block(*p, (u32)vfs_poll);
- stac();
- }
- clac();
-
- return PROC_MAX_BLOCK_IDS + 1;
-}
-
-res vfs_ready(const char *path)
-{
- if (!memory_readable(path))
- return -EFAULT;
-
- struct mount_info *m = vfs_find_mount_info(path);
- if (!m || !m->dev || !m->dev->vfs)
- return -ENOENT;
-
- if (!m->dev->vfs->ready)
- return -EINVAL;
-
- u32 len = strlen(m->path);
- if (len > 1)
- path += len;
-
- return m->dev->vfs->ready(path, m->dev);
-}
-
CLEAR void vfs_install(void)
{
mount_points = list_new();
}
/**
- * Device
- */
-
-PROTECTED static struct list *devices = NULL;
-
-CLEAR void device_add(struct device *dev)
-{
- dev->id = rand() + 1;
- list_add(devices, dev);
-}
-
-struct device *device_get_by_id(u32 id)
-{
- struct node *iterator = devices->head;
- while (iterator) {
- if (((struct device *)iterator->data)->id == id)
- return iterator->data;
- iterator = iterator->next;
- }
- return NULL;
-}
-
-struct device *device_get_by_name(const char *name)
-{
- struct node *iterator = devices->head;
- while (iterator) {
- if (!strcmp_user(((struct device *)iterator->data)->name, name))
- return iterator->data;
- iterator = iterator->next;
- }
- return NULL;
-}
-
-static res 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)
- return -ENOENT;
- if (!target->read)
- return -EINVAL;
- return target->read(buf, offset, count, dev);
-}
-
-static res devfs_ioctl(const char *path, u32 request, void *arg1, void *arg2, void *arg3,
- struct device *dev)
-{
- struct device *target = device_get_by_name(path + 1);
- if (!target)
- return -ENOENT;
- if (!target->ioctl)
- return -EINVAL;
- return target->ioctl(request, arg1, arg2, arg3, dev);
-}
-
-static res devfs_perm(const char *path, enum vfs_perm perm, struct device *dev)
-{
- UNUSED(path);
- UNUSED(perm);
- UNUSED(dev);
- return EOK;
-}
-
-static res devfs_ready(const char *path, struct device *dev)
-{
- UNUSED(dev);
-
- struct device *target = device_get_by_name(path + 1);
- if (!target)
- return -ENOENT;
- if (!target->ready)
- return -EINVAL;
- return target->ready();
-}
-
-CLEAR void device_install(void)
-{
- devices = list_new();
-
- struct vfs *vfs = zalloc(sizeof(*vfs));
- vfs->type = VFS_DEVFS;
- vfs->read = devfs_read;
- vfs->ioctl = devfs_ioctl;
- vfs->perm = devfs_perm;
- vfs->ready = devfs_ready;
- struct device *dev = zalloc(sizeof(*dev));
- dev->name = "dev";
- dev->type = DEV_CHAR;
- dev->vfs = vfs;
- device_add(dev);
- vfs_mount(dev, "/dev/");
-
- /* vfs_list_mounts(); */
-}
-
-/**
* EXT2
*/
// TODO: Remove malloc from ext2_buffer_read (attempt in #56cd63f199)
-static void *ext2_buffer_read(u32 block, struct device *dev)
+static void *ext2_buffer_read(u32 block, struct vfs_dev *dev)
{
void *buf = zalloc(BLOCK_SIZE);
dev->read(buf, block * SECTOR_COUNT, SECTOR_COUNT, dev);
return buf;
}
-static struct ext2_superblock *ext2_get_superblock(struct device *dev)
+static struct ext2_superblock *ext2_get_superblock(struct vfs_dev *dev)
{
struct ext2_superblock *sb = ext2_buffer_read(EXT2_SUPER, dev);
@@ -430,12 +270,12 @@ static struct ext2_superblock *ext2_get_superblock(struct device *dev)
return sb;
}
-static struct ext2_bgd *ext2_get_bgd(struct device *dev)
+static struct ext2_bgd *ext2_get_bgd(struct vfs_dev *dev)
{
return ext2_buffer_read(EXT2_SUPER + 1, dev);
}
-static struct ext2_inode *ext2_get_inode(u32 i, struct ext2_inode *in_buf, struct device *dev)
+static struct ext2_inode *ext2_get_inode(u32 i, struct ext2_inode *in_buf, struct vfs_dev *dev)
{
struct ext2_superblock *s = ext2_get_superblock(dev);
assert(s);
@@ -465,7 +305,7 @@ struct indirect_cache {
u8 data[BLOCK_SIZE];
};
static struct list *indirect_cache = NULL;
-static u32 ext2_read_indirect(u32 indirect, u32 block_num, struct device *dev)
+static u32 ext2_read_indirect(u32 indirect, u32 block_num, struct vfs_dev *dev)
{
void *data = NULL;
if (indirect_cache) {
@@ -496,7 +336,7 @@ static u32 ext2_read_indirect(u32 indirect, u32 block_num, struct device *dev)
}
static res ext2_read_inode(struct ext2_inode *in, void *buf, u32 offset, u32 count,
- struct device *dev)
+ struct vfs_dev *dev)
{
if (!in || !buf)
return -EINVAL;
@@ -563,7 +403,7 @@ static res ext2_read_inode(struct ext2_inode *in, void *buf, u32 offset, u32 cou
return copied;
}
-static u32 ext2_find_inode(const char *name, u32 dir_inode, struct device *dev)
+static u32 ext2_find_inode(const char *name, u32 dir_inode, struct vfs_dev *dev)
{
if ((signed)dir_inode <= 0)
return (unsigned)-1;
@@ -599,7 +439,7 @@ static u32 ext2_find_inode(const char *name, u32 dir_inode, struct device *dev)
}
static struct ext2_inode *ext2_find_inode_by_path(const char *path, struct ext2_inode *in_buf,
- struct device *dev)
+ struct vfs_dev *dev)
{
char *path_cp = strdup_user(path);
char *init = path_cp; // For freeing
@@ -640,7 +480,7 @@ static struct ext2_inode *ext2_find_inode_by_path(const char *path, struct ext2_
return ext2_get_inode(inode, in_buf, dev);
}
-res ext2_read(const char *path, void *buf, u32 offset, u32 count, struct device *dev)
+res ext2_read(const char *path, void *buf, u32 offset, u32 count, struct vfs_dev *dev)
{
struct ext2_inode in = { 0 };
if (ext2_find_inode_by_path(path, &in, dev) == &in) {
@@ -649,7 +489,7 @@ res ext2_read(const char *path, void *buf, u32 offset, u32 count, struct device
return -ENOENT;
}
-res ext2_stat(const char *path, struct stat *buf, struct device *dev)
+res ext2_stat(const char *path, struct stat *buf, struct vfs_dev *dev)
{
struct ext2_inode in = { 0 };
if (ext2_find_inode_by_path(path, &in, dev) != &in)
@@ -666,7 +506,7 @@ res ext2_stat(const char *path, struct stat *buf, struct device *dev)
return EOK;
}
-res ext2_perm(const char *path, enum vfs_perm perm, struct device *dev)
+res ext2_perm(const char *path, enum vfs_perm perm, struct vfs_dev *dev)
{
struct ext2_inode in = { 0 };
if (ext2_find_inode_by_path(path, &in, dev) != &in)
@@ -683,10 +523,3 @@ res ext2_perm(const char *path, enum vfs_perm perm, struct device *dev)
return -EINVAL;
}
}
-
-res ext2_ready(const char *path, struct device *dev)
-{
- UNUSED(path);
- UNUSED(dev);
- return 1;
-}
diff --git a/kernel/features/io.c b/kernel/features/io.c
new file mode 100644
index 0000000..2bd925b
--- /dev/null
+++ b/kernel/features/io.c
@@ -0,0 +1,67 @@
+// MIT License, Copyright (c) 2021 Marvin Borner
+
+#include <assert.h>
+#include <def.h>
+#include <io.h>
+#include <list.h>
+#include <mm.h>
+#include <rand.h>
+#include <str.h>
+
+PROTECTED static struct io_dev *io_mappings[IO_MAX] = { 0 };
+
+static u8 io_type_valid(enum io_type io)
+{
+ return io > IO_MIN && io < IO_MAX;
+}
+
+static struct io_dev *io_get(enum io_type io)
+{
+ if (!io_type_valid(io))
+ return NULL;
+
+ return io_mappings[io];
+}
+
+CLEAR void io_add(enum io_type io, struct io_dev *dev)
+{
+ assert(io_type_valid(io) && !io_mappings[io]);
+ io_mappings[io] = dev;
+}
+
+res io_control(enum io_type io)
+{
+ if (!io_get(io))
+ return -ENOENT;
+
+ return -ENOENT;
+}
+
+res io_write(enum io_type io, void *buf, u32 offset, u32 count)
+{
+ if (!memory_readable(buf))
+ return -EFAULT;
+
+ if (!io_get(io))
+ return -ENOENT;
+}
+
+res io_read(enum io_type io, void *buf, u32 offset, u32 count)
+{
+ if (!memory_readable(buf))
+ return -EFAULT;
+
+ if (!io_get(io))
+ return -ENOENT;
+}
+
+res io_poll(enum io_type io)
+{
+ if (!io_get(io))
+ return -ENOENT;
+}
+
+CLEAR void io_install(void)
+{
+ // TODO: Install I/O devices by selecting best working driver
+}
diff --git a/kernel/features/load.c b/kernel/features/load.c
index 77bb680..cbe32ed 100644
--- a/kernel/features/load.c
+++ b/kernel/features/load.c
@@ -6,7 +6,7 @@
#include <load.h>
#include <mem.h>
#include <mm.h>
-#include <random.h>
+#include <rand.h>
#include <str.h>
res elf_load(const char *name, struct proc *proc)
diff --git a/kernel/features/mm.c b/kernel/features/mm.c
index 9a7a64e..d9e4577 100644
--- a/kernel/features/mm.c
+++ b/kernel/features/mm.c
@@ -9,7 +9,7 @@
#include <mem.h>
#include <mm.h>
#include <print.h>
-#include <random.h>
+#include <rand.h>
PROTECTED static struct page_dir kernel_dir ALIGNED(PAGE_SIZE) = { 0 };
static struct page_table kernel_tables[PAGE_KERNEL_COUNT] ALIGNED(PAGE_SIZE) = { 0 };
diff --git a/kernel/features/proc.c b/kernel/features/proc.c
index 3a6b429..a71919e 100644
--- a/kernel/features/proc.c
+++ b/kernel/features/proc.c
@@ -184,92 +184,6 @@ void proc_yield(struct regs *r)
scheduler(r);
}
-// TODO: Rewrite block/unblock mechanisms
-void proc_block(u32 id, enum proc_block_type type, u32 func_ptr)
-{
- u8 already_exists = 0;
- struct proc *p = proc_current();
-
- // Check if already exists
- for (u32 i = 0; i < p->block.id_cnt; i++) {
- if (p->block.ids[i].id == id && p->block.ids[i].type == type) {
- assert(p->block.ids[i].func_ptr == func_ptr);
- already_exists = 1;
- }
- }
-
- if (already_exists)
- goto end;
-
- assert(p->block.id_cnt + 1 < PROC_MAX_BLOCK_IDS);
-
- // Find slot
- struct proc_block_identifier *slot = NULL;
- for (u32 i = 0; i < PROC_MAX_BLOCK_IDS; i++) {
- if (p->block.ids[i].magic != PROC_BLOCK_MAGIC) {
- slot = &p->block.ids[i];
- break;
- }
- }
- assert(slot);
-
- slot->magic = PROC_BLOCK_MAGIC;
- slot->id = id;
- slot->type = type;
- slot->func_ptr = func_ptr;
- p->block.id_cnt++;
-
-end:
- proc_state(p, PROC_BLOCKED);
-}
-
-// TODO: Rewrite block/unblock mechanisms
-void proc_unblock(u32 id, enum proc_block_type type)
-{
- struct page_dir *dir_bak;
- memory_backup_dir(&dir_bak);
-
- struct node *proc_bak = current;
- if (!proc_bak)
- return;
-
- struct node *iterator = proc_list_blocked->head;
- while (iterator) {
- struct proc *p = iterator->data;
- struct proc_block *w = &p->block;
-
- if (!p || !w || w->id_cnt == 0) {
- iterator = iterator->next;
- continue;
- }
-
- current = list_first_data(proc_list_blocked, p);
- assert(w->id_cnt < PROC_MAX_BLOCK_IDS);
- for (u32 i = 0; i < w->id_cnt; i++) {
- if (w->ids[i].magic == PROC_BLOCK_MAGIC && w->ids[i].id == id &&
- w->ids[i].type == type) {
- struct regs *r = &p->regs;
- u32 (*func)(u32, u32, u32, u32) =
- (u32(*)(u32, u32, u32, u32))w->ids[i].func_ptr;
- if (w->ids[i].func_ptr) {
- memory_switch_dir(p->page_dir);
- r->eax = func(r->ebx, r->ecx, r->edx, r->esi);
- memory_switch_dir(dir_bak);
- }
- memset(&w->ids[i], 0, sizeof(w->ids[i]));
- p->block.id_cnt--;
- proc_state(p, PROC_RUNNING);
- break;
- }
- }
-
- iterator = iterator->next;
- }
-
- if (current != proc_bak)
- current = proc_bak;
-}
-
struct proc *proc_make(enum proc_priv priv)
{
struct proc *proc = zalloc(sizeof(*proc));
@@ -355,9 +269,10 @@ struct procfs_message {
u32 size;
};
-static res procfs_write(const char *path, void *buf, u32 offset, u32 count, struct device *dev)
+static res procfs_write(const char *path, void *buf, u32 offset, u32 count, struct vfs_dev *dev)
{
UNUSED(offset);
+ UNUSED(dev);
u32 pid = 0;
procfs_parse_path(&path, &pid);
@@ -378,7 +293,6 @@ static res procfs_write(const char *path, void *buf, u32 offset, u32 count, stru
msg->data = msg_data;
msg->size = count;
stack_push_bot(p->messages, msg); // TODO: Use offset
- proc_unblock(pid, PROC_BLOCK_MSG);
return count;
} else if (!memcmp_user(path, "io/", 3)) {
path += 3;
@@ -394,7 +308,6 @@ static res procfs_write(const char *path, void *buf, u32 offset, u32 count, stru
assert(stream->offset_write + count < STREAM_MAX_SIZE); // TODO: Resize
memcpy_user((char *)(stream->data + stream->offset_write), buf, count);
stream->offset_write += count;
- proc_unblock(dev->id, PROC_BLOCK_DEV);
return count;
}
}
@@ -402,7 +315,7 @@ static res procfs_write(const char *path, void *buf, u32 offset, u32 count, stru
return -ENOENT;
}
-static res procfs_read(const char *path, void *buf, u32 offset, u32 count, struct device *dev)
+static res procfs_read(const char *path, void *buf, u32 offset, u32 count, struct vfs_dev *dev)
{
(void)dev;
u32 pid = 0;
@@ -458,34 +371,7 @@ static res procfs_read(const char *path, void *buf, u32 offset, u32 count, struc
return -ENOENT;
}
-static res procfs_block(const char *path, u32 func_ptr, struct device *dev)
-{
- u32 pid = 0;
- procfs_parse_path(&path, &pid);
-
- if (pid) {
- struct proc *p = proc_from_pid(pid);
- stac();
- if (!p || path[0] != '/') {
- clac();
- return -ENOENT;
- }
- clac();
-
- path++;
- if (!memcmp_user(path, "msg", 4)) {
- proc_block(pid, PROC_BLOCK_MSG, func_ptr);
- return EOK;
- } else {
- proc_block(dev->id, PROC_BLOCK_DEV, func_ptr);
- return EOK;
- }
- }
-
- return -ENOENT;
-}
-
-static res procfs_perm(const char *path, enum vfs_perm perm, struct device *dev)
+static res procfs_perm(const char *path, enum vfs_perm perm, struct vfs_dev *dev)
{
(void)path;
(void)dev;
@@ -496,38 +382,6 @@ static res procfs_perm(const char *path, enum vfs_perm perm, struct device *dev)
return EOK;
}
-static res procfs_ready(const char *path, struct device *dev)
-{
- (void)dev;
-
- u32 pid = 0;
- procfs_parse_path(&path, &pid);
-
- if (pid) {
- struct proc *p = proc_from_pid(pid);
- stac();
- if (!p || path[0] != '/') {
- clac();
- return -ENOENT;
- }
- clac();
-
- path++;
- if (!memcmp_user(path, "msg", 4)) {
- return stack_empty(p->messages) == 0;
- } else if (!memcmp_user(path, "io/", 3)) {
- path += 3;
- enum stream_defaults id = procfs_stream(path);
- if (id == STREAM_UNKNOWN)
- return -ENOENT;
- struct stream *stream = &p->streams[id];
- return stream->data[stream->offset_read] != 0;
- }
- }
-
- return 1;
-}
-
extern void proc_jump_userspace(void);
u32 _esp, _eip;
@@ -547,15 +401,13 @@ NORETURN void proc_init(void)
vfs->type = VFS_PROCFS;
vfs->read = procfs_read;
vfs->write = procfs_write;
- vfs->block = procfs_block;
vfs->perm = procfs_perm;
- vfs->ready = procfs_ready;
vfs->data = NULL;
- struct device *dev = zalloc(sizeof(*dev));
+ struct vfs_dev *dev = zalloc(sizeof(*dev));
dev->name = "proc";
dev->type = DEV_CHAR;
dev->vfs = vfs;
- device_add(dev);
+ vfs_add_dev(dev);
assert(vfs_mount(dev, "/proc/") == EOK);
// Idle proc
diff --git a/kernel/features/syscall.c b/kernel/features/syscall.c
index 65777d8..1e21edd 100644
--- a/kernel/features/syscall.c
+++ b/kernel/features/syscall.c
@@ -4,6 +4,7 @@
#include <errno.h>
#include <fs.h>
#include <interrupts.h>
+#include <io.h>
#include <load.h>
#include <mem.h>
#include <mm.h>
@@ -23,6 +24,7 @@ static void syscall_handler(struct regs *r)
/* printf("[SYSCALL] %d from %s\n", num, proc_current()->name); */
switch (num) {
+ // Memory operations
case SYS_ALLOC: {
r->eax = memory_sys_alloc(proc_current()->page_dir, r->ebx, (u32 *)r->ecx,
(u32 *)r->edx, (u8)r->esi);
@@ -37,20 +39,14 @@ static void syscall_handler(struct regs *r)
(u32 *)r->edx);
break;
}
+
+ // File operations
case SYS_STAT: {
r->eax = vfs_stat((char *)r->ebx, (struct stat *)r->ecx);
break;
}
case SYS_READ: {
- if (vfs_ready((char *)r->ebx)) {
- r->eax = (u32)vfs_read((char *)r->ebx, (void *)r->ecx, r->edx, r->esi);
- } else {
- res wait = vfs_block((char *)r->ebx, (u32)vfs_read);
- if (wait != 0)
- r->eax = wait;
- else
- proc_yield(r);
- }
+ r->eax = vfs_read((char *)r->ebx, (void *)r->ecx, r->edx, r->esi);
break;
}
case SYS_WRITE: {
@@ -62,13 +58,22 @@ static void syscall_handler(struct regs *r)
(void *)r->edi);
break;
}
- case SYS_POLL: {
- res ret = vfs_poll((const char **)r->ebx);
- r->eax = ret;
- if (ret == PROC_MAX_BLOCK_IDS + 1)
- proc_yield(r);
+
+ // I/O operations
+ case SYS_IOPOLL: {
+ r->eax = io_poll((void *)r->ebx);
break;
}
+ case SYS_IOREAD: {
+ r->eax = io_read(r->ebx, (void *)r->ecx, r->edx, r->esi);
+ break;
+ }
+ case SYS_IOWRITE: {
+ r->eax = io_write(r->ebx, (void *)r->ecx, r->edx, r->esi);
+ break;
+ }
+
+ // Process operations
case SYS_EXEC: {
char *path = (char *)r->ebx;
struct proc *proc = proc_make(PROC_PRIV_NONE);
@@ -86,6 +91,12 @@ static void syscall_handler(struct regs *r)
proc_exit(proc_current(), r, (s32)r->ebx);
break;
}
+ case SYS_YIELD: {
+ proc_yield(r);
+ break;
+ }
+
+ // System operations
case SYS_BOOT: { // TODO: Move
if (r->ebx != SYS_BOOT_MAGIC) {
r->eax = -EINVAL;
@@ -113,10 +124,7 @@ static void syscall_handler(struct regs *r)
}
break;
}
- case SYS_YIELD: {
- proc_yield(r);
- break;
- }
+
// TODO: Reimplement network functions using VFS
default: {
printf("Unknown syscall %d!\n", num);
diff --git a/kernel/inc/fs.h b/kernel/inc/fs.h
index 1a78072..93a8a5c 100644
--- a/kernel/inc/fs.h
+++ b/kernel/inc/fs.h
@@ -13,67 +13,60 @@
enum dev_type { DEV_BLOCK, DEV_CHAR };
-struct device {
+struct vfs_dev {
u32 id;
const char *name;
enum dev_type type;
struct vfs *vfs;
void *data;
- res (*read)(void *buf, u32 offset, u32 count, struct device *dev) NONNULL;
- res (*write)(void *buf, u32 offset, u32 count, struct device *dev) NONNULL;
- res (*ioctl)(u32 request, void *arg1, void *arg2, void *arg3, struct device *dev)
+ res (*read)(void *buf, u32 offset, u32 count, struct vfs_dev *dev) NONNULL;
+ res (*write)(void *buf, u32 offset, u32 count, struct vfs_dev *dev) NONNULL;
+ res (*ioctl)(u32 request, void *arg1, void *arg2, void *arg3, struct vfs_dev *dev)
ATTR((nonnull(5)));
- res (*ready)(void);
};
-void device_install(void);
-
-void device_add(struct device *dev) NONNULL;
-
/**
* VFS
*/
-enum vfs_type { VFS_DEVFS, VFS_TMPFS, VFS_PROCFS, VFS_EXT2 };
+enum vfs_type { VFS_TMPFS, VFS_PROCFS, VFS_EXT2 };
enum vfs_perm { VFS_EXEC, VFS_WRITE, VFS_READ };
struct vfs {
enum vfs_type type;
int flags;
void *data;
- res (*read)(const char *path, void *buf, u32 offset, u32 count, struct device *dev) NONNULL;
+ res (*read)(const char *path, void *buf, u32 offset, u32 count,
+ struct vfs_dev *dev) NONNULL;
res (*write)(const char *path, void *buf, u32 offset, u32 count,
- struct device *dev) NONNULL;
+ struct vfs_dev *dev) NONNULL;
res (*ioctl)(const char *path, u32 request, void *arg1, void *arg2, void *arg3,
- struct device *dev) ATTR((nonnull(1, 6)));
- res (*stat)(const char *path, struct stat *buf, struct device *dev) NONNULL;
- res (*block)(const char *path, u32 func_ptr, struct device *dev) NONNULL;
- res (*ready)(const char *path, struct device *dev) NONNULL;
- res (*perm)(const char *path, enum vfs_perm perm, struct device *dev) NONNULL;
+ struct vfs_dev *dev) ATTR((nonnull(1, 6)));
+ res (*stat)(const char *path, struct stat *buf, struct vfs_dev *dev) NONNULL;
+ res (*block)(const char *path, u32 func_ptr, struct vfs_dev *dev) NONNULL;
+ res (*perm)(const char *path, enum vfs_perm perm, struct vfs_dev *dev) NONNULL;
};
struct mount_info {
const char *path;
- struct device *dev;
+ struct vfs_dev *dev;
};
void vfs_install(void);
-u8 vfs_mounted(struct device *dev, const char *path) NONNULL;
-res vfs_mount(struct device *dev, const char *path) NONNULL;
+u8 vfs_mounted(struct vfs_dev *dev, const char *path) NONNULL;
+res vfs_mount(struct vfs_dev *dev, const char *path) NONNULL;
-struct device *vfs_find_dev(const char *path) NONNULL;
+struct vfs_dev *vfs_find_dev(const char *path) NONNULL;
+void vfs_add_dev(struct vfs_dev *dev) NONNULL;
res vfs_read(const char *path, void *buf, u32 offset, u32 count) NONNULL;
res vfs_write(const char *path, void *buf, u32 offset, u32 count) NONNULL;
res vfs_ioctl(const char *path, u32 request, void *arg1, void *arg2, void *arg3) ATTR((nonnull(1)));
res vfs_stat(const char *path, struct stat *buf) NONNULL;
-res vfs_block(const char *path, u32 func_ptr) NONNULL;
-res vfs_poll(const char **files) NONNULL;
-res vfs_ready(const char *path) NONNULL;
-struct device *device_get_by_name(const char *name) NONNULL;
-struct device *device_get_by_id(u32 id) NONNULL;
+struct vfs_dev *device_get_by_name(const char *name) NONNULL;
+struct vfs_dev *device_get_by_id(u32 id) NONNULL;
/**
* EXT2
@@ -177,9 +170,8 @@ struct ext2_file {
u32 curr_block_pos;
};
-res ext2_read(const char *path, void *buf, u32 offset, u32 count, struct device *dev) NONNULL;
-res ext2_stat(const char *path, struct stat *buf, struct device *dev) NONNULL;
-res ext2_perm(const char *path, enum vfs_perm perm, struct device *dev) NONNULL;
-res ext2_ready(const char *path, struct device *dev) NONNULL;
+res ext2_read(const char *path, void *buf, u32 offset, u32 count, struct vfs_dev *dev) NONNULL;
+res ext2_stat(const char *path, struct stat *buf, struct vfs_dev *dev) NONNULL;
+res ext2_perm(const char *path, enum vfs_perm perm, struct vfs_dev *dev) NONNULL;
#endif
diff --git a/kernel/inc/io.h b/kernel/inc/io.h
new file mode 100644
index 0000000..e05a419
--- /dev/null
+++ b/kernel/inc/io.h
@@ -0,0 +1,30 @@
+// MIT License, Copyright (c) 2021 Marvin Borner
+
+#ifndef DEV_H
+#define DEV_H
+
+#include <def.h>
+#include <sys.h>
+
+enum io_type {
+ IO_MIN,
+ IO_FRAMEBUFFER,
+ IO_NETWORK,
+ IO_KEYBOARD,
+ IO_MOUSE,
+ IO_BUS,
+ IO_MAX,
+};
+
+struct io_dev {
+ const char *name;
+};
+
+void io_install(void);
+
+res io_control();
+res io_write(enum io_type io, void *buf, u32 offset, u32 count);
+res io_read(enum io_type io, void *buf, u32 offset, u32 count);
+res io_poll();
+
+#endif
diff --git a/kernel/inc/proc.h b/kernel/inc/proc.h
index b8912a5..754693f 100644
--- a/kernel/inc/proc.h
+++ b/kernel/inc/proc.h
@@ -22,27 +22,11 @@
#define RING(regs) ((regs->cs) & 3)
-#define PROC_MAX_BLOCK_IDS 16
-#define PROC_BLOCK_MAGIC 0x00528491
-
#define STREAM_MAX_SIZE 4096
enum stream_defaults { STREAM_IN, STREAM_OUT, STREAM_ERR, STREAM_LOG, STREAM_UNKNOWN = -1 };
enum proc_priv { PROC_PRIV_NONE, PROC_PRIV_ROOT, PROC_PRIV_KERNEL };
enum proc_state { PROC_RUNNING, PROC_BLOCKED };
-enum proc_block_type { PROC_BLOCK_DEV, PROC_BLOCK_MSG };
-
-struct proc_block_identifier {
- u32 magic;
- u32 id;
- enum proc_block_type type;
- u32 func_ptr;
-};
-
-struct proc_block {
- struct proc_block_identifier ids[PROC_MAX_BLOCK_IDS];
- u32 id_cnt;
-};
struct stream {
u32 offset_read;
@@ -59,7 +43,6 @@ struct proc {
struct stream streams[4];
struct page_dir *page_dir;
struct regs regs;
- struct proc_block block; // dev_id
enum proc_priv priv;
enum proc_state state;
struct stack *messages;
@@ -92,8 +75,6 @@ void proc_yield(struct regs *r) NONNULL;
void proc_set_quantum(struct proc *proc, u32 value);
void proc_reset_quantum(struct proc *proc);
void proc_state(struct proc *proc, enum proc_state state);
-void proc_block(u32 id, enum proc_block_type type, u32 func_ptr);
-void proc_unblock(u32 id, enum proc_block_type type);
struct proc *proc_make(enum proc_priv priv);
void proc_stack_push(struct proc *proc, u32 data) NONNULL;
diff --git a/kernel/main.c b/kernel/main.c
index ab5fde8..fe1f75f 100644
--- a/kernel/main.c
+++ b/kernel/main.c
@@ -13,7 +13,7 @@
#include <mouse.h>
#include <net.h>
#include <pci.h>
-#include <random.h>
+#include <rand.h>
#include <rtc.h>
#include <serial.h>
#include <syscall.h>
@@ -47,7 +47,6 @@ int kernel_main(struct boot_info *boot)
// Install drivers
vfs_install();
- device_install();
ata_install();
pci_install();
interrupts_install();
diff --git a/libs/libc/crt/crt0.c b/libs/libc/crt/crt0.c
index d264d61..03e569c 100644
--- a/libs/libc/crt/crt0.c
+++ b/libs/libc/crt/crt0.c
@@ -2,7 +2,7 @@
#include <assert.h>
#include <def.h>
-#include <random.h>
+#include <rand.h>
#include <sys.h>
#ifdef USER
diff --git a/libs/libc/inc/random.h b/libs/libc/inc/rand.h
index 1b41df5..1b41df5 100644
--- a/libs/libc/inc/random.h
+++ b/libs/libc/inc/rand.h
diff --git a/libs/libc/inc/sys.h b/libs/libc/inc/sys.h
index a9ebf2e..e7acc49 100644
--- a/libs/libc/inc/sys.h
+++ b/libs/libc/inc/sys.h
@@ -19,10 +19,12 @@ enum sys {
SYS_SHACCESS, // Access shared memory
SYS_FREE, // Free memory
SYS_STAT, // Get file information
- SYS_READ, // Read file
+ SYS_READ, // Read file (non-blocking)
SYS_WRITE, // Write to file
- SYS_IOCTL, // Interact with a file/device
- SYS_POLL, // Wait for multiple files
+ SYS_IOCTL, // Interact with an I/O device
+ SYS_IOPOLL, // Block proc until I/O device is ready
+ SYS_IOREAD, // Read data from I/O device (blocking)
+ SYS_IOWRITE, // Write data to I/O device
SYS_EXEC, // Execute path
SYS_EXIT, // Exit current process
SYS_BOOT, // Boot functions (e.g. reboot/shutdown)
@@ -68,7 +70,6 @@ res read(const char *path, void *buf, u32 offset, u32 count) NONNULL;
res write(const char *path, const void *buf, u32 offset, u32 count) NONNULL;
res ioctl(const char *path, ...) NONNULL;
res stat(const char *path, struct stat *buf) NONNULL;
-res poll(const char **files) NONNULL;
res exec(const char *path, ...) ATTR((nonnull(1))) SENTINEL;
res yield(void);
res boot(u32 cmd);
diff --git a/libs/libc/random.c b/libs/libc/random.c
index 2cd7f93..268a21f 100644
--- a/libs/libc/random.c
+++ b/libs/libc/random.c
@@ -2,7 +2,7 @@
#include <def.h>
#include <mem.h>
-#include <random.h>
+#include <rand.h>
#ifdef KERNEL
#include <cpu.h>
diff --git a/libs/libc/sys.c b/libs/libc/sys.c
index 70dcc97..7563fb4 100644
--- a/libs/libc/sys.c
+++ b/libs/libc/sys.c
@@ -128,11 +128,6 @@ res stat(const char *path, struct stat *buf)
return sys2(SYS_STAT, (int)path, (int)buf);
}
-res poll(const char **files)
-{
- return sys1(SYS_POLL, (int)files);
-}
-
res exec(const char *path, ...)
{
va_list ap;