// Peer
// ----
-static DragonnetPeer *dragonnet_peer_accept(int sock, struct sockaddr_in6 addr,
- DragonnetListener *l)
+static bool dragonnet_peer_init_accepted(DragonnetPeer *p, int sock,
+ struct sockaddr_in6 addr, DragonnetListener *l)
{
- DragonnetPeer *p = malloc(sizeof *p);
p->mu = malloc(sizeof *p->mu);
pthread_rwlock_init(p->mu, NULL);
pthread_rwlock_wrlock(p->mu);
p->laddr = l->laddr;
p->raddr = dragonnet_addr_parse_sock(addr);
- if (p != NULL)
- pthread_rwlock_unlock(p->mu);
+ if (setsockopt(p->sock, SOL_SOCKET, SO_RCVTIMEO, &dragonnet_timeout,
+ sizeof dragonnet_timeout) < 0) {
+ perror("setsockopt");
+ return false;
+ }
+
+ if (setsockopt(p->sock, SOL_SOCKET, SO_SNDTIMEO, &dragonnet_timeout,
+ sizeof dragonnet_timeout) < 0) {
+ perror("setsockopt");
+ return false;
+ }
+
+ pthread_rwlock_unlock(p->mu);
+ return true;
+}
+
+static DragonnetPeer *dragonnet_peer_accept(int sock, struct sockaddr_in6 addr,
+ DragonnetListener *l)
+{
+ DragonnetPeer *p = malloc(sizeof *p);
+ if (!dragonnet_peer_init_accepted(p, sock, addr, l)) {
+ dragonnet_peer_delete(p);
+ return NULL;
+ }
return p;
}
// Listener
// --------
-DragonnetListener *dragonnet_listener_new(char *addr, void (*on_connect)(DragonnetPeer *p))
+DragonnetListener *dragonnet_listener_new(char *addr,
+ void (*on_connect)(DragonnetPeer *p))
{
DragonnetListener *l = malloc(sizeof *l);
l->mu = malloc(sizeof *l->mu);
l->sock = socket(AF_INET6, SOCK_STREAM, 0);
l->on_connect = on_connect;
- int flag = 1;
- if (setsockopt(l->sock, SOL_SOCKET, SO_REUSEADDR, &flag, sizeof flag) < 0) {
+ int so_reuseaddr = 1;
+ if (setsockopt(l->sock, SOL_SOCKET, SO_REUSEADDR, &so_reuseaddr,
+ sizeof so_reuseaddr) < 0) {
perror("setsockopt");
dragonnet_listener_delete(l);
return NULL;
assert(l->state == DRAGONNET_LISTENER_ACTIVE);
close(l->sock);
+ l->sock = 0;
l->state++;
pthread_rwlock_unlock(l->mu);