The documentation of kqueue is quite decent but it lacks some examples. After reading its main manual pages (kqueue(9)
and kevent(9)
), I wasn’t sure about how it worked, so I had to write a test program to verify its behavior.
Let’s start by analyzing the test program to later see its full code. The program will monitor changes to the /tmp/foo
file and will print messages whenever it is deleted, modified or their attributes change. The program finishes when the file being monitoring is deleted.
The steps to use kqueue are the following:
Call
kqueue(2)
to create a new kernel event queue. The descriptor it returns will be later used bykevent(2)
.Open the file to monitor and keep its descriptor around. We’ll need this to attach an event monitor to it.
Initialize a vector of
struct kevent
elements that describes the changes to monitor. Since we are only monitoring a single file, we need a one-element vector. This vector is filled up with calls to theEV_SET
macro. This macro takes: the descriptor of the kqueue, the descriptor of the file to monitor (ident
), thefilter
to apply to it, several flags and optional arguments to the filter. Note that an entry in this table is identified by itsident/filter
pair.Call the
kevent(2)
function. This system call takes the list of changes to monitor we constructed before and does not return until at least one event is received (or when an associated timeout is exhausted). The function returns the number of changes received and stores information about them in another vector ofstruct kevent
elements (we’ll only get notifications of one event at a time, hence we don’t use a vector, but a simple variable).Interpret the results. If
kevent(2)
returned a number greater than 0, we have to inspect the output vector and see which events were received. Each filter has its semantics about the results. For example, we are using theEVFILT_VNODE
filter, which takes a list of conditions to monitor in thefflags
field and modifies it to include only the conditions that triggered the filter.
With these concepts clear and with help of the manual pages, you should be able to interpret the following code easily:
#include <sys/event.h>
#include <sys/time.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int
main(void)
{
int f, kq, nev;
struct kevent change;
struct kevent event;
kq = kqueue();
if (kq == -1)
perror("kqueue");
f = open("/tmp/foo", O_RDONLY);
if (f == -1)
perror("open");
EV_SET(&change, f, EVFILT_VNODE,
EV_ADD | EV_ENABLE | EV_ONESHOT,
NOTE_DELETE | NOTE_EXTEND | NOTE_WRITE | NOTE_ATTRIB,
0, 0);
for (;;) {
nev = kevent(kq, &change, 1, &event, 1, NULL);
if (nev == -1)
perror("kevent");
else if (nev > 0) {
if (event.fflags & NOTE_DELETE) {
printf("File deletedn");
break;
}
if (event.fflags & NOTE_EXTEND ||
event.fflags & NOTE_WRITE)
printf("File modifiedn");
if (event.fflags & NOTE_ATTRIB)
printf("File attributes modifiedn");
}
}
close(kq);
close(f);
return EXIT_SUCCESS;
}
Now compile and run the program in one terminal. In another one, modify the /tmp/foo
file and see how our test program shows the events! If you delete the file, the program will terminate. (Note that we are not monitoring all possible events; we’d watch for file renames, as well as other conditions if we needed to.)