3#include <sys/resource.h>
11#include <cosmos/error/InternalError.hxx>
12#include <cosmos/error/UsageError.hxx>
13#include <cosmos/formatting.hxx>
14#include <cosmos/fs/filesystem.hxx>
15#include <cosmos/io/EventFile.hxx>
16#include <cosmos/private/Scheduler.hxx>
17#include <cosmos/private/cosmos.hxx>
18#include <cosmos/proc/ChildCloner.hxx>
19#include <cosmos/proc/ProcessFile.hxx>
20#include <cosmos/proc/SigSet.hxx>
21#include <cosmos/proc/SubProc.hxx>
22#include <cosmos/proc/clone.hxx>
23#include <cosmos/proc/process.hxx>
24#include <cosmos/proc/types.hxx>
25#include <cosmos/utils.hxx>
32 auto setup_env(
const StringVector &vars) {
35 for (
const auto &var: vars) {
36 ret.push_back(var.c_str());
39 ret.push_back(
nullptr);
44 auto setup_argv(
const StringVector &args) {
47 for (
const auto &arg: args) {
48 ret.push_back(arg.c_str());
51 ret.push_back(
nullptr);
56 void print_child_error(
const std::string_view context,
const std::string_view error) {
57 std::cerr <<
"[" << proc::get_own_pid() <<
"]" << context <<
": " << error << std::endl;
65class UnsafeProcessFile :
80 "attempted to run a sub process w/o specifying an executable path and/or argv0"
84 if (!cosmos::running_on_valgrind) {
101 clone_args.setPidFD(pidfd);
102 clone_args.setFlags({CloneFlag::CLEAR_SIGHAND, CloneFlag::PIDFD});
104 if (
auto pid = proc::clone(clone_args); pid != std::nullopt) {
133 if (
auto child = proc::fork(); child != std::nullopt) {
136 auto pidfile = UnsafeProcessFile{pid};
137 auto pidfd = pidfile.fd();
139 pidfile.invalidate();
143 signal::send(pid, signal::KILL);
144 }
catch (
const std::exception &ex) {
145 std::cerr <<
"WARNING: failed to kill half-ready child process\n";
147 (void)proc::wait(pid);
155 }
catch (
const std::exception &ex) {
156 print_child_error(
"post fork/ev wait", ex.what());
167 auto argv = setup_argv(
m_argv);
170 proc::exec(argv[0], &argv);
172 auto envp = setup_env(
m_env.value());
173 proc::exec(argv[0], &argv, &envp);
176 print_child_error(
"post fork/exec", ce.
what());
191 SigSet sigs{SigSet::filled};
192 signal::unblock(sigs);
196 if (m_post_fork_cb) {
197 m_post_fork_cb(*
this);
202 std::visit([](
auto &&sched_settings) {
203 sched_settings.apply(ProcessID::SELF);
205 }
catch(
const std::exception &ex) {
208 print_child_error(
"sched_setscheduler", ex.what());
214 if (!
m_cwd.empty()) {
215 fs::change_dir(
m_cwd);
223 fd.setCloseOnExec(
false);
228 if (redirect.invalid())
237 o <<
"Arguments: " << proc.
getArgs() <<
"\n";
238 if (!proc.
getCWD().empty())
239 o <<
"CWD: " << proc.
getCWD() <<
"\n";
Sub process creation facility.
StringVector m_argv
Argument vector including argv0 denoting the executable name (which can be different than m_executabl...
const auto & getCWD() const
Returns the currently set CWD for sub process execution.
void postFork()
Performs settings needed after forking i.e. in the child process but before exec()'ing.
std::string m_executable
Path to the child process executable to run.
std::optional< StringVector > m_env
Explicit environment child environment variables, if any.
void resetSignals()
restore a default signal mask in child process context.
std::string m_cwd
Path to an explicit working directory, if any.
std::optional< SchedulerSettingsVariant > m_sched_settings
Scheduler policy settings, if any.
FileDescriptor m_stderr
File descriptor to use as child's stderr.
void redirectFD(FileDescriptor orig, FileDescriptor redirect)
Redirects the given orig file descriptor to redirect (used in child context).
const auto & getArgs() const
Returns the currently configured argument vector.
FileDescriptor m_stdout
File descriptor to use as child's stdin.
FileDescriptor m_stdin
File descriptor to use as child's stdin.
SubProc run()
Clone a new process and execute the currently configured program.
std::vector< FileDescriptor > m_inherit_fds
Additional file descriptors to inherit to the child process.
Base class for libcosmos exceptions.
const char * what() const override
Implementation of the std::exception interface.
Wrapper around an eventfd FileDescriptor.
Counter wait()
Wait for the counter to become non-zero.
void signal(const Counter increment=Counter{1})
Signal the eventfd by adding the given value to the counter.
Thin Wrapper around OS file descriptors.
void duplicate(const FileDescriptor new_fd, const CloseOnExec cloexec=CloseOnExec{true}) const
Get a duplicate file descriptor that will further be known as new_fd.
Strong template type to wrap boolean values in a named type.
A specialized FileDescriptor for pidfds.
ProcessFile(const ProcessID pid, const OpenFlags flags=OpenFlags{})
Creates a new coupling to the given process ID.
A bit set of signal numbers for use in system calls.
Represents a child process created via ChildCloner.
Exception type for logical usage errors within the application.
ExitStatus
Represents an exit status code from a child process.
std::vector< const char * > CStringVector
A vector of plain const char*.
Argument struct for proc::clone().