blob: f7ee424cb5059f1fb78dd07be3304e304faa5b99 [file] [log] [blame]
Neels Hofmeyr17518fe2017-06-20 04:35:06 +02001/*! \file select.c
2 * select filedescriptor handling.
3 * Taken from:
Harald Welteec8b4502010-02-20 20:34:29 +01004 * userspace logging daemon for the iptables ULOG target
Neels Hofmeyr17518fe2017-06-20 04:35:06 +02005 * of the linux 2.4 netfilter subsystem. */
6/*
Harald Welteec8b4502010-02-20 20:34:29 +01007 * (C) 2000-2009 by Harald Welte <laforge@gnumonks.org>
8 *
9 * This program is free software; you can redistribute it and/or modify
Harald Welte9d92f0e2010-10-31 13:56:45 +010010 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
Harald Welteec8b4502010-02-20 20:34:29 +010013 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
Jaroslav Škarvada2b82c1c2015-11-11 16:02:54 +010021 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
22 * MA 02110-1301, USA.
Harald Welteec8b4502010-02-20 20:34:29 +010023 */
24
25#include <fcntl.h>
Holger Hans Peter Freyther43558312010-08-06 06:48:43 +080026#include <stdio.h>
Harald Welte9e166e82014-03-10 17:28:33 +010027#include <string.h>
Philipp Maierb2888532016-12-09 14:07:18 +010028#include <stdbool.h>
Holger Hans Peter Freyther43558312010-08-06 06:48:43 +080029
Pablo Neira Ayuso83419342011-03-22 16:36:13 +010030#include <osmocom/core/select.h>
31#include <osmocom/core/linuxlist.h>
32#include <osmocom/core/timer.h>
Harald Welteec8b4502010-02-20 20:34:29 +010033
Harald Welte54844802010-02-20 22:23:08 +010034#include "../config.h"
35
Harald Welteec8b4502010-02-20 20:34:29 +010036#ifdef HAVE_SYS_SELECT_H
Harald Welte1d604d82017-05-17 15:32:35 +010037#include <sys/select.h>
Harald Welteec8b4502010-02-20 20:34:29 +010038
Harald Welteba6988b2011-08-17 12:46:48 +020039/*! \addtogroup select
40 * @{
Neels Hofmeyr87e45502017-06-20 00:17:59 +020041 * select() loop abstraction
Neels Hofmeyr17518fe2017-06-20 04:35:06 +020042 *
43 * \file select.c */
Harald Welteba6988b2011-08-17 12:46:48 +020044
Harald Welteec8b4502010-02-20 20:34:29 +010045static int maxfd = 0;
Pablo Neira Ayusof7f89d02011-05-07 12:42:40 +020046static LLIST_HEAD(osmo_fds);
Harald Welteec8b4502010-02-20 20:34:29 +010047static int unregistered_count;
48
Philipp Maierb2888532016-12-09 14:07:18 +010049
Neels Hofmeyr87e45502017-06-20 00:17:59 +020050/*! Check if a file descriptor is already registered
Philipp Maierb2888532016-12-09 14:07:18 +010051 * \param[in] fd osmocom file descriptor to be checked
52 * \returns true if registered; otherwise false
53 */
54bool osmo_fd_is_registered(struct osmo_fd *fd)
55{
56 struct osmo_fd *entry;
57 llist_for_each_entry(entry, &osmo_fds, list) {
58 if (entry == fd) {
59 return true;
60 }
61 }
62
63 return false;
64}
65
Neels Hofmeyr87e45502017-06-20 00:17:59 +020066/*! Register a new file descriptor with select loop abstraction
Harald Welteba6988b2011-08-17 12:46:48 +020067 * \param[in] fd osmocom file descriptor to be registered
Harald Welte2d2e2cc2016-04-25 12:11:20 +020068 * \returns 0 on success; negative in case of error
Harald Welteba6988b2011-08-17 12:46:48 +020069 */
Pablo Neira Ayusof7f89d02011-05-07 12:42:40 +020070int osmo_fd_register(struct osmo_fd *fd)
Harald Welteec8b4502010-02-20 20:34:29 +010071{
72 int flags;
73
74 /* make FD nonblocking */
75 flags = fcntl(fd->fd, F_GETFL);
76 if (flags < 0)
77 return flags;
78 flags |= O_NONBLOCK;
79 flags = fcntl(fd->fd, F_SETFL, flags);
80 if (flags < 0)
81 return flags;
82
Harald Welte32e1f232011-06-26 13:07:18 +020083 /* set close-on-exec flag */
84 flags = fcntl(fd->fd, F_GETFD);
85 if (flags < 0)
86 return flags;
87 flags |= FD_CLOEXEC;
88 flags = fcntl(fd->fd, F_SETFD, flags);
89 if (flags < 0)
90 return flags;
91
Harald Welteec8b4502010-02-20 20:34:29 +010092 /* Register FD */
93 if (fd->fd > maxfd)
94 maxfd = fd->fd;
95
Holger Hans Peter Freyther43558312010-08-06 06:48:43 +080096#ifdef BSC_FD_CHECK
Philipp Maierb2888532016-12-09 14:07:18 +010097 if (osmo_fd_is_registered(fd)) {
98 fprintf(stderr, "Adding a osmo_fd that is already in the list.\n");
99 return 0;
Holger Hans Peter Freyther43558312010-08-06 06:48:43 +0800100 }
101#endif
102
Pablo Neira Ayusof7f89d02011-05-07 12:42:40 +0200103 llist_add_tail(&fd->list, &osmo_fds);
Harald Welteec8b4502010-02-20 20:34:29 +0100104
105 return 0;
106}
107
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200108/*! Unregister a file descriptor from select loop abstraction
Harald Welteba6988b2011-08-17 12:46:48 +0200109 * \param[in] fd osmocom file descriptor to be unregistered
110 */
Pablo Neira Ayusof7f89d02011-05-07 12:42:40 +0200111void osmo_fd_unregister(struct osmo_fd *fd)
Harald Welteec8b4502010-02-20 20:34:29 +0100112{
Philipp Maierb2888532016-12-09 14:07:18 +0100113 /* Note: when fd is inside the osmo_fds list (not registered before)
114 * this function will crash! If in doubt, check file descriptor with
115 * osmo_fd_is_registered() */
Harald Welteec8b4502010-02-20 20:34:29 +0100116 unregistered_count++;
117 llist_del(&fd->list);
118}
119
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200120/*! Populate the fd_sets and return the highest fd number
Holger Hans Peter Freytherff206412017-03-07 14:28:38 +0100121 * \param[in] _rset The readfds to populate
122 * \param[in] _wset The wrtiefds to populate
123 * \param[in] _eset The errorfds to populate
124 *
125 * \returns The highest file descriptor seen or 0 on an empty list
126 */
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100127inline int osmo_fd_fill_fds(void *_rset, void *_wset, void *_eset)
Harald Welteec8b4502010-02-20 20:34:29 +0100128{
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100129 fd_set *readset = _rset, *writeset = _wset, *exceptset = _eset;
130 struct osmo_fd *ufd;
Holger Hans Peter Freytherff206412017-03-07 14:28:38 +0100131 int highfd = 0;
Harald Welteec8b4502010-02-20 20:34:29 +0100132
Pablo Neira Ayusof7f89d02011-05-07 12:42:40 +0200133 llist_for_each_entry(ufd, &osmo_fds, list) {
Harald Welteec8b4502010-02-20 20:34:29 +0100134 if (ufd->when & BSC_FD_READ)
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100135 FD_SET(ufd->fd, readset);
Harald Welteec8b4502010-02-20 20:34:29 +0100136
137 if (ufd->when & BSC_FD_WRITE)
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100138 FD_SET(ufd->fd, writeset);
Harald Welteec8b4502010-02-20 20:34:29 +0100139
140 if (ufd->when & BSC_FD_EXCEPT)
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100141 FD_SET(ufd->fd, exceptset);
Holger Hans Peter Freytherff206412017-03-07 14:28:38 +0100142
143 if (ufd->fd > highfd)
144 highfd = ufd->fd;
Harald Welteec8b4502010-02-20 20:34:29 +0100145 }
146
Holger Hans Peter Freytherff206412017-03-07 14:28:38 +0100147 return highfd;
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100148}
Harald Welteec8b4502010-02-20 20:34:29 +0100149
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100150inline int osmo_fd_disp_fds(void *_rset, void *_wset, void *_eset)
151{
152 struct osmo_fd *ufd, *tmp;
153 int work = 0;
154 fd_set *readset = _rset, *writeset = _wset, *exceptset = _eset;
Harald Welteec8b4502010-02-20 20:34:29 +0100155
Harald Welteec8b4502010-02-20 20:34:29 +0100156restart:
157 unregistered_count = 0;
Pablo Neira Ayusof7f89d02011-05-07 12:42:40 +0200158 llist_for_each_entry_safe(ufd, tmp, &osmo_fds, list) {
Harald Welteec8b4502010-02-20 20:34:29 +0100159 int flags = 0;
160
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100161 if (FD_ISSET(ufd->fd, readset)) {
Harald Welteec8b4502010-02-20 20:34:29 +0100162 flags |= BSC_FD_READ;
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100163 FD_CLR(ufd->fd, readset);
Harald Welteec8b4502010-02-20 20:34:29 +0100164 }
165
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100166 if (FD_ISSET(ufd->fd, writeset)) {
Harald Welteec8b4502010-02-20 20:34:29 +0100167 flags |= BSC_FD_WRITE;
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100168 FD_CLR(ufd->fd, writeset);
Harald Welteec8b4502010-02-20 20:34:29 +0100169 }
170
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100171 if (FD_ISSET(ufd->fd, exceptset)) {
Harald Welteec8b4502010-02-20 20:34:29 +0100172 flags |= BSC_FD_EXCEPT;
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100173 FD_CLR(ufd->fd, exceptset);
Harald Welteec8b4502010-02-20 20:34:29 +0100174 }
175
176 if (flags) {
177 work = 1;
178 ufd->cb(ufd, flags);
179 }
Holger Hans Peter Freyther92e1e702014-04-09 17:24:00 +0200180 /* ugly, ugly hack. If more than one filedescriptor was
Harald Welteec8b4502010-02-20 20:34:29 +0100181 * unregistered, they might have been consecutive and
182 * llist_for_each_entry_safe() is no longer safe */
Holger Hans Peter Freyther23ba4742010-04-11 17:33:19 +0200183 /* this seems to happen with the last element of the list as well */
184 if (unregistered_count >= 1)
Harald Welteec8b4502010-02-20 20:34:29 +0100185 goto restart;
186 }
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100187
Harald Welteec8b4502010-02-20 20:34:29 +0100188 return work;
189}
190
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200191/*! select main loop integration
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100192 * \param[in] polling should we pollonly (1) or block on select (0)
Harald Welte2d2e2cc2016-04-25 12:11:20 +0200193 * \returns 0 if no fd handled; 1 if fd handled; negative in case of error
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100194 */
195int osmo_select_main(int polling)
196{
197 fd_set readset, writeset, exceptset;
198 int rc;
199 struct timeval no_time = {0, 0};
200
201 FD_ZERO(&readset);
202 FD_ZERO(&writeset);
203 FD_ZERO(&exceptset);
204
205 /* prepare read and write fdsets */
206 osmo_fd_fill_fds(&readset, &writeset, &exceptset);
207
Holger Hans Peter Freyther61f28882016-03-21 09:55:05 +0100208 if (!polling)
209 osmo_timers_prepare();
210 rc = select(maxfd+1, &readset, &writeset, &exceptset, polling ? &no_time : osmo_timers_nearest());
211 if (rc < 0)
212 return 0;
213
214 /* fire timers */
215 osmo_timers_update();
216
217 /* call registered callback functions */
218 return osmo_fd_disp_fds(&readset, &writeset, &exceptset);
219}
220
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200221/*! find an osmo_fd based on the integer fd
Harald Welte2d2e2cc2016-04-25 12:11:20 +0200222 * \param[in] fd file descriptor to use as search key
223 * \returns \ref osmo_fd for \ref fd; NULL in case it doesn't exist */
Harald Welte6c33ae22016-03-19 21:17:58 +0100224struct osmo_fd *osmo_fd_get_by_fd(int fd)
225{
226 struct osmo_fd *ofd;
227
228 llist_for_each_entry(ofd, &osmo_fds, list) {
229 if (ofd->fd == fd)
230 return ofd;
231 }
232 return NULL;
233}
234
Sylvain Munautdca7d2c2012-04-18 21:53:23 +0200235/*! @} */
Harald Welteba6988b2011-08-17 12:46:48 +0200236
Harald Welteec8b4502010-02-20 20:34:29 +0100237#endif /* _HAVE_SYS_SELECT_H */