2023-03-23 21:42:24 +00:00
|
|
|
#include "memory/MemoryManager.h"
|
|
|
|
#include "sys/Syscall.h"
|
|
|
|
#include "thread/Scheduler.h"
|
|
|
|
#include <bits/waitpid.h>
|
|
|
|
|
|
|
|
Result<u64> sys_waitpid(Registers*, SyscallArgs args)
|
|
|
|
{
|
|
|
|
pid_t pid = (pid_t)args[0];
|
|
|
|
int* status_ptr = (int*)args[1];
|
|
|
|
int options = (int)args[2];
|
|
|
|
|
2023-03-24 16:37:04 +00:00
|
|
|
Thread* current = Scheduler::current();
|
2023-03-23 21:42:24 +00:00
|
|
|
|
2023-03-24 16:37:04 +00:00
|
|
|
Thread* thread;
|
|
|
|
|
|
|
|
if (pid > 0)
|
|
|
|
{
|
|
|
|
thread = TRY(Result<Thread*>::from_option(Scheduler::find_by_pid(pid), ESRCH));
|
|
|
|
|
2023-05-04 20:58:04 +00:00
|
|
|
if (thread->parent && thread->parent != current) return err(ECHILD);
|
2023-03-24 16:37:04 +00:00
|
|
|
|
2023-05-04 21:03:31 +00:00
|
|
|
if (options & WNOHANG) return err(EAGAIN);
|
|
|
|
|
|
|
|
if (thread->state != ThreadState::Exited) kernel_wait(pid);
|
|
|
|
check(thread->state == ThreadState::Exited);
|
2023-03-24 16:37:04 +00:00
|
|
|
}
|
|
|
|
else if (pid == -1)
|
2023-03-23 21:42:24 +00:00
|
|
|
{
|
2023-04-28 13:19:01 +00:00
|
|
|
if (!Scheduler::has_children(current)) return err(ECHILD);
|
2023-03-24 16:37:04 +00:00
|
|
|
|
2023-05-04 21:03:31 +00:00
|
|
|
auto child = Scheduler::find_exited_child(current);
|
|
|
|
if (!child.has_value())
|
2023-03-24 16:37:04 +00:00
|
|
|
{
|
|
|
|
if (options & WNOHANG) return err(EAGAIN);
|
|
|
|
|
2023-05-04 21:03:31 +00:00
|
|
|
kernel_wait(pid);
|
|
|
|
check(current->child_being_waited_for.value_or(-1) != -1);
|
|
|
|
|
|
|
|
thread = TRY(Result<Thread*>::from_option(Scheduler::find_by_pid(*current->child_being_waited_for), ESRCH));
|
|
|
|
check(thread->state == ThreadState::Exited);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
thread = child.value();
|
2023-03-23 21:42:24 +00:00
|
|
|
}
|
2023-03-24 16:37:04 +00:00
|
|
|
else
|
|
|
|
return err(ENOTSUP);
|
2023-03-23 21:42:24 +00:00
|
|
|
|
2023-05-04 21:03:31 +00:00
|
|
|
current->child_being_waited_for = {};
|
|
|
|
|
2023-03-23 21:42:24 +00:00
|
|
|
int status = (int)thread->status;
|
2023-03-24 19:59:07 +00:00
|
|
|
u64 id = thread->id;
|
2023-03-23 21:42:24 +00:00
|
|
|
|
2023-05-20 10:48:07 +00:00
|
|
|
current->user_ticks_children += thread->user_ticks_self;
|
|
|
|
current->kernel_ticks_children += thread->kernel_ticks_self;
|
|
|
|
|
2023-03-23 21:42:24 +00:00
|
|
|
thread->state = ThreadState::Dying;
|
2023-05-04 21:06:00 +00:00
|
|
|
Scheduler::signal_reap_thread();
|
2023-03-23 21:42:24 +00:00
|
|
|
|
|
|
|
if (status_ptr)
|
|
|
|
if (!MemoryManager::copy_to_user_typed(status_ptr, &status)) return err(EFAULT);
|
|
|
|
|
2023-03-24 19:59:07 +00:00
|
|
|
return id;
|
2023-03-23 21:42:24 +00:00
|
|
|
}
|