xref: /netbsd-src/external/gpl3/gcc/dist/libcody/netserver.cc (revision b1e838363e3c6fc78a55519254d99869742dd33c)
1*b1e83836Smrg // CODYlib		-*- mode:c++ -*-
2*b1e83836Smrg // Copyright (C) 2020 Nathan Sidwell, nathan@acm.org
3*b1e83836Smrg // License: Apache v2.0
4*b1e83836Smrg 
5*b1e83836Smrg // Cody
6*b1e83836Smrg #include "internal.hh"
7*b1e83836Smrg #if CODY_NETWORKING
8*b1e83836Smrg // C
9*b1e83836Smrg #include <cerrno>
10*b1e83836Smrg #include <cstring>
11*b1e83836Smrg // OS
12*b1e83836Smrg #include <netdb.h>
13*b1e83836Smrg #include <unistd.h>
14*b1e83836Smrg #include <arpa/inet.h>
15*b1e83836Smrg #include <netinet/in.h>
16*b1e83836Smrg #include <sys/un.h>
17*b1e83836Smrg 
18*b1e83836Smrg #ifndef AI_NUMERICSERV
19*b1e83836Smrg #define AI_NUMERICSERV 0
20*b1e83836Smrg #endif
21*b1e83836Smrg 
22*b1e83836Smrg // Server-side networking helpers
23*b1e83836Smrg 
24*b1e83836Smrg namespace Cody {
25*b1e83836Smrg 
ListenSocket(char const ** e,sockaddr const * addr,socklen_t len,unsigned backlog)26*b1e83836Smrg int ListenSocket (char const **e, sockaddr const *addr, socklen_t len,
27*b1e83836Smrg 		  unsigned backlog)
28*b1e83836Smrg {
29*b1e83836Smrg   char const *errstr = nullptr;
30*b1e83836Smrg 
31*b1e83836Smrg   int fd = socket (addr->sa_family, SOCK_STREAM, 0);
32*b1e83836Smrg   if (fd < 0)
33*b1e83836Smrg     {
34*b1e83836Smrg       errstr = "creating socket";
35*b1e83836Smrg 
36*b1e83836Smrg     fail:;
37*b1e83836Smrg       int err = errno;
38*b1e83836Smrg       if (e)
39*b1e83836Smrg 	*e = errstr;
40*b1e83836Smrg       if (fd >= 0)
41*b1e83836Smrg 	close (fd);
42*b1e83836Smrg       errno = err;
43*b1e83836Smrg       return -1;
44*b1e83836Smrg     }
45*b1e83836Smrg 
46*b1e83836Smrg   if (bind (fd, addr, len) < 0)
47*b1e83836Smrg     {
48*b1e83836Smrg       errstr = "binding socket";
49*b1e83836Smrg       goto fail;
50*b1e83836Smrg     }
51*b1e83836Smrg 
52*b1e83836Smrg   if (listen (fd, backlog ? backlog : 17) < 0)
53*b1e83836Smrg     {
54*b1e83836Smrg       errstr = "listening socket";
55*b1e83836Smrg       goto fail;
56*b1e83836Smrg     }
57*b1e83836Smrg 
58*b1e83836Smrg   return fd;
59*b1e83836Smrg }
60*b1e83836Smrg 
ListenLocal(char const ** e,char const * name,unsigned backlog)61*b1e83836Smrg int ListenLocal (char const **e, char const *name, unsigned backlog)
62*b1e83836Smrg {
63*b1e83836Smrg   sockaddr_un addr;
64*b1e83836Smrg   size_t len = strlen (name);
65*b1e83836Smrg 
66*b1e83836Smrg   if (len >= sizeof (addr.sun_path))
67*b1e83836Smrg     {
68*b1e83836Smrg       errno = ENAMETOOLONG;
69*b1e83836Smrg       return -1;
70*b1e83836Smrg     }
71*b1e83836Smrg 
72*b1e83836Smrg   memset (&addr, 0, offsetof (sockaddr_un, sun_path));
73*b1e83836Smrg   addr.sun_family = AF_UNIX;
74*b1e83836Smrg   memcpy (addr.sun_path, name, len + 1);
75*b1e83836Smrg 
76*b1e83836Smrg   return ListenSocket (e, (sockaddr *)&addr, sizeof (addr), backlog);
77*b1e83836Smrg }
78*b1e83836Smrg 
ListenInet6(char const ** e,char const * name,int port,unsigned backlog)79*b1e83836Smrg int ListenInet6 (char const **e, char const *name, int port, unsigned backlog)
80*b1e83836Smrg {
81*b1e83836Smrg   addrinfo *addrs = nullptr;
82*b1e83836Smrg   int fd = -1;
83*b1e83836Smrg   char const *errstr = nullptr;
84*b1e83836Smrg 
85*b1e83836Smrg   fd = socket (AF_INET6, SOCK_STREAM, 0);
86*b1e83836Smrg   if (fd < 0)
87*b1e83836Smrg     {
88*b1e83836Smrg       errstr = "creating socket";
89*b1e83836Smrg 
90*b1e83836Smrg     fail:;
91*b1e83836Smrg       int err = errno;
92*b1e83836Smrg       if (e)
93*b1e83836Smrg 	*e = errstr;
94*b1e83836Smrg       if (fd >= 0)
95*b1e83836Smrg 	close (fd);
96*b1e83836Smrg       if (addrs)
97*b1e83836Smrg 	freeaddrinfo (addrs);
98*b1e83836Smrg       errno = err;
99*b1e83836Smrg       return -1;
100*b1e83836Smrg     }
101*b1e83836Smrg 
102*b1e83836Smrg   addrinfo hints;
103*b1e83836Smrg   hints.ai_flags = AI_NUMERICSERV;
104*b1e83836Smrg   hints.ai_family = AF_INET6;
105*b1e83836Smrg   hints.ai_socktype = SOCK_STREAM;
106*b1e83836Smrg   hints.ai_protocol = 0;
107*b1e83836Smrg   hints.ai_addrlen = 0;
108*b1e83836Smrg   hints.ai_addr = nullptr;
109*b1e83836Smrg   hints.ai_canonname = nullptr;
110*b1e83836Smrg   hints.ai_next = nullptr;
111*b1e83836Smrg 
112*b1e83836Smrg   /* getaddrinfo requires a port number, but is quite happy to accept
113*b1e83836Smrg      invalid ones.  So don't rely on it.  */
114*b1e83836Smrg   if (int err = getaddrinfo (name, "0", &hints, &addrs))
115*b1e83836Smrg     {
116*b1e83836Smrg       errstr = gai_strerror (err);
117*b1e83836Smrg       // What's the best errno to set?
118*b1e83836Smrg       errno = 0;
119*b1e83836Smrg       goto fail;
120*b1e83836Smrg     }
121*b1e83836Smrg 
122*b1e83836Smrg   sockaddr_in6 addr;
123*b1e83836Smrg   memset (&addr, 0, sizeof (addr));
124*b1e83836Smrg   addr.sin6_family = AF_INET6;
125*b1e83836Smrg 
126*b1e83836Smrg   for (struct addrinfo *next = addrs; next; next = next->ai_next)
127*b1e83836Smrg     if (next->ai_family == AF_INET6
128*b1e83836Smrg 	&& next->ai_socktype == SOCK_STREAM)
129*b1e83836Smrg       {
130*b1e83836Smrg 	sockaddr_in6 *in6 = (sockaddr_in6 *)next->ai_addr;
131*b1e83836Smrg 	in6->sin6_port = htons (port);
132*b1e83836Smrg 	if (ntohs (in6->sin6_port) != port)
133*b1e83836Smrg 	  errno = EINVAL;
134*b1e83836Smrg 	else if (!bind (fd, next->ai_addr, next->ai_addrlen))
135*b1e83836Smrg 	  goto listen;
136*b1e83836Smrg       }
137*b1e83836Smrg 
138*b1e83836Smrg   errstr = "binding socket";
139*b1e83836Smrg   goto fail;
140*b1e83836Smrg 
141*b1e83836Smrg  listen:;
142*b1e83836Smrg   freeaddrinfo (addrs);
143*b1e83836Smrg   addrs = nullptr;
144*b1e83836Smrg 
145*b1e83836Smrg   if (listen (fd, backlog ? backlog : 17) < 0)
146*b1e83836Smrg     {
147*b1e83836Smrg       errstr = "listening socket";
148*b1e83836Smrg       goto fail;
149*b1e83836Smrg     }
150*b1e83836Smrg 
151*b1e83836Smrg   return fd;
152*b1e83836Smrg }
153*b1e83836Smrg 
154*b1e83836Smrg }
155*b1e83836Smrg #endif
156