1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
|
/*
* This source code is released into the public domain.
*/
module;
/*
* Exec providers, mostly used for spawn().
*/
#include <expected>
#include <string>
export module nihil.posix:exec;
import nihil.error;
import :argv;
import :fd;
namespace nihil {
/*
* A concept to mark spawn executors.
*/
export struct exec_tag{};
export template<typename T>
concept executor =
requires (T e) {
std::same_as<exec_tag, typename std::remove_cvref_t<T>::tag>;
{ e.exec() };
};
/*
* fexecv: use a file descriptor and an argument vector to call ::fexecve().
* This is the lowest-level executor which all others are implemented
* in terms of.
*
* TODO: Should have a way to pass the environment (envp).
*/
export struct fexecv final {
using tag = exec_tag;
fexecv(fd &&execfd, argv &&args) noexcept;
[[nodiscard]] auto exec(this fexecv &self)
-> std::expected<void, error>;
// Movable
fexecv(fexecv &&) noexcept;
auto operator=(this fexecv &, fexecv &&) noexcept -> fexecv&;
// Not copyable (because we hold the open fd object)
fexecv(fexecv const &) = delete;
auto operator=(this fexecv &, fexecv const &) -> fexecv& = delete;
private:
fd m_execfd;
argv m_args;
};
/*
* execv: equivalent to fexecv(), except the command is passed as
* a pathname instead of a file descriptor. Does not search $PATH.
*/
export [[nodiscard]] auto execv(std::string_view path, argv &&argv)
-> std::expected<fexecv, error>;
/*
* execvp: equivalent to fexecv(), except the command is passed as
* a filename instead of a file descriptor. If the filename is not
* an absolute path, it will be searched for in $PATH.
*/
export [[nodiscard]] auto execvp(std::string_view file, argv &&argv)
-> std::expected<fexecv, error>;
/*
* execl: equivalent to execv, except the arguments are passed as a
* variadic pack of string-like objects.
*/
export [[nodiscard]] auto execl(std::string_view path, auto &&...args)
-> std::expected<fexecv, error>
{
return execv(path, argv({std::string_view(args)...}));
}
/*
* execlp: equivalent to execvp, except the arguments are passed as a
* variadic pack of string-like objects.
*/
export [[nodiscard]] auto execlp(std::string_view file, auto &&...args)
-> std::expected<fexecv, error>
{
return execvp(file, argv({std::string_view(args)...}));
}
/*
* shell: run the process by invoking /bin/sh -c with the single argument,
* equivalent to system(3).
*/
export [[nodiscard]] auto shell(std::string_view const &command)
-> std::expected<fexecv, error>;
} // namespace nihil
|