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>
18namespace cosmos::proc {
28UserID get_real_user_id() {
29 return UserID{::getuid()};
32UserID get_effective_user_id() {
33 return UserID{::geteuid()};
36GroupID get_real_group_id() {
37 return GroupID{::getgid()};
40GroupID get_effective_group_id() {
41 return GroupID{::getegid()};
44ProcessGroupID get_own_process_group() {
45 return ProcessGroupID{getpgrp()};
51 if (res == ProcessID::INVALID) {
52 cosmos_throw (ApiError(
"setsid()"));
60 std::optional<ChildData> wait(
const idtype_t wait_type,
const id_t
id,
const WaitFlags flags) {
64 if (::waitid(wait_type,
id, si.raw(), flags.raw()) != 0) {
65 cosmos_throw (ApiError(
"wait()"));
68 if (flags[WaitFlag::NO_HANG] && si.raw()->si_pid == 0) {
74 si.raw()->si_errno = EINVAL;
75 return si.childData();
78 template <
typename STYPE>
79 CStringVector to_cstring_vector(
const std::vector<STYPE> &in_vector) {
82 for (
const auto &str: in_vector) {
83 ret.push_back(str.data());
86 ret.push_back(
nullptr);
90 using ExecFunc = std::function<int(
const char*,
char *
const[],
char *
const[])>;
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"));
101 const std::array<const char*, 2> defargs = {path.raw(),
nullptr};
108 auto casted_args =
const_cast<char**
>(args ? args->data() : defargs.data());
109 auto casted_env =
const_cast<char**
>(env ? env->data() : environ);
111 exec_func(path.raw(), casted_args, casted_env);
116std::optional<ChildData> wait(
const ProcessID pid,
const WaitFlags flags) {
117 return wait(P_PID, to_integral(pid), flags);
120std::optional<ChildData> wait(
const ProcessGroupID pgid,
const WaitFlags flags) {
121 return wait(P_PGID, to_integral(pgid), flags);
124std::optional<ChildData> wait(
const WaitFlags flags) {
125 return wait(P_ALL, 0, flags);
128std::optional<ChildData> wait(
const PidFD fd,
const WaitFlags flags) {
129 return wait(P_PIDFD, to_integral(fd.raw()), flags);
132void exec(
const SysString path,
const CStringVector *args,
const CStringVector *env) {
134 exec_wrapper(::execvpe, path, args, env);
136 cosmos_throw (ApiError(
"execvpe()"));
139void exec(
const SysString path,
140 const StringViewVector &args,
const StringViewVector *env) {
142 const auto args_vector = to_cstring_vector(args);
145 exec(path, &args_vector);
147 const auto env_vector = to_cstring_vector(*env);
148 exec(path, &args_vector, &env_vector);
152void exec(
const SysString path,
153 const StringVector &args,
const StringVector *env) {
155 const auto args_vector = to_cstring_vector(args);
158 exec(path, &args_vector);
160 const auto env_vector = to_cstring_vector(*env);
161 exec(path, &args_vector, &env_vector);
165void exec_at(
const DirFD dir_fd,
const SysString path,
166 const CStringVector *args,
const CStringVector *env,
167 const FollowSymlinks follow_symlinks) {
169 auto exec_at_wrapper = [dir_fd, follow_symlinks](
170 const char *pathname,
char *
const argv[],
char *
const envp[]) ->
int {
172 to_integral(dir_fd.raw()), pathname, argv, envp,
173 follow_symlinks ? 0 : AT_SYMLINK_NOFOLLOW);
176 exec_wrapper(exec_at_wrapper, path, args, env);
178 cosmos_throw (ApiError(
"execveat()"));
181void fexec(
const FileDescriptor fd,
const CStringVector *args,
const CStringVector *env) {
183 auto exec_at_wrapper = [fd](
184 const char *pathname,
char *
const argv[],
char *
const envp[]) ->
int {
186 return ::execveat(to_integral(fd.raw()),
"", argv, envp, AT_EMPTY_PATH);
189 exec_wrapper(exec_at_wrapper,
"", args, env);
191 cosmos_throw (ApiError(
"fexecve()"));
194void exit(ExitStatus status) {
197 _exit(to_integral(status));
200std::optional<SysString> get_env_var(
const SysString name) {
201 if (
auto res = std::getenv(name.raw()); res !=
nullptr) {
208void set_env_var(
const SysString name,
const SysString val,
const OverwriteEnv overwrite) {
210 const auto res = ::setenv(name.raw(), val.raw(), overwrite ? 1 : 0);
213 cosmos_throw (ApiError(
"setenv()"));
217void clear_env_var(
const SysString name) {
218 const auto res = ::unsetenv(name.raw());
221 cosmos_throw (ApiError(
"unsetenv()"));
225std::optional<ProcessID> fork() {
228 if (res == ProcessID::INVALID) {
229 cosmos_throw (ApiError(
"fork()"));
230 }
else if (res == ProcessID::CHILD) {
244 o << cosmos::to_integral(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;
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;
ExitStatus
Represents an exit status code from a child process.
std::vector< const char * > CStringVector
A vector of plain const char*.
Additional data found in SigInfo with SIGCHILD.
std::optional< ExitStatus > status
Contains the process's exit status, if applicable.
std::optional< Signal > signal
Contains the signal number that caused the child process to change state.
Event event
The kind of child process event that occurred.
Event
Types of SIGCHLD events that can occur.