libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
ChildCloner.cxx
1// Linux
2#include <sched.h>
3#include <sys/resource.h>
4#include <sys/types.h>
5#include <unistd.h>
6
7// C++
8#include <iostream>
9
10// cosmos
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>
26
27namespace cosmos {
28
29namespace {
30
31 // creates a vector of string pointers suitable to pass as envp to execve() and friends
32 auto setup_env(const StringVector &vars) {
33 CStringVector ret;
34
35 for (const auto &var: vars) {
36 ret.push_back(var.c_str());
37 }
38
39 ret.push_back(nullptr);
40
41 return ret;
42 }
43
44 auto setup_argv(const StringVector &args) {
45 CStringVector ret;
46
47 for (const auto &arg: args) {
48 ret.push_back(arg.c_str());
49 }
50
51 ret.push_back(nullptr);
52
53 return ret;
54 }
55
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;
58 }
59
60} // end anon ns
61
62namespace {
63
64// ProcessFile with the ability to take ownership of the contained FD
65class UnsafeProcessFile :
66 public ProcessFile {
67public:
69
70 void invalidate() {
71 m_fd.reset();
72 }
73};
74
75} // end anon ns
76
78 if (m_executable.empty() || m_argv.empty()) {
79 cosmos_throw (UsageError(
80 "attempted to run a sub process w/o specifying an executable path and/or argv0"
81 ));
82 }
83
84 if (!cosmos::running_on_valgrind) {
85 // use clone3() instead of fork():
86 //
87 // clone() allows us to get a pid fd for the child in a race free
88 // fashion
89 //
90 // clone3() has fork() semantics and is easier to use for this case
91 // than the older clone syscalls.
92 //
93 // NOTE: clone3() is not yet supported in Valgrind, which
94 // means that is isn't possible to run programs that employ
95 // this system call through Valgrind anymore. clone2() is
96 // annoying to use because it doesn't have fork() semantics
97 // though ... maybe we can sit this out until Valgrind gets
98 // support for clone3().
99 PidFD pidfd;
100 CloneArgs clone_args;
101 clone_args.setPidFD(pidfd);
102 clone_args.setFlags({CloneFlag::CLEAR_SIGHAND, CloneFlag::PIDFD});
103
104 if (auto pid = proc::clone(clone_args); pid != std::nullopt) {
105 // parent process with child pid
106 return SubProc{*pid, pidfd};
107 }
108 } else {
109 // An alternative is using regular fork() and creating a
110 // pidfd from the child PID. As long as no one is collecting
111 // the child status in parallel in another thread via any of
112 // the wait() functions this is race free.
113 //
114 // This has some hairy error situations, though:
115 //
116 // - creating the pidfd after fork() can fail, in this case we
117 // cannot fulfill the API and need to end the child process
118 // again.
119 // - the child process may exit before we have a chance to
120 // create the pidfd. Basically a pidfd can also be obtained
121 // for a zombie child process. But this is not possible if
122 // the SIGCHLD handler is set so SIG_IGN. Since we cannot
123 // guarantee this in a general purpose library we need to
124 // deal with the worst case.
125 //
126 // To cover these situations we use an EventFile. The child
127 // process will not exec() before the parent process signals.
128 // If something goes wrong in the parent process then the
129 // child process is killed.
130
131 EventFile ev;
132
133 if (auto child = proc::fork(); child != std::nullopt) {
134 auto pid = *child;
135 try {
136 auto pidfile = UnsafeProcessFile{pid};
137 auto pidfd = pidfile.fd();
138 ev.signal();
139 pidfile.invalidate();
140 return SubProc{pid, pidfd};
141 } catch (...) {
142 try {
143 signal::send(pid, signal::KILL);
144 } catch (const std::exception &ex) {
145 std::cerr << "WARNING: failed to kill half-ready child process\n";
146 }
147 (void)proc::wait(pid);
148 throw;
149 }
150 } else {
151 try {
152 // wait for the parent to signal us to
153 // continue to exec()
154 ev.wait();
155 } catch (const std::exception &ex) {
156 print_child_error("post fork/ev wait", ex.what());
157 proc::exit(ExitStatus{3});
158 }
159 }
160 }
161
162 // the child process -- let's do something!
163
164 try {
165 postFork();
166
167 auto argv = setup_argv(m_argv);
168
169 if (!m_env) {
170 proc::exec(argv[0], &argv);
171 } else {
172 auto envp = setup_env(m_env.value());
173 proc::exec(argv[0], &argv, &envp);
174 }
175 } catch (const CosmosError &ce) {
176 print_child_error("post fork/exec", ce.what());
177 // use something else than "1" which might help debugging this
178 // situation a bit.
179 proc::exit(ExitStatus{3});
180 }
181
182 // should never be reached
183 return SubProc();
184}
185
187 /*
188 * the blocked signal mask is inherited via execve(), thus we need to
189 * initialize defaults here again.
190 */
191 SigSet sigs{SigSet::filled};
192 signal::unblock(sigs);
193}
194
196 if (m_post_fork_cb) {
197 m_post_fork_cb(*this);
198 }
199
200 if (m_sched_settings) {
201 try {
202 std::visit([](auto &&sched_settings) {
203 sched_settings.apply(ProcessID::SELF);
204 }, *m_sched_settings);
205 } catch(const std::exception &ex) {
206 // treat this as non-critical, the process can still
207 // run, even if not prioritized.
208 print_child_error("sched_setscheduler", ex.what());
209 }
210 }
211
212 resetSignals();
213
214 if (!m_cwd.empty()) {
215 fs::change_dir(m_cwd);
216 }
217
218 redirectFD(cosmos::stdout, m_stdout);
219 redirectFD(cosmos::stderr, m_stderr);
220 redirectFD(cosmos::stdin, m_stdin);
221
222 for (auto fd: m_inherit_fds) {
223 fd.setCloseOnExec(false);
224 }
225}
226
228 if (redirect.invalid())
229 return;
230
231 redirect.duplicate(orig, CloseOnExec(false));
232}
233
234} // end ns
235
236std::ostream& operator<<(std::ostream &o, const cosmos::ChildCloner &proc) {
237 o << "Arguments: " << proc.getArgs() << "\n";
238 if (!proc.getCWD().empty())
239 o << "CWD: " << proc.getCWD() << "\n";
240
241 return o;
242}
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.
Definition EventFile.hxx:42
Counter wait()
Wait for the counter to become non-zero.
Definition EventFile.cxx:19
void signal(const Counter increment=Counter{1})
Signal the eventfd by adding the given value to the counter.
Definition EventFile.cxx:32
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.
Definition utils.hxx:50
A specialized FileDescriptor for pidfds.
Definition PidFD.hxx:36
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.
Definition SigSet.hxx:25
Represents a child process created via ChildCloner.
Definition SubProc.hxx:30
Exception type for logical usage errors within the application.
ExitStatus
Represents an exit status code from a child process.
Definition types.hxx:43
std::vector< const char * > CStringVector
A vector of plain const char*.
Definition string.hxx:22
Argument struct for proc::clone().
Definition clone.hxx:61