libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
cosmos::ProcessFile Class Reference

Wrapper around a PidFD. More...

#include <ProcessFile.hxx>

Public Types

enum class  OpenFlag : unsigned int { NONBLOCK = PIDFD_NONBLOCK }
 
using OpenFlags = BitMask<OpenFlag>
 

Public Member Functions

 ProcessFile (const ProcessID pid, const OpenFlags flags=OpenFlags{})
 Creates a new coupling to the given process ID.
 
 ProcessFile (const PidFD fd)
 Wraps the given PidFD and takes ownership of it.
 
 ProcessFile (const ProcessFile &)=delete
 
ProcessFileoperator= (const ProcessFile &)=delete
 
 ProcessFile (ProcessFile &&other) noexcept
 
ProcessFileoperator= (ProcessFile &&other) noexcept
 
bool open () const
 Returns whether a pidfd is currently open.
 
void close ()
 
PidFD fd () const
 Returns the raw PidFD file descriptor.
 
void sendSignal (const Signal sig) const
 Send a signal to the represented process.
 
FileDescriptor dupFD (const FileNum targetfd) const
 Duplicate a file descriptor from the target process into the current process.
 
std::optional< ChildDatawait (const WaitFlags flags=WaitFlags{WaitFlag::WAIT_FOR_EXITED})
 Wait for the child process to exit.
 

Protected Attributes

PidFD m_fd
 

Detailed Description

Wrapper around a PidFD.

This wraps a PidFD just like a File object wraps a FileDescriptor. It adds lifetime handling i.e. closes the PidFD when no longer needed and also offers domain specific operations that can be performed on the PidFD.

Definition at line 22 of file ProcessFile.hxx.

Member Typedef Documentation

◆ OpenFlags

Definition at line 29 of file ProcessFile.hxx.

Member Enumeration Documentation

◆ OpenFlag

enum class cosmos::ProcessFile::OpenFlag : unsigned int
strong
Enumerator
NONBLOCK 

open the file descriptor in non-blocking mode - proc::wait() will never block.

Definition at line 25 of file ProcessFile.hxx.

25 : unsigned int {
26 NONBLOCK = PIDFD_NONBLOCK
27 };
@ NONBLOCK
open the file descriptor in non-blocking mode - proc::wait() will never block.

Constructor & Destructor Documentation

◆ ProcessFile() [1/3]

cosmos::ProcessFile::ProcessFile ( const ProcessID pid,
const OpenFlags flags = OpenFlags{} )
explicit

Creates a new coupling to the given process ID.

Note that creating a PidFD this way is often subject to race conditions i.e. the process with the given pid might be replaced by a different one than you expect.

It can be safe if pid is a child process of the calling process and no other thread is calling any of the wait family of functions to cleanup the child process in case it exits.

Definition at line 31 of file ProcessFile.cxx.

31 {
32 auto fd = pidfd_open(to_integral(pid), flags.raw());
33
34 if (fd == -1) {
35 cosmos_throw (ApiError("pidfd_open()"));
36 }
37
38 m_fd.setFD(FileNum{fd});
39}
void setFD(const FileNum fd)
Assigns a new primitive file descriptor to the object.
PidFD fd() const
Returns the raw PidFD file descriptor.

◆ ProcessFile() [2/3]

cosmos::ProcessFile::ProcessFile ( const PidFD fd)
inlineexplicit

Wraps the given PidFD and takes ownership of it.

The given fd will be owned by the new ProcessFile object. This means that ProcessFile will close() it if it deems this necessary.

The only way currently to obtain fd is via proc::clone().

Definition at line 52 of file ProcessFile.hxx.

52 :
53 m_fd{fd}
54 {}

◆ ProcessFile() [3/3]

cosmos::ProcessFile::ProcessFile ( ProcessFile && other)
inlinenoexcept

Definition at line 61 of file ProcessFile.hxx.

61 {
62 *this = std::move(other);
63 }

◆ ~ProcessFile()

cosmos::ProcessFile::~ProcessFile ( )

Definition at line 41 of file ProcessFile.cxx.

41 {
42 const auto orig_fd = m_fd.raw();
43
44 try {
45 close();
46 } catch (const std::exception &e) {
47 noncritical_error(sprintf("%s: failed to close fd(%d)", __FUNCTION__, to_integral(orig_fd)), e);
48 }
49}
FileNum raw() const
Returns the primitive file descriptor contained in the object.

Member Function Documentation

◆ close()

void cosmos::ProcessFile::close ( )
inline

Definition at line 78 of file ProcessFile.hxx.

78 {
79 m_fd.close();
80 }
void close()
Explicitly close the contained FD.

◆ dupFD()

FileDescriptor cosmos::ProcessFile::dupFD ( const FileNum targetfd) const

Duplicate a file descriptor from the target process into the current process.

This operation is similar to file descriptor passing over UNIX domain sockets. It doesn't require a socket connection though and also doesn't require the cooperation of the process the file descriptor is obtained from.

The operation requires PTRACE_MODE_ATTACH_REALCREDS credentials though, which roughly means the target process needs to run under the same user as the current process, or the current process needs to be privileged.

targetfd is the file descriptor number in the target process that should be duplicated into the current process.

The returned file descriptor will have the close-on-exec flag set.

The caller is responsible for closing the returned file descriptor at the appropriate time. It is best to wrap the file descriptor in a more specialized, managed type.

Definition at line 51 of file ProcessFile.cxx.

51 {
52 auto fd = pidfd_getfd(to_integral(m_fd.raw()), to_integral(targetfd), 0);
53
54 if (fd == -1) {
55 cosmos_throw (ApiError("pidfd_getfd()"));
56 }
57
58 return FileDescriptor{FileNum{fd}};
59}
FileNum
Primitive file descriptor.
Definition types.hxx:32

◆ fd()

PidFD cosmos::ProcessFile::fd ( ) const
inline

Returns the raw PidFD file descriptor.

You can use this for some operations like cosmos::proc::wait(). Make sure not to close the returned object, as ProcessFile is the owner of the file.

Definition at line 88 of file ProcessFile.hxx.

88{ return m_fd; }

◆ open()

bool cosmos::ProcessFile::open ( ) const
inline

Returns whether a pidfd is currently open.

Definition at line 74 of file ProcessFile.hxx.

74 {
75 return m_fd.valid();
76 }
bool valid() const
Returns whether currently a valid file descriptor number is assigned.

◆ operator=()

ProcessFile & cosmos::ProcessFile::operator= ( ProcessFile && other)
inlinenoexcept

Definition at line 65 of file ProcessFile.hxx.

65 {
66 m_fd = other.m_fd;
67 other.m_fd.reset();
68 return *this;
69 }

◆ sendSignal()

void cosmos::ProcessFile::sendSignal ( const Signal sig) const
inline

Send a signal to the represented process.

Definition at line 91 of file ProcessFile.hxx.

91 {
92 signal::send(m_fd, sig);
93 }

◆ wait()

std::optional< ChildData > cosmos::ProcessFile::wait ( const WaitFlags flags = WaitFlags{WaitFlag::WAIT_FOR_EXITED})
inline

Wait for the child process to exit.

See also
proc::wait(const PidFD, const WaitFlags).

Definition at line 122 of file ProcessFile.hxx.

122 {WaitFlag::WAIT_FOR_EXITED}) {
123 return proc::wait(m_fd, flags);
124 }

Member Data Documentation

◆ m_fd

PidFD cosmos::ProcessFile::m_fd
protected

Definition at line 128 of file ProcessFile.hxx.


The documentation for this class was generated from the following files: