Kernel, libc: Implement waitpid()
FIXME: exec() is now doing weird page table stuff. But at least it works, no panics :)
This commit is contained in:
parent
bb00e3c112
commit
671f2a2de3
@ -11,7 +11,6 @@ CFLAGS := -Wall -Wextra -Werror -Os -fno-asynchronous-unwind-tables
|
|||||||
$(APPS_BIN)/%: $(APPS_SRC)/%.c
|
$(APPS_BIN)/%: $(APPS_SRC)/%.c
|
||||||
@mkdir -p $(@D)
|
@mkdir -p $(@D)
|
||||||
$(CC) $(CFLAGS) -o $@ $^
|
$(CC) $(CFLAGS) -o $@ $^
|
||||||
$(STRIP) $@
|
|
||||||
|
|
||||||
build: $(REAL_APPS)
|
build: $(REAL_APPS)
|
||||||
|
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
|
#include <sys/wait.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
|
||||||
int print_version()
|
int print_version()
|
||||||
@ -162,7 +163,7 @@ int main()
|
|||||||
}
|
}
|
||||||
if (child == 0)
|
if (child == 0)
|
||||||
{
|
{
|
||||||
msleep(500);
|
sleep(2);
|
||||||
printf("I am the child (PID %ld), my parent is PID %ld!!\n", getpid(), getppid());
|
printf("I am the child (PID %ld), my parent is PID %ld!!\n", getpid(), getppid());
|
||||||
execv("/bin/sym", NULL);
|
execv("/bin/sym", NULL);
|
||||||
perror("execv");
|
perror("execv");
|
||||||
@ -170,14 +171,16 @@ int main()
|
|||||||
}
|
}
|
||||||
else { printf("Success!! Got PID %ld\n", child); }
|
else { printf("Success!! Got PID %ld\n", child); }
|
||||||
|
|
||||||
jmp_buf env;
|
int status;
|
||||||
int val = setjmp(env);
|
pid_t result;
|
||||||
if (val == 0) { printf("Returning from setjmp!\n"); }
|
while ((result = waitpid(child, &status, 0)) == 0) // Child has not yet exited
|
||||||
else
|
|
||||||
{
|
{
|
||||||
printf("Returning from longjmp! val=%d\n", val);
|
msleep(100);
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
if (result < 0)
|
||||||
longjmp(env, 3);
|
{
|
||||||
|
perror("waitpid");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
if (WIFEXITED(status)) { printf("Child process %ld exited with code %d\n", result, WEXITSTATUS(status)); }
|
||||||
}
|
}
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
#define EPERM 1
|
#define EPERM 1
|
||||||
#define ENOENT 2
|
#define ENOENT 2
|
||||||
|
#define ESRCH 3
|
||||||
#define ENOEXEC 8
|
#define ENOEXEC 8
|
||||||
#define EBADF 9
|
#define EBADF 9
|
||||||
#define ENOMEM 12
|
#define ENOMEM 12
|
||||||
|
@ -20,6 +20,7 @@
|
|||||||
#define SYS_clock 15
|
#define SYS_clock 15
|
||||||
#define SYS_mkdir 16
|
#define SYS_mkdir 16
|
||||||
#define SYS_fork 17
|
#define SYS_fork 17
|
||||||
|
#define SYS_waitpid 18
|
||||||
|
|
||||||
namespace Syscall
|
namespace Syscall
|
||||||
{
|
{
|
||||||
@ -45,4 +46,5 @@ void sys_fcntl(Context* context, int fd, int command, uintptr_t arg);
|
|||||||
void sys_mprotect(Context* context, void* address, size_t size, int prot);
|
void sys_mprotect(Context* context, void* address, size_t size, int prot);
|
||||||
void sys_clock(Context* context);
|
void sys_clock(Context* context);
|
||||||
void sys_mkdir(Context* context, const char* filename);
|
void sys_mkdir(Context* context, const char* filename);
|
||||||
void sys_fork(Context* context);
|
void sys_fork(Context* context);
|
||||||
|
void sys_waitpid(Context* context, long pid, int* wstatus, int options);
|
@ -14,6 +14,7 @@ struct Task
|
|||||||
Idle,
|
Idle,
|
||||||
Running,
|
Running,
|
||||||
Sleeping,
|
Sleeping,
|
||||||
|
Dying,
|
||||||
Exited
|
Exited
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -126,6 +126,9 @@ void AddressSpace::clear()
|
|||||||
pages_freed++;
|
pages_freed++;
|
||||||
PMM::free_page(pdp);
|
PMM::free_page(pdp);
|
||||||
}
|
}
|
||||||
|
memset(m_pml4, 0, PAGE_SIZE);
|
||||||
|
|
||||||
|
VMM::install_kernel_page_directory_into_address_space(*this);
|
||||||
|
|
||||||
kdbgln("Reclaimed %ld pages from address space!", pages_freed);
|
kdbgln("Reclaimed %ld pages from address space!", pages_freed);
|
||||||
}
|
}
|
||||||
|
@ -130,6 +130,7 @@ void* PMM::request_pages(uint64_t count)
|
|||||||
void PMM::free_page(void* address)
|
void PMM::free_page(void* address)
|
||||||
{
|
{
|
||||||
uint64_t index = (uint64_t)address / PAGE_SIZE;
|
uint64_t index = (uint64_t)address / PAGE_SIZE;
|
||||||
|
if (index > (bitmap_size * 8)) return;
|
||||||
if (!bitmap_read(index)) return;
|
if (!bitmap_read(index)) return;
|
||||||
bitmap_set(index, false);
|
bitmap_set(index, false);
|
||||||
used_mem -= PAGE_SIZE;
|
used_mem -= PAGE_SIZE;
|
||||||
|
@ -48,8 +48,8 @@ void dump_registers(Context* context)
|
|||||||
void fatal_dump_registers(Context* context)
|
void fatal_dump_registers(Context* context)
|
||||||
{
|
{
|
||||||
printf("-- Possibly Relevant Registers:\n");
|
printf("-- Possibly Relevant Registers:\n");
|
||||||
printf("rbp: %lx, rsp: %lx, rip: %lx, cs: %lx, ss: %lx, rflags: %lx, cr2: %lx\n", context->rbp, context->rsp,
|
printf("rbp: %lx, rsp: %lx, rip: %lx, cs: %lx, ss: %lx, rflags: %lx, cr2: %lx, error code: %lx\n", context->rbp,
|
||||||
context->rip, context->cs, context->ss, context->rflags, context->cr2);
|
context->rsp, context->rip, context->cs, context->ss, context->rflags, context->cr2, context->error_code);
|
||||||
}
|
}
|
||||||
|
|
||||||
[[noreturn]] void __panic_fatal_stub(Context* context)
|
[[noreturn]] void __panic_fatal_stub(Context* context)
|
||||||
|
@ -29,6 +29,7 @@ void Syscall::entry(Context* context)
|
|||||||
case SYS_clock: sys_clock(context); break;
|
case SYS_clock: sys_clock(context); break;
|
||||||
case SYS_mkdir: sys_mkdir(context, (const char*)context->rdi); break;
|
case SYS_mkdir: sys_mkdir(context, (const char*)context->rdi); break;
|
||||||
case SYS_fork: sys_fork(context); break;
|
case SYS_fork: sys_fork(context); break;
|
||||||
|
case SYS_waitpid: sys_waitpid(context, (long)context->rdi, (int*)context->rsi, (int)context->rdx); break;
|
||||||
default: context->rax = -ENOSYS; break;
|
default: context->rax = -ENOSYS; break;
|
||||||
}
|
}
|
||||||
VMM::exit_syscall_context();
|
VMM::exit_syscall_context();
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
#include "errno.h"
|
||||||
|
#include "memory/VMM.h"
|
||||||
#include "thread/Scheduler.h"
|
#include "thread/Scheduler.h"
|
||||||
|
|
||||||
void sys_exit(Context* context, int status)
|
void sys_exit(Context* context, int status)
|
||||||
@ -18,4 +20,24 @@ void sys_sleep(Context* context, uint64_t ms)
|
|||||||
task->task_sleep = ms;
|
task->task_sleep = ms;
|
||||||
task->state = task->Sleeping;
|
task->state = task->Sleeping;
|
||||||
Scheduler::task_yield(context);
|
Scheduler::task_yield(context);
|
||||||
|
}
|
||||||
|
|
||||||
|
void sys_waitpid(Context* context, long pid, int* wstatus,
|
||||||
|
int) // FIXME: Use the value in options and block if WNOHANG has not been specified.
|
||||||
|
{
|
||||||
|
Task* child = Scheduler::find_by_pid(pid); // FIXME: Wait for any child process if PID is -1.
|
||||||
|
if (!child)
|
||||||
|
{
|
||||||
|
context->rax = -ESRCH;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (child->state != child->Dying) // FIXME: This should block if WNOHANG has not been specified.
|
||||||
|
{
|
||||||
|
context->rax = 0;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
uint64_t phys = VMM::get_physical((uint64_t)wstatus);
|
||||||
|
if (phys != (uint64_t)-1) { *(int*)phys = (int)(child->exit_status & 0xff); }
|
||||||
|
child->state = child->Exited;
|
||||||
|
context->rax = (long)child->id;
|
||||||
}
|
}
|
@ -260,7 +260,9 @@ void Scheduler::task_exit(Context* context, int64_t status)
|
|||||||
ASSERT(Interrupts::is_in_handler());
|
ASSERT(Interrupts::is_in_handler());
|
||||||
kdbgln("exit: task %ld finished running, used %ld ms of cpu time", sched_current_task->id,
|
kdbgln("exit: task %ld finished running, used %ld ms of cpu time", sched_current_task->id,
|
||||||
sched_current_task->cpu_time);
|
sched_current_task->cpu_time);
|
||||||
sched_current_task->state = sched_current_task->Exited;
|
if (sched_current_task->id == 1) sched_current_task->state = sched_current_task->Exited;
|
||||||
|
else
|
||||||
|
sched_current_task->state = sched_current_task->Dying;
|
||||||
sched_current_task->exit_status = status;
|
sched_current_task->exit_status = status;
|
||||||
task_yield(context);
|
task_yield(context);
|
||||||
}
|
}
|
||||||
|
@ -6,6 +6,7 @@ extern int errno;
|
|||||||
|
|
||||||
#define EPERM 1 // Operation not permitted
|
#define EPERM 1 // Operation not permitted
|
||||||
#define ENOENT 2 // No such file or directory
|
#define ENOENT 2 // No such file or directory
|
||||||
|
#define ESRCH 3 // No such process
|
||||||
#define ENOEXEC 8 // Exec format error
|
#define ENOEXEC 8 // Exec format error
|
||||||
#define EBADF 9 // Bad file descriptor
|
#define EBADF 9 // Bad file descriptor
|
||||||
#define ENOMEM 12 // Cannot allocate memory
|
#define ENOMEM 12 // Cannot allocate memory
|
||||||
@ -18,6 +19,6 @@ extern int errno;
|
|||||||
#define ENOSPC 28 // No space left on device
|
#define ENOSPC 28 // No space left on device
|
||||||
#define EPIPE 32 // Broken pipe. Not implemented.
|
#define EPIPE 32 // Broken pipe. Not implemented.
|
||||||
#define ENOSYS 38 // Function not implemented
|
#define ENOSYS 38 // Function not implemented
|
||||||
#define ENOTSUP 95 // Operation not supported.
|
#define ENOTSUP 95 // Operation not supported
|
||||||
|
|
||||||
#endif
|
#endif
|
@ -19,6 +19,7 @@
|
|||||||
#define SYS_clock 15
|
#define SYS_clock 15
|
||||||
#define SYS_mkdir 16
|
#define SYS_mkdir 16
|
||||||
#define SYS_fork 17
|
#define SYS_fork 17
|
||||||
|
#define SYS_waitpid 18
|
||||||
|
|
||||||
#ifndef __want_syscalls
|
#ifndef __want_syscalls
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
|
27
libs/libc/include/sys/wait.h
Normal file
27
libs/libc/include/sys/wait.h
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
#ifndef _SYS_WAIT_H
|
||||||
|
#define _SYS_WAIT_H
|
||||||
|
|
||||||
|
#include <sys/types.h>
|
||||||
|
|
||||||
|
/* Has the child process exited by calling exit() or _exit()? */
|
||||||
|
#define WIFEXITED(status) ((status) || 1)
|
||||||
|
|
||||||
|
/* What was the child's exit status? */
|
||||||
|
#define WEXITSTATUS(status) (char)((status)&0xff)
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C"
|
||||||
|
{
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Waits for the child process to finish running. */
|
||||||
|
pid_t waitpid(pid_t pid, int* wstatus, int options);
|
||||||
|
|
||||||
|
/* Waits for any child process to finish running. */
|
||||||
|
pid_t wait(int* wstatus);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
@ -193,6 +193,7 @@ extern "C"
|
|||||||
{
|
{
|
||||||
case EPERM: return "Operation not permitted";
|
case EPERM: return "Operation not permitted";
|
||||||
case EINVAL: return "Invalid argument";
|
case EINVAL: return "Invalid argument";
|
||||||
|
case ESRCH: return "No such process";
|
||||||
case ENOMEM: return "Cannot allocate memory";
|
case ENOMEM: return "Cannot allocate memory";
|
||||||
case ENOSYS: return "Function not implemented";
|
case ENOSYS: return "Function not implemented";
|
||||||
case ENOENT: return "No such file or directory";
|
case ENOENT: return "No such file or directory";
|
||||||
|
16
libs/libc/src/sys/wait.cpp
Normal file
16
libs/libc/src/sys/wait.cpp
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
#include <sys/syscall.h>
|
||||||
|
#include <sys/wait.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
extern "C"
|
||||||
|
{
|
||||||
|
pid_t waitpid(pid_t pid, int* wstatus, int options)
|
||||||
|
{
|
||||||
|
return syscall(SYS_waitpid, pid, wstatus, options);
|
||||||
|
}
|
||||||
|
|
||||||
|
pid_t wait(int* wstatus)
|
||||||
|
{
|
||||||
|
return waitpid(-1, wstatus, 0);
|
||||||
|
}
|
||||||
|
}
|
@ -64,6 +64,7 @@ extern "C"
|
|||||||
case SYS_write:
|
case SYS_write:
|
||||||
case SYS_read:
|
case SYS_read:
|
||||||
case SYS_mprotect:
|
case SYS_mprotect:
|
||||||
|
case SYS_waitpid:
|
||||||
case SYS_mmap: {
|
case SYS_mmap: {
|
||||||
arg arg0 = va_arg(ap, arg);
|
arg arg0 = va_arg(ap, arg);
|
||||||
arg arg1 = va_arg(ap, arg);
|
arg arg1 = va_arg(ap, arg);
|
||||||
|
Loading…
Reference in New Issue
Block a user