mirror of
https://github.com/bjc/prosody.git
synced 2025-04-04 21:57:45 +03:00
util.poll: Add support for the poll() API
Might be better than select(), more portable than epoll.
This commit is contained in:
parent
04aa101da3
commit
af95bb77e6
3 changed files with 133 additions and 2 deletions
1
CHANGES
1
CHANGES
|
@ -56,6 +56,7 @@ TRUNK
|
||||||
- MUC: support for XEP-0421 occupant identifiers
|
- MUC: support for XEP-0421 occupant identifiers
|
||||||
- `prosodyctl check connectivity` via observe.jabber.network
|
- `prosodyctl check connectivity` via observe.jabber.network
|
||||||
- libunbound for DNS queries
|
- libunbound for DNS queries
|
||||||
|
- The POSIX poll() API used by server_epoll on \*nix other than Linux
|
||||||
|
|
||||||
## Changes
|
## Changes
|
||||||
|
|
||||||
|
|
|
@ -17,10 +17,12 @@ end
|
||||||
|
|
||||||
local record lib
|
local record lib
|
||||||
new : function () : state
|
new : function () : state
|
||||||
ENOENT : integer
|
|
||||||
EEXIST : integer
|
EEXIST : integer
|
||||||
|
EMFILE : integer
|
||||||
|
ENOENT : integer
|
||||||
enum api_backend
|
enum api_backend
|
||||||
"epoll"
|
"epoll"
|
||||||
|
"poll"
|
||||||
"select"
|
"select"
|
||||||
end
|
end
|
||||||
api : api_backend
|
api : api_backend
|
||||||
|
|
130
util-src/poll.c
130
util-src/poll.c
|
@ -1,7 +1,7 @@
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Lua polling library
|
* Lua polling library
|
||||||
* Copyright (C) 2017-2018 Kim Alvefur
|
* Copyright (C) 2017-2022 Kim Alvefur
|
||||||
*
|
*
|
||||||
* This project is MIT licensed. Please see the
|
* This project is MIT licensed. Please see the
|
||||||
* COPYING file in the source package for more information.
|
* COPYING file in the source package for more information.
|
||||||
|
@ -15,6 +15,9 @@
|
||||||
#if defined(__linux__)
|
#if defined(__linux__)
|
||||||
#define USE_EPOLL
|
#define USE_EPOLL
|
||||||
#define POLL_BACKEND "epoll"
|
#define POLL_BACKEND "epoll"
|
||||||
|
#elif defined(__unix__)
|
||||||
|
#define USE_POLL
|
||||||
|
#define POLL_BACKEND "poll"
|
||||||
#else
|
#else
|
||||||
#define USE_SELECT
|
#define USE_SELECT
|
||||||
#define POLL_BACKEND "select"
|
#define POLL_BACKEND "select"
|
||||||
|
@ -26,6 +29,12 @@
|
||||||
#define MAX_EVENTS 64
|
#define MAX_EVENTS 64
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef USE_POLL
|
||||||
|
#include <poll.h>
|
||||||
|
#ifndef MAX_EVENTS
|
||||||
|
#define MAX_EVENTS 10000
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
#include <sys/select.h>
|
#include <sys/select.h>
|
||||||
#endif
|
#endif
|
||||||
|
@ -51,6 +60,10 @@ typedef struct Lpoll_state {
|
||||||
int epoll_fd;
|
int epoll_fd;
|
||||||
struct epoll_event events[MAX_EVENTS];
|
struct epoll_event events[MAX_EVENTS];
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef USE_POLL
|
||||||
|
nfds_t count;
|
||||||
|
struct pollfd events[MAX_EVENTS];
|
||||||
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
fd_set wantread;
|
fd_set wantread;
|
||||||
fd_set wantwrite;
|
fd_set wantwrite;
|
||||||
|
@ -99,6 +112,32 @@ static int Ladd(lua_State *L) {
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef USE_POLL
|
||||||
|
|
||||||
|
for(nfds_t i = 0; i < state->count; i++) {
|
||||||
|
if(state->events[i].fd == fd) {
|
||||||
|
luaL_pushfail(L);
|
||||||
|
lua_pushstring(L, strerror(EEXIST));
|
||||||
|
lua_pushinteger(L, EEXIST);
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(state->count >= MAX_EVENTS) {
|
||||||
|
luaL_pushfail(L);
|
||||||
|
lua_pushstring(L, strerror(EMFILE));
|
||||||
|
lua_pushinteger(L, EMFILE);
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
state->events[state->count].fd = fd;
|
||||||
|
state->events[state->count].events = (wantread ? POLLIN : 0) | (wantwrite ? POLLOUT : 0);
|
||||||
|
state->events[state->count].revents = 0;
|
||||||
|
state->count++;
|
||||||
|
|
||||||
|
lua_pushboolean(L, 1);
|
||||||
|
return 1;
|
||||||
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
|
|
||||||
if(fd > FD_SETSIZE) {
|
if(fd > FD_SETSIZE) {
|
||||||
|
@ -173,6 +212,27 @@ static int Lset(lua_State *L) {
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef USE_POLL
|
||||||
|
int wantread = lua_toboolean(L, 3);
|
||||||
|
int wantwrite = lua_toboolean(L, 4);
|
||||||
|
|
||||||
|
for(nfds_t i = 0; i < state->count; i++) {
|
||||||
|
struct pollfd *event = &state->events[i];
|
||||||
|
|
||||||
|
if(event->fd == fd) {
|
||||||
|
event->events = (wantread ? POLLIN : 0) | (wantwrite ? POLLOUT : 0);
|
||||||
|
lua_pushboolean(L, 1);
|
||||||
|
return 1;
|
||||||
|
} else if(event->fd == -1) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
luaL_pushfail(L);
|
||||||
|
lua_pushstring(L, strerror(ENOENT));
|
||||||
|
lua_pushinteger(L, ENOENT);
|
||||||
|
return 3;
|
||||||
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
|
|
||||||
if(!FD_ISSET(fd, &state->all)) {
|
if(!FD_ISSET(fd, &state->all)) {
|
||||||
|
@ -232,6 +292,40 @@ static int Ldel(lua_State *L) {
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef USE_POLL
|
||||||
|
|
||||||
|
if(state->count == 0) {
|
||||||
|
luaL_pushfail(L);
|
||||||
|
lua_pushstring(L, strerror(ENOENT));
|
||||||
|
lua_pushinteger(L, ENOENT);
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Move the last item on top of the removed one
|
||||||
|
*/
|
||||||
|
struct pollfd *last = &state->events[state->count - 1];
|
||||||
|
|
||||||
|
for(nfds_t i = 0; i < state->count; i++) {
|
||||||
|
struct pollfd *event = &state->events[i];
|
||||||
|
|
||||||
|
if(event->fd == fd) {
|
||||||
|
event->fd = last->fd;
|
||||||
|
event->events = last->events;
|
||||||
|
event->revents = last->revents;
|
||||||
|
last->fd = -1;
|
||||||
|
state->count--;
|
||||||
|
|
||||||
|
lua_pushboolean(L, 1);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
luaL_pushfail(L);
|
||||||
|
lua_pushstring(L, strerror(ENOENT));
|
||||||
|
lua_pushinteger(L, ENOENT);
|
||||||
|
return 3;
|
||||||
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
|
|
||||||
if(!FD_ISSET(fd, &state->all)) {
|
if(!FD_ISSET(fd, &state->all)) {
|
||||||
|
@ -269,6 +363,22 @@ static int Lpushevent(lua_State *L, struct Lpoll_state *state) {
|
||||||
return 3;
|
return 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
#ifdef USE_POLL
|
||||||
|
|
||||||
|
for(int i = state->processed - 1; i >= 0; i--) {
|
||||||
|
struct pollfd *event = &state->events[i];
|
||||||
|
|
||||||
|
if(event->fd != -1 && event->revents != 0) {
|
||||||
|
lua_pushinteger(L, event->fd);
|
||||||
|
lua_pushboolean(L, event->revents & (POLLIN | POLLHUP | POLLERR));
|
||||||
|
lua_pushboolean(L, event->revents & POLLOUT);
|
||||||
|
event->revents = 0;
|
||||||
|
state->processed = i;
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
|
|
||||||
|
@ -307,6 +417,9 @@ static int Lwait(lua_State *L) {
|
||||||
#ifdef USE_EPOLL
|
#ifdef USE_EPOLL
|
||||||
ret = epoll_wait(state->epoll_fd, state->events, MAX_EVENTS, timeout * 1000);
|
ret = epoll_wait(state->epoll_fd, state->events, MAX_EVENTS, timeout * 1000);
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef USE_POLL
|
||||||
|
ret = poll(state->events, state->count, timeout * 1000);
|
||||||
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
/*
|
/*
|
||||||
* select(2) mutates the fd_sets passed to it so in order to not
|
* select(2) mutates the fd_sets passed to it so in order to not
|
||||||
|
@ -349,6 +462,9 @@ static int Lwait(lua_State *L) {
|
||||||
#ifdef USE_EPOLL
|
#ifdef USE_EPOLL
|
||||||
state->processed = ret;
|
state->processed = ret;
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef USE_POLL
|
||||||
|
state->processed = state->count;
|
||||||
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
state->processed = -1;
|
state->processed = -1;
|
||||||
#endif
|
#endif
|
||||||
|
@ -420,6 +536,17 @@ static int Lnew(lua_State *L) {
|
||||||
|
|
||||||
state->epoll_fd = epoll_fd;
|
state->epoll_fd = epoll_fd;
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef USE_POLL
|
||||||
|
state->processed = -1;
|
||||||
|
state->count = 0;
|
||||||
|
|
||||||
|
for(nfds_t i = 0; i < MAX_EVENTS; i++) {
|
||||||
|
state->events[i].fd = -1;
|
||||||
|
state->events[i].events = 0;
|
||||||
|
state->events[i].revents = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
FD_ZERO(&state->wantread);
|
FD_ZERO(&state->wantread);
|
||||||
FD_ZERO(&state->wantwrite);
|
FD_ZERO(&state->wantwrite);
|
||||||
|
@ -482,6 +609,7 @@ int luaopen_util_poll(lua_State *L) {
|
||||||
lua_setfield(L, -2, #named_error);
|
lua_setfield(L, -2, #named_error);
|
||||||
|
|
||||||
push_errno(EEXIST);
|
push_errno(EEXIST);
|
||||||
|
push_errno(EMFILE);
|
||||||
push_errno(ENOENT);
|
push_errno(ENOENT);
|
||||||
|
|
||||||
lua_pushliteral(L, POLL_BACKEND);
|
lua_pushliteral(L, POLL_BACKEND);
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue