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
,signalfd
oreventfd
, the system callsread
,write
andclose
are 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
/signalfd
in one part of a program and close it in a different part wheresys/timerfd.h
isn't included. The context would leak. Luckily, software such as libinput behaves very nicely and puts alltimerfd
related 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 usingEPOLLET
will 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 thesendmsg
call will returnEOPNOTSUPP
. In most cases sharingepoll
/timerfd
/signalfd
is a bad idea anyway, but there are some legitimate use cases (for example sharing semaphoreeventfd
s, issue #23).
The following operating systems are supported:
- FreeBSD >= 11.3
- NetBSD >= 9.1
- OpenBSD >= 6.7
- 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
- Add support for NetBSD 9.1.
- On FreeBSD, add missing
sys/signal.h
include that resulted insigset_t
errors (#21).
- Lift limit of 32 descriptors in
epoll_wait(2)
. - Implement
EPOLLPRI
usingEVFILT_EXCEPT
, if available. If it is not available, add logic toEVFILT_READ
handling that will work ifSO_OOBINLINE
is 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_WRITE
events together to more closely match epoll semantics. - Add support for NetBSD, OpenBSD and DragonFlyBSD.
- Implement
epoll_pwait(2)
.