2020-01-18 09:38:21 +01:00
|
|
|
/*
|
2021-01-11 09:52:18 +01:00
|
|
|
* Copyright (c) 2018-2021, Andreas Kling <kling@serenityos.org>
|
2020-01-18 09:38:21 +01:00
|
|
|
*
|
2021-04-22 01:24:48 -07:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-01-18 09:38:21 +01:00
|
|
|
*/
|
|
|
|
|
2019-06-07 11:43:58 +02:00
|
|
|
#include <AK/StdLibExtras.h>
|
2019-02-17 00:13:47 +01:00
|
|
|
#include <AK/StringBuilder.h>
|
2019-05-18 02:00:01 +02:00
|
|
|
#include <AK/Time.h>
|
2019-06-07 11:43:58 +02:00
|
|
|
#include <AK/Types.h>
|
2020-07-04 17:37:36 -06:00
|
|
|
#include <Kernel/API/Syscall.h>
|
2021-06-21 17:34:09 +02:00
|
|
|
#include <Kernel/Arch/x86/InterruptDisabler.h>
|
2020-11-06 10:09:51 +02:00
|
|
|
#include <Kernel/CoreDump.h>
|
2021-01-25 16:07:10 +01:00
|
|
|
#include <Kernel/Debug.h>
|
2019-11-28 20:59:11 +01:00
|
|
|
#include <Kernel/Devices/NullDevice.h>
|
2019-05-30 18:58:59 +02:00
|
|
|
#include <Kernel/FileSystem/Custody.h>
|
2019-06-07 11:43:58 +02:00
|
|
|
#include <Kernel/FileSystem/FileDescription.h>
|
|
|
|
#include <Kernel/FileSystem/VirtualFileSystem.h>
|
2019-08-07 21:52:43 +02:00
|
|
|
#include <Kernel/KBufferBuilder.h>
|
2019-06-07 19:29:34 +02:00
|
|
|
#include <Kernel/KSyms.h>
|
2019-11-28 20:59:11 +01:00
|
|
|
#include <Kernel/Module.h>
|
2021-06-23 21:54:41 +02:00
|
|
|
#include <Kernel/Panic.h>
|
2020-02-16 02:01:42 +01:00
|
|
|
#include <Kernel/PerformanceEventBuffer.h>
|
2021-05-07 01:41:13 -07:00
|
|
|
#include <Kernel/PerformanceManager.h>
|
2019-06-07 19:29:34 +02:00
|
|
|
#include <Kernel/Process.h>
|
Kernel: Introduce the new ProcFS design
The new ProcFS design consists of two main parts:
1. The representative ProcFS class, which is derived from the FS class.
The ProcFS and its inodes are much more lean - merely 3 classes to
represent the common type of inodes - regular files, symbolic links and
directories. They're backed by a ProcFSExposedComponent object, which
is responsible for the functional operation behind the scenes.
2. The backend of the ProcFS - the ProcFSComponentsRegistrar class
and all derived classes from the ProcFSExposedComponent class. These
together form the entire backend and handle all the functions you can
expect from the ProcFS.
The ProcFSExposedComponent derived classes split to 3 types in the
manner of lifetime in the kernel:
1. Persistent objects - this category includes all basic objects, like
the root folder, /proc/bus folder, main blob files in the root folders,
etc. These objects are persistent and cannot die ever.
2. Semi-persistent objects - this category includes all PID folders,
and subdirectories to the PID folders. It also includes exposed objects
like the unveil JSON'ed blob. These object are persistent as long as the
the responsible process they represent is still alive.
3. Dynamic objects - this category includes files in the subdirectories
of a PID folder, like /proc/PID/fd/* or /proc/PID/stacks/*. Essentially,
these objects are always created dynamically and when no longer in need
after being used, they're deallocated.
Nevertheless, the new allocated backend objects and inodes try to use
the same InodeIndex if possible - this might change only when a thread
dies and a new thread is born with a new thread stack, or when a file
descriptor is closed and a new one within the same file descriptor
number is opened. This is needed to actually be able to do something
useful with these objects.
The new design assures that many ProcFS instances can be used at once,
with one backend for usage for all instances.
2021-06-12 04:23:58 +03:00
|
|
|
#include <Kernel/ProcessExposed.h>
|
2020-11-06 10:09:51 +02:00
|
|
|
#include <Kernel/RTC.h>
|
2021-06-22 17:40:16 +02:00
|
|
|
#include <Kernel/Sections.h>
|
2020-05-16 03:02:14 +04:30
|
|
|
#include <Kernel/StdLib.h>
|
2020-02-16 02:01:42 +01:00
|
|
|
#include <Kernel/TTY/TTY.h>
|
2019-10-20 10:11:40 -06:00
|
|
|
#include <Kernel/Thread.h>
|
2021-01-17 08:51:41 +01:00
|
|
|
#include <Kernel/VM/AnonymousVMObject.h>
|
2020-02-16 01:33:41 +01:00
|
|
|
#include <Kernel/VM/PageDirectory.h>
|
2020-09-05 15:52:14 -06:00
|
|
|
#include <Kernel/VM/PrivateInodeVMObject.h>
|
2020-02-28 20:29:14 +01:00
|
|
|
#include <Kernel/VM/SharedInodeVMObject.h>
|
2019-06-07 11:43:58 +02:00
|
|
|
#include <LibC/errno_numbers.h>
|
2020-01-02 13:01:41 +01:00
|
|
|
#include <LibC/limits.h>
|
2018-10-16 11:01:38 +02:00
|
|
|
|
2020-02-16 01:27:42 +01:00
|
|
|
namespace Kernel {
|
|
|
|
|
2021-02-14 00:53:53 +01:00
|
|
|
static void create_signal_trampoline();
|
2019-07-19 17:01:16 +02:00
|
|
|
|
2020-06-28 15:34:31 -06:00
|
|
|
RecursiveSpinLock g_processes_lock;
|
|
|
|
static Atomic<pid_t> next_pid;
|
2021-06-06 14:40:03 -07:00
|
|
|
READONLY_AFTER_INIT Process::List* g_processes;
|
2021-02-14 17:39:25 +01:00
|
|
|
READONLY_AFTER_INIT String* g_hostname;
|
|
|
|
READONLY_AFTER_INIT Lock* g_hostname_lock;
|
|
|
|
READONLY_AFTER_INIT HashMap<String, OwnPtr<Module>>* g_modules;
|
|
|
|
READONLY_AFTER_INIT Region* g_signal_trampoline_region;
|
2018-10-26 14:56:21 +02:00
|
|
|
|
2020-08-08 17:32:34 +02:00
|
|
|
ProcessID Process::allocate_pid()
|
2019-12-22 11:51:24 +01:00
|
|
|
{
|
2020-08-08 17:32:34 +02:00
|
|
|
// Overflow is UB, and negative PIDs wreck havoc.
|
|
|
|
// TODO: Handle PID overflow
|
|
|
|
// For example: Use an Atomic<u32>, mask the most significant bit,
|
2020-08-08 22:04:20 +02:00
|
|
|
// retry if PID is already taken as a PID, taken as a TID,
|
|
|
|
// takes as a PGID, taken as a SID, or zero.
|
2020-06-28 15:34:31 -06:00
|
|
|
return next_pid.fetch_add(1, AK::MemoryOrder::memory_order_acq_rel);
|
2019-12-22 11:51:24 +01:00
|
|
|
}
|
|
|
|
|
2021-02-19 18:41:50 +01:00
|
|
|
UNMAP_AFTER_INIT void Process::initialize()
|
2018-10-16 11:01:38 +02:00
|
|
|
{
|
2019-11-28 20:59:11 +01:00
|
|
|
g_modules = new HashMap<String, OwnPtr<Module>>;
|
|
|
|
|
2020-06-28 15:34:31 -06:00
|
|
|
next_pid.store(0, AK::MemoryOrder::memory_order_release);
|
2021-06-06 14:40:03 -07:00
|
|
|
g_processes = new Process::List();
|
2021-06-03 03:21:04 -07:00
|
|
|
g_process_groups = new ProcessGroup::List();
|
2020-07-30 23:38:15 +02:00
|
|
|
g_hostname = new String("courage");
|
|
|
|
g_hostname_lock = new Lock;
|
2019-07-19 17:01:16 +02:00
|
|
|
|
2021-02-14 00:53:53 +01:00
|
|
|
create_signal_trampoline();
|
2018-11-02 14:06:48 +01:00
|
|
|
}
|
|
|
|
|
2020-08-08 17:32:34 +02:00
|
|
|
Vector<ProcessID> Process::all_pids()
|
2019-02-03 12:33:11 +01:00
|
|
|
{
|
2020-08-08 17:32:34 +02:00
|
|
|
Vector<ProcessID> pids;
|
2020-06-28 15:34:31 -06:00
|
|
|
ScopedSpinLock lock(g_processes_lock);
|
2021-06-06 14:15:48 -07:00
|
|
|
pids.ensure_capacity(g_processes->size_slow());
|
2019-08-08 14:40:13 +02:00
|
|
|
for (auto& process : *g_processes)
|
|
|
|
pids.append(process.pid());
|
2019-02-03 12:33:11 +01:00
|
|
|
return pids;
|
|
|
|
}
|
|
|
|
|
2020-08-01 20:04:56 -06:00
|
|
|
NonnullRefPtrVector<Process> Process::all_processes()
|
2018-10-23 12:44:46 +02:00
|
|
|
{
|
2020-08-01 20:04:56 -06:00
|
|
|
NonnullRefPtrVector<Process> processes;
|
2020-06-28 15:34:31 -06:00
|
|
|
ScopedSpinLock lock(g_processes_lock);
|
2021-06-06 14:15:48 -07:00
|
|
|
processes.ensure_capacity(g_processes->size_slow());
|
2019-08-08 14:40:13 +02:00
|
|
|
for (auto& process : *g_processes)
|
2020-08-01 20:04:56 -06:00
|
|
|
processes.append(NonnullRefPtr<Process>(process));
|
2018-11-01 13:15:46 +01:00
|
|
|
return processes;
|
2018-10-23 12:44:46 +02:00
|
|
|
}
|
|
|
|
|
2019-02-27 12:32:53 +01:00
|
|
|
bool Process::in_group(gid_t gid) const
|
|
|
|
{
|
2021-03-10 20:07:08 +01:00
|
|
|
return this->gid() == gid || extra_gids().contains_slow(gid);
|
2019-02-27 12:32:53 +01:00
|
|
|
}
|
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
void Process::kill_threads_except_self()
|
2018-10-28 09:57:22 +01:00
|
|
|
{
|
2020-07-30 23:38:15 +02:00
|
|
|
InterruptDisabler disabler;
|
2018-10-28 09:57:22 +01:00
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
if (thread_count() <= 1)
|
|
|
|
return;
|
2019-12-25 21:50:13 +01:00
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
auto current_thread = Thread::current();
|
|
|
|
for_each_thread([&](Thread& thread) {
|
2021-05-16 02:36:52 -07:00
|
|
|
if (&thread == current_thread)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (auto state = thread.state(); state == Thread::State::Dead
|
|
|
|
|| state == Thread::State::Dying)
|
|
|
|
return;
|
2019-12-25 21:50:13 +01:00
|
|
|
|
2020-09-25 21:44:43 -06:00
|
|
|
// We need to detach this thread in case it hasn't been joined
|
|
|
|
thread.detach();
|
2020-07-30 23:38:15 +02:00
|
|
|
thread.set_should_die();
|
|
|
|
});
|
2020-02-28 20:47:27 +01:00
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
big_lock().clear_waiters();
|
2020-01-07 19:29:18 +01:00
|
|
|
}
|
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
void Process::kill_all_threads()
|
2019-12-30 15:11:25 -05:00
|
|
|
{
|
2020-07-30 23:38:15 +02:00
|
|
|
for_each_thread([&](Thread& thread) {
|
2020-09-25 21:44:43 -06:00
|
|
|
// We need to detach this thread in case it hasn't been joined
|
|
|
|
thread.detach();
|
2020-07-30 23:38:15 +02:00
|
|
|
thread.set_should_die();
|
|
|
|
});
|
2019-12-30 15:11:25 -05:00
|
|
|
}
|
|
|
|
|
Kernel: Introduce the new ProcFS design
The new ProcFS design consists of two main parts:
1. The representative ProcFS class, which is derived from the FS class.
The ProcFS and its inodes are much more lean - merely 3 classes to
represent the common type of inodes - regular files, symbolic links and
directories. They're backed by a ProcFSExposedComponent object, which
is responsible for the functional operation behind the scenes.
2. The backend of the ProcFS - the ProcFSComponentsRegistrar class
and all derived classes from the ProcFSExposedComponent class. These
together form the entire backend and handle all the functions you can
expect from the ProcFS.
The ProcFSExposedComponent derived classes split to 3 types in the
manner of lifetime in the kernel:
1. Persistent objects - this category includes all basic objects, like
the root folder, /proc/bus folder, main blob files in the root folders,
etc. These objects are persistent and cannot die ever.
2. Semi-persistent objects - this category includes all PID folders,
and subdirectories to the PID folders. It also includes exposed objects
like the unveil JSON'ed blob. These object are persistent as long as the
the responsible process they represent is still alive.
3. Dynamic objects - this category includes files in the subdirectories
of a PID folder, like /proc/PID/fd/* or /proc/PID/stacks/*. Essentially,
these objects are always created dynamically and when no longer in need
after being used, they're deallocated.
Nevertheless, the new allocated backend objects and inodes try to use
the same InodeIndex if possible - this might change only when a thread
dies and a new thread is born with a new thread stack, or when a file
descriptor is closed and a new one within the same file descriptor
number is opened. This is needed to actually be able to do something
useful with these objects.
The new design assures that many ProcFS instances can be used at once,
with one backend for usage for all instances.
2021-06-12 04:23:58 +03:00
|
|
|
void Process::register_new(Process& process)
|
|
|
|
{
|
|
|
|
// Note: this is essentially the same like process->ref()
|
|
|
|
RefPtr<Process> new_process = process;
|
|
|
|
ScopedSpinLock lock(g_processes_lock);
|
|
|
|
g_processes->prepend(process);
|
|
|
|
ProcFSComponentsRegistrar::the().register_new_process(process);
|
|
|
|
}
|
|
|
|
|
2020-09-27 08:53:35 -06:00
|
|
|
RefPtr<Process> Process::create_user_process(RefPtr<Thread>& first_thread, const String& path, uid_t uid, gid_t gid, ProcessID parent_pid, int& error, Vector<String>&& arguments, Vector<String>&& environment, TTY* tty)
|
2018-10-24 09:48:24 +02:00
|
|
|
{
|
2020-07-30 23:38:15 +02:00
|
|
|
auto parts = path.split('/');
|
|
|
|
if (arguments.is_empty()) {
|
|
|
|
arguments.append(parts.last());
|
2020-01-11 12:47:47 +01:00
|
|
|
}
|
2020-07-30 23:38:15 +02:00
|
|
|
RefPtr<Custody> cwd;
|
|
|
|
{
|
|
|
|
ScopedSpinLock lock(g_processes_lock);
|
2020-08-01 20:04:56 -06:00
|
|
|
if (auto parent = Process::from_pid(parent_pid)) {
|
2020-07-30 23:38:15 +02:00
|
|
|
cwd = parent->m_cwd;
|
2019-12-29 00:54:10 -05:00
|
|
|
}
|
2019-12-09 19:12:38 +01:00
|
|
|
}
|
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
if (!cwd)
|
|
|
|
cwd = VFS::the().root_custody();
|
2020-01-30 21:46:45 +01:00
|
|
|
|
2021-05-14 04:55:43 -07:00
|
|
|
auto process = Process::create(first_thread, parts.take_last(), uid, gid, parent_pid, false, move(cwd), nullptr, tty);
|
2020-09-05 15:52:14 -06:00
|
|
|
if (!first_thread)
|
|
|
|
return {};
|
2021-06-22 21:22:17 +03:00
|
|
|
if (!process->m_fds.try_resize(process->m_fds.max_open())) {
|
2021-04-30 03:18:10 -07:00
|
|
|
first_thread = nullptr;
|
|
|
|
return {};
|
|
|
|
}
|
2020-07-30 23:38:15 +02:00
|
|
|
auto& device_to_use_as_tty = tty ? (CharacterDevice&)*tty : NullDevice::the();
|
|
|
|
auto description = device_to_use_as_tty.open(O_RDWR).value();
|
|
|
|
process->m_fds[0].set(*description);
|
|
|
|
process->m_fds[1].set(*description);
|
|
|
|
process->m_fds[2].set(*description);
|
2020-01-30 21:46:45 +01:00
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
error = process->exec(path, move(arguments), move(environment));
|
|
|
|
if (error != 0) {
|
2021-01-09 00:11:15 +01:00
|
|
|
dbgln("Failed to exec {}: {}", path, error);
|
2020-09-27 08:53:35 -06:00
|
|
|
first_thread = nullptr;
|
2020-08-01 20:04:56 -06:00
|
|
|
return {};
|
2019-08-29 20:57:02 +02:00
|
|
|
}
|
|
|
|
|
Kernel: Introduce the new ProcFS design
The new ProcFS design consists of two main parts:
1. The representative ProcFS class, which is derived from the FS class.
The ProcFS and its inodes are much more lean - merely 3 classes to
represent the common type of inodes - regular files, symbolic links and
directories. They're backed by a ProcFSExposedComponent object, which
is responsible for the functional operation behind the scenes.
2. The backend of the ProcFS - the ProcFSComponentsRegistrar class
and all derived classes from the ProcFSExposedComponent class. These
together form the entire backend and handle all the functions you can
expect from the ProcFS.
The ProcFSExposedComponent derived classes split to 3 types in the
manner of lifetime in the kernel:
1. Persistent objects - this category includes all basic objects, like
the root folder, /proc/bus folder, main blob files in the root folders,
etc. These objects are persistent and cannot die ever.
2. Semi-persistent objects - this category includes all PID folders,
and subdirectories to the PID folders. It also includes exposed objects
like the unveil JSON'ed blob. These object are persistent as long as the
the responsible process they represent is still alive.
3. Dynamic objects - this category includes files in the subdirectories
of a PID folder, like /proc/PID/fd/* or /proc/PID/stacks/*. Essentially,
these objects are always created dynamically and when no longer in need
after being used, they're deallocated.
Nevertheless, the new allocated backend objects and inodes try to use
the same InodeIndex if possible - this might change only when a thread
dies and a new thread is born with a new thread stack, or when a file
descriptor is closed and a new one within the same file descriptor
number is opened. This is needed to actually be able to do something
useful with these objects.
The new design assures that many ProcFS instances can be used at once,
with one backend for usage for all instances.
2021-06-12 04:23:58 +03:00
|
|
|
register_new(*process);
|
2020-07-30 23:38:15 +02:00
|
|
|
error = 0;
|
|
|
|
return process;
|
2018-10-24 09:48:24 +02:00
|
|
|
}
|
|
|
|
|
2020-09-05 15:52:14 -06:00
|
|
|
RefPtr<Process> Process::create_kernel_process(RefPtr<Thread>& first_thread, String&& name, void (*entry)(void*), void* entry_data, u32 affinity)
|
2019-08-12 19:33:24 +02:00
|
|
|
{
|
2021-05-14 04:55:43 -07:00
|
|
|
auto process = Process::create(first_thread, move(name), (uid_t)0, (gid_t)0, ProcessID(0), true);
|
|
|
|
if (!first_thread || !process)
|
2020-09-05 15:52:14 -06:00
|
|
|
return {};
|
2021-06-23 21:54:41 +02:00
|
|
|
#if ARCH(I386)
|
2021-06-26 19:57:16 +02:00
|
|
|
first_thread->regs().eip = (FlatPtr)entry;
|
|
|
|
first_thread->regs().esp = FlatPtr(entry_data); // entry function argument is expected to be in regs.esp
|
2021-06-23 21:54:41 +02:00
|
|
|
#else
|
2021-06-26 19:57:16 +02:00
|
|
|
first_thread->regs().rip = (FlatPtr)entry;
|
2021-06-27 19:49:19 +02:00
|
|
|
first_thread->regs().rdi = FlatPtr(entry_data); // entry function argument is expected to be in regs.rdi
|
2021-06-23 21:54:41 +02:00
|
|
|
#endif
|
2019-12-30 15:11:25 -05:00
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
if (process->pid() != 0) {
|
Kernel: Introduce the new ProcFS design
The new ProcFS design consists of two main parts:
1. The representative ProcFS class, which is derived from the FS class.
The ProcFS and its inodes are much more lean - merely 3 classes to
represent the common type of inodes - regular files, symbolic links and
directories. They're backed by a ProcFSExposedComponent object, which
is responsible for the functional operation behind the scenes.
2. The backend of the ProcFS - the ProcFSComponentsRegistrar class
and all derived classes from the ProcFSExposedComponent class. These
together form the entire backend and handle all the functions you can
expect from the ProcFS.
The ProcFSExposedComponent derived classes split to 3 types in the
manner of lifetime in the kernel:
1. Persistent objects - this category includes all basic objects, like
the root folder, /proc/bus folder, main blob files in the root folders,
etc. These objects are persistent and cannot die ever.
2. Semi-persistent objects - this category includes all PID folders,
and subdirectories to the PID folders. It also includes exposed objects
like the unveil JSON'ed blob. These object are persistent as long as the
the responsible process they represent is still alive.
3. Dynamic objects - this category includes files in the subdirectories
of a PID folder, like /proc/PID/fd/* or /proc/PID/stacks/*. Essentially,
these objects are always created dynamically and when no longer in need
after being used, they're deallocated.
Nevertheless, the new allocated backend objects and inodes try to use
the same InodeIndex if possible - this might change only when a thread
dies and a new thread is born with a new thread stack, or when a file
descriptor is closed and a new one within the same file descriptor
number is opened. This is needed to actually be able to do something
useful with these objects.
The new design assures that many ProcFS instances can be used at once,
with one backend for usage for all instances.
2021-06-12 04:23:58 +03:00
|
|
|
register_new(*process);
|
2019-12-30 15:11:25 -05:00
|
|
|
}
|
|
|
|
|
2020-10-25 20:22:59 -06:00
|
|
|
ScopedSpinLock lock(g_scheduler_lock);
|
2020-07-30 23:38:15 +02:00
|
|
|
first_thread->set_affinity(affinity);
|
|
|
|
first_thread->set_state(Thread::State::Runnable);
|
|
|
|
return process;
|
|
|
|
}
|
2019-12-30 15:11:25 -05:00
|
|
|
|
2021-03-10 19:59:46 +01:00
|
|
|
void Process::protect_data()
|
|
|
|
{
|
2021-07-07 10:29:19 -06:00
|
|
|
m_protected_data_refs.unref([&]() {
|
|
|
|
MM.set_page_writable_direct(VirtualAddress { this }, false);
|
|
|
|
});
|
2021-03-10 19:59:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Process::unprotect_data()
|
|
|
|
{
|
2021-07-07 10:29:19 -06:00
|
|
|
m_protected_data_refs.ref([&]() {
|
|
|
|
MM.set_page_writable_direct(VirtualAddress { this }, true);
|
|
|
|
});
|
2021-03-10 19:59:46 +01:00
|
|
|
}
|
|
|
|
|
2021-05-14 04:55:43 -07:00
|
|
|
RefPtr<Process> Process::create(RefPtr<Thread>& first_thread, const String& name, uid_t uid, gid_t gid, ProcessID ppid, bool is_kernel_process, RefPtr<Custody> cwd, RefPtr<Custody> executable, TTY* tty, Process* fork_parent)
|
|
|
|
{
|
2021-06-20 10:21:16 +02:00
|
|
|
auto process = adopt_ref_if_nonnull(new (nothrow) Process(name, uid, gid, ppid, is_kernel_process, move(cwd), move(executable), tty));
|
2021-05-14 04:55:43 -07:00
|
|
|
if (!process)
|
|
|
|
return {};
|
|
|
|
auto result = process->attach_resources(first_thread, fork_parent);
|
|
|
|
if (result.is_error())
|
|
|
|
return {};
|
|
|
|
return process;
|
|
|
|
}
|
|
|
|
|
|
|
|
Process::Process(const String& name, uid_t uid, gid_t gid, ProcessID ppid, bool is_kernel_process, RefPtr<Custody> cwd, RefPtr<Custody> executable, TTY* tty)
|
2020-07-30 23:38:15 +02:00
|
|
|
: m_name(move(name))
|
2020-09-10 09:46:24 -06:00
|
|
|
, m_is_kernel_process(is_kernel_process)
|
2020-07-30 23:38:15 +02:00
|
|
|
, m_executable(move(executable))
|
|
|
|
, m_cwd(move(cwd))
|
|
|
|
, m_tty(tty)
|
2020-11-29 16:05:27 -07:00
|
|
|
, m_wait_block_condition(*this)
|
2020-07-30 23:38:15 +02:00
|
|
|
{
|
2021-03-11 14:46:27 +01:00
|
|
|
// Ensure that we protect the process data when exiting the constructor.
|
|
|
|
ProtectedDataMutationScope scope { *this };
|
|
|
|
|
2021-03-11 13:13:05 +01:00
|
|
|
m_pid = allocate_pid();
|
|
|
|
m_ppid = ppid;
|
|
|
|
m_uid = uid;
|
|
|
|
m_gid = gid;
|
|
|
|
m_euid = uid;
|
|
|
|
m_egid = gid;
|
|
|
|
m_suid = uid;
|
|
|
|
m_sgid = gid;
|
2021-03-10 19:59:46 +01:00
|
|
|
|
|
|
|
dbgln_if(PROCESS_DEBUG, "Created new process {}({})", m_name, this->pid().value());
|
2021-05-14 04:55:43 -07:00
|
|
|
}
|
2019-12-30 15:11:25 -05:00
|
|
|
|
2021-05-14 04:55:43 -07:00
|
|
|
KResult Process::attach_resources(RefPtr<Thread>& first_thread, Process* fork_parent)
|
|
|
|
{
|
2021-02-08 15:45:40 +01:00
|
|
|
m_space = Space::create(*this, fork_parent ? &fork_parent->space() : nullptr);
|
2021-05-14 04:55:43 -07:00
|
|
|
if (!m_space)
|
|
|
|
return ENOMEM;
|
2019-12-30 15:11:25 -05:00
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
if (fork_parent) {
|
|
|
|
// NOTE: fork() doesn't clone all threads; the thread that called fork() becomes the only thread in the new process.
|
|
|
|
first_thread = Thread::current()->clone(*this);
|
2021-05-14 04:55:43 -07:00
|
|
|
if (!first_thread)
|
|
|
|
return ENOMEM;
|
2020-07-30 23:38:15 +02:00
|
|
|
} else {
|
|
|
|
// NOTE: This non-forked code path is only taken when the kernel creates a process "manually" (at boot.)
|
2021-02-07 18:13:51 +01:00
|
|
|
auto thread_or_error = Thread::try_create(*this);
|
2021-05-14 04:55:43 -07:00
|
|
|
if (thread_or_error.is_error())
|
|
|
|
return thread_or_error.error();
|
2021-02-07 18:13:51 +01:00
|
|
|
first_thread = thread_or_error.release_value();
|
2020-09-25 21:44:43 -06:00
|
|
|
first_thread->detach();
|
2019-12-30 15:11:25 -05:00
|
|
|
}
|
2021-05-14 04:55:43 -07:00
|
|
|
return KSuccess;
|
2019-08-12 19:33:24 +02:00
|
|
|
}
|
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
Process::~Process()
|
2019-12-09 19:12:38 +01:00
|
|
|
{
|
2021-03-11 13:13:05 +01:00
|
|
|
unprotect_data();
|
|
|
|
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(thread_count() == 0); // all threads should have been finalized
|
|
|
|
VERIFY(!m_alarm_timer);
|
2020-11-29 16:05:27 -07:00
|
|
|
|
2021-05-07 01:41:13 -07:00
|
|
|
PerformanceManager::add_process_exit_event(*this);
|
2021-04-25 23:42:36 +02:00
|
|
|
|
2020-11-29 16:05:27 -07:00
|
|
|
{
|
2021-04-25 23:42:36 +02:00
|
|
|
ScopedSpinLock processes_lock(g_processes_lock);
|
2021-06-06 14:40:03 -07:00
|
|
|
if (m_list_node.is_in_list())
|
|
|
|
g_processes->remove(*this);
|
2020-11-29 16:05:27 -07:00
|
|
|
}
|
2020-07-30 23:38:15 +02:00
|
|
|
}
|
2020-01-30 21:46:45 +01:00
|
|
|
|
Kernel: Mark compilation-unit-only functions as static
This enables a nice warning in case a function becomes dead code. Also, in case
of signal_trampoline_dummy, marking it external (non-static) prevents it from
being 'optimized away', which would lead to surprising and weird linker errors.
I found these places by using -Wmissing-declarations.
The Kernel still shows these issues, which I think are false-positives,
but don't want to touch:
- Kernel/Arch/i386/CPU.cpp:1081:17: void Kernel::enter_thread_context(Kernel::Thread*, Kernel::Thread*)
- Kernel/Arch/i386/CPU.cpp:1170:17: void Kernel::context_first_init(Kernel::Thread*, Kernel::Thread*, Kernel::TrapFrame*)
- Kernel/Arch/i386/CPU.cpp:1304:16: u32 Kernel::do_init_context(Kernel::Thread*, u32)
- Kernel/Arch/i386/CPU.cpp:1347:17: void Kernel::pre_init_finished()
- Kernel/Arch/i386/CPU.cpp:1360:17: void Kernel::post_init_finished()
No idea, not gonna touch it.
- Kernel/init.cpp:104:30: void Kernel::init()
- Kernel/init.cpp:167:30: void Kernel::init_ap(u32, Kernel::Processor*)
- Kernel/init.cpp:184:17: void Kernel::init_finished(u32)
Called by boot.S.
- Kernel/init.cpp:383:16: int Kernel::__cxa_atexit(void (*)(void*), void*, void*)
- Kernel/StdLib.cpp:285:19: void __cxa_pure_virtual()
- Kernel/StdLib.cpp:300:19: void __stack_chk_fail()
- Kernel/StdLib.cpp:305:19: void __stack_chk_fail_local()
Not sure how to tell the compiler that the compiler is already using them.
Also, maybe __cxa_atexit should go into StdLib.cpp?
- Kernel/Modules/TestModule.cpp:31:17: void module_init()
- Kernel/Modules/TestModule.cpp:40:17: void module_fini()
Could maybe go into a new header. This would also provide type-checking for new modules.
2020-08-10 21:12:13 +02:00
|
|
|
// Make sure the compiler doesn't "optimize away" this function:
|
2021-04-29 14:54:15 +02:00
|
|
|
extern void signal_trampoline_dummy() __attribute__((used));
|
2020-12-20 16:09:48 -07:00
|
|
|
void signal_trampoline_dummy()
|
2020-04-12 20:22:26 +02:00
|
|
|
{
|
2021-02-25 16:18:36 +01:00
|
|
|
#if ARCH(I386)
|
2020-07-30 23:38:15 +02:00
|
|
|
// The trampoline preserves the current eax, pushes the signal code and
|
|
|
|
// then calls the signal handler. We do this because, when interrupting a
|
|
|
|
// blocking syscall, that syscall may return some special error code in eax;
|
|
|
|
// This error code would likely be overwritten by the signal handler, so it's
|
2020-10-02 22:14:37 +01:00
|
|
|
// necessary to preserve it here.
|
2020-07-30 23:38:15 +02:00
|
|
|
asm(
|
|
|
|
".intel_syntax noprefix\n"
|
|
|
|
"asm_signal_trampoline:\n"
|
|
|
|
"push ebp\n"
|
|
|
|
"mov ebp, esp\n"
|
|
|
|
"push eax\n" // we have to store eax 'cause it might be the return value from a syscall
|
|
|
|
"sub esp, 4\n" // align the stack to 16 bytes
|
|
|
|
"mov eax, [ebp+12]\n" // push the signal code
|
|
|
|
"push eax\n"
|
|
|
|
"call [ebp+8]\n" // call the signal handler
|
|
|
|
"add esp, 8\n"
|
|
|
|
"mov eax, %P0\n"
|
|
|
|
"int 0x82\n" // sigreturn syscall
|
|
|
|
"asm_signal_trampoline_end:\n"
|
|
|
|
".att_syntax" ::"i"(Syscall::SC_sigreturn));
|
2021-03-04 17:50:05 +01:00
|
|
|
#elif ARCH(X86_64)
|
2021-06-29 10:31:25 +02:00
|
|
|
// The trampoline preserves the current rax, pushes the signal code and
|
|
|
|
// then calls the signal handler. We do this because, when interrupting a
|
|
|
|
// blocking syscall, that syscall may return some special error code in eax;
|
|
|
|
// This error code would likely be overwritten by the signal handler, so it's
|
|
|
|
// necessary to preserve it here.
|
|
|
|
asm(
|
|
|
|
".intel_syntax noprefix\n"
|
|
|
|
"asm_signal_trampoline:\n"
|
|
|
|
"push rbp\n"
|
|
|
|
"mov rbp, rsp\n"
|
|
|
|
"push rax\n" // we have to store rax 'cause it might be the return value from a syscall
|
|
|
|
"sub rsp, 8\n" // align the stack to 16 bytes
|
|
|
|
"mov rdi, [rbp+24]\n" // push the signal code
|
|
|
|
"call [rbp+16]\n" // call the signal handler
|
|
|
|
"add rsp, 8\n"
|
|
|
|
"mov rax, %P0\n"
|
|
|
|
"int 0x82\n" // sigreturn syscall
|
|
|
|
"asm_signal_trampoline_end:\n"
|
|
|
|
".att_syntax" ::"i"(Syscall::SC_sigreturn));
|
2021-02-25 16:18:36 +01:00
|
|
|
#endif
|
2020-07-30 23:38:15 +02:00
|
|
|
}
|
2020-04-12 20:22:26 +02:00
|
|
|
|
2021-06-26 06:27:33 +02:00
|
|
|
extern "C" char const asm_signal_trampoline[];
|
|
|
|
extern "C" char const asm_signal_trampoline_end[];
|
2020-04-12 20:22:26 +02:00
|
|
|
|
2021-02-14 00:53:53 +01:00
|
|
|
void create_signal_trampoline()
|
2020-07-30 23:38:15 +02:00
|
|
|
{
|
|
|
|
// NOTE: We leak this region.
|
2021-02-14 01:25:22 +01:00
|
|
|
g_signal_trampoline_region = MM.allocate_kernel_region(PAGE_SIZE, "Signal trampolines", Region::Access::Read | Region::Access::Write).leak_ptr();
|
2021-02-14 00:53:53 +01:00
|
|
|
g_signal_trampoline_region->set_syscall_region(true);
|
2020-04-12 20:22:26 +02:00
|
|
|
|
2021-06-26 06:27:33 +02:00
|
|
|
size_t trampoline_size = asm_signal_trampoline_end - asm_signal_trampoline;
|
2020-04-12 20:22:26 +02:00
|
|
|
|
2021-02-14 00:53:53 +01:00
|
|
|
u8* code_ptr = (u8*)g_signal_trampoline_region->vaddr().as_ptr();
|
2021-06-26 06:27:33 +02:00
|
|
|
memcpy(code_ptr, asm_signal_trampoline, trampoline_size);
|
2020-04-12 20:22:26 +02:00
|
|
|
|
2021-02-14 00:53:53 +01:00
|
|
|
g_signal_trampoline_region->set_writable(false);
|
|
|
|
g_signal_trampoline_region->remap();
|
2020-04-12 20:22:26 +02:00
|
|
|
}
|
|
|
|
|
2021-06-28 18:32:25 +02:00
|
|
|
void Process::crash(int signal, FlatPtr ip, bool out_of_memory)
|
2019-12-09 19:12:38 +01:00
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(!is_dead());
|
|
|
|
VERIFY(Process::current() == this);
|
2020-07-30 23:38:15 +02:00
|
|
|
|
|
|
|
if (out_of_memory) {
|
2021-01-09 00:11:15 +01:00
|
|
|
dbgln("\033[31;1mOut of memory\033[m, killing: {}", *this);
|
2020-07-30 23:38:15 +02:00
|
|
|
} else {
|
2021-06-28 18:43:18 +02:00
|
|
|
if (ip >= KERNEL_BASE && g_kernel_symbols_available) {
|
2021-06-28 18:32:25 +02:00
|
|
|
auto* symbol = symbolicate_kernel_address(ip);
|
2021-07-06 12:35:26 +02:00
|
|
|
dbgln("\033[31;1m{:p} {} +{}\033[0m\n", ip, (symbol ? symbol->name : "(k?)"), (symbol ? ip - symbol->address : 0));
|
2020-07-30 23:38:15 +02:00
|
|
|
} else {
|
2021-06-28 18:32:25 +02:00
|
|
|
dbgln("\033[31;1m{:p} (?)\033[0m\n", ip);
|
2019-12-29 13:16:53 +01:00
|
|
|
}
|
2020-07-30 23:38:15 +02:00
|
|
|
dump_backtrace();
|
2019-12-09 19:12:38 +01:00
|
|
|
}
|
2021-03-11 14:24:08 +01:00
|
|
|
{
|
|
|
|
ProtectedDataMutationScope scope { *this };
|
|
|
|
m_termination_signal = signal;
|
|
|
|
}
|
2020-12-18 11:22:21 +01:00
|
|
|
set_dump_core(!out_of_memory);
|
2021-02-08 15:45:40 +01:00
|
|
|
space().dump_regions();
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(is_user_process());
|
2020-07-30 23:38:15 +02:00
|
|
|
die();
|
|
|
|
// We can not return from here, as there is nowhere
|
|
|
|
// to unwind to, so die right away.
|
|
|
|
Thread::current()->die_if_needed();
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2019-12-09 19:12:38 +01:00
|
|
|
}
|
|
|
|
|
2020-08-08 17:32:34 +02:00
|
|
|
RefPtr<Process> Process::from_pid(ProcessID pid)
|
2018-10-26 09:54:29 +02:00
|
|
|
{
|
2020-07-30 23:38:15 +02:00
|
|
|
ScopedSpinLock lock(g_processes_lock);
|
|
|
|
for (auto& process : *g_processes) {
|
2020-08-08 17:32:34 +02:00
|
|
|
process.pid();
|
2020-07-30 23:38:15 +02:00
|
|
|
if (process.pid() == pid)
|
|
|
|
return &process;
|
|
|
|
}
|
2020-08-01 20:04:56 -06:00
|
|
|
return {};
|
2018-10-26 09:54:29 +02:00
|
|
|
}
|
|
|
|
|
2021-06-22 21:22:17 +03:00
|
|
|
const Process::FileDescriptionAndFlags& Process::FileDescriptions::at(size_t i) const
|
|
|
|
{
|
|
|
|
ScopedSpinLock lock(m_fds_lock);
|
|
|
|
return m_fds_metadatas[i];
|
|
|
|
}
|
|
|
|
Process::FileDescriptionAndFlags& Process::FileDescriptions::at(size_t i)
|
|
|
|
{
|
|
|
|
ScopedSpinLock lock(m_fds_lock);
|
|
|
|
return m_fds_metadatas[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
RefPtr<FileDescription> Process::FileDescriptions::file_description(int fd) const
|
2020-04-26 15:59:47 +10:00
|
|
|
{
|
Kernel: Introduce the new ProcFS design
The new ProcFS design consists of two main parts:
1. The representative ProcFS class, which is derived from the FS class.
The ProcFS and its inodes are much more lean - merely 3 classes to
represent the common type of inodes - regular files, symbolic links and
directories. They're backed by a ProcFSExposedComponent object, which
is responsible for the functional operation behind the scenes.
2. The backend of the ProcFS - the ProcFSComponentsRegistrar class
and all derived classes from the ProcFSExposedComponent class. These
together form the entire backend and handle all the functions you can
expect from the ProcFS.
The ProcFSExposedComponent derived classes split to 3 types in the
manner of lifetime in the kernel:
1. Persistent objects - this category includes all basic objects, like
the root folder, /proc/bus folder, main blob files in the root folders,
etc. These objects are persistent and cannot die ever.
2. Semi-persistent objects - this category includes all PID folders,
and subdirectories to the PID folders. It also includes exposed objects
like the unveil JSON'ed blob. These object are persistent as long as the
the responsible process they represent is still alive.
3. Dynamic objects - this category includes files in the subdirectories
of a PID folder, like /proc/PID/fd/* or /proc/PID/stacks/*. Essentially,
these objects are always created dynamically and when no longer in need
after being used, they're deallocated.
Nevertheless, the new allocated backend objects and inodes try to use
the same InodeIndex if possible - this might change only when a thread
dies and a new thread is born with a new thread stack, or when a file
descriptor is closed and a new one within the same file descriptor
number is opened. This is needed to actually be able to do something
useful with these objects.
The new design assures that many ProcFS instances can be used at once,
with one backend for usage for all instances.
2021-06-12 04:23:58 +03:00
|
|
|
ScopedSpinLock lock(m_fds_lock);
|
2020-07-30 23:38:15 +02:00
|
|
|
if (fd < 0)
|
|
|
|
return nullptr;
|
2021-06-22 21:22:17 +03:00
|
|
|
if (static_cast<size_t>(fd) < m_fds_metadatas.size())
|
|
|
|
return m_fds_metadatas[fd].description();
|
2020-07-30 23:38:15 +02:00
|
|
|
return nullptr;
|
2020-04-26 15:59:47 +10:00
|
|
|
}
|
|
|
|
|
2021-06-22 21:22:17 +03:00
|
|
|
int Process::FileDescriptions::fd_flags(int fd) const
|
2018-11-02 20:41:58 +01:00
|
|
|
{
|
Kernel: Introduce the new ProcFS design
The new ProcFS design consists of two main parts:
1. The representative ProcFS class, which is derived from the FS class.
The ProcFS and its inodes are much more lean - merely 3 classes to
represent the common type of inodes - regular files, symbolic links and
directories. They're backed by a ProcFSExposedComponent object, which
is responsible for the functional operation behind the scenes.
2. The backend of the ProcFS - the ProcFSComponentsRegistrar class
and all derived classes from the ProcFSExposedComponent class. These
together form the entire backend and handle all the functions you can
expect from the ProcFS.
The ProcFSExposedComponent derived classes split to 3 types in the
manner of lifetime in the kernel:
1. Persistent objects - this category includes all basic objects, like
the root folder, /proc/bus folder, main blob files in the root folders,
etc. These objects are persistent and cannot die ever.
2. Semi-persistent objects - this category includes all PID folders,
and subdirectories to the PID folders. It also includes exposed objects
like the unveil JSON'ed blob. These object are persistent as long as the
the responsible process they represent is still alive.
3. Dynamic objects - this category includes files in the subdirectories
of a PID folder, like /proc/PID/fd/* or /proc/PID/stacks/*. Essentially,
these objects are always created dynamically and when no longer in need
after being used, they're deallocated.
Nevertheless, the new allocated backend objects and inodes try to use
the same InodeIndex if possible - this might change only when a thread
dies and a new thread is born with a new thread stack, or when a file
descriptor is closed and a new one within the same file descriptor
number is opened. This is needed to actually be able to do something
useful with these objects.
The new design assures that many ProcFS instances can be used at once,
with one backend for usage for all instances.
2021-06-12 04:23:58 +03:00
|
|
|
ScopedSpinLock lock(m_fds_lock);
|
2020-07-30 23:38:15 +02:00
|
|
|
if (fd < 0)
|
|
|
|
return -1;
|
2021-06-22 21:22:17 +03:00
|
|
|
if (static_cast<size_t>(fd) < m_fds_metadatas.size())
|
|
|
|
return m_fds_metadatas[fd].flags();
|
2020-07-30 23:38:15 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2018-11-10 23:29:07 +01:00
|
|
|
|
2021-06-22 21:22:17 +03:00
|
|
|
void Process::FileDescriptions::enumerate(Function<void(const FileDescriptionAndFlags&)> callback) const
|
2019-11-14 20:58:23 +01:00
|
|
|
{
|
Kernel: Introduce the new ProcFS design
The new ProcFS design consists of two main parts:
1. The representative ProcFS class, which is derived from the FS class.
The ProcFS and its inodes are much more lean - merely 3 classes to
represent the common type of inodes - regular files, symbolic links and
directories. They're backed by a ProcFSExposedComponent object, which
is responsible for the functional operation behind the scenes.
2. The backend of the ProcFS - the ProcFSComponentsRegistrar class
and all derived classes from the ProcFSExposedComponent class. These
together form the entire backend and handle all the functions you can
expect from the ProcFS.
The ProcFSExposedComponent derived classes split to 3 types in the
manner of lifetime in the kernel:
1. Persistent objects - this category includes all basic objects, like
the root folder, /proc/bus folder, main blob files in the root folders,
etc. These objects are persistent and cannot die ever.
2. Semi-persistent objects - this category includes all PID folders,
and subdirectories to the PID folders. It also includes exposed objects
like the unveil JSON'ed blob. These object are persistent as long as the
the responsible process they represent is still alive.
3. Dynamic objects - this category includes files in the subdirectories
of a PID folder, like /proc/PID/fd/* or /proc/PID/stacks/*. Essentially,
these objects are always created dynamically and when no longer in need
after being used, they're deallocated.
Nevertheless, the new allocated backend objects and inodes try to use
the same InodeIndex if possible - this might change only when a thread
dies and a new thread is born with a new thread stack, or when a file
descriptor is closed and a new one within the same file descriptor
number is opened. This is needed to actually be able to do something
useful with these objects.
The new design assures that many ProcFS instances can be used at once,
with one backend for usage for all instances.
2021-06-12 04:23:58 +03:00
|
|
|
ScopedSpinLock lock(m_fds_lock);
|
2021-06-22 21:22:17 +03:00
|
|
|
for (auto& file_description_metadata : m_fds_metadatas) {
|
|
|
|
callback(file_description_metadata);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Process::FileDescriptions::change_each(Function<void(FileDescriptionAndFlags&)> callback)
|
|
|
|
{
|
|
|
|
ScopedSpinLock lock(m_fds_lock);
|
|
|
|
for (auto& file_description_metadata : m_fds_metadatas) {
|
|
|
|
callback(file_description_metadata);
|
2020-01-10 19:15:01 +01:00
|
|
|
}
|
2021-06-22 21:22:17 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t Process::FileDescriptions::open_count() const
|
|
|
|
{
|
|
|
|
size_t count = 0;
|
|
|
|
enumerate([&](auto& file_description_metadata) {
|
|
|
|
if (file_description_metadata.is_valid())
|
|
|
|
++count;
|
|
|
|
});
|
2020-07-30 23:38:15 +02:00
|
|
|
return count;
|
2019-04-09 01:10:00 +02:00
|
|
|
}
|
2019-04-22 18:44:45 +02:00
|
|
|
|
2021-06-22 21:22:17 +03:00
|
|
|
int Process::FileDescriptions::allocate(int first_candidate_fd)
|
2019-07-22 20:01:11 +02:00
|
|
|
{
|
Kernel: Introduce the new ProcFS design
The new ProcFS design consists of two main parts:
1. The representative ProcFS class, which is derived from the FS class.
The ProcFS and its inodes are much more lean - merely 3 classes to
represent the common type of inodes - regular files, symbolic links and
directories. They're backed by a ProcFSExposedComponent object, which
is responsible for the functional operation behind the scenes.
2. The backend of the ProcFS - the ProcFSComponentsRegistrar class
and all derived classes from the ProcFSExposedComponent class. These
together form the entire backend and handle all the functions you can
expect from the ProcFS.
The ProcFSExposedComponent derived classes split to 3 types in the
manner of lifetime in the kernel:
1. Persistent objects - this category includes all basic objects, like
the root folder, /proc/bus folder, main blob files in the root folders,
etc. These objects are persistent and cannot die ever.
2. Semi-persistent objects - this category includes all PID folders,
and subdirectories to the PID folders. It also includes exposed objects
like the unveil JSON'ed blob. These object are persistent as long as the
the responsible process they represent is still alive.
3. Dynamic objects - this category includes files in the subdirectories
of a PID folder, like /proc/PID/fd/* or /proc/PID/stacks/*. Essentially,
these objects are always created dynamically and when no longer in need
after being used, they're deallocated.
Nevertheless, the new allocated backend objects and inodes try to use
the same InodeIndex if possible - this might change only when a thread
dies and a new thread is born with a new thread stack, or when a file
descriptor is closed and a new one within the same file descriptor
number is opened. This is needed to actually be able to do something
useful with these objects.
The new design assures that many ProcFS instances can be used at once,
with one backend for usage for all instances.
2021-06-12 04:23:58 +03:00
|
|
|
ScopedSpinLock lock(m_fds_lock);
|
2021-06-22 21:22:17 +03:00
|
|
|
for (size_t i = first_candidate_fd; i < max_open(); ++i) {
|
|
|
|
if (!m_fds_metadatas[i])
|
2020-07-30 23:38:15 +02:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
return -EMFILE;
|
2019-07-22 20:01:11 +02:00
|
|
|
}
|
|
|
|
|
2021-02-28 02:18:48 +01:00
|
|
|
Time kgettimeofday()
|
2019-07-19 21:08:26 +10:00
|
|
|
{
|
2021-02-28 02:18:48 +01:00
|
|
|
return TimeManagement::now();
|
2019-07-19 21:08:26 +10:00
|
|
|
}
|
|
|
|
|
2020-11-29 16:05:27 -07:00
|
|
|
siginfo_t Process::wait_info()
|
2019-08-02 23:18:47 +10:00
|
|
|
{
|
2021-02-21 02:16:16 -08:00
|
|
|
siginfo_t siginfo {};
|
2020-07-30 23:38:15 +02:00
|
|
|
siginfo.si_signo = SIGCHLD;
|
2020-11-29 16:05:27 -07:00
|
|
|
siginfo.si_pid = pid().value();
|
|
|
|
siginfo.si_uid = uid();
|
2020-05-28 21:12:13 +03:00
|
|
|
|
2020-11-29 16:05:27 -07:00
|
|
|
if (m_termination_signal) {
|
|
|
|
siginfo.si_status = m_termination_signal;
|
2020-07-30 23:38:15 +02:00
|
|
|
siginfo.si_code = CLD_KILLED;
|
|
|
|
} else {
|
2020-11-29 16:05:27 -07:00
|
|
|
siginfo.si_status = m_termination_status;
|
2020-07-30 23:38:15 +02:00
|
|
|
siginfo.si_code = CLD_EXITED;
|
2020-01-11 19:08:35 +03:00
|
|
|
}
|
2020-07-30 23:38:15 +02:00
|
|
|
return siginfo;
|
2019-07-25 21:02:19 +02:00
|
|
|
}
|
2019-07-29 07:26:01 +02:00
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
Custody& Process::current_directory()
|
2019-10-13 11:41:55 -03:00
|
|
|
{
|
2020-07-30 23:38:15 +02:00
|
|
|
if (!m_cwd)
|
|
|
|
m_cwd = VFS::the().root_custody();
|
|
|
|
return *m_cwd;
|
2019-10-13 11:41:55 -03:00
|
|
|
}
|
2019-11-02 19:34:06 +01:00
|
|
|
|
2021-05-29 16:59:40 +02:00
|
|
|
KResultOr<NonnullOwnPtr<KString>> Process::get_syscall_path_argument(char const* user_path, size_t path_length) const
|
2019-11-23 10:48:07 +03:00
|
|
|
{
|
2020-07-30 23:38:15 +02:00
|
|
|
if (path_length == 0)
|
2021-01-20 23:11:17 +01:00
|
|
|
return EINVAL;
|
2020-07-30 23:38:15 +02:00
|
|
|
if (path_length > PATH_MAX)
|
2021-01-20 23:11:17 +01:00
|
|
|
return ENAMETOOLONG;
|
2021-05-29 16:59:40 +02:00
|
|
|
auto string_or_error = try_copy_kstring_from_user(user_path, path_length);
|
|
|
|
if (string_or_error.is_error())
|
|
|
|
return string_or_error.error();
|
|
|
|
return string_or_error.release_value();
|
2019-11-23 10:48:07 +03:00
|
|
|
}
|
|
|
|
|
2021-05-29 16:59:40 +02:00
|
|
|
KResultOr<NonnullOwnPtr<KString>> Process::get_syscall_path_argument(Syscall::StringArgument const& path) const
|
2019-11-02 19:34:06 +01:00
|
|
|
{
|
2020-07-30 23:38:15 +02:00
|
|
|
return get_syscall_path_argument(path.characters, path.length);
|
2019-11-02 19:34:06 +01:00
|
|
|
}
|
|
|
|
|
2021-01-11 18:53:45 +01:00
|
|
|
bool Process::dump_core()
|
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(is_dumpable());
|
|
|
|
VERIFY(should_core_dump());
|
2021-03-10 19:59:46 +01:00
|
|
|
dbgln("Generating coredump for pid: {}", pid().value());
|
|
|
|
auto coredump_path = String::formatted("/tmp/coredump/{}_{}_{}", name(), pid().value(), RTC::now());
|
2021-01-11 18:53:45 +01:00
|
|
|
auto coredump = CoreDump::create(*this, coredump_path);
|
|
|
|
if (!coredump)
|
|
|
|
return false;
|
|
|
|
return !coredump->write().is_error();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Process::dump_perfcore()
|
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(is_dumpable());
|
|
|
|
VERIFY(m_perf_event_buffer);
|
2021-03-10 19:59:46 +01:00
|
|
|
dbgln("Generating perfcore for pid: {}", pid().value());
|
|
|
|
auto description_or_error = VFS::the().open(String::formatted("perfcore.{}", pid().value()), O_CREAT | O_EXCL, 0400, current_directory(), UidAndGid { uid(), gid() });
|
2021-01-11 18:53:45 +01:00
|
|
|
if (description_or_error.is_error())
|
|
|
|
return false;
|
|
|
|
auto& description = description_or_error.value();
|
2021-03-02 19:01:02 +01:00
|
|
|
KBufferBuilder builder;
|
|
|
|
if (!m_perf_event_buffer->to_json(builder))
|
2021-01-11 18:53:45 +01:00
|
|
|
return false;
|
|
|
|
|
2021-03-02 19:01:02 +01:00
|
|
|
auto json = builder.build();
|
|
|
|
if (!json)
|
|
|
|
return false;
|
2021-01-11 18:53:45 +01:00
|
|
|
auto json_buffer = UserOrKernelBuffer::for_kernel_buffer(json->data());
|
2021-05-22 22:07:40 +02:00
|
|
|
if (description->write(json_buffer, json->size()).is_error())
|
|
|
|
return false;
|
|
|
|
dbgln("Wrote perfcore to {}", description->absolute_path());
|
|
|
|
return true;
|
2021-01-11 18:53:45 +01:00
|
|
|
}
|
|
|
|
|
2020-12-08 21:18:45 -07:00
|
|
|
void Process::finalize()
|
2020-03-13 01:17:14 +02:00
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(Thread::current() == g_finalizer);
|
2021-01-12 22:30:52 +01:00
|
|
|
|
2021-02-07 15:33:24 +03:30
|
|
|
dbgln_if(PROCESS_DEBUG, "Finalizing process {}", *this);
|
2020-03-13 01:17:14 +02:00
|
|
|
|
2021-01-11 18:53:45 +01:00
|
|
|
if (is_dumpable()) {
|
|
|
|
if (m_should_dump_core)
|
|
|
|
dump_core();
|
2021-05-17 15:28:46 +02:00
|
|
|
if (m_perf_event_buffer) {
|
2021-01-11 18:53:45 +01:00
|
|
|
dump_perfcore();
|
2021-05-17 15:28:46 +02:00
|
|
|
TimeManagement::the().disable_profile_timer();
|
|
|
|
}
|
2020-03-13 01:17:14 +02:00
|
|
|
}
|
2019-11-02 19:34:06 +01:00
|
|
|
|
2021-01-15 20:29:13 +01:00
|
|
|
m_threads_for_coredump.clear();
|
|
|
|
|
2020-12-01 15:44:52 -07:00
|
|
|
if (m_alarm_timer)
|
|
|
|
TimerQueue::the().cancel_timer(m_alarm_timer.release_nonnull());
|
2020-07-30 23:38:15 +02:00
|
|
|
m_fds.clear();
|
|
|
|
m_tty = nullptr;
|
|
|
|
m_executable = nullptr;
|
|
|
|
m_cwd = nullptr;
|
|
|
|
m_root_directory = nullptr;
|
|
|
|
m_root_directory_relative_to_global_root = nullptr;
|
2021-01-15 20:21:03 +01:00
|
|
|
m_arguments.clear();
|
|
|
|
m_environment.clear();
|
2020-03-03 05:12:39 +01:00
|
|
|
|
Kernel: Introduce the new ProcFS design
The new ProcFS design consists of two main parts:
1. The representative ProcFS class, which is derived from the FS class.
The ProcFS and its inodes are much more lean - merely 3 classes to
represent the common type of inodes - regular files, symbolic links and
directories. They're backed by a ProcFSExposedComponent object, which
is responsible for the functional operation behind the scenes.
2. The backend of the ProcFS - the ProcFSComponentsRegistrar class
and all derived classes from the ProcFSExposedComponent class. These
together form the entire backend and handle all the functions you can
expect from the ProcFS.
The ProcFSExposedComponent derived classes split to 3 types in the
manner of lifetime in the kernel:
1. Persistent objects - this category includes all basic objects, like
the root folder, /proc/bus folder, main blob files in the root folders,
etc. These objects are persistent and cannot die ever.
2. Semi-persistent objects - this category includes all PID folders,
and subdirectories to the PID folders. It also includes exposed objects
like the unveil JSON'ed blob. These object are persistent as long as the
the responsible process they represent is still alive.
3. Dynamic objects - this category includes files in the subdirectories
of a PID folder, like /proc/PID/fd/* or /proc/PID/stacks/*. Essentially,
these objects are always created dynamically and when no longer in need
after being used, they're deallocated.
Nevertheless, the new allocated backend objects and inodes try to use
the same InodeIndex if possible - this might change only when a thread
dies and a new thread is born with a new thread stack, or when a file
descriptor is closed and a new one within the same file descriptor
number is opened. This is needed to actually be able to do something
useful with these objects.
The new design assures that many ProcFS instances can be used at once,
with one backend for usage for all instances.
2021-06-12 04:23:58 +03:00
|
|
|
// Note: We need to remove the references from the ProcFS registrar
|
|
|
|
// If we don't do it here, we can't drop the object later, and we can't
|
|
|
|
// do this from the destructor because the state of the object doesn't
|
|
|
|
// allow us to take references anymore.
|
|
|
|
ProcFSComponentsRegistrar::the().unregister_process(*this);
|
|
|
|
|
2020-11-29 16:05:27 -07:00
|
|
|
m_dead = true;
|
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
{
|
2020-08-08 17:32:34 +02:00
|
|
|
// FIXME: PID/TID BUG
|
2021-03-10 20:09:49 +01:00
|
|
|
if (auto parent_thread = Thread::from_tid(ppid().value())) {
|
2020-11-29 16:05:27 -07:00
|
|
|
if (!(parent_thread->m_signal_action_data[SIGCHLD].flags & SA_NOCLDWAIT))
|
2020-07-30 23:38:15 +02:00
|
|
|
parent_thread->send_signal(SIGCHLD, this);
|
2020-11-29 16:05:27 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
ScopedSpinLock processses_lock(g_processes_lock);
|
|
|
|
if (!!ppid()) {
|
|
|
|
if (auto parent = Process::from_pid(ppid())) {
|
|
|
|
parent->m_ticks_in_user_for_dead_children += m_ticks_in_user + m_ticks_in_user_for_dead_children;
|
|
|
|
parent->m_ticks_in_kernel_for_dead_children += m_ticks_in_kernel + m_ticks_in_kernel_for_dead_children;
|
2019-11-02 19:34:06 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-07-30 23:38:15 +02:00
|
|
|
|
2020-12-08 21:18:45 -07:00
|
|
|
unblock_waiters(Thread::WaitBlocker::UnblockFlags::Terminated);
|
2020-11-29 16:05:27 -07:00
|
|
|
|
2021-02-08 15:45:40 +01:00
|
|
|
m_space->remove_all_regions({});
|
2020-07-30 23:38:15 +02:00
|
|
|
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(ref_count() > 0);
|
2020-11-29 16:05:27 -07:00
|
|
|
// WaitBlockCondition::finalize will be in charge of dropping the last
|
|
|
|
// reference if there are still waiters around, or whenever the last
|
|
|
|
// waitable states are consumed. Unless there is no parent around
|
|
|
|
// anymore, in which case we'll just drop it right away.
|
|
|
|
m_wait_block_condition.finalize();
|
|
|
|
}
|
|
|
|
|
2020-12-08 19:04:05 -07:00
|
|
|
void Process::disowned_by_waiter(Process& process)
|
|
|
|
{
|
|
|
|
m_wait_block_condition.disowned_by_waiter(process);
|
|
|
|
}
|
|
|
|
|
2020-12-08 21:18:45 -07:00
|
|
|
void Process::unblock_waiters(Thread::WaitBlocker::UnblockFlags flags, u8 signal)
|
2020-11-29 16:05:27 -07:00
|
|
|
{
|
|
|
|
if (auto parent = Process::from_pid(ppid()))
|
2020-12-08 21:18:45 -07:00
|
|
|
parent->m_wait_block_condition.unblock(*this, flags, signal);
|
2019-11-02 19:34:06 +01:00
|
|
|
}
|
2019-11-09 22:18:16 +01:00
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
void Process::die()
|
2019-11-09 22:18:16 +01:00
|
|
|
{
|
2020-07-30 23:38:15 +02:00
|
|
|
// Let go of the TTY, otherwise a slave PTY may keep the master PTY from
|
|
|
|
// getting an EOF when the last process using the slave PTY dies.
|
|
|
|
// If the master PTY owner relies on an EOF to know when to wait() on a
|
|
|
|
// slave owner, we have to allow the PTY pair to be torn down.
|
|
|
|
m_tty = nullptr;
|
|
|
|
|
2021-06-14 15:47:19 +02:00
|
|
|
VERIFY(m_threads_for_coredump.is_empty());
|
2021-01-15 20:29:13 +01:00
|
|
|
for_each_thread([&](auto& thread) {
|
2021-01-28 08:41:18 +01:00
|
|
|
m_threads_for_coredump.append(thread);
|
2021-01-15 20:29:13 +01:00
|
|
|
});
|
|
|
|
|
2021-02-25 17:19:31 +02:00
|
|
|
{
|
|
|
|
ScopedSpinLock lock(g_processes_lock);
|
2021-06-06 14:40:03 -07:00
|
|
|
for (auto it = g_processes->begin(); it != g_processes->end();) {
|
|
|
|
auto& process = *it;
|
|
|
|
++it;
|
|
|
|
if (process.has_tracee_thread(pid())) {
|
|
|
|
dbgln_if(PROCESS_DEBUG, "Process {} ({}) is attached by {} ({}) which will exit", process.name(), process.pid(), name(), pid());
|
|
|
|
process.stop_tracing();
|
|
|
|
auto err = process.send_signal(SIGSTOP, this);
|
2021-02-25 17:19:31 +02:00
|
|
|
if (err.is_error())
|
2021-06-06 14:40:03 -07:00
|
|
|
dbgln("Failed to send the SIGSTOP signal to {} ({})", process.name(), process.pid());
|
2021-02-25 17:19:31 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
kill_all_threads();
|
2019-11-09 22:18:16 +01:00
|
|
|
}
|
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
void Process::terminate_due_to_signal(u8 signal)
|
2019-12-22 21:29:47 +01:00
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_INTERRUPTS_DISABLED();
|
|
|
|
VERIFY(signal < 32);
|
|
|
|
VERIFY(Process::current() == this);
|
2021-01-09 00:11:15 +01:00
|
|
|
dbgln("Terminating {} due to signal {}", *this, signal);
|
2021-03-11 14:24:08 +01:00
|
|
|
{
|
|
|
|
ProtectedDataMutationScope scope { *this };
|
|
|
|
m_termination_status = 0;
|
|
|
|
m_termination_signal = signal;
|
|
|
|
}
|
2020-07-30 23:38:15 +02:00
|
|
|
die();
|
2019-12-22 21:29:47 +01:00
|
|
|
}
|
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
KResult Process::send_signal(u8 signal, Process* sender)
|
2019-12-22 21:29:47 +01:00
|
|
|
{
|
2020-08-09 15:23:31 +02:00
|
|
|
// Try to send it to the "obvious" main thread:
|
2021-03-10 19:59:46 +01:00
|
|
|
auto receiver_thread = Thread::from_tid(pid().value());
|
2020-08-09 15:23:31 +02:00
|
|
|
// If the main thread has died, there may still be other threads:
|
|
|
|
if (!receiver_thread) {
|
|
|
|
// The first one should be good enough.
|
|
|
|
// Neither kill(2) nor kill(3) specify any selection precedure.
|
|
|
|
for_each_thread([&receiver_thread](Thread& thread) -> IterationDecision {
|
|
|
|
receiver_thread = &thread;
|
|
|
|
return IterationDecision::Break;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (receiver_thread) {
|
|
|
|
receiver_thread->send_signal(signal, sender);
|
2020-07-30 23:38:15 +02:00
|
|
|
return KSuccess;
|
|
|
|
}
|
2021-01-20 23:11:17 +01:00
|
|
|
return ESRCH;
|
2020-07-30 23:38:15 +02:00
|
|
|
}
|
2019-12-22 21:29:47 +01:00
|
|
|
|
2020-11-16 20:51:34 -07:00
|
|
|
RefPtr<Thread> Process::create_kernel_thread(void (*entry)(void*), void* entry_data, u32 priority, const String& name, u32 affinity, bool joinable)
|
2020-07-30 23:38:15 +02:00
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY((priority >= THREAD_PRIORITY_MIN) && (priority <= THREAD_PRIORITY_MAX));
|
2020-04-26 02:40:24 -07:00
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
// FIXME: Do something with guard pages?
|
2020-04-26 02:40:24 -07:00
|
|
|
|
2021-02-07 18:13:51 +01:00
|
|
|
auto thread_or_error = Thread::try_create(*this);
|
|
|
|
if (thread_or_error.is_error())
|
2020-09-05 15:52:14 -06:00
|
|
|
return {};
|
2020-04-26 02:40:24 -07:00
|
|
|
|
2021-02-07 14:13:30 -07:00
|
|
|
auto thread = thread_or_error.release_value();
|
2020-07-30 23:38:15 +02:00
|
|
|
thread->set_name(name);
|
|
|
|
thread->set_affinity(affinity);
|
|
|
|
thread->set_priority(priority);
|
2020-09-25 21:44:43 -06:00
|
|
|
if (!joinable)
|
|
|
|
thread->detach();
|
2020-04-26 02:40:24 -07:00
|
|
|
|
2021-06-26 19:57:16 +02:00
|
|
|
auto& regs = thread->regs();
|
2021-06-23 21:54:41 +02:00
|
|
|
#if ARCH(I386)
|
2021-06-26 19:57:16 +02:00
|
|
|
regs.eip = (FlatPtr)entry;
|
|
|
|
regs.esp = FlatPtr(entry_data); // entry function argument is expected to be in regs.rsp
|
2021-06-23 21:54:41 +02:00
|
|
|
#else
|
2021-06-26 19:57:16 +02:00
|
|
|
regs.rip = (FlatPtr)entry;
|
|
|
|
regs.rsp = FlatPtr(entry_data); // entry function argument is expected to be in regs.rsp
|
2021-06-23 21:54:41 +02:00
|
|
|
#endif
|
2019-12-22 21:29:47 +01:00
|
|
|
|
2020-10-25 20:22:59 -06:00
|
|
|
ScopedSpinLock lock(g_scheduler_lock);
|
2020-07-30 23:38:15 +02:00
|
|
|
thread->set_state(Thread::State::Runnable);
|
|
|
|
return thread;
|
2019-12-22 21:29:47 +01:00
|
|
|
}
|
2019-12-30 19:23:13 +01:00
|
|
|
|
2020-07-30 23:38:15 +02:00
|
|
|
void Process::FileDescriptionAndFlags::clear()
|
2019-12-30 19:23:13 +01:00
|
|
|
{
|
Kernel: Introduce the new ProcFS design
The new ProcFS design consists of two main parts:
1. The representative ProcFS class, which is derived from the FS class.
The ProcFS and its inodes are much more lean - merely 3 classes to
represent the common type of inodes - regular files, symbolic links and
directories. They're backed by a ProcFSExposedComponent object, which
is responsible for the functional operation behind the scenes.
2. The backend of the ProcFS - the ProcFSComponentsRegistrar class
and all derived classes from the ProcFSExposedComponent class. These
together form the entire backend and handle all the functions you can
expect from the ProcFS.
The ProcFSExposedComponent derived classes split to 3 types in the
manner of lifetime in the kernel:
1. Persistent objects - this category includes all basic objects, like
the root folder, /proc/bus folder, main blob files in the root folders,
etc. These objects are persistent and cannot die ever.
2. Semi-persistent objects - this category includes all PID folders,
and subdirectories to the PID folders. It also includes exposed objects
like the unveil JSON'ed blob. These object are persistent as long as the
the responsible process they represent is still alive.
3. Dynamic objects - this category includes files in the subdirectories
of a PID folder, like /proc/PID/fd/* or /proc/PID/stacks/*. Essentially,
these objects are always created dynamically and when no longer in need
after being used, they're deallocated.
Nevertheless, the new allocated backend objects and inodes try to use
the same InodeIndex if possible - this might change only when a thread
dies and a new thread is born with a new thread stack, or when a file
descriptor is closed and a new one within the same file descriptor
number is opened. This is needed to actually be able to do something
useful with these objects.
The new design assures that many ProcFS instances can be used at once,
with one backend for usage for all instances.
2021-06-12 04:23:58 +03:00
|
|
|
// FIXME: Verify Process::m_fds_lock is locked!
|
2020-07-30 23:50:31 +02:00
|
|
|
m_description = nullptr;
|
|
|
|
m_flags = 0;
|
Kernel: Introduce the new ProcFS design
The new ProcFS design consists of two main parts:
1. The representative ProcFS class, which is derived from the FS class.
The ProcFS and its inodes are much more lean - merely 3 classes to
represent the common type of inodes - regular files, symbolic links and
directories. They're backed by a ProcFSExposedComponent object, which
is responsible for the functional operation behind the scenes.
2. The backend of the ProcFS - the ProcFSComponentsRegistrar class
and all derived classes from the ProcFSExposedComponent class. These
together form the entire backend and handle all the functions you can
expect from the ProcFS.
The ProcFSExposedComponent derived classes split to 3 types in the
manner of lifetime in the kernel:
1. Persistent objects - this category includes all basic objects, like
the root folder, /proc/bus folder, main blob files in the root folders,
etc. These objects are persistent and cannot die ever.
2. Semi-persistent objects - this category includes all PID folders,
and subdirectories to the PID folders. It also includes exposed objects
like the unveil JSON'ed blob. These object are persistent as long as the
the responsible process they represent is still alive.
3. Dynamic objects - this category includes files in the subdirectories
of a PID folder, like /proc/PID/fd/* or /proc/PID/stacks/*. Essentially,
these objects are always created dynamically and when no longer in need
after being used, they're deallocated.
Nevertheless, the new allocated backend objects and inodes try to use
the same InodeIndex if possible - this might change only when a thread
dies and a new thread is born with a new thread stack, or when a file
descriptor is closed and a new one within the same file descriptor
number is opened. This is needed to actually be able to do something
useful with these objects.
The new design assures that many ProcFS instances can be used at once,
with one backend for usage for all instances.
2021-06-12 04:23:58 +03:00
|
|
|
m_global_procfs_inode_index = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Process::FileDescriptionAndFlags::refresh_inode_index()
|
|
|
|
{
|
|
|
|
// FIXME: Verify Process::m_fds_lock is locked!
|
|
|
|
m_global_procfs_inode_index = ProcFSComponentsRegistrar::the().allocate_inode_index();
|
2019-12-30 19:23:13 +01:00
|
|
|
}
|
2019-12-30 20:10:00 +01:00
|
|
|
|
2020-07-30 23:50:31 +02:00
|
|
|
void Process::FileDescriptionAndFlags::set(NonnullRefPtr<FileDescription>&& description, u32 flags)
|
2019-12-30 20:10:00 +01:00
|
|
|
{
|
Kernel: Introduce the new ProcFS design
The new ProcFS design consists of two main parts:
1. The representative ProcFS class, which is derived from the FS class.
The ProcFS and its inodes are much more lean - merely 3 classes to
represent the common type of inodes - regular files, symbolic links and
directories. They're backed by a ProcFSExposedComponent object, which
is responsible for the functional operation behind the scenes.
2. The backend of the ProcFS - the ProcFSComponentsRegistrar class
and all derived classes from the ProcFSExposedComponent class. These
together form the entire backend and handle all the functions you can
expect from the ProcFS.
The ProcFSExposedComponent derived classes split to 3 types in the
manner of lifetime in the kernel:
1. Persistent objects - this category includes all basic objects, like
the root folder, /proc/bus folder, main blob files in the root folders,
etc. These objects are persistent and cannot die ever.
2. Semi-persistent objects - this category includes all PID folders,
and subdirectories to the PID folders. It also includes exposed objects
like the unveil JSON'ed blob. These object are persistent as long as the
the responsible process they represent is still alive.
3. Dynamic objects - this category includes files in the subdirectories
of a PID folder, like /proc/PID/fd/* or /proc/PID/stacks/*. Essentially,
these objects are always created dynamically and when no longer in need
after being used, they're deallocated.
Nevertheless, the new allocated backend objects and inodes try to use
the same InodeIndex if possible - this might change only when a thread
dies and a new thread is born with a new thread stack, or when a file
descriptor is closed and a new one within the same file descriptor
number is opened. This is needed to actually be able to do something
useful with these objects.
The new design assures that many ProcFS instances can be used at once,
with one backend for usage for all instances.
2021-06-12 04:23:58 +03:00
|
|
|
// FIXME: Verify Process::m_fds_lock is locked!
|
2020-07-30 23:50:31 +02:00
|
|
|
m_description = move(description);
|
|
|
|
m_flags = flags;
|
Kernel: Introduce the new ProcFS design
The new ProcFS design consists of two main parts:
1. The representative ProcFS class, which is derived from the FS class.
The ProcFS and its inodes are much more lean - merely 3 classes to
represent the common type of inodes - regular files, symbolic links and
directories. They're backed by a ProcFSExposedComponent object, which
is responsible for the functional operation behind the scenes.
2. The backend of the ProcFS - the ProcFSComponentsRegistrar class
and all derived classes from the ProcFSExposedComponent class. These
together form the entire backend and handle all the functions you can
expect from the ProcFS.
The ProcFSExposedComponent derived classes split to 3 types in the
manner of lifetime in the kernel:
1. Persistent objects - this category includes all basic objects, like
the root folder, /proc/bus folder, main blob files in the root folders,
etc. These objects are persistent and cannot die ever.
2. Semi-persistent objects - this category includes all PID folders,
and subdirectories to the PID folders. It also includes exposed objects
like the unveil JSON'ed blob. These object are persistent as long as the
the responsible process they represent is still alive.
3. Dynamic objects - this category includes files in the subdirectories
of a PID folder, like /proc/PID/fd/* or /proc/PID/stacks/*. Essentially,
these objects are always created dynamically and when no longer in need
after being used, they're deallocated.
Nevertheless, the new allocated backend objects and inodes try to use
the same InodeIndex if possible - this might change only when a thread
dies and a new thread is born with a new thread stack, or when a file
descriptor is closed and a new one within the same file descriptor
number is opened. This is needed to actually be able to do something
useful with these objects.
The new design assures that many ProcFS instances can be used at once,
with one backend for usage for all instances.
2021-06-12 04:23:58 +03:00
|
|
|
m_global_procfs_inode_index = ProcFSComponentsRegistrar::the().allocate_inode_index();
|
2019-12-30 20:10:00 +01:00
|
|
|
}
|
2020-01-10 23:14:04 +01:00
|
|
|
|
|
|
|
Custody& Process::root_directory()
|
|
|
|
{
|
|
|
|
if (!m_root_directory)
|
|
|
|
m_root_directory = VFS::the().root_custody();
|
|
|
|
return *m_root_directory;
|
|
|
|
}
|
|
|
|
|
2020-01-12 21:42:01 +03:00
|
|
|
Custody& Process::root_directory_relative_to_global_root()
|
2020-01-10 23:48:44 +01:00
|
|
|
{
|
2020-01-12 21:42:01 +03:00
|
|
|
if (!m_root_directory_relative_to_global_root)
|
|
|
|
m_root_directory_relative_to_global_root = root_directory();
|
|
|
|
return *m_root_directory_relative_to_global_root;
|
2020-01-10 23:48:44 +01:00
|
|
|
}
|
|
|
|
|
2020-01-10 23:14:04 +01:00
|
|
|
void Process::set_root_directory(const Custody& root)
|
|
|
|
{
|
|
|
|
m_root_directory = root;
|
|
|
|
}
|
2020-01-11 20:48:43 +01:00
|
|
|
|
2020-02-16 02:01:42 +01:00
|
|
|
void Process::set_tty(TTY* tty)
|
|
|
|
{
|
|
|
|
m_tty = tty;
|
|
|
|
}
|
|
|
|
|
2021-05-12 21:44:07 -07:00
|
|
|
KResult Process::start_tracing_from(ProcessID tracer)
|
2020-12-08 21:18:45 -07:00
|
|
|
{
|
2021-05-12 21:44:07 -07:00
|
|
|
auto thread_tracer = ThreadTracer::create(tracer);
|
|
|
|
if (!thread_tracer)
|
|
|
|
return ENOMEM;
|
|
|
|
m_tracer = move(thread_tracer);
|
|
|
|
return KSuccess;
|
2020-12-08 21:18:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void Process::stop_tracing()
|
|
|
|
{
|
|
|
|
m_tracer = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Process::tracer_trap(Thread& thread, const RegisterState& regs)
|
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(m_tracer.ptr());
|
2020-12-08 21:18:45 -07:00
|
|
|
m_tracer->set_regs(regs);
|
|
|
|
thread.send_urgent_signal_to_self(SIGTRAP);
|
|
|
|
}
|
|
|
|
|
2021-03-02 16:55:54 +01:00
|
|
|
bool Process::create_perf_events_buffer_if_needed()
|
2021-01-11 09:52:18 +01:00
|
|
|
{
|
2021-03-02 16:55:54 +01:00
|
|
|
if (!m_perf_event_buffer) {
|
|
|
|
m_perf_event_buffer = PerformanceEventBuffer::try_create_with_size(4 * MiB);
|
2021-04-25 23:42:36 +02:00
|
|
|
m_perf_event_buffer->add_process(*this, ProcessEventType::Create);
|
2021-03-02 16:55:54 +01:00
|
|
|
}
|
|
|
|
return !!m_perf_event_buffer;
|
2021-01-11 09:52:18 +01:00
|
|
|
}
|
2021-01-22 23:24:33 -07:00
|
|
|
|
2021-04-18 21:10:05 -07:00
|
|
|
void Process::delete_perf_events_buffer()
|
|
|
|
{
|
|
|
|
if (m_perf_event_buffer)
|
|
|
|
m_perf_event_buffer = nullptr;
|
|
|
|
}
|
|
|
|
|
2021-01-22 23:24:33 -07:00
|
|
|
bool Process::remove_thread(Thread& thread)
|
|
|
|
{
|
2021-03-11 14:12:55 +01:00
|
|
|
ProtectedDataMutationScope scope { *this };
|
2021-01-22 23:24:33 -07:00
|
|
|
auto thread_cnt_before = m_thread_count.fetch_sub(1, AK::MemoryOrder::memory_order_acq_rel);
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(thread_cnt_before != 0);
|
2021-01-22 23:24:33 -07:00
|
|
|
ScopedSpinLock thread_list_lock(m_thread_list_lock);
|
|
|
|
m_thread_list.remove(thread);
|
|
|
|
return thread_cnt_before == 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Process::add_thread(Thread& thread)
|
|
|
|
{
|
2021-03-11 14:12:55 +01:00
|
|
|
ProtectedDataMutationScope scope { *this };
|
2021-01-22 23:24:33 -07:00
|
|
|
bool is_first = m_thread_count.fetch_add(1, AK::MemoryOrder::memory_order_relaxed) == 0;
|
|
|
|
ScopedSpinLock thread_list_lock(m_thread_list_lock);
|
|
|
|
m_thread_list.append(thread);
|
|
|
|
return is_first;
|
|
|
|
}
|
|
|
|
|
2021-03-10 22:42:07 +01:00
|
|
|
void Process::set_dumpable(bool dumpable)
|
|
|
|
{
|
2021-03-11 13:13:05 +01:00
|
|
|
if (dumpable == m_dumpable)
|
2021-03-10 22:42:07 +01:00
|
|
|
return;
|
2021-03-11 13:13:05 +01:00
|
|
|
ProtectedDataMutationScope scope { *this };
|
|
|
|
m_dumpable = dumpable;
|
2021-03-10 22:42:07 +01:00
|
|
|
}
|
|
|
|
|
2021-04-04 20:11:54 +02:00
|
|
|
void Process::set_coredump_metadata(const String& key, String value)
|
|
|
|
{
|
|
|
|
m_coredump_metadata.set(key, move(value));
|
|
|
|
}
|
|
|
|
|
2020-02-16 01:27:42 +01:00
|
|
|
}
|