]> git.lizzy.rs Git - dragonnet.git/blobdiff - listen.c
Fix getnameinfo failure
[dragonnet.git] / listen.c
index 5571806ed9893f75b403312f8807b9ce6e186553..abafb01445203720a94496c679c07fedf448f441 100644 (file)
--- a/listen.c
+++ b/listen.c
@@ -1,7 +1,12 @@
+#define _GNU_SOURCE
 #include <assert.h>
+#include <dragonnet/addr.h>
 #include <dragonnet/listen.h>
 #include <dragonnet/recv.h>
+#include <errno.h>
+#include <features.h>
 #include <netdb.h>
+#include <signal.h>
 #include <stdio.h>
 #include <stdlib.h>
 #include <sys/socket.h>
 // ----
 
 static bool dragonnet_peer_init_accepted(DragonnetPeer *p, int sock,
-               struct sockaddr_in6 addr, DragonnetListener *l)
+               char *addr, DragonnetListener *l)
 {
        pthread_mutex_init(&p->mtx, NULL);
 
        p->sock = sock;
-       p->laddr = l->laddr;
-       p->raddr = dragonnet_addr_parse_sock(addr);
+       p->address = addr;
        p->on_disconnect = l->on_disconnect;
        p->on_recv = l->on_recv;
        p->on_recv_type = l->on_recv_type;
@@ -26,13 +30,14 @@ static bool dragonnet_peer_init_accepted(DragonnetPeer *p, int sock,
        return true;
 }
 
-static DragonnetPeer *dragonnet_peer_accept(int sock, struct sockaddr_in6 addr,
+static DragonnetPeer *dragonnet_peer_accept(int sock, char *addr,
                DragonnetListener *l)
 {
        DragonnetPeer *p = malloc(sizeof *p);
        if (!dragonnet_peer_init_accepted(p, sock, addr, l)) {
                pthread_mutex_destroy(&p->mtx);
                free(p);
+               free(addr);
                return NULL;
        }
 
@@ -45,11 +50,15 @@ static DragonnetPeer *dragonnet_peer_accept(int sock, struct sockaddr_in6 addr,
 
 DragonnetListener *dragonnet_listener_new(char *addr)
 {
+       struct addrinfo *info = dragonnet_str2addr(addr);
+       if (!info)
+               return NULL;
+
        DragonnetListener *l = malloc(sizeof *l);
-       pthread_rwlock_init(&l->mu, NULL);
 
        l->active = true;
-       l->sock = socket(AF_INET6, SOCK_STREAM, 0);
+       l->sock = socket(info->ai_family, info->ai_socktype, info->ai_protocol);
+       l->address = dragonnet_addr2str(info->ai_addr, info->ai_addrlen);
        l->on_connect = NULL;
        l->on_disconnect = NULL;
        l->on_recv = NULL;
@@ -59,19 +68,20 @@ DragonnetListener *dragonnet_listener_new(char *addr)
        if (setsockopt(l->sock, SOL_SOCKET, SO_REUSEADDR, &so_reuseaddr,
                        sizeof so_reuseaddr) < 0) {
                perror("setsockopt");
+               freeaddrinfo(info);
                dragonnet_listener_delete(l);
                return NULL;
        }
 
-       l->laddr = dragonnet_addr_parse_str(addr);
-       struct sockaddr_in6 ai_addr = dragonnet_addr_sock(l->laddr);
-
-       if (bind(l->sock, (const struct sockaddr *) &ai_addr, sizeof ai_addr) < 0) {
+       if (bind(l->sock, info->ai_addr, info->ai_addrlen) < 0) {
                perror("bind");
+               freeaddrinfo(info);
                dragonnet_listener_delete(l);
                return NULL;
        }
 
+       freeaddrinfo(info);
+
        if (listen(l->sock, 10) < 0) {
                perror("listen");
                dragonnet_listener_delete(l);
@@ -83,29 +93,29 @@ DragonnetListener *dragonnet_listener_new(char *addr)
 
 static void *listener_main(void *g_listener)
 {
+#ifdef __GLIBC__
+       pthread_setname_np(pthread_self(), "listen");
+#endif
+
        DragonnetListener *l = (DragonnetListener *) g_listener;
 
        while (l->active) {
-               struct sockaddr_in6 clt_addr;
+               struct sockaddr_storage clt_addr;
                socklen_t clt_addrlen = sizeof clt_addr;
 
-               pthread_rwlock_rdlock(&l->mu);
-               int sock = l->sock;
-               pthread_rwlock_unlock(&l->mu);
-
-               int clt_sock = accept(sock, (struct sockaddr *) &clt_addr, &clt_addrlen);
+               int clt_sock = accept(l->sock, (struct sockaddr *) &clt_addr, &clt_addrlen);
                if (clt_sock < 0) {
-                       perror("accept");
+                       if (errno != EINTR)
+                               perror("accept");
                        continue;
                }
 
-               DragonnetPeer *p = dragonnet_peer_accept(clt_sock, clt_addr, l);
+               char *clt_addstr =  dragonnet_addr2str((struct sockaddr *) &clt_addr, clt_addrlen);
+               DragonnetPeer *p = dragonnet_peer_accept(clt_sock, clt_addstr, l);
                if (p == NULL)
                        continue;
 
-               pthread_rwlock_rdlock(&l->mu);
                void (*on_connect)(DragonnetPeer *) = l->on_connect;
-               pthread_rwlock_unlock(&l->mu);
 
                if (on_connect != NULL)
                        on_connect(p);
@@ -124,12 +134,14 @@ void dragonnet_listener_run(DragonnetListener *l)
 void dragonnet_listener_close(DragonnetListener *l)
 {
        l->active = false;
-       pthread_cancel(l->accept_thread);
+
+       pthread_kill(l->accept_thread, SIGINT);
        pthread_join(l->accept_thread, NULL);
 }
 
 void dragonnet_listener_delete(DragonnetListener *l)
 {
-       pthread_rwlock_destroy(&l->mu);
+       free(l->on_recv_type);
+       free(l->address);
        free(l);
 }