net: speed up select fdwatch lookups
Some checks failed
build / Linux asan (push) Has been cancelled
build / Linux release (push) Has been cancelled
build / FreeBSD build (push) Has been cancelled

This commit is contained in:
server
2026-04-14 07:08:23 +02:00
parent 4f1e7b0e9c
commit 3272537376
2 changed files with 92 additions and 16 deletions

View File

@@ -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;

View File

@@ -308,6 +308,46 @@ void TestFdwatchBackendMetadata()
fdwatch_delete(fdw);
}
void TestFdwatchSlotReuseAfterDelete()
{
int sockets_a[2] = { -1, -1 };
int sockets_b[2] = { -1, -1 };
Expect(socketpair(AF_UNIX, SOCK_STREAM, 0, sockets_a) == 0, "socketpair A failed");
Expect(socketpair(AF_UNIX, SOCK_STREAM, 0, sockets_b) == 0, "socketpair B failed");
LPFDWATCH fdw = fdwatch_new(64);
Expect(fdw != nullptr, "fdwatch_new for slot reuse failed");
int marker_a = 11;
int marker_b = 22;
fdwatch_add_fd(fdw, sockets_a[1], &marker_a, FDW_READ, false);
fdwatch_add_fd(fdw, sockets_b[1], &marker_b, FDW_READ, false);
fdwatch_del_fd(fdw, sockets_a[1]);
const uint8_t byte = 0x51;
Expect(write(sockets_b[0], &byte, sizeof(byte)) == sizeof(byte), "socketpair B write failed");
timeval timeout {};
timeout.tv_sec = 0;
timeout.tv_usec = 200000;
const int num_events = fdwatch(fdw, &timeout);
Expect(num_events == 1, "Expected one read event after slot reuse");
Expect(fdwatch_get_client_data(fdw, 0) == &marker_b, "Unexpected client data after slot reuse");
Expect(fdwatch_check_event(fdw, sockets_b[1], 0) == FDW_READ, "Expected FDW_READ after slot reuse");
uint8_t read_back = 0;
Expect(read(sockets_b[1], &read_back, sizeof(read_back)) == sizeof(read_back), "socketpair B read failed");
Expect(read_back == byte, "Read payload mismatch after slot reuse");
fdwatch_delete(fdw);
close(sockets_a[0]);
close(sockets_a[1]);
close(sockets_b[0]);
close(sockets_b[1]);
}
}
int main()
@@ -319,6 +359,7 @@ int main()
TestSocketAuthWireFlow();
TestFdwatchBackendMetadata();
TestFdwatchReadAndOneshotWrite();
TestFdwatchSlotReuseAfterDelete();
std::cout << "metin smoke tests passed\n";
return 0;
}