/* This file is part of the YAZ toolkit. * Copyright (C) Index Data * See the file LICENSE for details. */ /** * \file * \brief Select, poll wrapper */ #if HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #if HAVE_SYS_TYPES_H #include #endif #if HAVE_SYS_TIME_H #include #endif #if HAVE_SYS_POLL_H #include #endif #if HAVE_SYS_SELECT_H #include #endif #ifdef WIN32 #if FD_SETSIZE < 512 #define FD_SETSIZE 512 #endif #include #endif /* Note that yaz_poll_select is limited as to how many file descriptors it can multiplex due to its use of select() which in turn uses the statically defined fd_set type to be a bitmap of the file descriptors to check. On Ubuntu 6.06 (and almost certainly on Debian, and probably on all Linuxes, and maybe all Unixes) this is by default set to 1024 (though it may be possible to override this using a #define before including -- I've not tried this). 1024 file descriptors is a lot, but not enough in all cases, e.g. when running IRSpy on a large target database. So you should ensure that YAZ uses ZOOM_yaz_poll_poll() when possible. */ int yaz_poll_select(struct yaz_poll_fd *fds, int num_fds, int sec, int nsec) { struct timeval tv; fd_set input, output, except; int i, r; int max_fd = 0; FD_ZERO(&input); FD_ZERO(&output); FD_ZERO(&except); assert(num_fds >= 0); for (i = 0; i < num_fds; i++) { enum yaz_poll_mask mask = fds[i].input_mask; int fd = fds[i].fd; /* Timeout events */ if (fd < 0) continue; if (mask & yaz_poll_read) FD_SET(fd, &input); if (mask & yaz_poll_write) FD_SET(fd, &output); if (mask & yaz_poll_except) FD_SET(fd, &except); if (max_fd < fd) max_fd = fd; } tv.tv_sec = sec; tv.tv_usec = nsec / 1000; r = select(max_fd+1, &input, &output, &except, (sec == -1 ? 0 : &tv)); if (r >= 0) { for (i = 0; i < num_fds; i++) { enum yaz_poll_mask mask = yaz_poll_none; int fd = fds[i].fd; if (!r) yaz_poll_add(mask, yaz_poll_timeout); else if (fd >= 0) { if (FD_ISSET(fd, &input)) yaz_poll_add(mask, yaz_poll_read); if (FD_ISSET(fd, &output)) yaz_poll_add(mask, yaz_poll_write); if (FD_ISSET(fd, &except)) yaz_poll_add(mask, yaz_poll_except); } fds[i].output_mask = mask; } } return r; } #if HAVE_SYS_POLL_H int yaz_poll_poll(struct yaz_poll_fd *fds, int num_fds, int sec, int nsec) { int i, r; struct pollfd *pollfds = 0; if (num_fds > 0) pollfds = (struct pollfd *) xmalloc(num_fds * sizeof *pollfds); assert(num_fds >= 0); for (i = 0; i < num_fds; i++) { enum yaz_poll_mask mask = fds[i].input_mask; int fd = fds[i].fd; short poll_events = 0; if (mask & yaz_poll_read) poll_events += POLLIN; if (mask & yaz_poll_write) poll_events += POLLOUT; if (mask & yaz_poll_except) poll_events += POLLERR; pollfds[i].fd = fd; pollfds[i].events = poll_events; pollfds[i].revents = 0; } r = poll(pollfds, num_fds, sec == -1 ? -1 : sec*1000 + nsec/1000000); if (r >= 0) { for (i = 0; i < num_fds; i++) { enum yaz_poll_mask mask = yaz_poll_none; if (!r) yaz_poll_add(mask, yaz_poll_timeout); else { if (pollfds[i].revents & POLLIN) yaz_poll_add(mask, yaz_poll_read); if (pollfds[i].revents & POLLOUT) yaz_poll_add(mask, yaz_poll_write); if (pollfds[i].revents & ~(POLLIN | POLLOUT)) { yaz_poll_add(mask, yaz_poll_except); } } fds[i].output_mask = mask; } } xfree(pollfds); return r; } #endif int yaz_poll(struct yaz_poll_fd *fds, int num_fds, int sec, int nsec) { #if HAVE_SYS_POLL_H return yaz_poll_poll(fds, num_fds, sec, nsec); #else return yaz_poll_select(fds, num_fds, sec, nsec); #endif } /* * Local variables: * c-basic-offset: 4 * c-file-style: "Stroustrup" * indent-tabs-mode: nil * End: * vim: shiftwidth=4 tabstop=8 expandtab */