xref: /freebsd/contrib/lib9p/transport/socket.c (revision 6132212808e8dccedc9e5d85fea4390c2f38059a)
1 /*
2  * Copyright 2016 Jakub Klama <jceel@FreeBSD.org>
3  * All rights reserved
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted providing that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
15  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
16  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
18  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
22  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
23  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
24  * POSSIBILITY OF SUCH DAMAGE.
25  *
26  */
27 
28 #include <stdlib.h>
29 #include <errno.h>
30 #include <string.h>
31 #include <unistd.h>
32 #include <pthread.h>
33 #include <assert.h>
34 #include <sys/types.h>
35 #ifdef __APPLE__
36 # include "../apple_endian.h"
37 #else
38 # include <sys/endian.h>
39 #endif
40 #include <sys/socket.h>
41 #include <sys/event.h>
42 #include <sys/uio.h>
43 #include <netdb.h>
44 #include "../lib9p.h"
45 #include "../lib9p_impl.h"
46 #include "../log.h"
47 #include "socket.h"
48 
49 struct l9p_socket_softc
50 {
51 	struct l9p_connection *ls_conn;
52 	struct sockaddr ls_sockaddr;
53 	socklen_t ls_socklen;
54 	pthread_t ls_thread;
55 	int ls_fd;
56 };
57 
58 static int l9p_socket_readmsg(struct l9p_socket_softc *, void **, size_t *);
59 static int l9p_socket_get_response_buffer(struct l9p_request *,
60     struct iovec *, size_t *, void *);
61 static int l9p_socket_send_response(struct l9p_request *, const struct iovec *,
62     const size_t, const size_t, void *);
63 static void l9p_socket_drop_response(struct l9p_request *, const struct iovec *,
64     size_t, void *);
65 static void *l9p_socket_thread(void *);
66 static ssize_t xread(int, void *, size_t);
67 static ssize_t xwrite(int, void *, size_t);
68 
69 int
70 l9p_start_server(struct l9p_server *server, const char *host, const char *port)
71 {
72 	struct addrinfo *res, *res0, hints;
73 	struct kevent kev[2];
74 	struct kevent event[2];
75 	int err, kq, i, val, evs, nsockets = 0;
76 	int sockets[2];
77 
78 	memset(&hints, 0, sizeof(hints));
79 	hints.ai_family = PF_UNSPEC;
80 	hints.ai_socktype = SOCK_STREAM;
81 	err = getaddrinfo(host, port, &hints, &res0);
82 
83 	if (err)
84 		return (-1);
85 
86 	for (res = res0; res; res = res->ai_next) {
87 		int s = socket(res->ai_family, res->ai_socktype,
88 		    res->ai_protocol);
89 
90 		val = 1;
91 		setsockopt(s, SOL_SOCKET, SO_REUSEADDR, &val, sizeof(val));
92 
93 		if (s < 0)
94 			continue;
95 
96 		if (bind(s, res->ai_addr, res->ai_addrlen) < 0) {
97 			close(s);
98 			continue;
99 		}
100 
101 		sockets[nsockets] = s;
102 		EV_SET(&kev[nsockets++], s, EVFILT_READ, EV_ADD | EV_ENABLE, 0,
103 		    0, 0);
104 		listen(s, 10);
105 	}
106 
107 	if (nsockets < 1) {
108 		L9P_LOG(L9P_ERROR, "bind(): %s", strerror(errno));
109 		return(-1);
110 	}
111 
112 	kq = kqueue();
113 
114 	if (kevent(kq, kev, nsockets, NULL, 0, NULL) < 0) {
115 		L9P_LOG(L9P_ERROR, "kevent(): %s", strerror(errno));
116 		return (-1);
117 	}
118 
119 	for (;;) {
120 		evs = kevent(kq, NULL, 0, event, nsockets, NULL);
121 		if (evs < 0) {
122 			if (errno == EINTR)
123 				continue;
124 
125 			L9P_LOG(L9P_ERROR, "kevent(): %s", strerror(errno));
126 			return (-1);
127 		}
128 
129 		for (i = 0; i < evs; i++) {
130 			struct sockaddr client_addr;
131 			socklen_t client_addr_len = sizeof(client_addr);
132 			int news = accept((int)event[i].ident, &client_addr,
133 			    &client_addr_len);
134 
135 			if (news < 0) {
136 				L9P_LOG(L9P_WARNING, "accept(): %s",
137 				    strerror(errno));
138 				continue;
139 			}
140 
141 			l9p_socket_accept(server, news, &client_addr,
142 			    client_addr_len);
143 		}
144 	}
145 
146 }
147 
148 void
149 l9p_socket_accept(struct l9p_server *server, int conn_fd,
150     struct sockaddr *client_addr, socklen_t client_addr_len)
151 {
152 	struct l9p_socket_softc *sc;
153 	struct l9p_connection *conn;
154 	char host[NI_MAXHOST + 1];
155 	char serv[NI_MAXSERV + 1];
156 	int err;
157 
158 	err = getnameinfo(client_addr, client_addr_len, host, NI_MAXHOST, serv,
159 	    NI_MAXSERV, NI_NUMERICHOST | NI_NUMERICSERV);
160 
161 	if (err != 0) {
162 		L9P_LOG(L9P_WARNING, "cannot look up client name: %s",
163 		    gai_strerror(err));
164 	} else {
165 		L9P_LOG(L9P_INFO, "new connection from %s:%s", host, serv);
166 	}
167 
168 	if (l9p_connection_init(server, &conn) != 0) {
169 		L9P_LOG(L9P_ERROR, "cannot create new connection");
170 		return;
171 	}
172 
173 	sc = l9p_calloc(1, sizeof(*sc));
174 	sc->ls_conn = conn;
175 	sc->ls_fd = conn_fd;
176 
177 	/*
178 	 * Fill in transport handler functions and aux argument.
179 	 */
180 	conn->lc_lt.lt_aux = sc;
181 	conn->lc_lt.lt_get_response_buffer = l9p_socket_get_response_buffer;
182 	conn->lc_lt.lt_send_response = l9p_socket_send_response;
183 	conn->lc_lt.lt_drop_response = l9p_socket_drop_response;
184 
185 	err = pthread_create(&sc->ls_thread, NULL, l9p_socket_thread, sc);
186 	if (err) {
187 		L9P_LOG(L9P_ERROR,
188 		    "pthread_create (for connection from %s:%s): error %s",
189 		    host, serv, strerror(err));
190 		l9p_connection_close(sc->ls_conn);
191 		free(sc);
192 	}
193 }
194 
195 static void *
196 l9p_socket_thread(void *arg)
197 {
198 	struct l9p_socket_softc *sc = (struct l9p_socket_softc *)arg;
199 	struct iovec iov;
200 	void *buf;
201 	size_t length;
202 
203 	for (;;) {
204 		if (l9p_socket_readmsg(sc, &buf, &length) != 0)
205 			break;
206 
207 		iov.iov_base = buf;
208 		iov.iov_len = length;
209 		l9p_connection_recv(sc->ls_conn, &iov, 1, NULL);
210 		free(buf);
211 	}
212 
213 	L9P_LOG(L9P_INFO, "connection closed");
214 	l9p_connection_close(sc->ls_conn);
215 	free(sc);
216 	return (NULL);
217 }
218 
219 static int
220 l9p_socket_readmsg(struct l9p_socket_softc *sc, void **buf, size_t *size)
221 {
222 	uint32_t msize;
223 	size_t toread;
224 	ssize_t ret;
225 	void *buffer;
226 	int fd = sc->ls_fd;
227 
228 	assert(fd > 0);
229 
230 	buffer = l9p_malloc(sizeof(uint32_t));
231 
232 	ret = xread(fd, buffer, sizeof(uint32_t));
233 	if (ret < 0) {
234 		L9P_LOG(L9P_ERROR, "read(): %s", strerror(errno));
235 		return (-1);
236 	}
237 
238 	if (ret != sizeof(uint32_t)) {
239 		if (ret == 0)
240 			L9P_LOG(L9P_DEBUG, "%p: EOF", (void *)sc->ls_conn);
241 		else
242 			L9P_LOG(L9P_ERROR,
243 			    "short read: %zd bytes of %zd expected",
244 			    ret, sizeof(uint32_t));
245 		return (-1);
246 	}
247 
248 	msize = le32toh(*(uint32_t *)buffer);
249 	toread = msize - sizeof(uint32_t);
250 	buffer = l9p_realloc(buffer, msize);
251 
252 	ret = xread(fd, (char *)buffer + sizeof(uint32_t), toread);
253 	if (ret < 0) {
254 		L9P_LOG(L9P_ERROR, "read(): %s", strerror(errno));
255 		return (-1);
256 	}
257 
258 	if (ret != (ssize_t)toread) {
259 		L9P_LOG(L9P_ERROR, "short read: %zd bytes of %zd expected",
260 		    ret, toread);
261 		return (-1);
262 	}
263 
264 	*size = msize;
265 	*buf = buffer;
266 	L9P_LOG(L9P_INFO, "%p: read complete message, buf=%p size=%d",
267 	    (void *)sc->ls_conn, buffer, msize);
268 
269 	return (0);
270 }
271 
272 static int
273 l9p_socket_get_response_buffer(struct l9p_request *req, struct iovec *iov,
274     size_t *niovp, void *arg __unused)
275 {
276 	size_t size = req->lr_conn->lc_msize;
277 	void *buf;
278 
279 	buf = l9p_malloc(size);
280 	iov[0].iov_base = buf;
281 	iov[0].iov_len = size;
282 
283 	*niovp = 1;
284 	return (0);
285 }
286 
287 static int
288 l9p_socket_send_response(struct l9p_request *req __unused,
289     const struct iovec *iov, const size_t niov __unused, const size_t iolen,
290     void *arg)
291 {
292 	struct l9p_socket_softc *sc = (struct l9p_socket_softc *)arg;
293 
294 	assert(sc->ls_fd >= 0);
295 
296 	L9P_LOG(L9P_DEBUG, "%p: sending reply, buf=%p, size=%d", arg,
297 	    iov[0].iov_base, iolen);
298 
299 	if (xwrite(sc->ls_fd, iov[0].iov_base, iolen) != (int)iolen) {
300 		L9P_LOG(L9P_ERROR, "short write: %s", strerror(errno));
301 		return (-1);
302 	}
303 
304 	free(iov[0].iov_base);
305 	return (0);
306 }
307 
308 static void
309 l9p_socket_drop_response(struct l9p_request *req __unused,
310     const struct iovec *iov, size_t niov __unused, void *arg)
311 {
312 
313 	L9P_LOG(L9P_DEBUG, "%p: drop buf=%p", arg, iov[0].iov_base);
314 	free(iov[0].iov_base);
315 }
316 
317 static ssize_t
318 xread(int fd, void *buf, size_t count)
319 {
320 	size_t done = 0;
321 	ssize_t ret;
322 
323 	while (done < count) {
324 		ret = read(fd, (char *)buf + done, count - done);
325 		if (ret < 0) {
326 			if (errno == EINTR)
327 				continue;
328 
329 			return (-1);
330 		}
331 
332 		if (ret == 0)
333 			return ((ssize_t)done);
334 
335 		done += (size_t)ret;
336 	}
337 
338 	return ((ssize_t)done);
339 }
340 
341 static ssize_t
342 xwrite(int fd, void *buf, size_t count)
343 {
344 	size_t done = 0;
345 	ssize_t ret;
346 
347 	while (done < count) {
348 		ret = write(fd, (char *)buf + done, count - done);
349 		if (ret < 0) {
350 			if (errno == EINTR)
351 				continue;
352 
353 			return (-1);
354 		}
355 
356 		if (ret == 0)
357 			return ((ssize_t)done);
358 
359 		done += (size_t)ret;
360 	}
361 
362 	return ((ssize_t)done);
363 }
364