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