xref: /freebsd/tests/sys/aio/aio_test.c (revision 788ca347b816afd83b2885e0c79aeeb88649b2ab)
1 /*-
2  * Copyright (c) 2004 Robert N. M. Watson
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided 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 AND CONTRIBUTORS ``AS IS'' AND
15  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18  * FOR ANY 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, STRICT
22  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24  * SUCH DAMAGE.
25  *
26  * $FreeBSD$
27  */
28 
29 /*
30  * Regression test to do some very basic AIO exercising on several types of
31  * file descriptors.  Currently, the tests consist of initializing a fixed
32  * size buffer with pseudo-random data, writing it to one fd using AIO, then
33  * reading it from a second descriptor using AIO.  For some targets, the same
34  * fd is used for write and read (i.e., file, md device), but for others the
35  * operation is performed on a peer (pty, socket, fifo, etc).  A timeout is
36  * initiated to detect undo blocking.  This test does not attempt to exercise
37  * error cases or more subtle asynchronous behavior, just make sure that the
38  * basic operations work on some basic object types.
39  */
40 
41 #include <sys/param.h>
42 #include <sys/module.h>
43 #include <sys/socket.h>
44 #include <sys/stat.h>
45 #include <sys/mdioctl.h>
46 
47 #include <aio.h>
48 #include <err.h>
49 #include <errno.h>
50 #include <fcntl.h>
51 #include <libutil.h>
52 #include <limits.h>
53 #include <stdint.h>
54 #include <stdio.h>
55 #include <stdlib.h>
56 #include <string.h>
57 #include <termios.h>
58 #include <unistd.h>
59 
60 #include <atf-c.h>
61 
62 #define	PATH_TEMPLATE	"aio.XXXXXXXXXX"
63 
64 /*
65  * GLOBAL_MAX sets the largest usable buffer size to be read and written, as
66  * it sizes ac_buffer in the aio_context structure.  It is also the default
67  * size for file I/O.  For other types, we use smaller blocks or we risk
68  * blocking (and we run in a single process/thread so that would be bad).
69  */
70 #define	GLOBAL_MAX	16384
71 
72 #define	BUFFER_MAX	GLOBAL_MAX
73 struct aio_context {
74 	int		 ac_read_fd, ac_write_fd;
75 	long		 ac_seed;
76 	char		 ac_buffer[GLOBAL_MAX];
77 	int		 ac_buflen;
78 	int		 ac_seconds;
79 	void		 (*ac_cleanup)(void *arg);
80 	void		*ac_cleanup_arg;
81 };
82 
83 static int	aio_timedout;
84 
85 static void
86 aio_available(void)
87 {
88 
89 	if (modfind("aio") == -1)
90 		atf_tc_skip("aio support not available in the kernel; "
91 		    "skipping testcases");
92 }
93 
94 /*
95  * Each test run specifies a timeout in seconds.  Use the somewhat obsoleted
96  * signal(3) and alarm(3) APIs to set this up.
97  */
98 static void
99 aio_timeout_signal(int sig __unused)
100 {
101 
102 	aio_timedout = 1;
103 }
104 
105 static void
106 aio_timeout_start(int seconds)
107 {
108 
109 	aio_timedout = 0;
110 	ATF_REQUIRE_MSG(signal(SIGALRM, aio_timeout_signal) != SIG_ERR,
111 	    "failed to set SIGALRM handler: %s", strerror(errno));
112 	alarm(seconds);
113 }
114 
115 static void
116 aio_timeout_stop(void)
117 {
118 
119 	ATF_REQUIRE_MSG(signal(SIGALRM, NULL) != SIG_ERR,
120 	    "failed to reset SIGALRM handler to default: %s", strerror(errno));
121 	alarm(0);
122 }
123 
124 /*
125  * Fill a buffer given a seed that can be fed into srandom() to initialize
126  * the PRNG in a repeatable manner.
127  */
128 static void
129 aio_fill_buffer(char *buffer, int len, long seed)
130 {
131 	char ch;
132 	int i;
133 
134 	srandom(seed);
135 	for (i = 0; i < len; i++) {
136 		ch = random() & 0xff;
137 		buffer[i] = ch;
138 	}
139 }
140 
141 /*
142  * Test that a buffer matches a given seed.  See aio_fill_buffer().  Return
143  * (1) on a match, (0) on a mismatch.
144  */
145 static int
146 aio_test_buffer(char *buffer, int len, long seed)
147 {
148 	char ch;
149 	int i;
150 
151 	srandom(seed);
152 	for (i = 0; i < len; i++) {
153 		ch = random() & 0xff;
154 		if (buffer[i] != ch)
155 			return (0);
156 	}
157 	return (1);
158 }
159 
160 /*
161  * Initialize a testing context given the file descriptors provided by the
162  * test setup.
163  */
164 static void
165 aio_context_init(struct aio_context *ac, int read_fd,
166     int write_fd, int buflen, int seconds, void (*cleanup)(void *),
167     void *cleanup_arg)
168 {
169 
170 	ATF_REQUIRE_MSG(buflen <= BUFFER_MAX,
171 	    "aio_context_init: buffer too large (%d > %d)",
172 	    buflen, BUFFER_MAX);
173 	bzero(ac, sizeof(*ac));
174 	ac->ac_read_fd = read_fd;
175 	ac->ac_write_fd = write_fd;
176 	ac->ac_buflen = buflen;
177 	srandomdev();
178 	ac->ac_seed = random();
179 	aio_fill_buffer(ac->ac_buffer, buflen, ac->ac_seed);
180 	ATF_REQUIRE_MSG(aio_test_buffer(ac->ac_buffer, buflen,
181 	    ac->ac_seed) != 0, "aio_test_buffer: internal error");
182 	ac->ac_seconds = seconds;
183 	ac->ac_cleanup = cleanup;
184 	ac->ac_cleanup_arg = cleanup_arg;
185 }
186 
187 /*
188  * Each tester can register a callback to clean up in the event the test
189  * fails.  Preserve the value of errno so that subsequent calls to errx()
190  * work properly.
191  */
192 static void
193 aio_cleanup(struct aio_context *ac)
194 {
195 	int error;
196 
197 	if (ac->ac_cleanup == NULL)
198 		return;
199 	error = errno;
200 	(ac->ac_cleanup)(ac->ac_cleanup_arg);
201 	errno = error;
202 }
203 
204 /*
205  * Perform a simple write test of our initialized data buffer to the provided
206  * file descriptor.
207  */
208 static void
209 aio_write_test(struct aio_context *ac)
210 {
211 	struct aiocb aio, *aiop;
212 	ssize_t len;
213 
214 	aio_available();
215 
216 	bzero(&aio, sizeof(aio));
217 	aio.aio_buf = ac->ac_buffer;
218 	aio.aio_nbytes = ac->ac_buflen;
219 	aio.aio_fildes = ac->ac_write_fd;
220 	aio.aio_offset = 0;
221 
222 	aio_timeout_start(ac->ac_seconds);
223 
224 	if (aio_write(&aio) < 0) {
225 		if (errno == EINTR) {
226 			if (aio_timedout) {
227 				aio_cleanup(ac);
228 				atf_tc_fail("aio_write timed out");
229 			}
230 		}
231 		aio_cleanup(ac);
232 		atf_tc_fail("aio_write failed: %s", strerror(errno));
233 	}
234 
235 	len = aio_waitcomplete(&aiop, NULL);
236 	if (len < 0) {
237 		if (errno == EINTR) {
238 			if (aio_timedout) {
239 				aio_cleanup(ac);
240 				atf_tc_fail("aio_waitcomplete timed out");
241 			}
242 		}
243 		aio_cleanup(ac);
244 		atf_tc_fail("aio_waitcomplete failed: %s", strerror(errno));
245 	}
246 
247 	aio_timeout_stop();
248 
249 	if (len != ac->ac_buflen) {
250 		aio_cleanup(ac);
251 		atf_tc_fail("aio_waitcomplete short write (%jd)",
252 		    (intmax_t)len);
253 	}
254 }
255 
256 /*
257  * Perform a simple read test of our initialized data buffer from the
258  * provided file descriptor.
259  */
260 static void
261 aio_read_test(struct aio_context *ac)
262 {
263 	struct aiocb aio, *aiop;
264 	ssize_t len;
265 
266 	aio_available();
267 
268 	bzero(ac->ac_buffer, ac->ac_buflen);
269 	bzero(&aio, sizeof(aio));
270 	aio.aio_buf = ac->ac_buffer;
271 	aio.aio_nbytes = ac->ac_buflen;
272 	aio.aio_fildes = ac->ac_read_fd;
273 	aio.aio_offset = 0;
274 
275 	aio_timeout_start(ac->ac_seconds);
276 
277 	if (aio_read(&aio) < 0) {
278 		if (errno == EINTR) {
279 			if (aio_timedout) {
280 				aio_cleanup(ac);
281 				atf_tc_fail("aio_write timed out");
282 			}
283 		}
284 		aio_cleanup(ac);
285 		atf_tc_fail("aio_read failed: %s", strerror(errno));
286 	}
287 
288 	len = aio_waitcomplete(&aiop, NULL);
289 	if (len < 0) {
290 		if (errno == EINTR) {
291 			if (aio_timedout) {
292 				aio_cleanup(ac);
293 				atf_tc_fail("aio_waitcomplete timed out");
294 			}
295 		}
296 		aio_cleanup(ac);
297 		atf_tc_fail("aio_waitcomplete failed: %s", strerror(errno));
298 	}
299 
300 	aio_timeout_stop();
301 
302 	if (len != ac->ac_buflen) {
303 		aio_cleanup(ac);
304 		atf_tc_fail("aio_waitcomplete short read (%jd)",
305 		    (intmax_t)len);
306 	}
307 
308 	if (aio_test_buffer(ac->ac_buffer, ac->ac_buflen, ac->ac_seed) == 0) {
309 		aio_cleanup(ac);
310 		atf_tc_fail("buffer mismatched");
311 	}
312 }
313 
314 /*
315  * Series of type-specific tests for AIO.  For now, we just make sure we can
316  * issue a write and then a read to each type.  We assume that once a write
317  * is issued, a read can follow.
318  */
319 
320 /*
321  * Test with a classic file.  Assumes we can create a moderate size temporary
322  * file.
323  */
324 struct aio_file_arg {
325 	int	 afa_fd;
326 	char	*afa_pathname;
327 };
328 
329 static void
330 aio_file_cleanup(void *arg)
331 {
332 	struct aio_file_arg *afa;
333 
334 	afa = arg;
335 	close(afa->afa_fd);
336 	unlink(afa->afa_pathname);
337 }
338 
339 #define	FILE_LEN	GLOBAL_MAX
340 #define	FILE_TIMEOUT	30
341 ATF_TC_WITHOUT_HEAD(aio_file_test);
342 ATF_TC_BODY(aio_file_test, tc)
343 {
344 	char pathname[PATH_MAX];
345 	struct aio_file_arg arg;
346 	struct aio_context ac;
347 	int fd;
348 
349 	aio_available();
350 
351 	strcpy(pathname, PATH_TEMPLATE);
352 	fd = mkstemp(pathname);
353 	ATF_REQUIRE_MSG(fd != -1, "mkstemp failed: %s", strerror(errno));
354 
355 	arg.afa_fd = fd;
356 	arg.afa_pathname = pathname;
357 
358 	aio_context_init(&ac, fd, fd, FILE_LEN,
359 	    FILE_TIMEOUT, aio_file_cleanup, &arg);
360 	aio_write_test(&ac);
361 	aio_read_test(&ac);
362 
363 	aio_file_cleanup(&arg);
364 }
365 
366 struct aio_fifo_arg {
367 	int	 afa_read_fd;
368 	int	 afa_write_fd;
369 	char	*afa_pathname;
370 };
371 
372 static void
373 aio_fifo_cleanup(void *arg)
374 {
375 	struct aio_fifo_arg *afa;
376 
377 	afa = arg;
378 	if (afa->afa_read_fd != -1)
379 		close(afa->afa_read_fd);
380 	if (afa->afa_write_fd != -1)
381 		close(afa->afa_write_fd);
382 	unlink(afa->afa_pathname);
383 }
384 
385 #define	FIFO_LEN	256
386 #define	FIFO_TIMEOUT	30
387 ATF_TC_WITHOUT_HEAD(aio_fifo_test);
388 ATF_TC_BODY(aio_fifo_test, tc)
389 {
390 	int error, read_fd = -1, write_fd = -1;
391 	struct aio_fifo_arg arg;
392 	char pathname[PATH_MAX];
393 	struct aio_context ac;
394 
395 	aio_available();
396 
397 	/*
398 	 * In theory, mkstemp() can return a name that is then collided with.
399 	 * Because this is a regression test, we treat that as a test failure
400 	 * rather than retrying.
401 	 */
402 	strcpy(pathname, PATH_TEMPLATE);
403 	ATF_REQUIRE_MSG(mkstemp(pathname) != -1,
404 	    "mkstemp failed: %s", strerror(errno));
405 	ATF_REQUIRE_MSG(unlink(pathname) == 0,
406 	    "unlink failed: %s", strerror(errno));
407 	ATF_REQUIRE_MSG(mkfifo(pathname, 0600) != -1,
408 	    "mkfifo failed: %s", strerror(errno));
409 	arg.afa_pathname = pathname;
410 	arg.afa_read_fd = -1;
411 	arg.afa_write_fd = -1;
412 
413 	read_fd = open(pathname, O_RDONLY | O_NONBLOCK);
414 	if (read_fd == -1) {
415 		error = errno;
416 		aio_fifo_cleanup(&arg);
417 		errno = error;
418 		atf_tc_fail("read_fd open failed: %s",
419 		    strerror(errno));
420 	}
421 	arg.afa_read_fd = read_fd;
422 
423 	write_fd = open(pathname, O_WRONLY);
424 	if (write_fd == -1) {
425 		error = errno;
426 		aio_fifo_cleanup(&arg);
427 		errno = error;
428 		atf_tc_fail("write_fd open failed: %s",
429 		    strerror(errno));
430 	}
431 	arg.afa_write_fd = write_fd;
432 
433 	aio_context_init(&ac, read_fd, write_fd, FIFO_LEN,
434 	    FIFO_TIMEOUT, aio_fifo_cleanup, &arg);
435 	aio_write_test(&ac);
436 	aio_read_test(&ac);
437 
438 	aio_fifo_cleanup(&arg);
439 }
440 
441 struct aio_unix_socketpair_arg {
442 	int	asa_sockets[2];
443 };
444 
445 static void
446 aio_unix_socketpair_cleanup(void *arg)
447 {
448 	struct aio_unix_socketpair_arg *asa;
449 
450 	asa = arg;
451 	close(asa->asa_sockets[0]);
452 	close(asa->asa_sockets[1]);
453 }
454 
455 #define	UNIX_SOCKETPAIR_LEN	256
456 #define	UNIX_SOCKETPAIR_TIMEOUT	30
457 ATF_TC_WITHOUT_HEAD(aio_unix_socketpair_test);
458 ATF_TC_BODY(aio_unix_socketpair_test, tc)
459 {
460 	struct aio_unix_socketpair_arg arg;
461 	struct aio_context ac;
462 	int sockets[2];
463 
464 	aio_available();
465 
466 	ATF_REQUIRE_MSG(socketpair(PF_UNIX, SOCK_STREAM, 0, sockets) != -1,
467 	    "socketpair failed: %s", strerror(errno));
468 
469 	arg.asa_sockets[0] = sockets[0];
470 	arg.asa_sockets[1] = sockets[1];
471 	aio_context_init(&ac, sockets[0],
472 	    sockets[1], UNIX_SOCKETPAIR_LEN, UNIX_SOCKETPAIR_TIMEOUT,
473 	    aio_unix_socketpair_cleanup, &arg);
474 	aio_write_test(&ac);
475 	aio_read_test(&ac);
476 
477 	aio_unix_socketpair_cleanup(&arg);
478 }
479 
480 struct aio_pty_arg {
481 	int	apa_read_fd;
482 	int	apa_write_fd;
483 };
484 
485 static void
486 aio_pty_cleanup(void *arg)
487 {
488 	struct aio_pty_arg *apa;
489 
490 	apa = arg;
491 	close(apa->apa_read_fd);
492 	close(apa->apa_write_fd);
493 };
494 
495 #define	PTY_LEN		256
496 #define	PTY_TIMEOUT	30
497 ATF_TC_WITHOUT_HEAD(aio_pty_test);
498 ATF_TC_BODY(aio_pty_test, tc)
499 {
500 	struct aio_pty_arg arg;
501 	struct aio_context ac;
502 	int read_fd, write_fd;
503 	struct termios ts;
504 	int error;
505 
506 	aio_available();
507 
508 	ATF_REQUIRE_MSG(openpty(&read_fd, &write_fd, NULL, NULL, NULL) == 0,
509 	    "openpty failed: %s", strerror(errno));
510 
511 	arg.apa_read_fd = read_fd;
512 	arg.apa_write_fd = write_fd;
513 
514 	if (tcgetattr(write_fd, &ts) < 0) {
515 		error = errno;
516 		aio_pty_cleanup(&arg);
517 		errno = error;
518 		atf_tc_fail("tcgetattr failed: %s", strerror(errno));
519 	}
520 	cfmakeraw(&ts);
521 	if (tcsetattr(write_fd, TCSANOW, &ts) < 0) {
522 		error = errno;
523 		aio_pty_cleanup(&arg);
524 		errno = error;
525 		atf_tc_fail("tcsetattr failed: %s", strerror(errno));
526 	}
527 	aio_context_init(&ac, read_fd, write_fd, PTY_LEN,
528 	    PTY_TIMEOUT, aio_pty_cleanup, &arg);
529 
530 	aio_write_test(&ac);
531 	aio_read_test(&ac);
532 
533 	aio_pty_cleanup(&arg);
534 }
535 
536 static void
537 aio_pipe_cleanup(void *arg)
538 {
539 	int *pipes = arg;
540 
541 	close(pipes[0]);
542 	close(pipes[1]);
543 }
544 
545 #define	PIPE_LEN	256
546 #define	PIPE_TIMEOUT	30
547 ATF_TC_WITHOUT_HEAD(aio_pipe_test);
548 ATF_TC_BODY(aio_pipe_test, tc)
549 {
550 	struct aio_context ac;
551 	int pipes[2];
552 
553 	aio_available();
554 
555 	ATF_REQUIRE_MSG(pipe(pipes) != -1,
556 	    "pipe failed: %s", strerror(errno));
557 
558 	aio_context_init(&ac, pipes[0], pipes[1], PIPE_LEN,
559 	    PIPE_TIMEOUT, aio_pipe_cleanup, pipes);
560 	aio_write_test(&ac);
561 	aio_read_test(&ac);
562 
563 	aio_pipe_cleanup(pipes);
564 }
565 
566 struct aio_md_arg {
567 	int	ama_mdctl_fd;
568 	int	ama_unit;
569 	int	ama_fd;
570 };
571 
572 static void
573 aio_md_cleanup(void *arg)
574 {
575 	struct aio_md_arg *ama;
576 	struct md_ioctl mdio;
577 	int error;
578 
579 	ama = arg;
580 
581 	if (ama->ama_fd != -1)
582 		close(ama->ama_fd);
583 
584 	if (ama->ama_unit != -1) {
585 		bzero(&mdio, sizeof(mdio));
586 		mdio.md_version = MDIOVERSION;
587 		mdio.md_unit = ama->ama_unit;
588 		if (ioctl(ama->ama_mdctl_fd, MDIOCDETACH, &mdio) == -1) {
589 			error = errno;
590 			close(ama->ama_mdctl_fd);
591 			errno = error;
592 			atf_tc_fail("ioctl MDIOCDETACH failed: %s",
593 			    strerror(errno));
594 		}
595 	}
596 
597 	close(ama->ama_mdctl_fd);
598 }
599 
600 #define	MD_LEN		GLOBAL_MAX
601 #define	MD_TIMEOUT	30
602 ATF_TC(aio_md_test);
603 ATF_TC_HEAD(aio_md_test, tc)
604 {
605 
606 	atf_tc_set_md_var(tc, "require.user", "root");
607 }
608 ATF_TC_BODY(aio_md_test, tc)
609 {
610 	int error, fd, mdctl_fd, unit;
611 	char pathname[PATH_MAX];
612 	struct aio_md_arg arg;
613 	struct aio_context ac;
614 	struct md_ioctl mdio;
615 
616 	aio_available();
617 
618 	mdctl_fd = open("/dev/" MDCTL_NAME, O_RDWR, 0);
619 	ATF_REQUIRE_MSG(mdctl_fd != -1,
620 	    "opening /dev/%s failed: %s", MDCTL_NAME, strerror(errno));
621 
622 	bzero(&mdio, sizeof(mdio));
623 	mdio.md_version = MDIOVERSION;
624 	mdio.md_type = MD_MALLOC;
625 	mdio.md_options = MD_AUTOUNIT | MD_COMPRESS;
626 	mdio.md_mediasize = GLOBAL_MAX;
627 	mdio.md_sectorsize = 512;
628 
629 	arg.ama_mdctl_fd = mdctl_fd;
630 	arg.ama_unit = -1;
631 	arg.ama_fd = -1;
632 	if (ioctl(mdctl_fd, MDIOCATTACH, &mdio) < 0) {
633 		error = errno;
634 		aio_md_cleanup(&arg);
635 		errno = error;
636 		atf_tc_fail("ioctl MDIOCATTACH failed: %s", strerror(errno));
637 	}
638 
639 	arg.ama_unit = unit = mdio.md_unit;
640 	snprintf(pathname, PATH_MAX, "/dev/md%d", unit);
641 	fd = open(pathname, O_RDWR);
642 	ATF_REQUIRE_MSG(fd != -1,
643 	    "opening %s failed: %s", pathname, strerror(errno));
644 	arg.ama_fd = fd;
645 
646 	aio_context_init(&ac, fd, fd, MD_LEN, MD_TIMEOUT,
647 	    aio_md_cleanup, &arg);
648 	aio_write_test(&ac);
649 	aio_read_test(&ac);
650 
651 	aio_md_cleanup(&arg);
652 }
653 
654 ATF_TP_ADD_TCS(tp)
655 {
656 
657 	ATF_TP_ADD_TC(tp, aio_file_test);
658 	ATF_TP_ADD_TC(tp, aio_fifo_test);
659 	ATF_TP_ADD_TC(tp, aio_unix_socketpair_test);
660 	ATF_TP_ADD_TC(tp, aio_pty_test);
661 	ATF_TP_ADD_TC(tp, aio_pipe_test);
662 	ATF_TP_ADD_TC(tp, aio_md_test);
663 
664 	return (atf_no_error());
665 }
666