libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
process.cxx
1// Linux
2#include <sys/types.h>
3#include <unistd.h>
4
5// C++
6#include <cstdlib>
7#include <functional>
8#include <ostream>
9
10// Cosmos
11#include <cosmos/error/ApiError.hxx>
12#include <cosmos/error/UsageError.hxx>
13#include <cosmos/proc/SigSet.hxx>
14#include <cosmos/proc/pidfd.h>
15#include <cosmos/proc/process.hxx>
16#include <cosmos/utils.hxx>
17
18namespace cosmos::proc {
19
20ProcessID get_own_pid() {
21 return ProcessID{::getpid()};
22}
23
24ProcessID get_parent_pid() {
25 return ProcessID{::getppid()};
26}
27
28UserID get_real_user_id() {
29 return UserID{::getuid()};
30}
31
32UserID get_effective_user_id() {
33 return UserID{::geteuid()};
34}
35
36GroupID get_real_group_id() {
37 return GroupID{::getgid()};
38}
39
40GroupID get_effective_group_id() {
41 return GroupID{::getegid()};
42}
43
44ProcessGroupID get_own_process_group() {
45 return ProcessGroupID{getpgrp()};
46}
47
48ProcessID create_new_session() {
49 auto res = ProcessID{::setsid()};
50
51 if (res == ProcessID::INVALID) {
52 cosmos_throw (ApiError("setsid()"));
53 }
54
55 return res;
56}
57
58namespace {
59
60 std::optional<ChildData> wait(const idtype_t wait_type, const id_t id, const WaitFlags flags) {
61 SigInfo si;
62 si.raw()->si_pid = 0;
63
64 if (::waitid(wait_type, id, si.raw(), flags.raw()) != 0) {
65 cosmos_throw (ApiError("wait()"));
66 }
67
68 if (flags[WaitFlag::NO_HANG] && si.raw()->si_pid == 0) {
69 return {};
70 }
71
72 // signify to SigInfo that this is a waitid() result, see
73 // SigInfo::childData() implementation.
74 si.raw()->si_errno = EINVAL;
75 return si.childData();
76 }
77
78 template <typename STYPE>
79 CStringVector to_cstring_vector(const std::vector<STYPE> &in_vector) {
80 CStringVector ret;
81
82 for (const auto &str: in_vector) {
83 ret.push_back(str.data());
84 }
85
86 ret.push_back(nullptr);
87 return ret;
88 }
89
90 using ExecFunc = std::function<int(const char*, char *const[], char *const[])>;
91
92 // wrapper to reuse execve style invocation logic
93 void exec_wrapper(ExecFunc exec_func, const SysString path,
94 const CStringVector *args, const CStringVector *env) {
95 if (args && args->back() != nullptr) {
96 cosmos_throw (UsageError("argument vector without nullptr terminator encountered"));
97 } else if(env && env->back() != nullptr) {
98 cosmos_throw (UsageError("environment vector without nullptr terminator encountered"));
99 }
100
101 const std::array<const char*, 2> defargs = {path.raw(), nullptr};
102
103 // the execve() signature uses `char* const []` declarations for argv
104 // end envp, POSIX specs say that this is only historically, since in
105 // C this could be expressed better without breaking compatibility.
106 // There is a kind of contract that the arrays aren't modified by the
107 // implementation. Thus const_cast it.
108 auto casted_args = const_cast<char**>(args ? args->data() : defargs.data());
109 auto casted_env = const_cast<char**>(env ? env->data() : environ);
110
111 exec_func(path.raw(), casted_args, casted_env);
112 }
113
114} // end anons ns
115
116std::optional<ChildData> wait(const ProcessID pid, const WaitFlags flags) {
117 return wait(P_PID, to_integral(pid), flags);
118}
119
120std::optional<ChildData> wait(const ProcessGroupID pgid, const WaitFlags flags) {
121 return wait(P_PGID, to_integral(pgid), flags);
122}
123
124std::optional<ChildData> wait(const WaitFlags flags) {
125 return wait(P_ALL, 0, flags);
126}
127
128std::optional<ChildData> wait(const PidFD fd, const WaitFlags flags) {
129 return wait(P_PIDFD, to_integral(fd.raw()), flags);
130}
131
132void exec(const SysString path, const CStringVector *args, const CStringVector *env) {
133
134 exec_wrapper(::execvpe, path, args, env);
135
136 cosmos_throw (ApiError("execvpe()"));
137}
138
139void exec(const SysString path,
140 const StringViewVector &args, const StringViewVector *env) {
141
142 const auto args_vector = to_cstring_vector(args);
143
144 if (!env) {
145 exec(path, &args_vector);
146 } else {
147 const auto env_vector = to_cstring_vector(*env);
148 exec(path, &args_vector, &env_vector);
149 }
150}
151
152void exec(const SysString path,
153 const StringVector &args, const StringVector *env) {
154
155 const auto args_vector = to_cstring_vector(args);
156
157 if (!env) {
158 exec(path, &args_vector);
159 } else {
160 const auto env_vector = to_cstring_vector(*env);
161 exec(path, &args_vector, &env_vector);
162 }
163}
164
165void exec_at(const DirFD dir_fd, const SysString path,
166 const CStringVector *args, const CStringVector *env,
167 const FollowSymlinks follow_symlinks) {
168
169 auto exec_at_wrapper = [dir_fd, follow_symlinks](
170 const char *pathname, char * const argv[], char * const envp[]) -> int {
171 return ::execveat(
172 to_integral(dir_fd.raw()), pathname, argv, envp,
173 follow_symlinks ? 0 : AT_SYMLINK_NOFOLLOW);
174 };
175
176 exec_wrapper(exec_at_wrapper, path, args, env);
177
178 cosmos_throw (ApiError("execveat()"));
179}
180
181void fexec(const FileDescriptor fd, const CStringVector *args, const CStringVector *env) {
182
183 auto exec_at_wrapper = [fd](
184 const char *pathname, char * const argv[], char * const envp[]) -> int {
185 (void)pathname;
186 return ::execveat(to_integral(fd.raw()), "", argv, envp, AT_EMPTY_PATH);
187 };
188
189 exec_wrapper(exec_at_wrapper, "", args, env);
190
191 cosmos_throw (ApiError("fexecve()"));
192}
193
194void exit(ExitStatus status) {
195 // Note: the glibc wrapper we use here ends the complete process, the
196 // actual Linux system call _exit only ends the calling thread.
197 _exit(to_integral(status));
198}
199
200std::optional<SysString> get_env_var(const SysString name) {
201 if (auto res = std::getenv(name.raw()); res != nullptr) {
202 return {res};
203 }
204
205 return {};
206}
207
208void set_env_var(const SysString name, const SysString val, const OverwriteEnv overwrite) {
209 // NOTE: this is POSIX, not existing in C specs
210 const auto res = ::setenv(name.raw(), val.raw(), overwrite ? 1 : 0);
211
212 if (res != 0) {
213 cosmos_throw (ApiError("setenv()"));
214 }
215}
216
217void clear_env_var(const SysString name) {
218 const auto res = ::unsetenv(name.raw());
219
220 if (res != 0) {
221 cosmos_throw (ApiError("unsetenv()"));
222 }
223}
224
225std::optional<ProcessID> fork() {
226 ProcessID res{::fork()};
227
228 if (res == ProcessID::INVALID) {
229 cosmos_throw (ApiError("fork()"));
230 } else if (res == ProcessID::CHILD) {
231 return {};
232 }
233
234 return res;
235}
236
237PidInfo cached_pids;
238
239} // end ns
240
241std::ostream& operator<<(std::ostream &o, const cosmos::ExitStatus status) {
244 o << cosmos::to_integral(status);
245 switch (status) {
246 case cosmos::ExitStatus::INVALID: o << " (INVALID)"; break;
247 case cosmos::ExitStatus::SUCCESS: o << " (SUCCESS)"; break;
248 case cosmos::ExitStatus::FAILURE: o << " (FAILURE)"; break;
249 default: o << " (other)"; break;
250 }
251 return o;
252}
253
254std::ostream& operator<<(std::ostream &o, const cosmos::ChildData &info) {
255 using Event = cosmos::ChildData::Event;
256
257 switch (info.event) {
258 case Event::EXITED: o << "Child exited with " << *info.status; break;
259 case Event::KILLED: o << "Child killed by " << *info.signal; break;
260 case Event::DUMPED: o << "Child killed by " << *info.signal << " (dumped core)"; break;
261 case Event::TRAPPED: o << "Child trapped"; break;
262 case Event::STOPPED: o << "Child stopped by " << *info.signal; break;
263 case Event::CONTINUED: o << "Child continued by " << *info.signal; break;
264 default: o << "Bad ChildData"; break;
265 }
266
267 return o;
268}
ProcessID
Definition types.hxx:25
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
Additional data found in SigInfo with SIGCHILD.
Definition SigInfo.hxx:321
std::optional< ExitStatus > status
Contains the process's exit status, if applicable.
Definition SigInfo.hxx:392
std::optional< Signal > signal
Contains the signal number that caused the child process to change state.
Definition SigInfo.hxx:399
Event event
The kind of child process event that occurred.
Definition SigInfo.hxx:383
Event
Types of SIGCHLD events that can occur.
Definition SigInfo.hxx:325