2 * Copyright (C) 2011 - David Goulet <david.goulet@polymtl.ca>
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License, version 2 only,
6 * as published by the Free Software Foundation.
8 * This program is distributed in the hope that it will be useful, but WITHOUT
9 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
10 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
13 * You should have received a copy of the GNU General Public License along
14 * with this program; if not, write to the Free Software Foundation, Inc.,
15 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
21 #include <sys/resource.h>
24 #include <common/defaults.h>
25 #include <common/error.h>
26 #include <common/macros.h>
27 #include <common/utils.h>
31 unsigned int poll_max_size
;
34 * Resize the epoll events structure of the new size.
36 * Return 0 on success or else -1 with the current events pointer untouched.
38 static int resize_poll_event(struct compat_poll_event_array
*array
,
45 /* Refuse to resize the array more than the max size. */
46 if (new_size
> poll_max_size
) {
50 ptr
= realloc(array
->events
, new_size
* sizeof(*ptr
));
52 PERROR("realloc epoll add");
56 array
->alloc_size
= new_size
;
65 * Update events with the current events object.
67 static int update_current_events(struct lttng_poll_event
*events
)
70 struct compat_poll_event_array
*current
, *wait
;
74 current
= &events
->current
;
77 wait
->nb_fd
= current
->nb_fd
;
78 if (events
->need_realloc
) {
79 ret
= resize_poll_event(wait
, current
->alloc_size
);
84 memcpy(wait
->events
, current
->events
,
85 current
->nb_fd
* sizeof(*current
->events
));
87 /* Update is done and realloc as well. */
88 events
->need_update
= 0;
89 events
->need_realloc
= 0;
98 * Create pollfd data structure.
100 int compat_poll_create(struct lttng_poll_event
*events
, int size
)
102 struct compat_poll_event_array
*current
, *wait
;
104 if (events
== NULL
|| size
<= 0) {
105 ERR("Wrong arguments for poll create");
109 /* Don't bust the limit here */
110 if (size
> poll_max_size
) {
111 size
= poll_max_size
;
114 /* Reset everything before begining the allocation. */
115 memset(events
, 0, sizeof(struct lttng_poll_event
));
117 /* Ease our life a bit. */
118 current
= &events
->current
;
119 wait
= &events
->wait
;
121 /* This *must* be freed by using lttng_poll_free() */
122 wait
->events
= zmalloc(size
* sizeof(struct pollfd
));
123 if (wait
->events
== NULL
) {
124 perror("zmalloc struct pollfd");
128 wait
->alloc_size
= wait
->init_size
= size
;
130 current
->events
= zmalloc(size
* sizeof(struct pollfd
));
131 if (current
->events
== NULL
) {
132 perror("zmalloc struct current pollfd");
136 current
->alloc_size
= current
->init_size
= size
;
145 * Add fd to pollfd data structure with requested events.
147 int compat_poll_add(struct lttng_poll_event
*events
, int fd
,
150 int new_size
, ret
, i
;
151 struct compat_poll_event_array
*current
;
153 if (events
== NULL
|| events
->current
.events
== NULL
|| fd
< 0) {
154 ERR("Bad compat poll add arguments");
158 /* Ease our life a bit. */
159 current
= &events
->current
;
161 /* Check if fd we are trying to add is already there. */
162 for (i
= 0; i
< current
->nb_fd
; i
++) {
163 /* Don't put back the fd we want to delete */
164 if (current
->events
[i
].fd
== fd
) {
170 /* Check for a needed resize of the array. */
171 if (current
->nb_fd
> current
->alloc_size
) {
172 /* Expand it by a power of two of the current size. */
173 new_size
= max_t(int,
174 1U << utils_get_count_order_u32(current
->nb_fd
),
175 current
->alloc_size
<< 1UL);
176 ret
= resize_poll_event(current
, new_size
);
180 events
->need_realloc
= 1;
183 current
->events
[current
->nb_fd
].fd
= fd
;
184 current
->events
[current
->nb_fd
].events
= req_events
;
186 events
->need_update
= 1;
188 DBG("fd %d of %d added to pollfd", fd
, current
->nb_fd
);
197 * Remove a fd from the pollfd structure.
199 int compat_poll_del(struct lttng_poll_event
*events
, int fd
)
201 int new_size
, i
, count
= 0, ret
;
202 struct compat_poll_event_array
*current
;
204 if (events
== NULL
|| events
->current
.events
== NULL
|| fd
< 0) {
205 ERR("Wrong arguments for poll del");
209 /* Ease our life a bit. */
210 current
= &events
->current
;
212 /* Check if we need to shrink it down. */
213 if ((current
->nb_fd
<< 1UL) <= current
->alloc_size
&&
214 current
->nb_fd
>= current
->init_size
) {
216 * Shrink if nb_fd multiplied by two is <= than the actual size and we
217 * are above the initial size.
219 new_size
= max_t(int,
220 utils_get_count_order_u32(current
->nb_fd
) >> 1U,
221 current
->alloc_size
>> 1U);
222 ret
= resize_poll_event(current
, new_size
);
226 events
->need_realloc
= 1;
229 for (i
= 0; i
< current
->nb_fd
; i
++) {
230 /* Don't put back the fd we want to delete */
231 if (current
->events
[i
].fd
!= fd
) {
232 current
->events
[count
].fd
= current
->events
[i
].fd
;
233 current
->events
[count
].events
= current
->events
[i
].events
;
239 events
->need_update
= 1;
248 * Wait on poll() with timeout. Blocking call.
250 int compat_poll_wait(struct lttng_poll_event
*events
, int timeout
)
254 if (events
== NULL
|| events
->current
.events
== NULL
) {
255 ERR("poll wait arguments error");
259 if (events
->current
.nb_fd
== 0) {
260 /* Return an invalid error to be consistent with epoll. */
265 if (events
->need_update
) {
266 ret
= update_current_events(events
);
273 ret
= poll(events
->wait
.events
, events
->wait
.nb_fd
, timeout
);
275 /* At this point, every error is fatal */
281 * poll() should always iterate on all FDs since we handle the pollset in
282 * user space and after poll returns, we have to try every fd for a match.
284 return events
->wait
.nb_fd
;
291 * Setup poll set maximum size.
293 void compat_poll_set_max_size(void)
299 poll_max_size
= DEFAULT_POLL_SIZE
;
301 ret
= getrlimit(RLIMIT_NOFILE
, &lim
);
303 perror("getrlimit poll RLIMIT_NOFILE");
307 poll_max_size
= lim
.rlim_cur
;
308 if (poll_max_size
== 0) {
309 /* Extra precaution */
310 poll_max_size
= DEFAULT_POLL_SIZE
;
313 DBG("poll set max size set to %u", poll_max_size
);
This page took 0.035187 seconds and 4 git commands to generate.