aboutsummaryrefslogtreecommitdiff
path: root/kernel
diff options
context:
space:
mode:
authorMarvin Borner2021-02-09 21:48:59 +0100
committerMarvin Borner2021-02-09 21:54:24 +0100
commit58b4e00736c0f2f13a3e96b5fcb3c00623ed35be (patch)
tree2286b0c26d6706f8da46f8c3d31ff3761fcfca15 /kernel
parent59894afa1bc0f4efc85917710adf2e93d7e17a5e (diff)
Quite many message queue fixes and impl start
Diffstat (limited to 'kernel')
-rw-r--r--kernel/features/proc.c75
-rw-r--r--kernel/features/syscall.c13
2 files changed, 52 insertions, 36 deletions
diff --git a/kernel/features/proc.c b/kernel/features/proc.c
index 1f402d8..5c2fab1 100644
--- a/kernel/features/proc.c
+++ b/kernel/features/proc.c
@@ -46,6 +46,14 @@ void scheduler(struct regs *regs)
current = proc_list->head;
} else {
current = idle_proc;
+ struct node *iterator = proc_list->head;
+ while (iterator) {
+ if (((struct proc *)iterator->data)->state == PROC_RUNNING) {
+ current = iterator;
+ break;
+ }
+ iterator = iterator->next;
+ }
}
memcpy(regs, &((struct proc *)current->data)->regs, sizeof(struct regs));
@@ -149,6 +157,8 @@ void proc_yield(struct regs *r)
void proc_enable_waiting(u32 id, enum proc_wait_type type)
{
+ struct proc *proc_bak = proc_current();
+
struct node *iterator = proc_list->head;
while (iterator) {
struct proc *p = iterator->data;
@@ -161,6 +171,7 @@ void proc_enable_waiting(u32 id, enum proc_wait_type type)
u8 dispatched = 0;
+ current = list_first_data(proc_list, p);
for (u32 i = 0; i < PROC_MAX_WAIT_IDS; i++) {
if (w->ids[i] == id) {
struct regs *r = &p->regs;
@@ -179,6 +190,9 @@ void proc_enable_waiting(u32 id, enum proc_wait_type type)
iterator = iterator->next;
}
+
+ if (current->data != proc_bak)
+ current = list_first_data(proc_list, proc_bak);
}
void proc_wait_for(u32 id, enum proc_wait_type type, s32 (*func)())
@@ -214,22 +228,28 @@ struct proc *proc_make(void)
return proc;
}
-s32 procfs_write(const char *path, void *buf, u32 offset, u32 count, struct device *dev)
+const char *procfs_parse_path(const char **path, u32 *pid)
{
- while (*path == '/')
- path++;
+ while (**path == '/')
+ (*path)++;
- int pid = 0;
- while (path[0] >= '0' && path[0] <= '9') {
- pid = pid * 10 + (path[0] - '0');
- path++;
+ while ((*path)[0] >= '0' && (*path)[0] <= '9') {
+ *pid = *pid * 10 + ((*path)[0] - '0');
+ (*path)++;
}
- if (!pid && !memcmp(path, "self/", 5)) {
- pid = proc_current()->pid;
- path += 4;
+ if (!*pid && !memcmp(*path, "self/", 5)) {
+ *pid = proc_current()->pid;
+ *path += 4;
}
+ return *path;
+}
+
+s32 procfs_write(const char *path, void *buf, u32 offset, u32 count, struct device *dev)
+{
+ u32 pid = 0;
+ procfs_parse_path(&path, &pid);
if (pid) {
struct proc *p = proc_from_pid(pid);
if (!p || path[0] != '/')
@@ -237,7 +257,8 @@ s32 procfs_write(const char *path, void *buf, u32 offset, u32 count, struct devi
path++;
if (!memcmp(path, "msg", 4)) {
- stack_push_bot(p->messages, buf);
+ stack_push_bot(p->messages, buf); // TODO: Use offset and count
+ proc_enable_waiting(dev->id, PROC_WAIT_DEV); // TODO: Better wakeup solution
return count;
}
}
@@ -249,20 +270,8 @@ s32 procfs_write(const char *path, void *buf, u32 offset, u32 count, struct devi
s32 procfs_read(const char *path, void *buf, u32 offset, u32 count, struct device *dev)
{
(void)dev;
-
- while (*path == '/')
- path++;
-
- int pid = 0;
- while (path[0] >= '0' && path[0] <= '9') {
- pid = pid * 10 + (path[0] - '0');
- path++;
- }
-
- if (!pid && !memcmp(path, "self/", 5)) {
- pid = proc_current()->pid;
- path += 4;
- }
+ u32 pid = 0;
+ procfs_parse_path(&path, &pid);
if (pid) {
struct proc *p = proc_from_pid(pid);
@@ -286,6 +295,7 @@ s32 procfs_read(const char *path, void *buf, u32 offset, u32 count, struct devic
return 0;
} else {
u8 *msg = stack_pop(p->messages);
+ printf("Pop: %s\n", msg);
memcpy(buf, msg + offset, count);
return count;
}
@@ -308,8 +318,21 @@ u8 procfs_perm(const char *path, enum vfs_perm perm, struct device *dev)
u8 procfs_ready(const char *path, struct device *dev)
{
- (void)path;
(void)dev;
+
+ u32 pid = 0;
+ procfs_parse_path(&path, &pid);
+
+ if (pid) {
+ struct proc *p = proc_from_pid(pid);
+ if (!p || path[0] != '/')
+ return -1;
+
+ path++;
+ if (!memcmp(path, "msg", 4))
+ return stack_empty(p->messages) == 0;
+ }
+
return 1;
}
diff --git a/kernel/features/syscall.c b/kernel/features/syscall.c
index 8140313..80c3e9e 100644
--- a/kernel/features/syscall.c
+++ b/kernel/features/syscall.c
@@ -12,14 +12,6 @@
#include <sys.h>
#include <timer.h>
-void syscall_yield(struct regs *r)
-{
- proc_yield(r);
- sti();
- while (1)
- hlt();
-}
-
void syscall_handler(struct regs *r)
{
enum sys num = r->eax;
@@ -49,7 +41,7 @@ void syscall_handler(struct regs *r)
r->eax = (u32)vfs_read((char *)r->ebx, (void *)r->ecx, r->edx, r->esi);
} else {
proc_wait_for(vfs_find_dev((char *)r->ebx)->id, PROC_WAIT_DEV, vfs_read);
- syscall_yield(r);
+ proc_yield(r);
}
break;
}
@@ -60,7 +52,7 @@ void syscall_handler(struct regs *r)
case SYS_POLL: {
s32 ret = vfs_poll((const char **)r->ebx);
if (ret == PROC_MAX_WAIT_IDS + 1)
- syscall_yield(r);
+ proc_yield(r);
else
r->eax = ret;
break;
@@ -80,6 +72,7 @@ void syscall_handler(struct regs *r)
((u32 *)proc->regs.useresp)[1] = (u32)argv;
if (r->eax)
proc_exit(proc, (int)r->eax);
+ proc_yield(r);
break;
}
case SYS_EXIT: {