libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
SubProc.cxx
1// cosmos
2#include <cosmos/error/ApiError.hxx>
3#include <cosmos/error/UsageError.hxx>
4#include <cosmos/io/Poller.hxx>
5#include <cosmos/private/cosmos.hxx>
6#include <cosmos/proc/process.hxx>
7#include <cosmos/proc/signal.hxx>
8#include <cosmos/proc/SubProc.hxx>
9
10namespace cosmos {
11
12SubProc::~SubProc() {
13 if (running()) {
14 fatal_error("destroying SubProc while child process still running");
15 }
16}
17
18void SubProc::kill(const Signal s) {
19 signal::send(m_child_fd, s);
20}
21
22void SubProc::reset() {
23 m_pid = ProcessID::INVALID;
25}
26
28
29 if (flags[WaitFlag::NO_HANG]) {
30 cosmos_throw (UsageError("cannot use NO_HANG with SubProc, use waitTimed() instead"));
31 }
32
33 try {
34 auto child = proc::wait(m_child_fd, flags);
35 if (!flags[WaitFlag::LEAVE_INFO] && (child->exited() || child->signaled())) {
36 reset();
37 }
38 return *child;
39 } catch (const ApiError &err) {
40 // for some reason the child was already collected, invalidate
41 // our state to prevent infinite loops / uncleanable SubProc
42 // objects
43 if (err.errnum() == Errno::NO_CHILD) {
44 try {
45 reset();
46 } catch(...) {
47 // FileDescriptor will be invalidated already in this case
48 }
49 }
50
51 throw;
52 }
53}
54
55std::optional<ChildData> SubProc::waitTimed(const IntervalTime max, const WaitFlags flags) {
56 Poller poller(8);
57
59
60 if (poller.wait(max).empty()) {
61 return std::nullopt;
62 }
63
64 return wait(flags);
65}
66
67SubProc& SubProc::operator=(SubProc &&other) noexcept {
68 if (running()) {
69 fatal_error("moving into SubProc object with still running child process");
70 }
71 m_pid = other.m_pid;
72 m_child_fd = other.m_child_fd;
73 other.m_pid = ProcessID::INVALID;
74 other.m_child_fd.reset();
75 return *this;
76}
77
78} // end ns
Specialized exception type used when system APIs fail.
Definition ApiError.hxx:18
auto errnum() const
Returns the plain errno stored in the exception.
Definition ApiError.hxx:34
A typesafe bit mask representation using class enums.
Definition BitMask.hxx:19
void close()
Explicitly close the contained FD.
Efficient file descriptor I/O event polling.
Definition Poller.hxx:58
std::vector< PollEvent > wait(const std::optional< IntervalTime > timeout={})
Wait for one of the monitored events to be ready.
Definition Poller.cxx:77
void addFD(const FileDescriptor fd, const MonitorFlags flags)
Start monitoring the given file descriptor using the given settings.
Definition Poller.cxx:63
@ INPUT
Monitor for read() operation becoming possible.
Represents a POSIX signal number and offers a minimal API around it.
Definition types.hxx:96
Represents a child process created via ChildCloner.
Definition SubProc.hxx:30
auto running() const
Returns whether a child process is still active.
Definition SubProc.hxx:62
void kill(const Signal signal)
Send the specified signal to the child process.
Definition SubProc.cxx:18
ChildData wait(const WaitFlags flags=WaitFlags{WaitFlag::WAIT_FOR_EXITED})
Performs a blocking wait until the child process exits.
Definition SubProc.cxx:27
std::optional< ChildData > waitTimed(const IntervalTime max, const WaitFlags flags=WaitFlags{WaitFlag::WAIT_FOR_EXITED})
Wait for sub process exit within a timeout in milliseconds.
Definition SubProc.cxx:55
PidFD m_child_fd
Pidfd referring to the active child, if any.
Definition SubProc.hxx:124
ProcessID m_pid
The pid of the child process, if any.
Definition SubProc.hxx:121
A C++ wrapper around the POSIX struct timespec coupled to a specific CLOCK type.
Definition types.hxx:57
Exception type for logical usage errors within the application.
Additional data found in SigInfo with SIGCHILD.
Definition SigInfo.hxx:321