1 /* SPDX-License-Identifier: LGPL-2.1+ */
11 #include <sys/epoll.h>
17 struct mainloop_handler
{
18 lxc_mainloop_callback_t callback
;
25 int lxc_mainloop(struct lxc_epoll_descr
*descr
, int timeout_ms
)
28 struct mainloop_handler
*handler
;
29 struct epoll_event events
[MAX_EVENTS
];
32 nfds
= epoll_wait(descr
->epfd
, events
, MAX_EVENTS
, timeout_ms
);
40 for (i
= 0; i
< nfds
; i
++) {
41 handler
= events
[i
].data
.ptr
;
43 /* If the handler returns a positive value, exit the
46 ret
= handler
->callback(handler
->fd
, events
[i
].events
,
47 handler
->data
, descr
);
48 if (ret
== LXC_MAINLOOP_ERROR
)
50 if (ret
== LXC_MAINLOOP_CLOSE
)
57 if (lxc_list_empty(&descr
->handlers
))
62 int lxc_mainloop_add_handler_events(struct lxc_epoll_descr
*descr
, int fd
,
64 lxc_mainloop_callback_t callback
,
67 __do_free
struct mainloop_handler
*handler
= NULL
;
68 __do_free
struct lxc_list
*item
= NULL
;
69 struct epoll_event ev
;
74 handler
= malloc(sizeof(*handler
));
78 handler
->callback
= callback
;
83 ev
.data
.ptr
= handler
;
85 if (epoll_ctl(descr
->epfd
, EPOLL_CTL_ADD
, fd
, &ev
) < 0)
88 item
= malloc(sizeof(*item
));
90 return ret_errno(ENOMEM
);
92 item
->elem
= move_ptr(handler
);
93 lxc_list_add(&descr
->handlers
, move_ptr(item
));
97 int lxc_mainloop_add_handler(struct lxc_epoll_descr
*descr
, int fd
,
98 lxc_mainloop_callback_t callback
, void *data
)
100 return lxc_mainloop_add_handler_events(descr
, fd
, EPOLLIN
, callback
,
104 int lxc_mainloop_del_handler(struct lxc_epoll_descr
*descr
, int fd
)
106 struct mainloop_handler
*handler
;
107 struct lxc_list
*iterator
;
109 lxc_list_for_each(iterator
, &descr
->handlers
) {
110 handler
= iterator
->elem
;
112 if (handler
->fd
== fd
) {
114 if (epoll_ctl(descr
->epfd
, EPOLL_CTL_DEL
, fd
, NULL
))
117 lxc_list_del(iterator
);
118 free(iterator
->elem
);
124 return ret_errno(EINVAL
);
127 int lxc_mainloop_open(struct lxc_epoll_descr
*descr
)
129 descr
->epfd
= epoll_create1(EPOLL_CLOEXEC
);
133 lxc_list_init(&descr
->handlers
);
137 void lxc_mainloop_close(struct lxc_epoll_descr
*descr
)
139 struct lxc_list
*iterator
, *next
;
141 iterator
= descr
->handlers
.next
;
142 while (iterator
!= &descr
->handlers
) {
143 next
= iterator
->next
;
145 lxc_list_del(iterator
);
146 free(iterator
->elem
);
151 close_prot_errno_disarm(descr
->epfd
);