xref: /freebsd/tests/sys/kern/socket_splice.c (revision 3a56015a2f5d630910177fa79a522bb95511ccf7)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause
3  *
4  * Copyright (c) 2024 Stormshield
5  */
6 
7 #include <sys/capsicum.h>
8 #include <sys/event.h>
9 #include <sys/filio.h>
10 #include <sys/socket.h>
11 #include <sys/wait.h>
12 
13 #include <netinet/in.h>
14 #include <netinet/tcp.h>
15 
16 #include <errno.h>
17 #include <poll.h>
18 #include <pthread.h>
19 #include <signal.h>
20 #include <stdio.h>
21 #include <stdlib.h>
22 #include <string.h>
23 #include <time.h>
24 
25 #include <atf-c.h>
26 
27 static void
28 checked_close(int fd)
29 {
30 	int error;
31 
32 	error = close(fd);
33 	ATF_REQUIRE_MSG(error == 0, "close failed: %s", strerror(errno));
34 }
35 
36 static int
37 fionread(int fd)
38 {
39 	int data, error;
40 
41 	data = 0;
42 	error = ioctl(fd, FIONREAD, &data);
43 	ATF_REQUIRE_MSG(error == 0, "ioctl failed: %s", strerror(errno));
44 	ATF_REQUIRE(data >= 0);
45 	return (data);
46 }
47 
48 static void
49 noblocking(int fd)
50 {
51 	int flags, error;
52 
53 	flags = fcntl(fd, F_GETFL);
54 	ATF_REQUIRE_MSG(flags != -1, "fcntl failed: %s", strerror(errno));
55 	flags |= O_NONBLOCK;
56 	error = fcntl(fd, F_SETFL, flags);
57 	ATF_REQUIRE_MSG(error == 0, "fcntl failed: %s", strerror(errno));
58 }
59 
60 /*
61  * Create a pair of connected TCP sockets, returned via the "out" array.
62  */
63 static void
64 tcp_socketpair(int out[2], int domain)
65 {
66 	struct sockaddr_in sin;
67 	struct sockaddr_in6 sin6;
68 	struct sockaddr *sinp;
69 	int error, sd[2];
70 
71 	sd[0] = socket(domain, SOCK_STREAM, 0);
72 	ATF_REQUIRE_MSG(sd[0] >= 0, "socket failed: %s", strerror(errno));
73 	sd[1] = socket(domain, SOCK_STREAM, 0);
74 	ATF_REQUIRE_MSG(sd[1] >= 0, "socket failed: %s", strerror(errno));
75 
76 	error = setsockopt(sd[0], IPPROTO_TCP, TCP_NODELAY, &(int){ 1 },
77 	    sizeof(int));
78 	ATF_REQUIRE_MSG(error == 0, "setsockopt failed: %s", strerror(errno));
79 	error = setsockopt(sd[1], IPPROTO_TCP, TCP_NODELAY, &(int){ 1 },
80 	    sizeof(int));
81 	ATF_REQUIRE_MSG(error == 0, "setsockopt failed: %s", strerror(errno));
82 
83 	if (domain == PF_INET) {
84 		memset(&sin, 0, sizeof(sin));
85 		sin.sin_family = AF_INET;
86 		sin.sin_len = sizeof(sin);
87 		sin.sin_addr.s_addr = htonl(INADDR_ANY);
88 		sin.sin_port = htons(0);
89 		sinp = (struct sockaddr *)&sin;
90 	} else {
91 		ATF_REQUIRE(domain == PF_INET6);
92 		memset(&sin6, 0, sizeof(sin6));
93 		sin6.sin6_family = AF_INET6;
94 		sin6.sin6_len = sizeof(sin6);
95 		sin6.sin6_addr = in6addr_any;
96 		sin6.sin6_port = htons(0);
97 		sinp = (struct sockaddr *)&sin6;
98 	}
99 
100 	error = bind(sd[0], sinp, sinp->sa_len);
101 	ATF_REQUIRE_MSG(error == 0, "bind failed: %s", strerror(errno));
102 	error = listen(sd[0], 1);
103 	ATF_REQUIRE_MSG(error == 0, "listen failed: %s", strerror(errno));
104 
105 	error = getsockname(sd[0], sinp, &(socklen_t){ sinp->sa_len });
106 	ATF_REQUIRE_MSG(error == 0, "getsockname failed: %s", strerror(errno));
107 
108 	error = connect(sd[1], sinp, sinp->sa_len);
109 	ATF_REQUIRE_MSG(error == 0, "connect failed: %s", strerror(errno));
110 	out[0] = accept(sd[0], NULL, NULL);
111 	ATF_REQUIRE_MSG(out[0] >= 0, "accept failed: %s", strerror(errno));
112 	checked_close(sd[0]);
113 	out[1] = sd[1];
114 }
115 
116 static void
117 tcp4_socketpair(int out[2])
118 {
119 	tcp_socketpair(out, PF_INET);
120 }
121 
122 static void
123 tcp6_socketpair(int out[2])
124 {
125 	tcp_socketpair(out, PF_INET6);
126 }
127 
128 static off_t
129 nspliced(int sd)
130 {
131 	off_t n;
132 	socklen_t len;
133 	int error;
134 
135 	len = sizeof(n);
136 	error = getsockopt(sd, SOL_SOCKET, SO_SPLICE, &n, &len);
137 	ATF_REQUIRE_MSG(error == 0, "getsockopt failed: %s", strerror(errno));
138 	ATF_REQUIRE_MSG(len == sizeof(n), "unexpected length: %d", len);
139 	return (n);
140 }
141 
142 /*
143  * Use a macro so that ATF_REQUIRE_MSG prints a useful line number.
144  */
145 #define check_nspliced(sd, n) do {					\
146 	off_t sofar;							\
147 									\
148 	sofar = nspliced(sd);						\
149 	ATF_REQUIRE_MSG(sofar == (off_t)n, "spliced %jd bytes, expected %jd", \
150 	    (intmax_t)sofar, (intmax_t)n);				\
151 } while (0)
152 
153 static void
154 splice_init(struct splice *sp, int fd, off_t max, struct timeval *tv)
155 {
156 	memset(sp, 0, sizeof(*sp));
157 	sp->sp_fd = fd;
158 	sp->sp_max = max;
159 	if (tv != NULL)
160 		sp->sp_idle = *tv;
161 	else
162 		sp->sp_idle.tv_sec = sp->sp_idle.tv_usec = 0;
163 }
164 
165 static void
166 unsplice(int fd)
167 {
168 	struct splice sp;
169 	int error;
170 
171 	splice_init(&sp, -1, 0, NULL);
172 	error = setsockopt(fd, SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
173 	ATF_REQUIRE_MSG(error == 0, "setsockopt failed: %s", strerror(errno));
174 }
175 
176 static void
177 unsplice_pair(int fd1, int fd2)
178 {
179 	unsplice(fd1);
180 	unsplice(fd2);
181 }
182 
183 static void
184 splice_pair(int fd1, int fd2, off_t max, struct timeval *tv)
185 {
186 	struct splice sp;
187 	int error;
188 
189 	splice_init(&sp, fd1, max, tv);
190 	error = setsockopt(fd2, SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
191 	ATF_REQUIRE_MSG(error == 0, "setsockopt failed: %s", strerror(errno));
192 
193 	splice_init(&sp, fd2, max, tv);
194 	error = setsockopt(fd1, SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
195 	ATF_REQUIRE_MSG(error == 0, "setsockopt failed: %s", strerror(errno));
196 }
197 
198 /*
199  * A structure representing a spliced pair of connections.  left[1] is
200  * bidirectionally spliced with right[0].
201  */
202 struct splice_conn {
203 	int left[2];
204 	int right[2];
205 };
206 
207 /*
208  * Initialize a splice connection with the given maximum number of bytes to
209  * splice and the given idle timeout.  For now we're forced to use TCP socket,
210  * but at some point it would be nice (and simpler) to use pairs of PF_LOCAL
211  * sockets.
212  */
213 static void
214 splice_conn_init_limits(struct splice_conn *sc, off_t max, struct timeval *tv)
215 {
216 	memset(sc, 0, sizeof(*sc));
217 	tcp4_socketpair(sc->left);
218 	tcp4_socketpair(sc->right);
219 	splice_pair(sc->left[1], sc->right[0], max, tv);
220 }
221 
222 static void
223 splice_conn_init(struct splice_conn *sc)
224 {
225 	splice_conn_init_limits(sc, 0, NULL);
226 }
227 
228 static void
229 splice_conn_check_empty(struct splice_conn *sc)
230 {
231 	int data;
232 
233 	data = fionread(sc->left[0]);
234 	ATF_REQUIRE_MSG(data == 0, "unexpected data on left[0]: %d", data);
235 	data = fionread(sc->left[1]);
236 	ATF_REQUIRE_MSG(data == 0, "unexpected data on left[1]: %d", data);
237 	data = fionread(sc->right[0]);
238 	ATF_REQUIRE_MSG(data == 0, "unexpected data on right[0]: %d", data);
239 	data = fionread(sc->right[1]);
240 	ATF_REQUIRE_MSG(data == 0, "unexpected data on right[1]: %d", data);
241 }
242 
243 static void
244 splice_conn_fini(struct splice_conn *sc)
245 {
246 	checked_close(sc->left[0]);
247 	checked_close(sc->left[1]);
248 	checked_close(sc->right[0]);
249 	checked_close(sc->right[1]);
250 }
251 
252 static void
253 splice_conn_noblocking(struct splice_conn *sc)
254 {
255 	noblocking(sc->left[0]);
256 	noblocking(sc->left[1]);
257 	noblocking(sc->right[0]);
258 	noblocking(sc->right[1]);
259 }
260 
261 /* Pass a byte through a pair of spliced connections. */
262 ATF_TC_WITHOUT_HEAD(splice_basic);
263 ATF_TC_BODY(splice_basic, tc)
264 {
265 	struct splice_conn sc;
266 	ssize_t n;
267 	char c;
268 
269 	splice_conn_init(&sc);
270 
271 	check_nspliced(sc.left[1], 0);
272 	check_nspliced(sc.right[0], 0);
273 
274 	/* Left-to-right. */
275 	c = 'M';
276 	n = write(sc.left[0], &c, 1);
277 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
278 	n = read(sc.right[1], &c, 1);
279 	ATF_REQUIRE_MSG(n == 1, "read failed: %s", strerror(errno));
280 	ATF_REQUIRE_MSG(c == 'M', "unexpected character: %c", c);
281 	check_nspliced(sc.left[1], 1);
282 	check_nspliced(sc.right[0], 0);
283 
284 	/* Right-to-left. */
285 	c = 'J';
286 	n = write(sc.right[1], &c, 1);
287 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
288 	n = read(sc.left[0], &c, 1);
289 	ATF_REQUIRE_MSG(n == 1, "read failed: %s", strerror(errno));
290 	ATF_REQUIRE_MSG(c == 'J', "unexpected character: %c", c);
291 	check_nspliced(sc.left[1], 1);
292 	check_nspliced(sc.right[0], 1);
293 
294 	/* Unsplice and verify that the byte counts haven't changed. */
295 	unsplice(sc.left[1]);
296 	unsplice(sc.right[0]);
297 	check_nspliced(sc.left[1], 1);
298 	check_nspliced(sc.right[0], 1);
299 
300 	splice_conn_fini(&sc);
301 }
302 
303 static void
304 remove_rights(int fd, const cap_rights_t *toremove)
305 {
306 	cap_rights_t rights;
307 	int error;
308 
309 	error = cap_rights_get(fd, &rights);
310 	ATF_REQUIRE_MSG(error == 0, "cap_rights_get failed: %s",
311 	    strerror(errno));
312 	cap_rights_remove(&rights, toremove);
313 	error = cap_rights_limit(fd, &rights);
314 	ATF_REQUIRE_MSG(error == 0, "cap_rights_limit failed: %s",
315 	    strerror(errno));
316 }
317 
318 /*
319  * Verify that splicing fails when the socket is missing the necessary rights.
320  */
321 ATF_TC_WITHOUT_HEAD(splice_capsicum);
322 ATF_TC_BODY(splice_capsicum, tc)
323 {
324 	struct splice sp;
325 	cap_rights_t rights;
326 	off_t n;
327 	int error, left[2], right[2];
328 
329 	tcp4_socketpair(left);
330 	tcp4_socketpair(right);
331 
332 	/*
333 	 * Make sure that we splice a socket that's missing recv rights.
334 	 */
335 	remove_rights(left[1], cap_rights_init(&rights, CAP_RECV));
336 	splice_init(&sp, right[0], 0, NULL);
337 	error = setsockopt(left[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
338 	ATF_REQUIRE_ERRNO(ENOTCAPABLE, error == -1);
339 
340 	/* Make sure we can still splice left[1] in the other direction. */
341 	splice_init(&sp, left[1], 0, NULL);
342 	error = setsockopt(right[0], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
343 	ATF_REQUIRE_MSG(error == 0, "setsockopt failed: %s", strerror(errno));
344 	splice_init(&sp, -1, 0, NULL);
345 	error = setsockopt(right[0], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
346 	ATF_REQUIRE_MSG(error == 0, "setsockopt failed: %s", strerror(errno));
347 
348 	/*
349 	 * Now remove send rights from left[1] and verify that splicing is no
350 	 * longer possible.
351 	 */
352 	remove_rights(left[1], cap_rights_init(&rights, CAP_SEND));
353 	splice_init(&sp, left[1], 0, NULL);
354 	error = setsockopt(right[0], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
355 	ATF_REQUIRE_ERRNO(ENOTCAPABLE, error == -1);
356 
357 	/*
358 	 * It's still ok to query the SO_SPLICE state though.
359 	 */
360 	n = -1;
361 	error = getsockopt(left[1], SOL_SOCKET, SO_SPLICE, &n,
362 	    &(socklen_t){ sizeof(n) });
363 	ATF_REQUIRE_MSG(error == 0, "getsockopt failed: %s", strerror(errno));
364 	ATF_REQUIRE(n == 0);
365 
366 	/*
367 	 * Make sure that we can unsplice a spliced pair without any rights
368 	 * other than CAP_SETSOCKOPT.
369 	 */
370 	splice_pair(left[0], right[1], 0, NULL);
371 	error = cap_rights_limit(left[0],
372 	    cap_rights_init(&rights, CAP_SETSOCKOPT));
373 	ATF_REQUIRE_MSG(error == 0, "cap_rights_limit failed: %s",
374 	    strerror(errno));
375 	unsplice(left[0]);
376 
377 	checked_close(left[0]);
378 	checked_close(left[1]);
379 	checked_close(right[0]);
380 	checked_close(right[1]);
381 }
382 
383 /*
384  * Check various error cases in splice configuration.
385  */
386 ATF_TC_WITHOUT_HEAD(splice_error);
387 ATF_TC_BODY(splice_error, tc)
388 {
389 	struct splice_conn sc;
390 	struct splice sp;
391 	char path[PATH_MAX];
392 	int error, fd, sd, usd[2];
393 
394 	memset(&sc, 0, sizeof(sc));
395 	tcp4_socketpair(sc.left);
396 	tcp4_socketpair(sc.right);
397 
398 	/* A negative byte limit is invalid. */
399 	splice_init(&sp, sc.right[0], -3, NULL);
400 	error = setsockopt(sc.left[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
401 	ATF_REQUIRE_ERRNO(EINVAL, error == -1);
402 
403 	/* Can't unsplice a never-spliced socket. */
404 	splice_init(&sp, -1, 0, NULL);
405 	error = setsockopt(sc.left[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
406 	ATF_REQUIRE_ERRNO(ENOTCONN, error == -1);
407 
408 	/* Can't double-unsplice a socket. */
409 	splice_init(&sp, sc.right[0], 0, NULL);
410 	error = setsockopt(sc.left[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
411 	ATF_REQUIRE_MSG(error == 0, "setsockopt failed: %s", strerror(errno));
412 	unsplice(sc.left[1]);
413 	splice_init(&sp, -1, 0, NULL);
414 	error = setsockopt(sc.left[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
415 	ATF_REQUIRE_ERRNO(ENOTCONN, error == -1);
416 
417 	/* Can't splice a spliced socket */
418 	splice_init(&sp, sc.right[0], 0, NULL);
419 	error = setsockopt(sc.left[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
420 	ATF_REQUIRE_MSG(error == 0, "setsockopt failed: %s", strerror(errno));
421 	splice_init(&sp, sc.right[1], 0, NULL);
422 	error = setsockopt(sc.left[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
423 	ATF_REQUIRE_ERRNO(EBUSY, error == -1);
424 	splice_init(&sp, sc.right[0], 0, NULL);
425 	error = setsockopt(sc.left[0], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
426 	ATF_REQUIRE_ERRNO(EBUSY, error == -1);
427 	splice_init(&sp, -1, 0, NULL);
428 	error = setsockopt(sc.left[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
429 
430 	/* Can't splice to a non-socket. */
431 	snprintf(path, sizeof(path), "/tmp/splice_error.XXXXXX");
432 	fd = mkstemp(path);
433 	ATF_REQUIRE_MSG(fd >= 0, "mkstemp failed: %s", strerror(errno));
434 	splice_init(&sp, fd, 0, NULL);
435 	error = setsockopt(sc.left[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
436 	ATF_REQUIRE_ERRNO(ENOTSOCK, error == -1);
437 
438 	/* Can't splice to an invalid fd. */
439 	checked_close(fd);
440 	splice_init(&sp, fd, 0, NULL);
441 	error = setsockopt(sc.left[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
442 	ATF_REQUIRE_ERRNO(EBADF, error == -1);
443 
444 	/* Can't splice a unix stream socket. */
445 	error = socketpair(AF_UNIX, SOCK_STREAM, 0, usd);
446 	ATF_REQUIRE_MSG(error == 0, "socketpair failed: %s", strerror(errno));
447 	splice_init(&sp, usd[0], 0, NULL);
448 	error = setsockopt(sc.left[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
449 	ATF_REQUIRE_ERRNO(EPROTONOSUPPORT, error == -1);
450 	error = setsockopt(usd[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
451 	ATF_REQUIRE_ERRNO(EPROTONOSUPPORT, error == -1);
452 	checked_close(usd[0]);
453 	checked_close(usd[1]);
454 
455 	/* Can't splice an unconnected TCP socket. */
456 	sd = socket(PF_INET, SOCK_STREAM, 0);
457 	ATF_REQUIRE_MSG(sd >= 0, "socket failed: %s", strerror(errno));
458 	splice_init(&sp, sd, 0, NULL);
459 	error = setsockopt(sc.left[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
460 	ATF_REQUIRE_ERRNO(ENOTCONN, error == -1);
461 	splice_init(&sp, sc.right[0], 0, NULL);
462 	error = setsockopt(sd, SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
463 	ATF_REQUIRE_ERRNO(ENOTCONN, error == -1);
464 
465 	splice_conn_fini(&sc);
466 }
467 
468 /*
469  * Make sure that kevent() doesn't report read I/O events on spliced sockets.
470  */
471 ATF_TC_WITHOUT_HEAD(splice_kevent);
472 ATF_TC_BODY(splice_kevent, tc)
473 {
474 	struct splice_conn sc;
475 	struct kevent kev;
476 	struct timespec ts;
477 	ssize_t n;
478 	int error, nev, kq;
479 	uint8_t b;
480 
481 	splice_conn_init(&sc);
482 
483 	kq = kqueue();
484 	ATF_REQUIRE_MSG(kq >= 0, "kqueue failed: %s", strerror(errno));
485 
486 	EV_SET(&kev, sc.left[1], EVFILT_READ, EV_ADD, 0, 0, NULL);
487 	error = kevent(kq, &kev, 1, NULL, 0, NULL);
488 	ATF_REQUIRE_MSG(error == 0, "kevent failed: %s", strerror(errno));
489 
490 	memset(&ts, 0, sizeof(ts));
491 	nev = kevent(kq, NULL, 0, &kev, 1, &ts);
492 	ATF_REQUIRE_MSG(nev >= 0, "kevent failed: %s", strerror(errno));
493 	ATF_REQUIRE(nev == 0);
494 
495 	b = 'M';
496 	n = write(sc.left[0], &b, 1);
497 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
498 	n = read(sc.right[1], &b, 1);
499 	ATF_REQUIRE_MSG(n == 1, "read failed: %s", strerror(errno));
500 	ATF_REQUIRE(b == 'M');
501 
502 	nev = kevent(kq, NULL, 0, &kev, 1, &ts);
503 	ATF_REQUIRE_MSG(nev >= 0, "kevent failed: %s", strerror(errno));
504 	ATF_REQUIRE(nev == 0);
505 
506 	b = 'J';
507 	n = write(sc.right[1], &b, 1);
508 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
509 	n = read(sc.left[0], &b, 1);
510 	ATF_REQUIRE_MSG(n == 1, "read failed: %s", strerror(errno));
511 	ATF_REQUIRE(b == 'J');
512 
513 	splice_conn_fini(&sc);
514 	checked_close(kq);
515 }
516 
517 /*
518  * Verify that a splice byte limit is applied.
519  */
520 ATF_TC_WITHOUT_HEAD(splice_limit_bytes);
521 ATF_TC_BODY(splice_limit_bytes, tc)
522 {
523 	struct splice_conn sc;
524 	ssize_t n;
525 	uint8_t b, buf[128];
526 
527 	splice_conn_init_limits(&sc, sizeof(buf) + 1, NULL);
528 
529 	memset(buf, 'A', sizeof(buf));
530 	for (size_t total = sizeof(buf); total > 0; total -= n) {
531 		n = write(sc.left[0], buf, total);
532 		ATF_REQUIRE_MSG(n > 0, "write failed: %s", strerror(errno));
533 	}
534 	for (size_t total = sizeof(buf); total > 0; total -= n) {
535 		n = read(sc.right[1], buf, sizeof(buf));
536 		ATF_REQUIRE_MSG(n > 0, "read failed: %s", strerror(errno));
537 	}
538 
539 	check_nspliced(sc.left[1], sizeof(buf));
540 	check_nspliced(sc.right[0], 0);
541 
542 	/* Trigger an unsplice by writing the last byte. */
543 	b = 'B';
544 	n = write(sc.left[0], &b, 1);
545 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
546 	n = read(sc.right[1], &b, 1);
547 	ATF_REQUIRE_MSG(n == 1, "read failed: %s", strerror(errno));
548 	ATF_REQUIRE(b == 'B');
549 
550 	/*
551 	 * The next byte should appear on the other side of the connection
552 	 * rather than the splice.
553 	 */
554 	b = 'C';
555 	n = write(sc.left[0], &b, 1);
556 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
557 	n = read(sc.left[1], &b, 1);
558 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
559 	ATF_REQUIRE(b == 'C');
560 
561 	splice_conn_check_empty(&sc);
562 
563 	splice_conn_fini(&sc);
564 }
565 
566 /*
567  * Verify that a splice timeout limit is applied.
568  */
569 ATF_TC_WITHOUT_HEAD(splice_limit_timeout);
570 ATF_TC_BODY(splice_limit_timeout, tc)
571 {
572 	struct splice_conn sc;
573 	ssize_t n;
574 	int error;
575 	uint8_t b, buf[128];
576 
577 	splice_conn_init_limits(&sc, 0,
578 	    &(struct timeval){ .tv_sec = 0, .tv_usec = 500000 /* 500ms */ });
579 
580 	/* Write some data through the splice. */
581 	memset(buf, 'A', sizeof(buf));
582 	for (size_t total = sizeof(buf); total > 0; total -= n) {
583 		n = write(sc.left[0], buf, total);
584 		ATF_REQUIRE_MSG(n > 0, "write failed: %s", strerror(errno));
585 	}
586 	for (size_t total = sizeof(buf); total > 0; total -= n) {
587 		n = read(sc.right[1], buf, sizeof(buf));
588 		ATF_REQUIRE_MSG(n > 0, "read failed: %s", strerror(errno));
589 	}
590 
591 	check_nspliced(sc.left[1], sizeof(buf));
592 	check_nspliced(sc.right[0], 0);
593 
594 	/* Wait for the splice to time out. */
595 	error = usleep(550000);
596 	ATF_REQUIRE_MSG(error == 0, "usleep failed: %s", strerror(errno));
597 
598 	/*
599 	 * The next byte should appear on the other side of the connection
600 	 * rather than the splice.
601 	 */
602 	b = 'C';
603 	n = write(sc.left[0], &b, 1);
604 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
605 	n = read(sc.left[1], &b, 1);
606 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
607 	ATF_REQUIRE(b == 'C');
608 
609 	splice_conn_fini(&sc);
610 }
611 
612 /*
613  * Make sure that listen() fails on spliced sockets, and that SO_SPLICE can't be
614  * used with listening sockets.
615  */
616 ATF_TC_WITHOUT_HEAD(splice_listen);
617 ATF_TC_BODY(splice_listen, tc)
618 {
619 	struct splice sp;
620 	struct splice_conn sc;
621 	int error, sd[3];
622 
623 	/*
624 	 * These should fail regardless since the sockets are connected, but it
625 	 * doesn't hurt to check.
626 	 */
627 	splice_conn_init(&sc);
628 	error = listen(sc.left[1], 1);
629 	ATF_REQUIRE_ERRNO(EINVAL, error == -1);
630 	error = listen(sc.right[0], 1);
631 	ATF_REQUIRE_ERRNO(EINVAL, error == -1);
632 	splice_conn_fini(&sc);
633 
634 	tcp4_socketpair(sd);
635 	sd[2] = socket(PF_INET, SOCK_STREAM, 0);
636 	ATF_REQUIRE_MSG(sd[2] >= 0, "socket failed: %s", strerror(errno));
637 	error = listen(sd[2], 1);
638 	ATF_REQUIRE_MSG(error == 0, "listen failed: %s", strerror(errno));
639 
640 	/*
641 	 * Make sure a listening socket can't be spliced in either direction.
642 	 */
643 	splice_init(&sp, sd[2], 0, NULL);
644 	error = setsockopt(sd[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
645 	ATF_REQUIRE_ERRNO(EINVAL, error == -1);
646 	splice_init(&sp, sd[1], 0, NULL);
647 	error = setsockopt(sd[2], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
648 	ATF_REQUIRE_ERRNO(EINVAL, error == -1);
649 
650 	/*
651 	 * Make sure we can't try to unsplice a listening socket.
652 	 */
653 	splice_init(&sp, -1, 0, NULL);
654 	error = setsockopt(sd[2], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
655 	ATF_REQUIRE_ERRNO(EINVAL, error == -1);
656 
657 	checked_close(sd[0]);
658 	checked_close(sd[1]);
659 	checked_close(sd[2]);
660 }
661 
662 static void
663 sigalarm(int sig __unused)
664 {
665 }
666 
667 /*
668  * Our SO_SPLICE implementation doesn't do anything to prevent loops.  We should
669  * however make sure that they are interruptible.
670  */
671 ATF_TC_WITHOUT_HEAD(splice_loop);
672 ATF_TC_BODY(splice_loop, tc)
673 {
674 	ssize_t n;
675 	int sd[2], status;
676 	pid_t child;
677 	char c;
678 
679 	tcp_socketpair(sd, PF_INET);
680 	splice_pair(sd[0], sd[1], 0, NULL);
681 
682 	/*
683 	 * Let the child process trigger an infinite loop.  It should still be
684 	 * possible to kill the child with a signal, causing the connection to
685 	 * be dropped and ending the loop.
686 	 */
687 	child = fork();
688 	ATF_REQUIRE_MSG(child >= 0, "fork failed: %s", strerror(errno));
689 	if (child == 0) {
690 		alarm(2);
691 		c = 42;
692 		n = write(sd[0], &c, 1);
693 		if (n != 1)
694 			_exit(2);
695 		c = 24;
696 		n = write(sd[1], &c, 1);
697 		if (n != 1)
698 			_exit(3);
699 
700 		for (;;) {
701 			/* Wait for SIGALARM. */
702 			sleep(100);
703 		}
704 
705 		_exit(0);
706 	} else {
707 		checked_close(sd[0]);
708 		checked_close(sd[1]);
709 
710 		child = waitpid(child, &status, 0);
711 		ATF_REQUIRE_MSG(child >= 0,
712 		    "waitpid failed: %s", strerror(errno));
713 		ATF_REQUIRE(WIFSIGNALED(status));
714 		ATF_REQUIRE(WTERMSIG(status) == SIGALRM);
715 	}
716 }
717 
718 /*
719  * Simple I/O test.
720  */
721 ATF_TC_WITHOUT_HEAD(splice_nonblock);
722 ATF_TC_BODY(splice_nonblock, tc)
723 {
724 	struct splice_conn sc;
725 	char buf[200];
726 	size_t sofar;
727 	ssize_t n;
728 
729 	splice_conn_init(&sc);
730 	splice_conn_noblocking(&sc);
731 
732 	memset(buf, 'A', sizeof(buf));
733 	for (sofar = 0;;) {
734 		n = write(sc.left[0], buf, sizeof(buf));
735 		if (n < 0) {
736 			ATF_REQUIRE_ERRNO(EAGAIN, n == -1);
737 			break;
738 		}
739 		sofar += n;
740 	}
741 
742 	while (sofar > 0) {
743 		n = read(sc.right[1], buf, sizeof(buf));
744 		if (n < 0) {
745 			ATF_REQUIRE_ERRNO(EAGAIN, n == -1);
746 			usleep(100);
747 		} else {
748 			for (size_t i = 0; i < (size_t)n; i++)
749 				ATF_REQUIRE(buf[i] == 'A');
750 			sofar -= n;
751 		}
752 	}
753 
754 	splice_conn_fini(&sc);
755 }
756 
757 ATF_TC_WITHOUT_HEAD(splice_resplice);
758 ATF_TC_BODY(splice_resplice, tc)
759 {
760 	struct splice_conn sc;
761 	ssize_t n;
762 	char c;
763 
764 	splice_conn_init(&sc);
765 
766 	/* Left-to-right. */
767 	c = 'M';
768 	n = write(sc.left[0], &c, 1);
769 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
770 	n = read(sc.right[1], &c, 1);
771 	ATF_REQUIRE_MSG(n == 1, "read failed: %s", strerror(errno));
772 	ATF_REQUIRE_MSG(c == 'M', "unexpected character: %c", c);
773 	check_nspliced(sc.left[1], 1);
774 	check_nspliced(sc.right[0], 0);
775 
776 	/* Right-to-left. */
777 	c = 'J';
778 	n = write(sc.right[1], &c, 1);
779 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
780 	n = read(sc.left[0], &c, 1);
781 	ATF_REQUIRE_MSG(n == 1, "read failed: %s", strerror(errno));
782 	ATF_REQUIRE_MSG(c == 'J', "unexpected character: %c", c);
783 	check_nspliced(sc.left[1], 1);
784 	check_nspliced(sc.right[0], 1);
785 
786 	/* Unsplice and verify that the byte counts haven't changed. */
787 	unsplice(sc.left[1]);
788 	unsplice(sc.right[0]);
789 	check_nspliced(sc.left[1], 1);
790 	check_nspliced(sc.right[0], 1);
791 
792 	/* Splice again, check that byte counts are reset. */
793 	splice_pair(sc.left[1], sc.right[0], 0, NULL);
794 	check_nspliced(sc.left[1], 0);
795 	check_nspliced(sc.right[0], 0);
796 
797 	/* Left-to-right. */
798 	c = 'M';
799 	n = write(sc.left[0], &c, 1);
800 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
801 	n = read(sc.right[1], &c, 1);
802 	ATF_REQUIRE_MSG(n == 1, "read failed: %s", strerror(errno));
803 	ATF_REQUIRE_MSG(c == 'M', "unexpected character: %c", c);
804 	check_nspliced(sc.left[1], 1);
805 	check_nspliced(sc.right[0], 0);
806 
807 	/* Right-to-left. */
808 	c = 'J';
809 	n = write(sc.right[1], &c, 1);
810 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
811 	n = read(sc.left[0], &c, 1);
812 	ATF_REQUIRE_MSG(n == 1, "read failed: %s", strerror(errno));
813 	ATF_REQUIRE_MSG(c == 'J', "unexpected character: %c", c);
814 	check_nspliced(sc.left[1], 1);
815 	check_nspliced(sc.right[0], 1);
816 
817 	splice_conn_fini(&sc);
818 }
819 
820 struct xfer_args {
821 	pthread_barrier_t *barrier;
822 	uint32_t bytes;
823 	int fd;
824 };
825 
826 static void *
827 xfer(void *arg)
828 {
829 	struct xfer_args *xfer;
830 	uint8_t *buf;
831 	size_t sz;
832 	ssize_t n;
833 	uint32_t resid;
834 	int error;
835 
836 	xfer = arg;
837 
838 	error = fcntl(xfer->fd, F_SETFL, O_NONBLOCK);
839 	ATF_REQUIRE_MSG(error == 0, "fcntl failed: %s", strerror(errno));
840 
841 	sz = MIN(xfer->bytes, 1024 * 1024);
842 	buf = malloc(sz);
843 	ATF_REQUIRE(buf != NULL);
844 	arc4random_buf(buf, sz);
845 
846 	pthread_barrier_wait(xfer->barrier);
847 
848 	for (resid = xfer->bytes; xfer->bytes > 0 || resid > 0;) {
849 		n = write(xfer->fd, buf, MIN(sz, xfer->bytes));
850 		if (n < 0) {
851 			ATF_REQUIRE_ERRNO(EAGAIN, n == -1);
852 			usleep(1000);
853 		} else {
854 			ATF_REQUIRE(xfer->bytes >= (size_t)n);
855 			xfer->bytes -= n;
856 		}
857 
858 		n = read(xfer->fd, buf, sz);
859 		if (n < 0) {
860 			ATF_REQUIRE_ERRNO(EAGAIN, n == -1);
861 			usleep(1000);
862 		} else {
863 			ATF_REQUIRE(resid >= (size_t)n);
864 			resid -= n;
865 		}
866 	}
867 
868 	free(buf);
869 	return (NULL);
870 }
871 
872 /*
873  * Use two threads to transfer data between two spliced connections.
874  */
875 ATF_TC_WITHOUT_HEAD(splice_throughput);
876 ATF_TC_BODY(splice_throughput, tc)
877 {
878 	struct xfer_args xfers[2];
879 	pthread_t thread[2];
880 	pthread_barrier_t barrier;
881 	struct splice_conn sc;
882 	uint32_t bytes;
883 	int error;
884 
885 	/* Transfer an amount between 1B and 1GB. */
886 	bytes = arc4random_uniform(1024 * 1024 * 1024) + 1;
887 	splice_conn_init(&sc);
888 
889 	error = pthread_barrier_init(&barrier, NULL, 2);
890 	ATF_REQUIRE(error == 0);
891 	xfers[0] = (struct xfer_args){
892 	    .barrier = &barrier,
893 	    .bytes = bytes,
894 	    .fd = sc.left[0]
895 	};
896 	xfers[1] = (struct xfer_args){
897 	    .barrier = &barrier,
898 	    .bytes = bytes,
899 	    .fd = sc.right[1]
900 	};
901 
902 	error = pthread_create(&thread[0], NULL, xfer, &xfers[0]);
903 	ATF_REQUIRE_MSG(error == 0,
904 	    "pthread_create failed: %s", strerror(errno));
905 	error = pthread_create(&thread[1], NULL, xfer, &xfers[1]);
906 	ATF_REQUIRE_MSG(error == 0,
907 	    "pthread_create failed: %s", strerror(errno));
908 
909 	error = pthread_join(thread[0], NULL);
910 	ATF_REQUIRE_MSG(error == 0,
911 	    "pthread_join failed: %s", strerror(errno));
912 	error = pthread_join(thread[1], NULL);
913 	ATF_REQUIRE_MSG(error == 0,
914 	    "pthread_join failed: %s", strerror(errno));
915 
916 	error = pthread_barrier_destroy(&barrier);
917 	ATF_REQUIRE(error == 0);
918 	splice_conn_fini(&sc);
919 }
920 
921 /*
922  * Make sure it's possible to splice v4 and v6 sockets together.
923  */
924 ATF_TC_WITHOUT_HEAD(splice_v4v6);
925 ATF_TC_BODY(splice_v4v6, tc)
926 {
927 	struct splice sp;
928 	ssize_t n;
929 	int sd4[2], sd6[2];
930 	int error;
931 	uint8_t b;
932 
933 	tcp4_socketpair(sd4);
934 	tcp6_socketpair(sd6);
935 
936 	splice_init(&sp, sd6[0], 0, NULL);
937 	error = setsockopt(sd4[1], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
938 	ATF_REQUIRE_MSG(error == 0, "setsockopt failed: %s", strerror(errno));
939 
940 	splice_init(&sp, sd4[1], 0, NULL);
941 	error = setsockopt(sd6[0], SOL_SOCKET, SO_SPLICE, &sp, sizeof(sp));
942 	ATF_REQUIRE_MSG(error == 0, "setsockopt failed: %s", strerror(errno));
943 
944 	b = 'M';
945 	n = write(sd4[0], &b, 1);
946 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
947 	n = read(sd6[1], &b, 1);
948 	ATF_REQUIRE_MSG(n == 1, "read failed: %s", strerror(errno));
949 	ATF_REQUIRE(b == 'M');
950 
951 	b = 'J';
952 	n = write(sd6[1], &b, 1);
953 	ATF_REQUIRE_MSG(n == 1, "write failed: %s", strerror(errno));
954 	n = read(sd4[0], &b, 1);
955 	ATF_REQUIRE_MSG(n == 1, "read failed: %s", strerror(errno));
956 	ATF_REQUIRE(b == 'J');
957 
958 	checked_close(sd4[0]);
959 	checked_close(sd4[1]);
960 	checked_close(sd6[0]);
961 	checked_close(sd6[1]);
962 }
963 
964 ATF_TP_ADD_TCS(tp)
965 {
966 	ATF_TP_ADD_TC(tp, splice_basic);
967 	ATF_TP_ADD_TC(tp, splice_capsicum);
968 	ATF_TP_ADD_TC(tp, splice_error);
969 	ATF_TP_ADD_TC(tp, splice_kevent);
970 	ATF_TP_ADD_TC(tp, splice_limit_bytes);
971 	ATF_TP_ADD_TC(tp, splice_limit_timeout);
972 	ATF_TP_ADD_TC(tp, splice_listen);
973 	ATF_TP_ADD_TC(tp, splice_loop);
974 	ATF_TP_ADD_TC(tp, splice_nonblock);
975 	ATF_TP_ADD_TC(tp, splice_resplice);
976 	ATF_TP_ADD_TC(tp, splice_throughput);
977 	ATF_TP_ADD_TC(tp, splice_v4v6);
978 	return (atf_no_error());
979 }
980