This is a small library that implements epoll on top of kqueue. It has been successfully used to port libinput, libevdev, Wayland and more software to FreeBSD: https://www.freshports.org/devel/libepoll-shim/
It may be useful for porting other software that uses epoll as well.
There are some tests inside test/. They should also compile under Linux and
can be used to verify proper epoll behavior.
However, this library contains some very ugly hacks and workarounds. For example:
-
When using
timerfd,signalfdoreventfd, the system callsread,writeandcloseare redefined as macros to internal helper functions. This is needed as there is some internal context that has to be free'd properly. This means that you shouldn't create atimerfd/signalfdin one part of a program and close it in a different part wheresys/timerfd.hisn't included. The context would leak. Luckily, software such as libinput behaves very nicely and puts alltimerfdrelated code in a single source file. -
There is limited support for file descriptors that lack support for kqueue but are supported by
poll(2). This includes graphics or sound devices under/dev. Those descriptors are handled in an outerpoll(2)loop. Edge triggering usingEPOLLETwill not work. -
Shimmed file descriptors cannot be shared between processes. On
fork()those fds are closed. When trying to pass a shimmed fd to another process thesendmsgcall will returnEOPNOTSUPP. In most cases sharingepoll/timerfd/signalfdis a bad idea anyway, but there are some legitimate use cases (for example sharing semaphoreeventfds, issue #23). When the OS natively supportseventfds (as is the case for FreeBSD >= 13) this library won't provideeventfdshims or thesys/eventfd.hheader. -
There is no proper notification mechanism for changes to the system
CLOCK_REALTIMEclock on BSD systems. Also,keventEVFILT_TIMERs use the system monotonic clock as reference. Therefore, in order to implement absolute (TFD_TIMER_ABSTIME)CLOCK_REALTIMEtimerfds or cancellation support (TFD_TIMER_CANCEL_ON_SET), a thread is spawned that periodically polls the system boot time for changes to the realtime clock.
The library is tested on the following operating systems:
- FreeBSD 12.2, 13.0
- NetBSD 9.1
- OpenBSD 7.0
- DragonFlyBSD 5.8.0
Be aware of some subtle kqueue bugs that may affect the emulated
epoll behavior. I've marked tests that hit those behaviors as "skipped".
Have a look at atf_tc_skip() calls in the tests.
Run the following commands to build libepoll-shim:
mkdir build
cd build
cmake .. -DCMAKE_BUILD_TYPE=RelWithDebInfo
cmake --build .
To run the tests:
ctest --output-on-failure
To install (as root):
cmake --build . --target install
- Fix compiler warning when using shimmed
fcntl.
- Allow setting
O_NONBLOCKflag withfcntlon created file descriptors. - Implement
TFD_TIMER_CANCEL_ON_SETfortimerfd. - Implement correction of absolute (
TFD_TIMER_ABSTIME)CLOCK_REALTIMEtimerfds when the system time is stepped.
- Fix compilation on FreeBSD < 12 (#28).
- Add
O_CLOEXEChandling to created file descriptors (PR #26, thanks arichardson!). Note that the shimmed file descriptors still won't work correctly afterexec(3). Therefore, not usingEPOLL_CLOEXEC/TFD_CLOEXEC/SFD_CLOEXEC/EFD_CLOEXECis strongly discouraged.
- Fix compilation on FreeBSD 12.1 (#25).
signalfdnow hooks into the signal disposition mechanism, just like on Linux. Note:pollandppollare also shimmed with macros in casesys/signalfd.his included to support some use cases seen in the wild. Many moressi_*fields are now set on the resultingstruct signalfd_siginfo.- More accurate timeout calculations for
epoll_wait/poll/ppoll. - Fix integer overflow on timerfd timeout field on 32-bit machines.
- Fix re-arming of timerfd timeouts on BSDs where EV_ADD of a EVFILT_TIMER doesn't do it.
- Add support for native
eventfds (provided by FreeBSD >= 13). Thesys/eventfd.hheader will not be installed in this case.
- Add support for NetBSD 9.1.
- On FreeBSD, add missing
sys/signal.hinclude that resulted insigset_terrors (#21).
- Lift limit of 32 descriptors in
epoll_wait(2). - Implement
EPOLLPRIusingEVFILT_EXCEPT, if available. If it is not available, add logic toEVFILT_READhandling that will work ifSO_OOBINLINEis set on the socket. - Implement
EPOLLONESHOT. - Implement edge triggering with
EPOLLET. - Add support for unlimited numbers of poll-only fds per epoll instance.
- Merge
EVFILT_READ/EVFILT_WRITEevents together to more closely match epoll semantics. - Add support for NetBSD, OpenBSD and DragonFlyBSD.
- Implement
epoll_pwait(2).