]>
git.saurik.com Git - redis.git/blob - src/ae_epoll.c
1 /* Linux epoll(2) based ae.c module
2 * Copyright (C) 2009-2010 Salvatore Sanfilippo - antirez@gmail.com
3 * Released under the BSD license. See the COPYING file for more info. */
7 typedef struct aeApiState
{
9 struct epoll_event events
[AE_SETSIZE
];
12 static int aeApiCreate(aeEventLoop
*eventLoop
) {
13 aeApiState
*state
= zmalloc(sizeof(aeApiState
));
15 if (!state
) return -1;
16 state
->epfd
= epoll_create(1024); /* 1024 is just an hint for the kernel */
17 if (state
->epfd
== -1) {
21 eventLoop
->apidata
= state
;
25 static void aeApiFree(aeEventLoop
*eventLoop
) {
26 aeApiState
*state
= eventLoop
->apidata
;
32 static int aeApiAddEvent(aeEventLoop
*eventLoop
, int fd
, int mask
) {
33 aeApiState
*state
= eventLoop
->apidata
;
34 struct epoll_event ee
;
35 /* If the fd was already monitored for some event, we need a MOD
36 * operation. Otherwise we need an ADD operation. */
37 int op
= eventLoop
->events
[fd
].mask
== AE_NONE
?
38 EPOLL_CTL_ADD
: EPOLL_CTL_MOD
;
41 mask
|= eventLoop
->events
[fd
].mask
; /* Merge old events */
42 if (mask
& AE_READABLE
) ee
.events
|= EPOLLIN
;
43 if (mask
& AE_WRITABLE
) ee
.events
|= EPOLLOUT
;
44 ee
.data
.u64
= 0; /* avoid valgrind warning */
46 if (epoll_ctl(state
->epfd
,op
,fd
,&ee
) == -1) return -1;
50 static void aeApiDelEvent(aeEventLoop
*eventLoop
, int fd
, int delmask
) {
51 aeApiState
*state
= eventLoop
->apidata
;
52 struct epoll_event ee
;
53 int mask
= eventLoop
->events
[fd
].mask
& (~delmask
);
56 if (mask
& AE_READABLE
) ee
.events
|= EPOLLIN
;
57 if (mask
& AE_WRITABLE
) ee
.events
|= EPOLLOUT
;
58 ee
.data
.u64
= 0; /* avoid valgrind warning */
60 if (mask
!= AE_NONE
) {
61 epoll_ctl(state
->epfd
,EPOLL_CTL_MOD
,fd
,&ee
);
63 /* Note, Kernel < 2.6.9 requires a non null event pointer even for
65 epoll_ctl(state
->epfd
,EPOLL_CTL_DEL
,fd
,&ee
);
69 static int aeApiPoll(aeEventLoop
*eventLoop
, struct timeval
*tvp
) {
70 aeApiState
*state
= eventLoop
->apidata
;
71 int retval
, numevents
= 0;
73 retval
= epoll_wait(state
->epfd
,state
->events
,AE_SETSIZE
,
74 tvp
? (tvp
->tv_sec
*1000 + tvp
->tv_usec
/1000) : -1);
79 for (j
= 0; j
< numevents
; j
++) {
81 struct epoll_event
*e
= state
->events
+j
;
83 if (e
->events
& EPOLLIN
) mask
|= AE_READABLE
;
84 if (e
->events
& EPOLLOUT
) mask
|= AE_WRITABLE
;
85 eventLoop
->fired
[j
].fd
= e
->data
.fd
;
86 eventLoop
->fired
[j
].mask
= mask
;
92 static char *aeApiName(void) {