Files
sx/library/modules/std/socket.sx
agra 59f90d2939 refactor(ffi-linkage): Phase 6.3 — migrate std/ #foreign→extern
Pure source rename across 11 std modules (~60 sites): cli/core/fmt/fs/log/
net/kqueue/process/socket/thread/time/trace. All fn-decl markers — bare
'#foreign;', '#foreign libc;'/'#foreign tlib;' (LIB ref), and
'#foreign libc "csym";' (LIB+rename) → the same 'extern …' tail (extern carries
the identical [LIB] ["csym"] axis). Plus 2 stale comment mentions (fmt/fs).
No class forms in std. These modules ARE host-corpus-exercised, so the empty
snapshot diff is direct validation. Suite green (647 corpus / 444 unit, 0
failed).
2026-06-15 04:35:52 +03:00

138 lines
4.7 KiB
Plaintext

// POSIX socket module (macOS only)
// sockaddr_in layout, errno symbol, and constants are platform-specific;
// per-OS selection is PLAN-HTTPZ C3.
libc :: #library "c";
// POSIX socket API
socket :: (domain: i32, kind: i32, protocol: i32) -> i32 extern libc;
setsockopt :: (fd: i32, level: i32, optname: i32, optval: *i32, optlen: u32) -> i32 extern libc;
bind :: (fd: i32, addr: *SockAddr, addrlen: u32) -> i32 extern libc;
listen :: (fd: i32, backlog: i32) -> i32 extern libc;
accept :: (fd: i32, addr: *SockAddr, addrlen: *u32) -> i32 extern libc;
connect :: (fd: i32, addr: *SockAddr, addrlen: u32) -> i32 extern libc;
read :: (fd: i32, buf: [*]u8, count: usize) -> isize extern libc;
write :: (fd: i32, buf: [*]u8, count: usize) -> isize extern libc;
close :: (fd: i32) -> i32 extern libc;
shutdown :: (fd: i32, how: i32) -> i32 extern libc;
socketpair :: (domain: i32, kind: i32, protocol: i32, fds: *i32) -> i32 extern libc;
fcntl :: (fd: i32, cmd: i32, ..args: []i32) -> i32 extern libc;
// Constants (macOS)
AF_UNIX :i32: 1;
AF_INET :i32: 2;
SOCK_STREAM :i32: 1;
SOL_SOCKET :i32: 0xFFFF;
SO_REUSEADDR :i32: 0x4;
SHUT_RD :i32: 0;
SHUT_WR :i32: 1;
SHUT_RDWR :i32: 2;
// macOS sockaddr_in (16 bytes, has sin_len field)
SockAddr :: struct {
sin_len: u8;
sin_family: u8;
sin_port: u16;
sin_addr: u32 = 0;
sin_zero: u64 = 0;
}
htons :: (port: i64) -> u16 {
cast(u16) (((port & 0xFF) << 8) | ((port >> 8) & 0xFF))
}
// ── nonblocking I/O (PLAN-HTTPZ S2) ──────────────────────────────────
// Typed wrappers over the readiness-loop syscall patterns: callers see
// WouldBlock / Closed / Fault, never a raw -1/errno pair. EINTR is
// retried internally — a readiness loop has nothing useful to do with
// an interrupted syscall except issue it again.
// errno resolves to a real function under the C macro: `__error` on
// darwin, `__errno_location` on linux (C3 selects per-OS).
errno_slot :: () -> *i32 extern libc "__error";
// fcntl file-status flags + errno values (macOS).
F_GETFL :i32: 3;
F_SETFL :i32: 4;
O_NONBLOCK :i32: 4;
EINTR :i32: 4;
EPIPE :i32: 32;
EAGAIN :i32: 35; // == EWOULDBLOCK on darwin (and linux's 11 == its EWOULDBLOCK)
EINPROGRESS :i32: 36;
ECONNABORTED :i32: 53;
ECONNRESET :i32: 54;
errno :: () -> i32 {
return errno_slot().*;
}
is_wouldblock :: (e: i32) -> bool {
return e == EAGAIN;
}
// Put `fd` in nonblocking mode (reads/writes/accepts return EAGAIN
// instead of parking the thread). False when fcntl refuses.
set_nonblocking :: (fd: i32) -> bool {
flags := fcntl(fd, F_GETFL);
if flags < 0 { return false; }
return fcntl(fd, F_SETFL, flags | O_NONBLOCK) >= 0;
}
// Failure classes for the _nb wrappers.
// WouldBlock — no data/space/pending connection right now; wait for
// readiness and retry.
// Closed — the peer is gone: EOF on read, EPIPE/ECONNRESET on
// write, ECONNRESET on read.
// Fault — any other errno (caller treats the fd as broken).
SockErr :: error {
WouldBlock,
Closed,
Fault,
}
// Accept one pending connection on a nonblocking listener. A connection
// that died between queueing and accept (ECONNABORTED) is skipped, not
// surfaced — the listener is fine.
accept_nb :: (fd: i32) -> (i32, !SockErr) {
while true {
c := accept(fd, null, null);
if c >= 0 { return c; }
e := errno();
if e == EINTR or e == ECONNABORTED { continue; }
if is_wouldblock(e) { raise error.WouldBlock; }
raise error.Fault;
}
raise error.Fault;
}
// Read up to `cap` bytes. Returns the byte count (> 0); an orderly EOF
// or a peer reset is Closed.
read_nb :: (fd: i32, buf: [*]u8, cap: usize) -> (i64, !SockErr) {
while true {
n := read(fd, buf, cap);
if n > 0 { return xx n; }
if n == 0 { raise error.Closed; }
e := errno();
if e == EINTR { continue; }
if is_wouldblock(e) { raise error.WouldBlock; }
if e == ECONNRESET { raise error.Closed; }
raise error.Fault;
}
raise error.Fault;
}
// Write up to `len` bytes, returning how many the kernel took (possibly
// fewer — the caller continues from there on the next writability).
write_nb :: (fd: i32, buf: [*]u8, len: usize) -> (i64, !SockErr) {
while true {
n := write(fd, buf, len);
if n >= 0 { return xx n; }
e := errno();
if e == EINTR { continue; }
if is_wouldblock(e) { raise error.WouldBlock; }
if e == EPIPE or e == ECONNRESET { raise error.Closed; }
raise error.Fault;
}
raise error.Fault;
}