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

Efficient file descriptor I/O event polling. More...

#include <Poller.hxx>

Classes

struct  PollEvent
 A single poll event as returned by wait(). More...
 

Public Types

enum class  MonitorFlag : uint32_t {
  INPUT = EPOLLIN , OUTPUT = EPOLLOUT , SOCKET_HANGUP = EPOLLRDHUP , EXCEPTIONS = EPOLLPRI ,
  EDGE_TRIGGERED = EPOLLET , ONESHOT = EPOLLONESHOT , STAY_AWAKE = EPOLLWAKEUP
}
 Flags used to declare interest in specific events and options in addFD() and modFD(). More...
 
enum class  Event : uint32_t {
  INPUT_READY = EPOLLIN , OUTPUT_READY = EPOLLOUT , SOCKET_HANGUP = EPOLLRDHUP , EXCEPTION_OCCURED = EPOLLPRI ,
  ERROR_OCCURED = EPOLLERR , HANGUP_OCCURED = EPOLLHUP
}
 Flags found in PollEvent that indicate the events that occurred on a file descriptor. More...
 
using MonitorFlags = BitMask<MonitorFlag>
 
using EventMask = BitMask<Event>
 

Public Member Functions

 Poller ()
 Creates a yet invalid Poller instance.
 
 Poller (size_t max_events)
 Creates a Poller instance ready for use.
 
 ~Poller ()
 Calls close()
 
 Poller (const Poller &)=delete
 Avoid copying due to the file descriptor member.
 
Polleroperator= (const Poller &)=delete
 
void create (size_t max_events=16)
 Actually create the poll file descriptor backing this object.
 
void close ()
 Closes a previously create()'d poll file descriptor again.
 
bool valid () const
 Returns whether currently a valid poll file descriptor exists.
 
void addFD (const FileDescriptor fd, const MonitorFlags flags)
 Start monitoring the given file descriptor using the given settings.
 
void modFD (const FileDescriptor fd, const MonitorFlags flags)
 Modify monitoring settings for an already monitored descriptor.
 
void delFD (const FileDescriptor fd)
 Remove a file descriptor from the set of monitored files.
 
std::vector< PollEventwait (const std::optional< IntervalTime > timeout={})
 Wait for one of the monitored events to be ready.
 

Protected Member Functions

int rawPollFD () const
 

Protected Attributes

FileDescriptor m_poll_fd
 
std::vector< PollEventm_events
 

Detailed Description

Efficient file descriptor I/O event polling.

This class provides a wrapper around the epoll() Linux specific file descriptor monitoring API. The API operates on a file descriptor of its own that references a set of monitored file descriptors.

A peculiarity of the API is that it can operate in a level triggered or an edge triggered fashion. The level triggered mode is the one known from classical APIs like select(). It means that a file descriptor will always be signaled as ready if currently one of the monitoring conditions is fulfilled. Edge triggered instead means that the condition is only signaled once for a single event and afterwards only triggered again if additional events occur, regardless of whether data was actually read/written to the monitored file descriptors or not.

The edge triggered approach can be more efficient as it requires less system calls on high I/O load. However, it also requires more care taken by the implementation of the userspace application. The general recommendation is that all monitored file descriptors should be operated in non-blocking mode in this case and once an event is signaled the respective file descriptor should be read from / written to until an EAGAIN result is encountered. Only then should the poll API be consulted again for waiting for further events.

Special care also is required when file descriptors that are monitored are closed within the application. The kernel monitors open file descriptions here, not only open file descriptors. This means if there exist copies of a file descriptor then closing one of the involved file descriptors will not end monitoring of the still open file description.

File descriptors signaled as being ready for non-blocking I/O could still be blocking for example in case of networking sockets, e.g. if a received packet has an invalid checksum and therefore nothing to return to userspace. To avoid such scenarios the application should use non-blocking file descriptors and react to EAGAIN results upon read/write.

The Poller FD is created, as usual, with the O_CLOEXEC flag set. Explicitly re-enable the flag should you require inheritance to unrelated sub processes.

Definition at line 58 of file Poller.hxx.

Member Typedef Documentation

◆ EventMask

Definition at line 97 of file Poller.hxx.

◆ MonitorFlags

Definition at line 79 of file Poller.hxx.

Member Enumeration Documentation

◆ Event

enum class cosmos::Poller::Event : uint32_t
strong

Flags found in PollEvent that indicate the events that occurred on a file descriptor.

Enumerator
INPUT_READY 
See also
MonitorFlag::INPUT
OUTPUT_READY 
See also
MonitorFlag::OUTPUT
SOCKET_HANGUP 
See also
MonitorFlag::SOCKET_HANGUP
EXCEPTION_OCCURED 
See also
MonitorFlag::EXCEPTIONS
ERROR_OCCURED 

An error condition occurred on the file descriptor (this is also reported for the write end of a pipe, if the read end is closed). This event is always reported independently of MonitorFlag.

HANGUP_OCCURED 

Socket or pipe peer has hung up. Data may still be pending though. This event is always reported independently of MonitorFlags.

Definition at line 82 of file Poller.hxx.

82 : uint32_t {
84 INPUT_READY = EPOLLIN,
86 OUTPUT_READY = EPOLLOUT,
88 SOCKET_HANGUP = EPOLLRDHUP,
90 EXCEPTION_OCCURED = EPOLLPRI,
92 ERROR_OCCURED = EPOLLERR,
94 HANGUP_OCCURED = EPOLLHUP
95 };
@ ERROR_OCCURED
An error condition occurred on the file descriptor (this is also reported for the write end of a pipe...
@ HANGUP_OCCURED
Socket or pipe peer has hung up. Data may still be pending though. This event is always reported inde...
@ SOCKET_HANGUP
Monitor for stream socket peer closed or shut down the write half of the connection (data may still b...

◆ MonitorFlag

enum class cosmos::Poller::MonitorFlag : uint32_t
strong

Flags used to declare interest in specific events and options in addFD() and modFD().

Enumerator
INPUT 

Monitor for read() operation becoming possible.

OUTPUT 

Monitor for write() operation becoming possible.

SOCKET_HANGUP 

Monitor for stream socket peer closed or shut down the write half of the connection (data may still be pending)

EXCEPTIONS 

Monitor for exceptional conditions occuring on the file descriptor, depending on the actual file type.

EDGE_TRIGGERED 

Operate in edge triggered mode instead of level triggered (which is the default)

ONESHOT 

Only report events once, then disable monitoring until this flag is set again using modFD()

STAY_AWAKE 

If the process has the CAP_BLOCK_SUSPEND capability then the system won't enter a suspend state until the process that received this event calls wait() again.

Definition at line 62 of file Poller.hxx.

62 : uint32_t {
64 INPUT = EPOLLIN,
66 OUTPUT = EPOLLOUT,
68 SOCKET_HANGUP = EPOLLRDHUP,
70 EXCEPTIONS = EPOLLPRI,
72 EDGE_TRIGGERED = EPOLLET,
74 ONESHOT = EPOLLONESHOT,
76 STAY_AWAKE = EPOLLWAKEUP
77 };
@ EXCEPTIONS
Monitor for exceptional conditions occuring on the file descriptor, depending on the actual file type...
@ OUTPUT
Monitor for write() operation becoming possible.
@ EDGE_TRIGGERED
Operate in edge triggered mode instead of level triggered (which is the default)
@ INPUT
Monitor for read() operation becoming possible.
@ ONESHOT
Only report events once, then disable monitoring until this flag is set again using modFD()
@ STAY_AWAKE
If the process has the CAP_BLOCK_SUSPEND capability then the system won't enter a suspend state until...

Constructor & Destructor Documentation

◆ Poller() [1/2]

cosmos::Poller::Poller ( )
inline

Creates a yet invalid Poller instance.

Definition at line 113 of file Poller.hxx.

113{}

◆ Poller() [2/2]

cosmos::Poller::Poller ( size_t max_events)
inlineexplicit

Creates a Poller instance ready for use.

See also
create()

Definition at line 119 of file Poller.hxx.

119 {
120 create(max_events);
121 }
void create(size_t max_events=16)
Actually create the poll file descriptor backing this object.
Definition Poller.cxx:29

◆ ~Poller()

cosmos::Poller::~Poller ( )

Calls close()

Definition at line 15 of file Poller.cxx.

15 {
16 try {
17 close();
18 } catch (const std::exception &ex) {
19 noncritical_error(
20 sprintf("%s: failed to close()", __FUNCTION__),
21 ex);
22 }
23}
void close()
Closes a previously create()'d poll file descriptor again.
Definition Poller.cxx:43

Member Function Documentation

◆ addFD()

void cosmos::Poller::addFD ( const FileDescriptor fd,
const MonitorFlags flags )

Start monitoring the given file descriptor using the given settings.

If currently no valid poll FD exists then this will throw an ApiError exception.

Adding the same file descriptor twice also causes an error. Use modFD() to modify monitoring settings for FDs already monitored.

Definition at line 63 of file Poller.cxx.

63 {
64 control(rawPollFD(), to_integral(fd.raw()), EPOLL_CTL_ADD, flags.raw());
65}

◆ close()

void cosmos::Poller::close ( )

Closes a previously create()'d poll file descriptor again.

Any monitoring that was setup previously will be dropped. A future call to create() can reestablish the Poller functionality.

In the (rare) case that closing fails then an ApiError is thrown.

If currently no valid poll file descriptor exist then this function does nothing.

Definition at line 43 of file Poller.cxx.

43 {
44 if (!valid())
45 return;
46
47 m_poll_fd.close();
48 m_events.clear();
49}
void close()
Explicitly close the contained FD.
bool valid() const
Returns whether currently a valid poll file descriptor exists.
Definition Poller.hxx:154

◆ create()

void cosmos::Poller::create ( size_t max_events = 16)

Actually create the poll file descriptor backing this object.

If the file descriptor already exists this does nothing.

If the creation fails then an ApiError is thrown.

Parameters
[in]max_eventsMaximum number of events that can be reported with a single call to wait().

Definition at line 29 of file Poller.cxx.

29 {
30 if (valid())
31 return;
32
33 auto pfd = epoll_create1(EPOLL_CLOEXEC);
34
35 if (pfd < 0) {
36 cosmos_throw (ApiError("epoll_create1()"));
37 }
38
39 m_poll_fd.setFD(FileNum{pfd});
40 m_events.resize(max_events);
41}
void setFD(const FileNum fd)
Assigns a new primitive file descriptor to the object.

◆ delFD()

void cosmos::Poller::delFD ( const FileDescriptor fd)

Remove a file descriptor from the set of monitored files.

If the given file descriptor is not currently monitored then this will throw an ApiError.

Definition at line 71 of file Poller.cxx.

71 {
72 if (epoll_ctl(rawPollFD(), EPOLL_CTL_DEL, to_integral(fd.raw()), nullptr) < 0) {
73 cosmos_throw (ApiError("epoll_ctl(EPOLL_CTL_DEL)"));
74 }
75}

◆ modFD()

void cosmos::Poller::modFD ( const FileDescriptor fd,
const MonitorFlags flags )

Modify monitoring settings for an already monitored descriptor.

If currently no valid poll FD exists then this will throw an ApiError exception.

Definition at line 67 of file Poller.cxx.

67 {
68 control(rawPollFD(), to_integral(fd.raw()), EPOLL_CTL_MOD, flags.raw());
69}

◆ rawPollFD()

int cosmos::Poller::rawPollFD ( ) const
protected

Definition at line 25 of file Poller.cxx.

25 {
26 return to_integral(m_poll_fd.raw());
27}
FileNum raw() const
Returns the primitive file descriptor contained in the object.

◆ valid()

bool cosmos::Poller::valid ( ) const
inline

Returns whether currently a valid poll file descriptor exists.

Definition at line 154 of file Poller.hxx.

154{ return m_poll_fd.valid(); }
bool valid() const
Returns whether currently a valid file descriptor number is assigned.

◆ wait()

std::vector< Poller::PollEvent > cosmos::Poller::wait ( const std::optional< IntervalTime > timeout = {})

Wait for one of the monitored events to be ready.

Parameters
[in]timeoutAn optional timeout interval to apply after which the call will return even if no events are ready. An empty vector is returned in the timeout case. A timeout of zero achieves a polling operation that will not block at all. This is a relative timeout specification which means that completing the timeout after an interruption is not possible.
Returns
The range of events that occurred, or an empty vector if the timeout occurred.

Definition at line 77 of file Poller.cxx.

77 {
78 while (true) {
79 const auto num_events = epoll_pwait2(
80 rawPollFD(), m_events.data(), m_events.size(), timeout ? &*timeout : nullptr, nullptr);
81
82 if (num_events < 0) {
83 if (auto_restart_syscalls && get_errno() == Errno::INTERRUPTED)
84 // transparent restart
85 continue;
86 cosmos_throw (ApiError("epoll_wait()"));
87 }
88
89 return std::vector<PollEvent>(m_events.begin(), m_events.begin() + num_events);
90 }
91}
Errno get_errno()
Wrapper that returns the Errno strongly typed representation of the current errno
Definition errno.hxx:111

Member Data Documentation

◆ m_events

std::vector<PollEvent> cosmos::Poller::m_events
protected

Definition at line 201 of file Poller.hxx.

◆ m_poll_fd

FileDescriptor cosmos::Poller::m_poll_fd
protected

Definition at line 200 of file Poller.hxx.


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