aboutsummaryrefslogtreecommitdiff
path: root/kernel/features/syscall.c
diff options
context:
space:
mode:
authorMarvin Borner2021-04-16 23:22:02 +0200
committerMarvin Borner2021-04-16 23:22:02 +0200
commit7485f7e441ca892876d9401380aa77610eb85f76 (patch)
tree065f910effe33f5c9d71cd9619dd25216282a13d /kernel/features/syscall.c
parent4d4e784770b576199b18f22100689125a18bfd9a (diff)
New elegant I/O blocking solution
This is done using an internal scheduler syscall (127). Very nice!
Diffstat (limited to 'kernel/features/syscall.c')
-rw-r--r--kernel/features/syscall.c32
1 files changed, 22 insertions, 10 deletions
diff --git a/kernel/features/syscall.c b/kernel/features/syscall.c
index 2c30a3b..94ea778 100644
--- a/kernel/features/syscall.c
+++ b/kernel/features/syscall.c
@@ -19,7 +19,6 @@
static void syscall_handler(struct regs *r)
{
enum sys num = r->eax;
- r->eax = EOK;
/* printf("[SYSCALL] %d from %s\n", num, proc_current()->name); */
@@ -56,16 +55,18 @@ static void syscall_handler(struct regs *r)
// I/O operations
case SYS_IOPOLL: {
- r->eax = io_poll((u32 *)r->ebx);
+ r->eax = io_poll((void *)r->ebx);
break;
}
case SYS_IOREAD: {
- res ret = io_read(r->ebx, (void *)r->ecx, r->edx, r->esi);
- if (ret == -EAGAIN) {
- io_block(r->ebx, proc_current(), r);
- } else {
- r->eax = ret;
+ res ready = io_ready(r->ebx);
+ if (ready == -EAGAIN) {
+ io_block(r->ebx, proc_current());
+ } else if (ready != EOK) {
+ r->eax = ready;
+ break;
}
+ r->eax = io_read(r->ebx, (void *)r->ecx, r->edx, r->esi);
break;
}
case SYS_IOWRITE: {
@@ -87,7 +88,7 @@ static void syscall_handler(struct regs *r)
} else {
// TODO: Reimplement argc,argv
proc_stack_push(proc, 0);
- proc_yield(r);
+ proc_yield();
}
break;
}
@@ -98,7 +99,7 @@ static void syscall_handler(struct regs *r)
}
case SYS_YIELD: {
r->eax = EOK;
- proc_yield(r);
+ proc_yield();
break;
}
@@ -140,8 +141,19 @@ static void syscall_handler(struct regs *r)
}
}
+// For kernel syscalls (internal)
+static void syscall_special_handler(struct regs *r)
+{
+ if (RING(r) != 0)
+ return;
+
+ scheduler(r);
+}
+
CLEAR void syscall_init(void)
{
- idt_set_gate(0x80, (u32)isr128, 0x08, 0x8E);
+ idt_set_gate(0x7f, (u32)isr127, 0x08, 0x8e);
+ idt_set_gate(0x80, (u32)isr128, 0x08, 0xee);
+ isr_install_handler(0x7f, syscall_special_handler);
isr_install_handler(0x80, syscall_handler);
}