From b1fd0299f0dfb69ab11e50a2d1b3c7714c2bb1fa Mon Sep 17 00:00:00 2001 From: lns Date: Wed, 11 Apr 2018 14:28:18 +0200 Subject: POTD skeleton. --- src/socket.c | 66 ++++++++++++++++++++++++++++++++++++++++++++++++------------ 1 file changed, 53 insertions(+), 13 deletions(-) (limited to 'src/socket.c') diff --git a/src/socket.c b/src/socket.c index 99e4477..7edbda2 100644 --- a/src/socket.c +++ b/src/socket.c @@ -1,3 +1,6 @@ +#include +#include +#include #include #include #include @@ -6,25 +9,62 @@ #include "socket.h" -int socket_init_in(psocket *psocket, const char *listen_addr, unsigned int listen_port) +static int socket_nonblock(psocket *psocket) { - struct in_addr addr = {0}; + int flags; - assert(psocket); - if (!inet_aton(listen_addr, &addr)) + flags = fcntl(psocket->fd, F_GETFL, 0); + if (flags < 0) + return 1; + flags |= O_NONBLOCK; + if (fcntl(psocket->fd, F_SETFL, flags) == -1) return 1; + return 0; +} + +int socket_init_in(psocket *psocket, const char *listen_addr, + const char *listen_port, struct addrinfo **results) +{ + struct addrinfo hints = {0}; + + assert(psocket); + assert(listen_addr || listen_port); /* getaddrinfo wants either node or service */ + hints.ai_family = AF_UNSPEC; /* IPV4 && IPV6 */ + hints.ai_socktype = SOCK_STREAM; /* TCP */ + hints.ai_flags = AI_PASSIVE; /* all interfaces */ + + return getaddrinfo(listen_addr, listen_port, &hints, results); +} + +int socket_bind_in(psocket *psocket, struct addrinfo *results) +{ + int fd = -1, rv; + struct addrinfo *rp = NULL; + + assert(psocket && results); - psocket->sock.sin_family = AF_INET; - psocket->sock.sin_addr = addr; - psocket->sock.sin_port = htons(listen_port); - psocket->fd = socket(AF_INET, SOCK_STREAM | SOCK_NONBLOCK, 0); - return psocket->fd < 0; + for (rp = results; rp != NULL; rp = rp->ai_next) { + fd = socket(rp->ai_family, rp->ai_socktype, rp->ai_protocol); + if (fd < 0) + continue; + rv = bind(fd, rp->ai_addr, rp->ai_addrlen); + if (!rv) + break; + close(fd); + } + + if (!rp) + return -1; + + psocket->fd = fd; + psocket->addr = *rp; + freeaddrinfo(results); + return socket_nonblock(psocket); } -int socket_bind_listen(psocket *psocket) +int socket_listen_in(psocket *psocket) { assert(psocket); - if (bind(psocket->fd, &psocket->sock, sizeof(psocket->sock)) < 0) - return 1; - return listen(psocket->fd, POTD_BACKLOG) < 0; + + return listen(psocket->fd, POTD_BACKLOG); } -- cgit v1.2.3