1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
|
/* kqueue interface, C mini-library
*
* Copyright 2012 Joey Hess <joey@kitenet.net>
*
* Licensed under the GNU GPL version 3 or higher.
*/
#include <stdio.h>
#include <dirent.h>
#include <fcntl.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/event.h>
#include <sys/time.h>
#include <errno.h>
/* The specified fds are added to the set of fds being watched for changes.
* Fds passed to prior calls still take effect, so it's most efficient to
* not pass the same fds repeatedly.
*/
signed int helper(const int kq, const int fdcnt, const int *fdlist, int nodelay) {
int i, nev;
struct kevent evlist[1];
struct kevent chlist[fdcnt];
struct timespec avoiddelay = {0, 0};
struct timespec *timeout = nodelay ? &avoiddelay : NULL;
for (i = 0; i < fdcnt; i++) {
EV_SET(&chlist[i], fdlist[i], EVFILT_VNODE,
EV_ADD | EV_ENABLE | EV_CLEAR,
NOTE_WRITE,
0, 0);
}
while ((nev = kevent(kq, chlist, fdcnt, evlist, 1, timeout))) {
if (!(nev == -1 && errno == EINTR)) {
break;
}
}
if (nev == 1)
return evlist[0].ident;
else
return -1;
}
/* Initializes a new, empty kqueue. */
int init_kqueue(const int fdcnt, const int *fdlist) {
int kq;
if ((kq = kqueue()) == -1) {
perror("kqueue");
exit(1);
}
return kq;
}
/* Adds fds to the set that should be watched. */
void addfds_kqueue(const int kq, const int fdcnt, const int *fdlist) {
helper(kq, fdcnt, fdlist, 1);
}
/* Waits for a change event on a kqueue.
*
* Returns the fd that changed, or -1 on error.
*/
signed int waitchange_kqueue(const int kq) {
return helper(kq, 0, NULL, 0);
}
/*
main () {
int list[1];
int kq;
list[0]=open(".", O_RDONLY);
kq = init_kqueue();
addfds_kqueue(kq, 1, list)
printf("change: %i\n", waitchange_kqueue(kq));
}
*/
|