const std = @import("std");
const net = std.net;
const os = std.os.linux;
const posix = std.posix;
pub fn get_errno(
sock: os.fd_t,
) posix.E {
var fr: [1024]u8 = undefined;
const e = os.ioctl(
sock,
os.SIOCSIFFLAGS,
@intFromPtr(&fr),
);
return posix.errno(e);
}
pub fn start_poll() !void {
var addr = net.Address.initIp4([4]u8{ 127, 0, 0, 1 }, 19690);
const listen_sock = try posix.socket(
addr.any.family,
posix.SOCK.STREAM,
posix.IPPROTO.TCP,
);
const len: posix.socklen_t = addr.getOsSockLen();
const lfd = os.bind(listen_sock, &addr.any, len);
std.debug.print(
"LFD: {any}\n",
.{lfd},
);
const listfd = os.listen(listen_sock, 5);
std.debug.print(
"ListFD: {any}\n",
.{listfd},
);
const epollfd = os.epoll_create1(0);
const rval = get_errno(listen_sock);
switch (rval) {
.SUCCESS => std.debug.print(
"Successfully created epoll",
.{},
),
else => std.debug.print(
"Some error: {any}\n",
.{rval},
),
}
if (epollfd < 0) {
@panic("Could not get epoll fd\n");
} else {
std.debug.print(
"Epoll created: {any}\n",
.{epollfd},
);
}
var events: [10]os.epoll_event = undefined;
var ev: os.epoll_event = undefined;
var conn_sock: usize = undefined;
ev.events |= os.EPOLL.IN;
ev.data.fd = listen_sock;
var epolladd = os.epoll_ctl(
@intCast(epollfd),
os.EPOLL.CTL_ADD,
listen_sock,
&ev,
);
if (epolladd < 0) {
@panic("Could not add listen_sock to epoll");
} else {
std.debug.print(
"Epoll added: {any}\n",
.{epolladd},
);
}
var msgbuff: [4096]u8 = undefined;
while (true) {
const nfds = os.epoll_wait(
@intCast(epollfd),
&events,
10,
-1,
);
if (nfds < 0) {
@panic("Could not wait on tasks.\n");
} else {
std.debug.print(
"Got fds after wait: {any}\n",
.{nfds},
);
}
for (0..nfds) |n| {
if (events[n].data.fd == listen_sock) {
// this the socket for the client used for getting and sending data
// this line is important; the len ptr must be set correctly
var client_addr_len: posix.socklen_t = addr.getOsSockLen();
conn_sock = os.accept(listen_sock, &addr.any, &client_addr_len);
if (conn_sock < 0) {
@panic("Could not accept on listening socket");
} else {
std.debug.print(
"Connection accepted: {any}\n",
.{conn_sock},
);
}
const fd_64: isize = @bitCast(conn_sock);
const fd_32: i32 = @intCast(fd_64);
const fd_t: os.fd_t = @as(os.fd_t, fd_32);
_ = os.fcntl(fd_t, os.F.SETFL, os.SOCK.NONBLOCK);
ev.events = os.EPOLL.IN | os.EPOLL.ET;
ev.data.fd = fd_t;
epolladd = os.epoll_ctl(@intCast(epollfd), os.EPOLL.CTL_ADD, fd_t, &ev);
if (epolladd < 0) {
@panic("Could not add connection socket");
} else {
// do something here
std.debug.print(
"Connect sock added to epoll: {any}\n",
.{epolladd},
);
const buff: []const u8 = "Hi there, thanks for connection to our server!\n";
const bytes_written = os.write(
fd_t,
@ptrCast(&buff[0]),
buff.len,
);
std.debug.print("Wrote {} bytes to socket\n", .{bytes_written});
}
continue;
}
if (events[n].events & os.EPOLL.IN != 0) {
@memset(&msgbuff, 0);
const bytes_read = os.read(
events[n].data.fd,
@ptrCast(&msgbuff[0]),
msgbuff.len,
);
std.debug.print("Got data. Total bytes received: {d}\nMessage: {sc}\n", .{ bytes_read, msgbuff[0..bytes_read] });
}
}
}
}
So here’s the full script. I am basically trying to learn how non-blocking I/O works. I read somewhere that Node.js event loop uses libuv
which I think use epoll
and other platform specific syscall for this. I was trying to create a minimal example.
Most of the epoll
related code is from man pages on epoll
. Hope that explains the usecase.