net: speed up select fdwatch lookups
This commit is contained in:
@@ -52,8 +52,10 @@ struct FdwatchSelectState
|
||||
fd_set rfd_set;
|
||||
fd_set wfd_set;
|
||||
socket_t* select_fds;
|
||||
int* fd_slot_by_fd;
|
||||
int* select_rfdidx;
|
||||
int nselect_fds;
|
||||
int max_fd_plus_one;
|
||||
fd_set working_rfd_set;
|
||||
fd_set working_wfd_set;
|
||||
void** fd_data;
|
||||
@@ -343,11 +345,19 @@ static LPFDWATCH fdwatch_new_select(int nfiles)
|
||||
FD_ZERO(&fdw->state.working_wfd_set);
|
||||
|
||||
CREATE(fdw->state.select_fds, socket_t, fdw->descriptor_limit);
|
||||
CREATE(fdw->state.fd_slot_by_fd, int, fdw->descriptor_limit);
|
||||
CREATE(fdw->state.select_rfdidx, int, fdw->descriptor_limit);
|
||||
CREATE(fdw->state.fd_rw, int, fdw->descriptor_limit);
|
||||
CREATE(fdw->state.fd_data, void*, fdw->descriptor_limit);
|
||||
|
||||
for (int i = 0; i < fdw->descriptor_limit; ++i)
|
||||
{
|
||||
fdw->state.fd_slot_by_fd[i] = -1;
|
||||
fdw->state.select_rfdidx[i] = -1;
|
||||
}
|
||||
|
||||
fdw->state.nselect_fds = 0;
|
||||
fdw->state.max_fd_plus_one = 1;
|
||||
return fdw;
|
||||
}
|
||||
|
||||
@@ -356,6 +366,7 @@ static void fdwatch_delete_select(LPFDWATCH fdw)
|
||||
free(fdw->state.fd_data);
|
||||
free(fdw->state.fd_rw);
|
||||
free(fdw->state.select_fds);
|
||||
free(fdw->state.fd_slot_by_fd);
|
||||
free(fdw->state.select_rfdidx);
|
||||
free(fdw);
|
||||
|
||||
@@ -366,26 +377,22 @@ static void fdwatch_delete_select(LPFDWATCH fdw)
|
||||
|
||||
static int fdwatch_get_fdidx_select(LPFDWATCH fdw, socket_t fd)
|
||||
{
|
||||
for (int i = 0; i < fdw->state.nselect_fds; ++i)
|
||||
{
|
||||
if (fdw->state.select_fds[i] == fd)
|
||||
return i;
|
||||
}
|
||||
if (fd < 0 || fd >= fdw->descriptor_limit)
|
||||
return -1;
|
||||
|
||||
return -1;
|
||||
return fdw->state.fd_slot_by_fd[fd];
|
||||
}
|
||||
|
||||
static int fdwatch_get_maxfd_select(LPFDWATCH fdw)
|
||||
static void fdwatch_recompute_maxfd_select(LPFDWATCH fdw)
|
||||
{
|
||||
socket_t max_fd = 0;
|
||||
fdw->state.max_fd_plus_one = 1;
|
||||
|
||||
for (int i = 0; i < fdw->state.nselect_fds; ++i)
|
||||
{
|
||||
if (fdw->state.select_fds[i] > max_fd)
|
||||
max_fd = fdw->state.select_fds[i];
|
||||
const int candidate = static_cast<int>(fdw->state.select_fds[i]) + 1;
|
||||
if (candidate > fdw->state.max_fd_plus_one)
|
||||
fdw->state.max_fd_plus_one = candidate;
|
||||
}
|
||||
|
||||
return static_cast<int>(max_fd) + 1;
|
||||
}
|
||||
|
||||
static int fdwatch_check_fd_select(LPFDWATCH fdw, socket_t fd);
|
||||
@@ -407,16 +414,27 @@ static void fdwatch_clear_fd_select(LPFDWATCH fdw, socket_t fd)
|
||||
|
||||
static void fdwatch_add_fd_select(LPFDWATCH fdw, socket_t fd, void* client_data, int rw, int oneshot)
|
||||
{
|
||||
if (fd < 0 || fd >= fdw->descriptor_limit)
|
||||
{
|
||||
sys_err("fdwatch_add_fd_select: descriptor %d exceeds select backend limit %d", fd, fdw->descriptor_limit);
|
||||
return;
|
||||
}
|
||||
|
||||
int idx = fdwatch_get_fdidx_select(fdw, fd);
|
||||
|
||||
if (idx < 0)
|
||||
{
|
||||
if (fdw->state.nselect_fds >= fdw->descriptor_limit)
|
||||
{
|
||||
sys_err("fdwatch_add_fd_select: descriptor table full (%d)", fdw->descriptor_limit);
|
||||
return;
|
||||
}
|
||||
|
||||
idx = fdw->state.nselect_fds;
|
||||
fdw->state.select_fds[fdw->state.nselect_fds++] = fd;
|
||||
fdw->state.fd_slot_by_fd[fd] = idx;
|
||||
fdw->state.fd_rw[idx] = 0;
|
||||
fdw->state.select_rfdidx[idx] = -1;
|
||||
}
|
||||
|
||||
fdw->state.fd_rw[idx] |= rw;
|
||||
@@ -431,6 +449,10 @@ static void fdwatch_add_fd_select(LPFDWATCH fdw, socket_t fd, void* client_data,
|
||||
|
||||
if (rw & FDW_WRITE)
|
||||
FD_SET(fd, &fdw->state.wfd_set);
|
||||
|
||||
const int fd_plus_one = static_cast<int>(fd) + 1;
|
||||
if (fd_plus_one > fdw->state.max_fd_plus_one)
|
||||
fdw->state.max_fd_plus_one = fd_plus_one;
|
||||
}
|
||||
|
||||
static void fdwatch_del_fd_select(LPFDWATCH fdw, socket_t fd)
|
||||
@@ -444,22 +466,35 @@ static void fdwatch_del_fd_select(LPFDWATCH fdw, socket_t fd)
|
||||
return;
|
||||
|
||||
--fdw->state.nselect_fds;
|
||||
const int last_idx = fdw->state.nselect_fds;
|
||||
const socket_t moved_fd = fdw->state.select_fds[last_idx];
|
||||
|
||||
fdw->state.select_fds[idx] = fdw->state.select_fds[fdw->state.nselect_fds];
|
||||
fdw->state.fd_data[idx] = fdw->state.fd_data[fdw->state.nselect_fds];
|
||||
fdw->state.fd_rw[idx] = fdw->state.fd_rw[fdw->state.nselect_fds];
|
||||
if (idx != last_idx)
|
||||
{
|
||||
fdw->state.select_fds[idx] = moved_fd;
|
||||
fdw->state.fd_data[idx] = fdw->state.fd_data[last_idx];
|
||||
fdw->state.fd_rw[idx] = fdw->state.fd_rw[last_idx];
|
||||
fdw->state.select_rfdidx[idx] = -1;
|
||||
fdw->state.fd_slot_by_fd[moved_fd] = idx;
|
||||
}
|
||||
|
||||
fdw->state.fd_slot_by_fd[fd] = -1;
|
||||
fdw->state.select_rfdidx[last_idx] = -1;
|
||||
|
||||
FD_CLR(fd, &fdw->state.rfd_set);
|
||||
FD_CLR(fd, &fdw->state.wfd_set);
|
||||
FD_CLR(fd, &fdw->state.working_rfd_set);
|
||||
FD_CLR(fd, &fdw->state.working_wfd_set);
|
||||
|
||||
if (static_cast<int>(fd) + 1 >= fdw->state.max_fd_plus_one)
|
||||
fdwatch_recompute_maxfd_select(fdw);
|
||||
}
|
||||
|
||||
static int fdwatch_select(LPFDWATCH fdw, struct timeval* timeout)
|
||||
{
|
||||
int r;
|
||||
int event_idx = 0;
|
||||
int max_fd = fdwatch_get_maxfd_select(fdw);
|
||||
int max_fd = fdw->state.max_fd_plus_one;
|
||||
struct timeval tv;
|
||||
|
||||
fdw->state.working_rfd_set = fdw->state.rfd_set;
|
||||
|
||||
Reference in New Issue
Block a user