xref: /freebsd-src/contrib/pf/libevent/select.c (revision fc515400ab05695df359c4b4d5fab9285a642699)
1*67ecd4f3SMax Laier /*	$OpenBSD: select.c,v 1.2 2002/06/25 15:50:15 mickey Exp $	*/
2*67ecd4f3SMax Laier 
3*67ecd4f3SMax Laier /*
4*67ecd4f3SMax Laier  * Copyright 2000-2002 Niels Provos <provos@citi.umich.edu>
5*67ecd4f3SMax Laier  * All rights reserved.
6*67ecd4f3SMax Laier  *
7*67ecd4f3SMax Laier  * Redistribution and use in source and binary forms, with or without
8*67ecd4f3SMax Laier  * modification, are permitted provided that the following conditions
9*67ecd4f3SMax Laier  * are met:
10*67ecd4f3SMax Laier  * 1. Redistributions of source code must retain the above copyright
11*67ecd4f3SMax Laier  *    notice, this list of conditions and the following disclaimer.
12*67ecd4f3SMax Laier  * 2. Redistributions in binary form must reproduce the above copyright
13*67ecd4f3SMax Laier  *    notice, this list of conditions and the following disclaimer in the
14*67ecd4f3SMax Laier  *    documentation and/or other materials provided with the distribution.
15*67ecd4f3SMax Laier  * 3. The name of the author may not be used to endorse or promote products
16*67ecd4f3SMax Laier  *    derived from this software without specific prior written permission.
17*67ecd4f3SMax Laier  *
18*67ecd4f3SMax Laier  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
19*67ecd4f3SMax Laier  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
20*67ecd4f3SMax Laier  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
21*67ecd4f3SMax Laier  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
22*67ecd4f3SMax Laier  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
23*67ecd4f3SMax Laier  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24*67ecd4f3SMax Laier  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25*67ecd4f3SMax Laier  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26*67ecd4f3SMax Laier  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
27*67ecd4f3SMax Laier  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*67ecd4f3SMax Laier  */
29*67ecd4f3SMax Laier #ifdef HAVE_CONFIG_H
30*67ecd4f3SMax Laier #include "config.h"
31*67ecd4f3SMax Laier #endif
32*67ecd4f3SMax Laier 
33*67ecd4f3SMax Laier #include <sys/types.h>
34*67ecd4f3SMax Laier #ifdef HAVE_SYS_TIME_H
35*67ecd4f3SMax Laier #include <sys/time.h>
36*67ecd4f3SMax Laier #else
37*67ecd4f3SMax Laier #include <sys/_time.h>
38*67ecd4f3SMax Laier #endif
39*67ecd4f3SMax Laier #include <sys/queue.h>
40*67ecd4f3SMax Laier #include <sys/tree.h>
41*67ecd4f3SMax Laier #include <signal.h>
42*67ecd4f3SMax Laier #include <stdio.h>
43*67ecd4f3SMax Laier #include <stdlib.h>
44*67ecd4f3SMax Laier #include <string.h>
45*67ecd4f3SMax Laier #include <unistd.h>
46*67ecd4f3SMax Laier #include <errno.h>
47*67ecd4f3SMax Laier #ifdef CHECK_INVARIANTS
48*67ecd4f3SMax Laier #include <assert.h>
49*67ecd4f3SMax Laier #endif
50*67ecd4f3SMax Laier 
51*67ecd4f3SMax Laier #include "event.h"
52*67ecd4f3SMax Laier #include "event-internal.h"
53*67ecd4f3SMax Laier #include "evsignal.h"
54*67ecd4f3SMax Laier #include "log.h"
55*67ecd4f3SMax Laier 
56*67ecd4f3SMax Laier #ifndef howmany
57*67ecd4f3SMax Laier #define        howmany(x, y)   (((x)+((y)-1))/(y))
58*67ecd4f3SMax Laier #endif
59*67ecd4f3SMax Laier 
60*67ecd4f3SMax Laier extern volatile sig_atomic_t evsignal_caught;
61*67ecd4f3SMax Laier 
62*67ecd4f3SMax Laier struct selectop {
63*67ecd4f3SMax Laier 	int event_fds;		/* Highest fd in fd set */
64*67ecd4f3SMax Laier 	int event_fdsz;
65*67ecd4f3SMax Laier 	fd_set *event_readset_in;
66*67ecd4f3SMax Laier 	fd_set *event_writeset_in;
67*67ecd4f3SMax Laier 	fd_set *event_readset_out;
68*67ecd4f3SMax Laier 	fd_set *event_writeset_out;
69*67ecd4f3SMax Laier 	struct event **event_r_by_fd;
70*67ecd4f3SMax Laier 	struct event **event_w_by_fd;
71*67ecd4f3SMax Laier };
72*67ecd4f3SMax Laier 
73*67ecd4f3SMax Laier void *select_init	(void);
74*67ecd4f3SMax Laier int select_add		(void *, struct event *);
75*67ecd4f3SMax Laier int select_del		(void *, struct event *);
76*67ecd4f3SMax Laier int select_recalc	(struct event_base *, void *, int);
77*67ecd4f3SMax Laier int select_dispatch	(struct event_base *, void *, struct timeval *);
78*67ecd4f3SMax Laier void select_dealloc     (void *);
79*67ecd4f3SMax Laier 
80*67ecd4f3SMax Laier const struct eventop selectops = {
81*67ecd4f3SMax Laier 	"select",
82*67ecd4f3SMax Laier 	select_init,
83*67ecd4f3SMax Laier 	select_add,
84*67ecd4f3SMax Laier 	select_del,
85*67ecd4f3SMax Laier 	select_recalc,
86*67ecd4f3SMax Laier 	select_dispatch,
87*67ecd4f3SMax Laier 	select_dealloc
88*67ecd4f3SMax Laier };
89*67ecd4f3SMax Laier 
90*67ecd4f3SMax Laier static int select_resize(struct selectop *sop, int fdsz);
91*67ecd4f3SMax Laier 
92*67ecd4f3SMax Laier void *
select_init(void)93*67ecd4f3SMax Laier select_init(void)
94*67ecd4f3SMax Laier {
95*67ecd4f3SMax Laier 	struct selectop *sop;
96*67ecd4f3SMax Laier 
97*67ecd4f3SMax Laier 	/* Disable select when this environment variable is set */
98*67ecd4f3SMax Laier 	if (getenv("EVENT_NOSELECT"))
99*67ecd4f3SMax Laier 		return (NULL);
100*67ecd4f3SMax Laier 
101*67ecd4f3SMax Laier 	if (!(sop = calloc(1, sizeof(struct selectop))))
102*67ecd4f3SMax Laier 		return (NULL);
103*67ecd4f3SMax Laier 
104*67ecd4f3SMax Laier 	select_resize(sop, howmany(32 + 1, NFDBITS)*sizeof(fd_mask));
105*67ecd4f3SMax Laier 
106*67ecd4f3SMax Laier 	evsignal_init();
107*67ecd4f3SMax Laier 
108*67ecd4f3SMax Laier 	return (sop);
109*67ecd4f3SMax Laier }
110*67ecd4f3SMax Laier 
111*67ecd4f3SMax Laier #ifdef CHECK_INVARIANTS
112*67ecd4f3SMax Laier static void
check_selectop(struct selectop * sop)113*67ecd4f3SMax Laier check_selectop(struct selectop *sop)
114*67ecd4f3SMax Laier {
115*67ecd4f3SMax Laier 	int i;
116*67ecd4f3SMax Laier 	for (i=0;i<=sop->event_fds;++i) {
117*67ecd4f3SMax Laier 		if (FD_ISSET(i, sop->event_readset_in)) {
118*67ecd4f3SMax Laier 			assert(sop->event_r_by_fd[i]);
119*67ecd4f3SMax Laier 			assert(sop->event_r_by_fd[i]->ev_events & EV_READ);
120*67ecd4f3SMax Laier 			assert(sop->event_r_by_fd[i]->ev_fd == i);
121*67ecd4f3SMax Laier 		} else {
122*67ecd4f3SMax Laier 			assert(! sop->event_r_by_fd[i]);
123*67ecd4f3SMax Laier 		}
124*67ecd4f3SMax Laier 		if (FD_ISSET(i, sop->event_writeset_in)) {
125*67ecd4f3SMax Laier 			assert(sop->event_w_by_fd[i]);
126*67ecd4f3SMax Laier 			assert(sop->event_w_by_fd[i]->ev_events & EV_WRITE);
127*67ecd4f3SMax Laier 			assert(sop->event_w_by_fd[i]->ev_fd == i);
128*67ecd4f3SMax Laier 		} else {
129*67ecd4f3SMax Laier 			assert(! sop->event_w_by_fd[i]);
130*67ecd4f3SMax Laier 		}
131*67ecd4f3SMax Laier 	}
132*67ecd4f3SMax Laier 
133*67ecd4f3SMax Laier }
134*67ecd4f3SMax Laier #else
135*67ecd4f3SMax Laier #define check_selectop(sop) do { (void) sop; } while (0)
136*67ecd4f3SMax Laier #endif
137*67ecd4f3SMax Laier 
138*67ecd4f3SMax Laier /*
139*67ecd4f3SMax Laier  * Called with the highest fd that we know about.  If it is 0, completely
140*67ecd4f3SMax Laier  * recalculate everything.
141*67ecd4f3SMax Laier  */
142*67ecd4f3SMax Laier 
143*67ecd4f3SMax Laier int
select_recalc(struct event_base * base,void * arg,int max)144*67ecd4f3SMax Laier select_recalc(struct event_base *base, void *arg, int max)
145*67ecd4f3SMax Laier {
146*67ecd4f3SMax Laier 	struct selectop *sop = arg;
147*67ecd4f3SMax Laier 
148*67ecd4f3SMax Laier 	check_selectop(sop);
149*67ecd4f3SMax Laier 
150*67ecd4f3SMax Laier 	return (0);
151*67ecd4f3SMax Laier }
152*67ecd4f3SMax Laier 
153*67ecd4f3SMax Laier int
select_dispatch(struct event_base * base,void * arg,struct timeval * tv)154*67ecd4f3SMax Laier select_dispatch(struct event_base *base, void *arg, struct timeval *tv)
155*67ecd4f3SMax Laier {
156*67ecd4f3SMax Laier 	int res, i;
157*67ecd4f3SMax Laier 	struct selectop *sop = arg;
158*67ecd4f3SMax Laier 
159*67ecd4f3SMax Laier 	check_selectop(sop);
160*67ecd4f3SMax Laier 
161*67ecd4f3SMax Laier 	memcpy(sop->event_readset_out, sop->event_readset_in,
162*67ecd4f3SMax Laier 	       sop->event_fdsz);
163*67ecd4f3SMax Laier 	memcpy(sop->event_writeset_out, sop->event_writeset_in,
164*67ecd4f3SMax Laier 	       sop->event_fdsz);
165*67ecd4f3SMax Laier 
166*67ecd4f3SMax Laier 	res = select(sop->event_fds + 1, sop->event_readset_out,
167*67ecd4f3SMax Laier 	    sop->event_writeset_out, NULL, tv);
168*67ecd4f3SMax Laier 
169*67ecd4f3SMax Laier 	check_selectop(sop);
170*67ecd4f3SMax Laier 
171*67ecd4f3SMax Laier 	if (res == -1) {
172*67ecd4f3SMax Laier 		if (errno != EINTR) {
173*67ecd4f3SMax Laier 			event_warn("select");
174*67ecd4f3SMax Laier 			return (-1);
175*67ecd4f3SMax Laier 		}
176*67ecd4f3SMax Laier 
177*67ecd4f3SMax Laier 		evsignal_process();
178*67ecd4f3SMax Laier 		return (0);
179*67ecd4f3SMax Laier 	} else if (evsignal_caught)
180*67ecd4f3SMax Laier 		evsignal_process();
181*67ecd4f3SMax Laier 
182*67ecd4f3SMax Laier 	event_debug(("%s: select reports %d", __func__, res));
183*67ecd4f3SMax Laier 
184*67ecd4f3SMax Laier 	check_selectop(sop);
185*67ecd4f3SMax Laier 	for (i = 0; i <= sop->event_fds; ++i) {
186*67ecd4f3SMax Laier 		struct event *r_ev = NULL, *w_ev = NULL;
187*67ecd4f3SMax Laier 		res = 0;
188*67ecd4f3SMax Laier 		if (FD_ISSET(i, sop->event_readset_out)) {
189*67ecd4f3SMax Laier 			r_ev = sop->event_r_by_fd[i];
190*67ecd4f3SMax Laier 			res |= EV_READ;
191*67ecd4f3SMax Laier 		}
192*67ecd4f3SMax Laier 		if (FD_ISSET(i, sop->event_writeset_out)) {
193*67ecd4f3SMax Laier 			w_ev = sop->event_w_by_fd[i];
194*67ecd4f3SMax Laier 			res |= EV_WRITE;
195*67ecd4f3SMax Laier 		}
196*67ecd4f3SMax Laier 		if (r_ev && (res & r_ev->ev_events)) {
197*67ecd4f3SMax Laier 			if (!(r_ev->ev_events & EV_PERSIST))
198*67ecd4f3SMax Laier 				event_del(r_ev);
199*67ecd4f3SMax Laier 			event_active(r_ev, res & r_ev->ev_events, 1);
200*67ecd4f3SMax Laier 		}
201*67ecd4f3SMax Laier 		if (w_ev && w_ev != r_ev && (res & w_ev->ev_events)) {
202*67ecd4f3SMax Laier 			if (!(w_ev->ev_events & EV_PERSIST))
203*67ecd4f3SMax Laier 				event_del(w_ev);
204*67ecd4f3SMax Laier 			event_active(w_ev, res & w_ev->ev_events, 1);
205*67ecd4f3SMax Laier 		}
206*67ecd4f3SMax Laier 	}
207*67ecd4f3SMax Laier 	check_selectop(sop);
208*67ecd4f3SMax Laier 
209*67ecd4f3SMax Laier 	return (0);
210*67ecd4f3SMax Laier }
211*67ecd4f3SMax Laier 
212*67ecd4f3SMax Laier 
213*67ecd4f3SMax Laier static int
select_resize(struct selectop * sop,int fdsz)214*67ecd4f3SMax Laier select_resize(struct selectop *sop, int fdsz)
215*67ecd4f3SMax Laier {
216*67ecd4f3SMax Laier 	int n_events, n_events_old;
217*67ecd4f3SMax Laier 
218*67ecd4f3SMax Laier 	fd_set *readset_in = NULL;
219*67ecd4f3SMax Laier 	fd_set *writeset_in = NULL;
220*67ecd4f3SMax Laier 	fd_set *readset_out = NULL;
221*67ecd4f3SMax Laier 	fd_set *writeset_out = NULL;
222*67ecd4f3SMax Laier 	struct event **r_by_fd = NULL;
223*67ecd4f3SMax Laier 	struct event **w_by_fd = NULL;
224*67ecd4f3SMax Laier 
225*67ecd4f3SMax Laier 	n_events = (fdsz/sizeof(fd_mask)) * NFDBITS;
226*67ecd4f3SMax Laier 	n_events_old = (sop->event_fdsz/sizeof(fd_mask)) * NFDBITS;
227*67ecd4f3SMax Laier 
228*67ecd4f3SMax Laier 	if (sop->event_readset_in)
229*67ecd4f3SMax Laier 		check_selectop(sop);
230*67ecd4f3SMax Laier 
231*67ecd4f3SMax Laier 	if ((readset_in = realloc(sop->event_readset_in, fdsz)) == NULL)
232*67ecd4f3SMax Laier 		goto error;
233*67ecd4f3SMax Laier 	sop->event_readset_in = readset_in;
234*67ecd4f3SMax Laier 	if ((readset_out = realloc(sop->event_readset_out, fdsz)) == NULL)
235*67ecd4f3SMax Laier 		goto error;
236*67ecd4f3SMax Laier 	sop->event_readset_out = readset_out;
237*67ecd4f3SMax Laier 	if ((writeset_in = realloc(sop->event_writeset_in, fdsz)) == NULL)
238*67ecd4f3SMax Laier 		goto error;
239*67ecd4f3SMax Laier 	sop->event_writeset_in = writeset_in;
240*67ecd4f3SMax Laier 	if ((writeset_out = realloc(sop->event_writeset_out, fdsz)) == NULL)
241*67ecd4f3SMax Laier 		goto error;
242*67ecd4f3SMax Laier 	sop->event_writeset_out = writeset_out;
243*67ecd4f3SMax Laier 	if ((r_by_fd = realloc(sop->event_r_by_fd,
244*67ecd4f3SMax Laier 		 n_events*sizeof(struct event*))) == NULL)
245*67ecd4f3SMax Laier 		goto error;
246*67ecd4f3SMax Laier 	sop->event_r_by_fd = r_by_fd;
247*67ecd4f3SMax Laier 	if ((w_by_fd = realloc(sop->event_w_by_fd,
248*67ecd4f3SMax Laier 		 n_events * sizeof(struct event*))) == NULL)
249*67ecd4f3SMax Laier 		goto error;
250*67ecd4f3SMax Laier 	sop->event_w_by_fd = w_by_fd;
251*67ecd4f3SMax Laier 
252*67ecd4f3SMax Laier 	memset((char *)sop->event_readset_in + sop->event_fdsz, 0,
253*67ecd4f3SMax Laier 	    fdsz - sop->event_fdsz);
254*67ecd4f3SMax Laier 	memset((char *)sop->event_writeset_in + sop->event_fdsz, 0,
255*67ecd4f3SMax Laier 	    fdsz - sop->event_fdsz);
256*67ecd4f3SMax Laier 	memset(sop->event_r_by_fd + n_events_old, 0,
257*67ecd4f3SMax Laier 	    (n_events-n_events_old) * sizeof(struct event*));
258*67ecd4f3SMax Laier 	memset(sop->event_w_by_fd + n_events_old, 0,
259*67ecd4f3SMax Laier 	    (n_events-n_events_old) * sizeof(struct event*));
260*67ecd4f3SMax Laier 
261*67ecd4f3SMax Laier 	sop->event_fdsz = fdsz;
262*67ecd4f3SMax Laier 	check_selectop(sop);
263*67ecd4f3SMax Laier 
264*67ecd4f3SMax Laier 	return (0);
265*67ecd4f3SMax Laier 
266*67ecd4f3SMax Laier  error:
267*67ecd4f3SMax Laier 	event_warn("malloc");
268*67ecd4f3SMax Laier 	return (-1);
269*67ecd4f3SMax Laier }
270*67ecd4f3SMax Laier 
271*67ecd4f3SMax Laier 
272*67ecd4f3SMax Laier int
select_add(void * arg,struct event * ev)273*67ecd4f3SMax Laier select_add(void *arg, struct event *ev)
274*67ecd4f3SMax Laier {
275*67ecd4f3SMax Laier 	struct selectop *sop = arg;
276*67ecd4f3SMax Laier 
277*67ecd4f3SMax Laier 	if (ev->ev_events & EV_SIGNAL)
278*67ecd4f3SMax Laier 		return (evsignal_add(ev));
279*67ecd4f3SMax Laier 
280*67ecd4f3SMax Laier 	check_selectop(sop);
281*67ecd4f3SMax Laier 	/*
282*67ecd4f3SMax Laier 	 * Keep track of the highest fd, so that we can calculate the size
283*67ecd4f3SMax Laier 	 * of the fd_sets for select(2)
284*67ecd4f3SMax Laier 	 */
285*67ecd4f3SMax Laier 	if (sop->event_fds < ev->ev_fd) {
286*67ecd4f3SMax Laier 		int fdsz = sop->event_fdsz;
287*67ecd4f3SMax Laier 
288*67ecd4f3SMax Laier 		if (fdsz < sizeof(fd_mask))
289*67ecd4f3SMax Laier 			fdsz = sizeof(fd_mask);
290*67ecd4f3SMax Laier 
291*67ecd4f3SMax Laier 		while (fdsz <
292*67ecd4f3SMax Laier 		    (howmany(ev->ev_fd + 1, NFDBITS) * sizeof(fd_mask)))
293*67ecd4f3SMax Laier 			fdsz *= 2;
294*67ecd4f3SMax Laier 
295*67ecd4f3SMax Laier 		if (fdsz != sop->event_fdsz) {
296*67ecd4f3SMax Laier 			if (select_resize(sop, fdsz)) {
297*67ecd4f3SMax Laier 				check_selectop(sop);
298*67ecd4f3SMax Laier 				return (-1);
299*67ecd4f3SMax Laier 			}
300*67ecd4f3SMax Laier 		}
301*67ecd4f3SMax Laier 
302*67ecd4f3SMax Laier 		sop->event_fds = ev->ev_fd;
303*67ecd4f3SMax Laier 	}
304*67ecd4f3SMax Laier 
305*67ecd4f3SMax Laier 	if (ev->ev_events & EV_READ) {
306*67ecd4f3SMax Laier 		FD_SET(ev->ev_fd, sop->event_readset_in);
307*67ecd4f3SMax Laier 		sop->event_r_by_fd[ev->ev_fd] = ev;
308*67ecd4f3SMax Laier 	}
309*67ecd4f3SMax Laier 	if (ev->ev_events & EV_WRITE) {
310*67ecd4f3SMax Laier 		FD_SET(ev->ev_fd, sop->event_writeset_in);
311*67ecd4f3SMax Laier 		sop->event_w_by_fd[ev->ev_fd] = ev;
312*67ecd4f3SMax Laier 	}
313*67ecd4f3SMax Laier 	check_selectop(sop);
314*67ecd4f3SMax Laier 
315*67ecd4f3SMax Laier 	return (0);
316*67ecd4f3SMax Laier }
317*67ecd4f3SMax Laier 
318*67ecd4f3SMax Laier /*
319*67ecd4f3SMax Laier  * Nothing to be done here.
320*67ecd4f3SMax Laier  */
321*67ecd4f3SMax Laier 
322*67ecd4f3SMax Laier int
select_del(void * arg,struct event * ev)323*67ecd4f3SMax Laier select_del(void *arg, struct event *ev)
324*67ecd4f3SMax Laier {
325*67ecd4f3SMax Laier 	struct selectop *sop = arg;
326*67ecd4f3SMax Laier 
327*67ecd4f3SMax Laier 	check_selectop(sop);
328*67ecd4f3SMax Laier 	if (ev->ev_events & EV_SIGNAL)
329*67ecd4f3SMax Laier 		return (evsignal_del(ev));
330*67ecd4f3SMax Laier 
331*67ecd4f3SMax Laier 	if (sop->event_fds < ev->ev_fd) {
332*67ecd4f3SMax Laier 		check_selectop(sop);
333*67ecd4f3SMax Laier 		return (0);
334*67ecd4f3SMax Laier 	}
335*67ecd4f3SMax Laier 
336*67ecd4f3SMax Laier 	if (ev->ev_events & EV_READ) {
337*67ecd4f3SMax Laier 		FD_CLR(ev->ev_fd, sop->event_readset_in);
338*67ecd4f3SMax Laier 		sop->event_r_by_fd[ev->ev_fd] = NULL;
339*67ecd4f3SMax Laier 	}
340*67ecd4f3SMax Laier 
341*67ecd4f3SMax Laier 	if (ev->ev_events & EV_WRITE) {
342*67ecd4f3SMax Laier 		FD_CLR(ev->ev_fd, sop->event_writeset_in);
343*67ecd4f3SMax Laier 		sop->event_w_by_fd[ev->ev_fd] = NULL;
344*67ecd4f3SMax Laier 	}
345*67ecd4f3SMax Laier 
346*67ecd4f3SMax Laier 	check_selectop(sop);
347*67ecd4f3SMax Laier 	return (0);
348*67ecd4f3SMax Laier }
349*67ecd4f3SMax Laier 
350*67ecd4f3SMax Laier void
select_dealloc(void * arg)351*67ecd4f3SMax Laier select_dealloc(void *arg)
352*67ecd4f3SMax Laier {
353*67ecd4f3SMax Laier 	struct selectop *sop = arg;
354*67ecd4f3SMax Laier 
355*67ecd4f3SMax Laier 	if (sop->event_readset_in)
356*67ecd4f3SMax Laier 		free(sop->event_readset_in);
357*67ecd4f3SMax Laier 	if (sop->event_writeset_in)
358*67ecd4f3SMax Laier 		free(sop->event_writeset_in);
359*67ecd4f3SMax Laier 	if (sop->event_readset_out)
360*67ecd4f3SMax Laier 		free(sop->event_readset_out);
361*67ecd4f3SMax Laier 	if (sop->event_writeset_out)
362*67ecd4f3SMax Laier 		free(sop->event_writeset_out);
363*67ecd4f3SMax Laier 	if (sop->event_r_by_fd)
364*67ecd4f3SMax Laier 		free(sop->event_r_by_fd);
365*67ecd4f3SMax Laier 	if (sop->event_w_by_fd)
366*67ecd4f3SMax Laier 		free(sop->event_w_by_fd);
367*67ecd4f3SMax Laier 
368*67ecd4f3SMax Laier 	memset(sop, 0, sizeof(struct selectop));
369*67ecd4f3SMax Laier 	free(sop);
370*67ecd4f3SMax Laier }
371