1 /*- 2 * Copyright (c) 2018 Enji Cooper. 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 27 #include <sys/cdefs.h> 28 __FBSDID("$FreeBSD$"); 29 30 #include <sys/param.h> 31 #include <sys/mman.h> 32 #include <sys/socket.h> 33 #include <sys/stat.h> 34 #include <sys/sysctl.h> 35 #include <sys/uio.h> 36 #include <err.h> 37 #include <errno.h> 38 #include <fcntl.h> 39 #include <netdb.h> 40 #include <paths.h> 41 #include <stdio.h> 42 #include <stdlib.h> 43 #include <string.h> 44 #include <unistd.h> 45 46 #include <atf-c.h> 47 48 const char DETERMINISTIC_PATTERN[] = 49 "The past is already gone, the future is not yet here. There's only one moment for you to live.\n"; 50 51 #define SOURCE_FILE "source" 52 #define DESTINATION_FILE "dest" 53 54 #define PORTRANGE_FIRST "net.inet.ip.portrange.first" 55 #define PORTRANGE_LAST "net.inet.ip.portrange.last" 56 57 static int portrange_first, portrange_last; 58 59 static int 60 get_int_via_sysctlbyname(const char *oidname) 61 { 62 size_t oldlen; 63 int int_value; 64 65 oldlen = sizeof(int_value); 66 67 ATF_REQUIRE_EQ_MSG(sysctlbyname(oidname, &int_value, &oldlen, NULL, 0), 68 0, "sysctlbyname(%s, ...) failed: %s", oidname, strerror(errno)); 69 ATF_REQUIRE_EQ_MSG(sizeof(int_value), oldlen, "sanity check failed"); 70 71 return (int_value); 72 } 73 74 static int 75 generate_random_port(int seed) 76 { 77 int random_port; 78 79 printf("Generating a random port with seed=%d\n", seed); 80 if (portrange_first == 0) { 81 portrange_first = get_int_via_sysctlbyname(PORTRANGE_FIRST); 82 printf("Port range lower bound: %d\n", portrange_first); 83 } 84 85 if (portrange_last == 0) { 86 portrange_last = get_int_via_sysctlbyname(PORTRANGE_LAST); 87 printf("Port range upper bound: %d\n", portrange_last); 88 } 89 90 srand((unsigned)seed); 91 92 random_port = rand() % (portrange_last - portrange_first) + 93 portrange_first; 94 95 printf("Random port generated: %d\n", random_port); 96 return (random_port); 97 } 98 99 static void 100 resolve_localhost(struct addrinfo **res, int domain, int type, int port) 101 { 102 const char *host; 103 char *serv; 104 struct addrinfo hints; 105 int error; 106 107 switch (domain) { 108 case AF_INET: 109 host = "127.0.0.1"; 110 break; 111 case AF_INET6: 112 host = "::1"; 113 break; 114 default: 115 atf_tc_fail("unhandled domain: %d", domain); 116 } 117 118 ATF_REQUIRE_MSG(asprintf(&serv, "%d", port) >= 0, 119 "asprintf failed: %s", strerror(errno)); 120 121 memset(&hints, 0, sizeof(hints)); 122 hints.ai_family = domain; 123 hints.ai_flags = AI_ADDRCONFIG|AI_NUMERICSERV|AI_NUMERICHOST; 124 hints.ai_socktype = type; 125 126 error = getaddrinfo(host, serv, &hints, res); 127 ATF_REQUIRE_EQ_MSG(error, 0, 128 "getaddrinfo failed: %s", gai_strerror(error)); 129 free(serv); 130 } 131 132 static int 133 make_socket(int domain, int type, int protocol) 134 { 135 int sock; 136 137 sock = socket(domain, type, protocol); 138 ATF_REQUIRE_MSG(sock != -1, "socket(%d, %d, 0) failed: %s", 139 domain, type, strerror(errno)); 140 141 return (sock); 142 } 143 144 static int 145 setup_client(int domain, int type, int port) 146 { 147 struct addrinfo *res; 148 char host[NI_MAXHOST+1]; 149 int error, sock; 150 151 resolve_localhost(&res, domain, type, port); 152 error = getnameinfo( 153 (const struct sockaddr*)res->ai_addr, res->ai_addrlen, 154 host, nitems(host) - 1, NULL, 0, NI_NUMERICHOST); 155 ATF_REQUIRE_EQ_MSG(error, 0, 156 "getnameinfo failed: %s", gai_strerror(error)); 157 printf( 158 "Will try to connect to host='%s', address_family=%d, " 159 "socket_type=%d\n", 160 host, res->ai_family, res->ai_socktype); 161 /* Avoid a double print when forked by flushing. */ 162 fflush(stdout); 163 sock = make_socket(res->ai_family, res->ai_socktype, res->ai_protocol); 164 error = connect(sock, (struct sockaddr*)res->ai_addr, res->ai_addrlen); 165 freeaddrinfo(res); 166 ATF_REQUIRE_EQ_MSG(error, 0, "connect failed: %s", strerror(errno)); 167 return (sock); 168 } 169 170 /* 171 * XXX: use linear probing to find a free port and eliminate `port` argument as 172 * a [const] int (it will need to be a pointer so it can be passed back out of 173 * the function and can influence which port `setup_client(..)` connects on. 174 */ 175 static int 176 setup_server(int domain, int type, int port) 177 { 178 struct addrinfo *res; 179 char host[NI_MAXHOST+1]; 180 int error, sock; 181 182 resolve_localhost(&res, domain, type, port); 183 sock = make_socket(res->ai_family, res->ai_socktype, res->ai_protocol); 184 185 error = getnameinfo( 186 (const struct sockaddr*)res->ai_addr, res->ai_addrlen, 187 host, nitems(host) - 1, NULL, 0, NI_NUMERICHOST); 188 ATF_REQUIRE_EQ_MSG(error, 0, 189 "getnameinfo failed: %s", gai_strerror(error)); 190 printf( 191 "Will try to bind socket to host='%s', address_family=%d, " 192 "socket_type=%d\n", 193 host, res->ai_family, res->ai_socktype); 194 /* Avoid a double print when forked by flushing. */ 195 fflush(stdout); 196 error = bind(sock, res->ai_addr, res->ai_addrlen); 197 freeaddrinfo(res); 198 ATF_REQUIRE_EQ_MSG(error, 0, "bind failed: %s", strerror(errno)); 199 error = listen(sock, 1); 200 ATF_REQUIRE_EQ_MSG(error, 0, "listen failed: %s", strerror(errno)); 201 202 return (sock); 203 } 204 205 /* 206 * This function is a helper routine for taking data being sent by `sendfile` via 207 * `server_sock`, and pushing the received stream out to a file, denoted by 208 * `dest_filename`. 209 */ 210 static void 211 server_cat(const char *dest_filename, int server_sock, size_t len) 212 { 213 char *buffer, *buf_window_ptr; 214 int recv_sock; 215 size_t buffer_size; 216 ssize_t received_bytes, recv_ret; 217 218 /* 219 * Ensure that there isn't excess data sent across the wire by 220 * capturing 10 extra bytes (plus 1 for nul). 221 */ 222 buffer_size = len + 10 + 1; 223 buffer = calloc(buffer_size, sizeof(char)); 224 if (buffer == NULL) 225 err(1, "malloc failed"); 226 227 recv_sock = accept(server_sock, NULL, 0); 228 if (recv_sock == -1) 229 err(1, "accept failed"); 230 231 buf_window_ptr = buffer; 232 received_bytes = 0; 233 do { 234 recv_ret = recv(recv_sock, buf_window_ptr, 235 buffer_size - received_bytes, 0); 236 if (recv_ret <= 0) 237 break; 238 buf_window_ptr += recv_ret; 239 received_bytes += recv_ret; 240 } while (received_bytes < buffer_size); 241 242 atf_utils_create_file(dest_filename, "%s", buffer); 243 244 (void)close(recv_sock); 245 (void)close(server_sock); 246 free(buffer); 247 248 if (received_bytes != len) 249 errx(1, "received unexpected data: %zd != %zd", received_bytes, 250 len); 251 } 252 253 static int 254 setup_tcp_server(int domain, int port) 255 { 256 257 return (setup_server(domain, SOCK_STREAM, port)); 258 } 259 260 static int 261 setup_tcp_client(int domain, int port) 262 { 263 264 return (setup_client(domain, SOCK_STREAM, port)); 265 } 266 267 static off_t 268 file_size_from_fd(int fd) 269 { 270 struct stat st; 271 272 ATF_REQUIRE_EQ_MSG(0, fstat(fd, &st), 273 "fstat failed: %s", strerror(errno)); 274 275 return (st.st_size); 276 } 277 278 /* 279 * NB: `nbytes` == 0 has special connotations given the sendfile(2) API 280 * contract. In short, "send the whole file" (paraphrased). 281 */ 282 static void 283 verify_source_and_dest(const char* dest_filename, int src_fd, off_t offset, 284 size_t nbytes) 285 { 286 char *dest_pointer, *src_pointer; 287 off_t dest_file_size, src_file_size; 288 size_t length; 289 int dest_fd; 290 291 atf_utils_cat_file(dest_filename, "dest_file: "); 292 293 dest_fd = open(dest_filename, O_RDONLY); 294 ATF_REQUIRE_MSG(dest_fd != -1, "open failed"); 295 296 dest_file_size = file_size_from_fd(dest_fd); 297 src_file_size = file_size_from_fd(src_fd); 298 299 /* 300 * Per sendfile(2), "send the whole file" (paraphrased). This means 301 * that we need to grab the file size, as passing in length = 0 with 302 * mmap(2) will result in a failure with EINVAL (length = 0 is invalid). 303 */ 304 length = (nbytes == 0) ? (size_t)(src_file_size - offset) : nbytes; 305 306 ATF_REQUIRE_EQ_MSG(dest_file_size, length, 307 "number of bytes written out to %s (%ju) doesn't match the " 308 "expected number of bytes (%zu)", dest_filename, dest_file_size, 309 length); 310 311 ATF_REQUIRE_EQ_MSG(0, lseek(src_fd, offset, SEEK_SET), 312 "lseek failed: %s", strerror(errno)); 313 314 dest_pointer = mmap(NULL, length, PROT_READ, MAP_PRIVATE, dest_fd, 0); 315 ATF_REQUIRE_MSG(dest_pointer != MAP_FAILED, "mmap failed: %s", 316 strerror(errno)); 317 318 printf("Will mmap in the source file from offset=%jd to length=%zu\n", 319 offset, length); 320 321 src_pointer = mmap(NULL, length, PROT_READ, MAP_PRIVATE, src_fd, offset); 322 ATF_REQUIRE_MSG(src_pointer != MAP_FAILED, "mmap failed: %s", 323 strerror(errno)); 324 325 ATF_REQUIRE_EQ_MSG(0, memcmp(src_pointer, dest_pointer, length), 326 "Contents of source and destination do not match. '%s' != '%s'", 327 src_pointer, dest_pointer); 328 329 (void)munmap(src_pointer, length); 330 (void)munmap(dest_pointer, length); 331 (void)close(dest_fd); 332 } 333 334 static void 335 fd_positive_file_test(int domain) 336 { 337 off_t offset; 338 size_t nbytes, pattern_size; 339 int client_sock, error, fd, port, server_sock; 340 pid_t server_pid; 341 342 pattern_size = strlen(DETERMINISTIC_PATTERN); 343 344 atf_utils_create_file(SOURCE_FILE, "%s", DETERMINISTIC_PATTERN); 345 fd = open(SOURCE_FILE, O_RDONLY); 346 ATF_REQUIRE_MSG(fd != -1, "open failed: %s", strerror(errno)); 347 348 port = generate_random_port(__LINE__ + domain); 349 server_sock = setup_tcp_server(domain, port); 350 client_sock = setup_tcp_client(domain, port); 351 352 server_pid = atf_utils_fork(); 353 if (server_pid == 0) { 354 (void)close(client_sock); 355 server_cat(DESTINATION_FILE, server_sock, pattern_size); 356 _exit(0); 357 } else 358 (void)close(server_sock); 359 360 nbytes = 0; 361 offset = 0; 362 error = sendfile(fd, client_sock, offset, nbytes, NULL, NULL, 363 SF_FLAGS(0, 0)); 364 ATF_REQUIRE_EQ_MSG(0, error, "sendfile failed: %s", strerror(errno)); 365 (void)close(client_sock); 366 367 atf_utils_wait(server_pid, 0, "", ""); 368 verify_source_and_dest(DESTINATION_FILE, fd, offset, nbytes); 369 370 (void)close(fd); 371 } 372 373 ATF_TC(fd_positive_file_v4); 374 ATF_TC_HEAD(fd_positive_file_v4, tc) 375 { 376 377 atf_tc_set_md_var(tc, "descr", 378 "Verify regular file as file descriptor support (IPv4)"); 379 } 380 ATF_TC_BODY(fd_positive_file_v4, tc) 381 { 382 383 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 384 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 385 386 fd_positive_file_test(AF_INET); 387 } 388 389 ATF_TC(fd_positive_file_v6); 390 ATF_TC_HEAD(fd_positive_file_v6, tc) 391 { 392 393 atf_tc_set_md_var(tc, "descr", 394 "Verify regular file as file descriptor support (IPv6)"); 395 } 396 ATF_TC_BODY(fd_positive_file_v6, tc) 397 { 398 399 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 400 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 401 402 fd_positive_file_test(AF_INET6); 403 } 404 405 static void 406 fd_positive_shm_test(int domain) 407 { 408 char *shm_pointer; 409 off_t offset; 410 size_t nbytes, pattern_size; 411 pid_t server_pid; 412 int client_sock, error, fd, port, server_sock; 413 414 pattern_size = strlen(DETERMINISTIC_PATTERN); 415 416 printf("pattern size: %zu\n", pattern_size); 417 418 fd = shm_open(SHM_ANON, O_RDWR|O_CREAT, 0600); 419 ATF_REQUIRE_MSG(fd != -1, "shm_open failed: %s", strerror(errno)); 420 ATF_REQUIRE_EQ_MSG(0, ftruncate(fd, pattern_size), 421 "ftruncate failed: %s", strerror(errno)); 422 shm_pointer = mmap(NULL, pattern_size, PROT_READ|PROT_WRITE, 423 MAP_SHARED, fd, 0); 424 ATF_REQUIRE_MSG(shm_pointer != MAP_FAILED, 425 "mmap failed: %s", strerror(errno)); 426 memcpy(shm_pointer, DETERMINISTIC_PATTERN, pattern_size); 427 ATF_REQUIRE_EQ_MSG(0, 428 memcmp(shm_pointer, DETERMINISTIC_PATTERN, pattern_size), 429 "memcmp showed data mismatch: '%s' != '%s'", 430 DETERMINISTIC_PATTERN, shm_pointer); 431 432 port = generate_random_port(__LINE__ + domain); 433 server_sock = setup_tcp_server(domain, port); 434 client_sock = setup_tcp_client(domain, port); 435 436 server_pid = atf_utils_fork(); 437 if (server_pid == 0) { 438 (void)close(client_sock); 439 server_cat(DESTINATION_FILE, server_sock, pattern_size); 440 _exit(0); 441 } else 442 (void)close(server_sock); 443 444 nbytes = 0; 445 offset = 0; 446 error = sendfile(fd, client_sock, offset, nbytes, NULL, NULL, 447 SF_FLAGS(0, 0)); 448 ATF_REQUIRE_EQ_MSG(0, error, "sendfile failed: %s", strerror(errno)); 449 (void)close(client_sock); 450 451 atf_utils_wait(server_pid, 0, "", ""); 452 verify_source_and_dest(DESTINATION_FILE, fd, offset, nbytes); 453 454 (void)munmap(shm_pointer, sizeof(DETERMINISTIC_PATTERN)); 455 (void)close(fd); 456 } 457 458 ATF_TC(fd_positive_shm_v4); 459 ATF_TC_HEAD(fd_positive_shm_v4, tc) 460 { 461 462 atf_tc_set_md_var(tc, "descr", 463 "Verify shared memory as file descriptor support (IPv4)"); 464 } 465 ATF_TC_BODY(fd_positive_shm_v4, tc) 466 { 467 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 468 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 469 470 fd_positive_shm_test(AF_INET); 471 } 472 473 ATF_TC(fd_positive_shm_v6); 474 ATF_TC_HEAD(fd_positive_shm_v6, tc) 475 { 476 477 atf_tc_set_md_var(tc, "descr", 478 "Verify shared memory as file descriptor support (IPv6))"); 479 } 480 ATF_TC_BODY(fd_positive_shm_v6, tc) 481 { 482 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 483 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 484 485 fd_positive_shm_test(AF_INET6); 486 } 487 488 static void 489 fd_negative_bad_fd_test(int domain) 490 { 491 int client_sock, error, fd, port, server_sock; 492 493 port = generate_random_port(__LINE__ + domain); 494 server_sock = setup_tcp_server(domain, port); 495 client_sock = setup_tcp_client(domain, port); 496 497 fd = -1; 498 499 error = sendfile(fd, client_sock, 0, 0, NULL, NULL, SF_FLAGS(0, 0)); 500 ATF_REQUIRE_ERRNO(EBADF, error == -1); 501 502 (void)close(client_sock); 503 (void)close(server_sock); 504 } 505 506 ATF_TC(fd_negative_bad_fd_v4); 507 ATF_TC_HEAD(fd_negative_bad_fd_v4, tc) 508 { 509 510 atf_tc_set_md_var(tc, "descr", 511 "Verify bad file descriptor returns EBADF (IPv4)"); 512 } 513 ATF_TC_BODY(fd_negative_bad_fd_v4, tc) 514 { 515 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 516 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 517 518 fd_negative_bad_fd_test(AF_INET); 519 } 520 521 ATF_TC(fd_negative_bad_fd_v6); 522 ATF_TC_HEAD(fd_negative_bad_fd_v6, tc) 523 { 524 525 atf_tc_set_md_var(tc, "descr", 526 "Verify bad file descriptor returns EBADF (IPv6)"); 527 } 528 ATF_TC_BODY(fd_negative_bad_fd_v6, tc) 529 { 530 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 531 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 532 533 fd_negative_bad_fd_test(AF_INET6); 534 } 535 536 static void 537 flags_test(int domain) 538 { 539 off_t offset; 540 size_t nbytes, pattern_size; 541 int client_sock, error, fd, i, port, server_sock; 542 pid_t server_pid; 543 int16_t number_pages = 10; 544 545 pattern_size = strlen(DETERMINISTIC_PATTERN); 546 547 struct testcase { 548 int16_t readahead_pages, flags; 549 } testcases[] = { 550 /* This is covered in `:fd_positive_file` */ 551 #if 0 552 { 553 .readahead_pages = 0, 554 .flags = 0 555 }, 556 #endif 557 { 558 .readahead_pages = 0, 559 .flags = SF_NOCACHE 560 }, 561 #ifdef SF_USER_READAHEAD 562 { 563 .readahead_pages = 0, 564 .flags = SF_NOCACHE|SF_USER_READAHEAD 565 }, 566 { 567 .readahead_pages = 0, 568 .flags = SF_USER_READAHEAD 569 }, 570 #endif 571 { 572 .readahead_pages = number_pages, 573 .flags = 0 574 }, 575 { 576 .readahead_pages = number_pages, 577 .flags = SF_NOCACHE 578 }, 579 #ifdef SF_USER_READAHEAD 580 { 581 .readahead_pages = number_pages, 582 .flags = SF_NOCACHE|SF_USER_READAHEAD 583 }, 584 #endif 585 { 586 .readahead_pages = number_pages, 587 .flags = SF_NOCACHE 588 }, 589 { 590 .readahead_pages = number_pages, 591 .flags = SF_NODISKIO 592 } 593 }; 594 595 atf_utils_create_file(SOURCE_FILE, "%s", DETERMINISTIC_PATTERN); 596 for (i = 0; i < nitems(testcases); i++) { 597 fd = open(SOURCE_FILE, O_RDONLY); 598 ATF_REQUIRE_MSG(fd != -1, "open failed: %s", strerror(errno)); 599 600 port = generate_random_port(i * __LINE__ + domain); 601 server_sock = setup_tcp_server(domain, port); 602 client_sock = setup_tcp_client(domain, port); 603 604 server_pid = atf_utils_fork(); 605 if (server_pid == 0) { 606 (void)close(client_sock); 607 server_cat(DESTINATION_FILE, server_sock, pattern_size); 608 _exit(0); 609 } else 610 (void)close(server_sock); 611 612 nbytes = 0; 613 offset = 0; 614 error = sendfile(fd, client_sock, offset, nbytes, NULL, NULL, 615 SF_FLAGS(testcases[i].readahead_pages, testcases[i].flags)); 616 ATF_CHECK_EQ_MSG(error, 0, "sendfile testcase #%d failed: %s", 617 i, strerror(errno)); 618 (void)close(client_sock); 619 620 atf_utils_wait(server_pid, 0, "", ""); 621 verify_source_and_dest(DESTINATION_FILE, fd, offset, nbytes); 622 623 (void)close(fd); 624 } 625 } 626 627 ATF_TC(flags_v4); 628 ATF_TC_HEAD(flags_v4, tc) 629 { 630 631 atf_tc_set_md_var(tc, "descr", "Verify flags functionality (IPv4)"); 632 } 633 ATF_TC_BODY(flags_v4, tc) 634 { 635 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 636 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 637 638 flags_test(AF_INET); 639 } 640 641 ATF_TC(flags_v6); 642 ATF_TC_HEAD(flags_v6, tc) 643 { 644 645 atf_tc_set_md_var(tc, "descr", "Verify flags functionality (IPv6)"); 646 } 647 ATF_TC_BODY(flags_v6, tc) 648 { 649 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 650 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 651 652 flags_test(AF_INET6); 653 } 654 655 static void 656 hdtr_positive_test(int domain) 657 { 658 struct iovec headers[1], trailers[1]; 659 struct testcase { 660 bool include_headers, include_trailers; 661 } testcases[] = { 662 /* This is covered in `:fd_positive_file` */ 663 #if 0 664 { 665 .include_headers = false, 666 .include_trailers = false 667 }, 668 #endif 669 { 670 .include_headers = true, 671 .include_trailers = false 672 }, 673 { 674 .include_headers = false, 675 .include_trailers = true 676 }, 677 { 678 .include_headers = true, 679 .include_trailers = true 680 } 681 }; 682 off_t offset; 683 size_t nbytes; 684 int client_sock, error, fd, fd2, i, port, rc, server_sock; 685 pid_t server_pid; 686 687 headers[0].iov_base = "This is a header"; 688 headers[0].iov_len = strlen(headers[0].iov_base); 689 trailers[0].iov_base = "This is a trailer"; 690 trailers[0].iov_len = strlen(trailers[0].iov_base); 691 offset = 0; 692 nbytes = 0; 693 694 for (i = 0; i < nitems(testcases); i++) { 695 struct sf_hdtr hdtr; 696 char *pattern; 697 698 if (testcases[i].include_headers) { 699 hdtr.headers = headers; 700 hdtr.hdr_cnt = nitems(headers); 701 } else { 702 hdtr.headers = NULL; 703 hdtr.hdr_cnt = 0; 704 } 705 706 if (testcases[i].include_trailers) { 707 hdtr.trailers = trailers; 708 hdtr.trl_cnt = nitems(trailers); 709 } else { 710 hdtr.trailers = NULL; 711 hdtr.trl_cnt = 0; 712 } 713 714 port = generate_random_port(i * __LINE__ + domain); 715 server_sock = setup_tcp_server(domain, port); 716 client_sock = setup_tcp_client(domain, port); 717 718 rc = asprintf(&pattern, "%s%s%s", 719 testcases[i].include_headers ? (char *)headers[0].iov_base : "", 720 DETERMINISTIC_PATTERN, 721 testcases[i].include_trailers ? (char *)trailers[0].iov_base : ""); 722 ATF_REQUIRE_MSG(rc != -1, "asprintf failed: %s", strerror(errno)); 723 724 atf_utils_create_file(SOURCE_FILE ".full", "%s", pattern); 725 atf_utils_create_file(SOURCE_FILE, "%s", DETERMINISTIC_PATTERN); 726 727 fd = open(SOURCE_FILE, O_RDONLY); 728 ATF_REQUIRE_MSG(fd != -1, "open failed: %s", strerror(errno)); 729 730 fd2 = open(SOURCE_FILE ".full", O_RDONLY); 731 ATF_REQUIRE_MSG(fd2 != -1, "open failed: %s", strerror(errno)); 732 733 server_pid = atf_utils_fork(); 734 if (server_pid == 0) { 735 (void)close(client_sock); 736 server_cat(DESTINATION_FILE, server_sock, 737 strlen(pattern)); 738 _exit(0); 739 } else 740 (void)close(server_sock); 741 742 error = sendfile(fd, client_sock, offset, nbytes, &hdtr, 743 NULL, SF_FLAGS(0, 0)); 744 ATF_CHECK_EQ_MSG(error, 0, "sendfile testcase #%d failed: %s", 745 i, strerror(errno)); 746 (void)close(client_sock); 747 748 atf_utils_wait(server_pid, 0, "", ""); 749 verify_source_and_dest(DESTINATION_FILE, fd2, offset, nbytes); 750 751 (void)close(fd); 752 (void)close(fd2); 753 free(pattern); 754 pattern = NULL; 755 } 756 } 757 758 ATF_TC(hdtr_positive_v4); 759 ATF_TC_HEAD(hdtr_positive_v4, tc) 760 { 761 762 atf_tc_set_md_var(tc, "descr", 763 "Verify positive hdtr functionality (IPv4)"); 764 } 765 ATF_TC_BODY(hdtr_positive_v4, tc) 766 { 767 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 768 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 769 770 hdtr_positive_test(AF_INET); 771 } 772 773 ATF_TC(hdtr_positive_v6); 774 ATF_TC_HEAD(hdtr_positive_v6, tc) 775 { 776 777 atf_tc_set_md_var(tc, "descr", 778 "Verify positive hdtr functionality (IPv6)"); 779 } 780 ATF_TC_BODY(hdtr_positive_v6, tc) 781 { 782 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 783 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 784 785 hdtr_positive_test(AF_INET); 786 } 787 788 static void 789 hdtr_negative_bad_pointers_test(int domain) 790 { 791 int client_sock, error, fd, port, server_sock; 792 struct sf_hdtr *hdtr1, hdtr2, hdtr3; 793 794 port = generate_random_port(__LINE__ + domain); 795 796 hdtr1 = (struct sf_hdtr*)-1; 797 798 memset(&hdtr2, 0, sizeof(hdtr2)); 799 hdtr2.hdr_cnt = 1; 800 hdtr2.headers = (struct iovec*)-1; 801 802 memset(&hdtr3, 0, sizeof(hdtr3)); 803 hdtr3.trl_cnt = 1; 804 hdtr3.trailers = (struct iovec*)-1; 805 806 fd = open(SOURCE_FILE, O_CREAT|O_RDWR, 0600); 807 ATF_REQUIRE_MSG(fd != -1, "open failed: %s", strerror(errno)); 808 809 server_sock = setup_tcp_server(domain, port); 810 client_sock = setup_tcp_client(domain, port); 811 812 error = sendfile(fd, client_sock, 0, 0, hdtr1, NULL, SF_FLAGS(0, 0)); 813 ATF_CHECK_ERRNO(EFAULT, error == -1); 814 815 error = sendfile(fd, client_sock, 0, 0, &hdtr2, NULL, SF_FLAGS(0, 0)); 816 ATF_CHECK_ERRNO(EFAULT, error == -1); 817 818 error = sendfile(fd, client_sock, 0, 0, &hdtr3, NULL, SF_FLAGS(0, 0)); 819 ATF_CHECK_ERRNO(EFAULT, error == -1); 820 821 (void)close(fd); 822 (void)close(client_sock); 823 (void)close(server_sock); 824 } 825 826 ATF_TC(hdtr_negative_bad_pointers_v4); 827 ATF_TC_HEAD(hdtr_negative_bad_pointers_v4, tc) 828 { 829 830 atf_tc_set_md_var(tc, "descr", 831 "Verify that bad pointers for hdtr storage result in EFAULT (IPv4)"); 832 } 833 ATF_TC_BODY(hdtr_negative_bad_pointers_v4, tc) 834 { 835 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 836 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 837 838 hdtr_negative_bad_pointers_test(AF_INET); 839 } 840 841 ATF_TC(hdtr_negative_bad_pointers_v6); 842 ATF_TC_HEAD(hdtr_negative_bad_pointers_v6, tc) 843 { 844 845 atf_tc_set_md_var(tc, "descr", 846 "Verify that bad pointers for hdtr storage result in EFAULT (IPv6)"); 847 } 848 ATF_TC_BODY(hdtr_negative_bad_pointers_v6, tc) 849 { 850 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 851 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 852 853 hdtr_negative_bad_pointers_test(AF_INET6); 854 } 855 856 static void 857 offset_negative_value_less_than_zero_test(int domain) 858 { 859 int client_sock, error, fd, port, server_sock; 860 861 port = generate_random_port(__LINE__ + domain); 862 server_sock = setup_tcp_server(domain, port); 863 client_sock = setup_tcp_client(domain, port); 864 865 fd = open(SOURCE_FILE, O_CREAT|O_RDWR, 0600); 866 ATF_REQUIRE_MSG(fd != -1, "open failed: %s", strerror(errno)); 867 868 error = sendfile(fd, client_sock, -1, 0, NULL, NULL, SF_FLAGS(0, 0)); 869 ATF_REQUIRE_ERRNO(EINVAL, error == -1); 870 871 (void)close(fd); 872 (void)close(client_sock); 873 (void)close(server_sock); 874 } 875 876 ATF_TC(offset_negative_value_less_than_zero_v4); 877 ATF_TC_HEAD(offset_negative_value_less_than_zero_v4, tc) 878 { 879 880 atf_tc_set_md_var(tc, "descr", 881 "Verify that a negative offset results in EINVAL (IPv4)"); 882 } 883 ATF_TC_BODY(offset_negative_value_less_than_zero_v4, tc) 884 { 885 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 886 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 887 888 offset_negative_value_less_than_zero_test(AF_INET); 889 } 890 891 ATF_TC(offset_negative_value_less_than_zero_v6); 892 ATF_TC_HEAD(offset_negative_value_less_than_zero_v6, tc) 893 { 894 895 atf_tc_set_md_var(tc, "descr", 896 "Verify that a negative offset results in EINVAL (IPv6)"); 897 } 898 ATF_TC_BODY(offset_negative_value_less_than_zero_v6, tc) 899 { 900 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 901 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 902 903 offset_negative_value_less_than_zero_test(AF_INET6); 904 } 905 906 static void 907 sbytes_positive_test(int domain) 908 { 909 size_t pattern_size = strlen(DETERMINISTIC_PATTERN); 910 off_t sbytes; 911 int client_sock, error, fd, port, server_sock; 912 913 port = generate_random_port(__LINE__ + domain); 914 server_sock = setup_tcp_server(domain, port); 915 client_sock = setup_tcp_client(domain, port); 916 917 atf_utils_create_file(SOURCE_FILE, "%s", DETERMINISTIC_PATTERN); 918 fd = open(SOURCE_FILE, O_RDONLY); 919 ATF_REQUIRE_MSG(fd != -1, "open failed: %s", strerror(errno)); 920 921 error = sendfile(fd, client_sock, 0, 0, NULL, &sbytes, SF_FLAGS(0, 0)); 922 ATF_CHECK_EQ_MSG(error, 0, "sendfile failed: %s", strerror(errno)); 923 924 (void)close(fd); 925 (void)close(client_sock); 926 (void)close(server_sock); 927 928 ATF_CHECK_EQ_MSG(pattern_size, sbytes, 929 "the value returned by sbytes does not match the expected pattern " 930 "size"); 931 } 932 933 ATF_TC(sbytes_positive_v4); 934 ATF_TC_HEAD(sbytes_positive_v4, tc) 935 { 936 937 atf_tc_set_md_var(tc, "descr", 938 "Verify positive `sbytes` functionality (IPv4)"); 939 } 940 ATF_TC_BODY(sbytes_positive_v4, tc) 941 { 942 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 943 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 944 945 sbytes_positive_test(AF_INET); 946 } 947 948 ATF_TC(sbytes_positive_v6); 949 ATF_TC_HEAD(sbytes_positive_v6, tc) 950 { 951 952 atf_tc_set_md_var(tc, "descr", 953 "Verify positive `sbytes` functionality (IPv6)"); 954 } 955 ATF_TC_BODY(sbytes_positive_v6, tc) 956 { 957 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 958 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 959 960 sbytes_positive_test(AF_INET6); 961 } 962 963 static void 964 sbytes_negative_test(int domain) 965 { 966 off_t *sbytes_p = (off_t*)-1; 967 int client_sock, error, fd, port, server_sock; 968 969 port = generate_random_port(__LINE__ + domain); 970 server_sock = setup_tcp_server(domain, port); 971 client_sock = setup_tcp_client(domain, port); 972 973 atf_utils_create_file(SOURCE_FILE, "%s", DETERMINISTIC_PATTERN); 974 fd = open(SOURCE_FILE, O_RDONLY); 975 ATF_REQUIRE_MSG(fd != -1, "open failed: %s", strerror(errno)); 976 977 atf_tc_expect_fail( 978 "bug 232210: EFAULT assert fails because copyout(9) call is not checked"); 979 980 error = sendfile(fd, client_sock, 0, 0, NULL, sbytes_p, SF_FLAGS(0, 0)); 981 ATF_REQUIRE_ERRNO(EFAULT, error == -1); 982 983 (void)close(fd); 984 (void)close(client_sock); 985 (void)close(server_sock); 986 } 987 988 ATF_TC(sbytes_negative_v4); 989 ATF_TC_HEAD(sbytes_negative_v4, tc) 990 { 991 992 atf_tc_set_md_var(tc, "descr", 993 "Verify negative `sbytes` functionality (IPv4)"); 994 } 995 ATF_TC_BODY(sbytes_negative_v4, tc) 996 { 997 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 998 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 999 1000 sbytes_negative_test(AF_INET); 1001 } 1002 1003 ATF_TC(sbytes_negative_v6); 1004 ATF_TC_HEAD(sbytes_negative_v6, tc) 1005 { 1006 1007 atf_tc_set_md_var(tc, "descr", 1008 "Verify negative `sbytes` functionality (IPv6)"); 1009 } 1010 ATF_TC_BODY(sbytes_negative_v6, tc) 1011 { 1012 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 1013 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 1014 1015 sbytes_negative_test(AF_INET6); 1016 } 1017 1018 static void 1019 s_negative_not_connected_socket_test(int domain) 1020 { 1021 int client_sock, error, fd, port; 1022 1023 port = generate_random_port(__LINE__ + domain); 1024 client_sock = setup_tcp_server(domain, port); 1025 1026 fd = open(SOURCE_FILE, O_CREAT|O_RDWR, 0600); 1027 ATF_REQUIRE_MSG(fd != -1, "open failed: %s", strerror(errno)); 1028 1029 error = sendfile(fd, client_sock, 0, 0, NULL, NULL, SF_FLAGS(0, 0)); 1030 ATF_REQUIRE_ERRNO(ENOTCONN, error == -1); 1031 1032 (void)close(fd); 1033 (void)close(client_sock); 1034 } 1035 1036 ATF_TC(s_negative_not_connected_socket_v4); 1037 ATF_TC_HEAD(s_negative_not_connected_socket_v4, tc) 1038 { 1039 1040 atf_tc_set_md_var(tc, "descr", 1041 "Verify that a non-connected SOCK_STREAM socket results in ENOTCONN (IPv4)"); 1042 } 1043 1044 ATF_TC_BODY(s_negative_not_connected_socket_v4, tc) 1045 { 1046 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 1047 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 1048 1049 s_negative_not_connected_socket_test(AF_INET); 1050 } 1051 1052 ATF_TC(s_negative_not_connected_socket_v6); 1053 ATF_TC_HEAD(s_negative_not_connected_socket_v6, tc) 1054 { 1055 1056 atf_tc_set_md_var(tc, "descr", 1057 "Verify that a non-connected SOCK_STREAM socket results in ENOTCONN (IPv6)"); 1058 } 1059 1060 ATF_TC_BODY(s_negative_not_connected_socket_v6, tc) 1061 { 1062 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 1063 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 1064 1065 s_negative_not_connected_socket_test(AF_INET6); 1066 } 1067 1068 ATF_TC(s_negative_not_descriptor); 1069 ATF_TC_HEAD(s_negative_not_descriptor, tc) 1070 { 1071 1072 atf_tc_set_md_var(tc, "descr", 1073 "Verify that an invalid file descriptor, e.g., -1, fails with EBADF"); 1074 } 1075 1076 ATF_TC_BODY(s_negative_not_descriptor, tc) 1077 { 1078 int client_sock, error, fd; 1079 1080 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 1081 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 1082 1083 client_sock = -1; 1084 1085 fd = open(SOURCE_FILE, O_CREAT|O_RDWR, 0600); 1086 ATF_REQUIRE_MSG(fd != -1, "open failed: %s", strerror(errno)); 1087 1088 error = sendfile(fd, client_sock, 0, 0, NULL, NULL, SF_FLAGS(0, 0)); 1089 ATF_REQUIRE_ERRNO(EBADF, error == -1); 1090 1091 (void)close(fd); 1092 } 1093 1094 ATF_TC(s_negative_not_socket_file_descriptor); 1095 ATF_TC_HEAD(s_negative_not_socket_file_descriptor, tc) 1096 { 1097 1098 atf_tc_set_md_var(tc, "descr", 1099 "Verify that a non-socket file descriptor fails with ENOTSOCK"); 1100 } 1101 1102 ATF_TC_BODY(s_negative_not_socket_file_descriptor, tc) 1103 { 1104 int client_sock, error, fd; 1105 1106 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 1107 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 1108 1109 fd = open(SOURCE_FILE, O_CREAT|O_RDWR, 0600); 1110 ATF_REQUIRE_MSG(fd != -1, "open failed: %s", strerror(errno)); 1111 1112 client_sock = open(_PATH_DEVNULL, O_WRONLY); 1113 ATF_REQUIRE_MSG(fd != -1, "open failed: %s", strerror(errno)); 1114 1115 error = sendfile(fd, client_sock, 0, 0, NULL, NULL, SF_FLAGS(0, 0)); 1116 ATF_REQUIRE_ERRNO(ENOTSOCK, error == -1); 1117 1118 (void)close(fd); 1119 (void)close(client_sock); 1120 } 1121 1122 static void 1123 s_negative_udp_socket_test(int domain) 1124 { 1125 int client_sock, error, fd, port; 1126 1127 port = generate_random_port(__LINE__ + domain); 1128 client_sock = setup_client(domain, SOCK_DGRAM, port); 1129 1130 fd = open(SOURCE_FILE, O_CREAT|O_RDWR, 0600); 1131 ATF_REQUIRE_MSG(fd != -1, "open failed: %s", strerror(errno)); 1132 1133 error = sendfile(fd, client_sock, 0, 0, NULL, NULL, SF_FLAGS(0, 0)); 1134 ATF_REQUIRE_ERRNO(EINVAL, error == -1); 1135 1136 (void)close(fd); 1137 (void)close(client_sock); 1138 } 1139 1140 ATF_TC(s_negative_udp_socket_v4); 1141 ATF_TC_HEAD(s_negative_udp_socket_v4, tc) 1142 { 1143 1144 atf_tc_set_md_var(tc, "descr", 1145 "Verify that a non-SOCK_STREAM type socket results in EINVAL (IPv4)"); 1146 } 1147 ATF_TC_BODY(s_negative_udp_socket_v4, tc) 1148 { 1149 1150 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 1151 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 1152 1153 s_negative_udp_socket_test(AF_INET); 1154 } 1155 1156 ATF_TC(s_negative_udp_socket_v6); 1157 ATF_TC_HEAD(s_negative_udp_socket_v6, tc) 1158 { 1159 1160 atf_tc_set_md_var(tc, "descr", 1161 "Verify that a non-SOCK_STREAM type socket results in EINVAL (IPv6)"); 1162 } 1163 ATF_TC_BODY(s_negative_udp_socket_v6, tc) 1164 { 1165 1166 if (atf_tc_get_config_var_as_bool_wd(tc, "qemu", false)) 1167 atf_tc_skip("Sendfile(4) unimplemented. https://github.com/qemu-bsd-user/qemu-bsd-user/issues/25"); 1168 1169 s_negative_udp_socket_test(AF_INET6); 1170 } 1171 1172 ATF_TP_ADD_TCS(tp) 1173 { 1174 1175 ATF_TP_ADD_TC(tp, fd_positive_file_v4); 1176 ATF_TP_ADD_TC(tp, fd_positive_file_v6); 1177 ATF_TP_ADD_TC(tp, fd_positive_shm_v4); 1178 ATF_TP_ADD_TC(tp, fd_positive_shm_v6); 1179 ATF_TP_ADD_TC(tp, fd_negative_bad_fd_v4); 1180 ATF_TP_ADD_TC(tp, fd_negative_bad_fd_v6); 1181 ATF_TP_ADD_TC(tp, flags_v4); 1182 ATF_TP_ADD_TC(tp, flags_v6); 1183 /* 1184 * TODO: the negative case for SF_NODISKIO (returns EBUSY if file in 1185 * use) is not covered yet. 1186 * 1187 * Need to lock a file in a subprocess in write mode, then try and 1188 * send the data in read mode with sendfile. 1189 * 1190 * This should work with FFS/UFS, but there are no guarantees about 1191 * other filesystem implementations of sendfile(2), e.g., ZFS. 1192 */ 1193 ATF_TP_ADD_TC(tp, hdtr_positive_v4); 1194 ATF_TP_ADD_TC(tp, hdtr_positive_v6); 1195 ATF_TP_ADD_TC(tp, hdtr_negative_bad_pointers_v4); 1196 ATF_TP_ADD_TC(tp, hdtr_negative_bad_pointers_v6); 1197 ATF_TP_ADD_TC(tp, offset_negative_value_less_than_zero_v4); 1198 ATF_TP_ADD_TC(tp, offset_negative_value_less_than_zero_v6); 1199 ATF_TP_ADD_TC(tp, sbytes_positive_v4); 1200 ATF_TP_ADD_TC(tp, sbytes_positive_v6); 1201 ATF_TP_ADD_TC(tp, sbytes_negative_v4); 1202 ATF_TP_ADD_TC(tp, sbytes_negative_v6); 1203 ATF_TP_ADD_TC(tp, s_negative_not_connected_socket_v4); 1204 ATF_TP_ADD_TC(tp, s_negative_not_connected_socket_v6); 1205 ATF_TP_ADD_TC(tp, s_negative_not_descriptor); 1206 ATF_TP_ADD_TC(tp, s_negative_not_socket_file_descriptor); 1207 ATF_TP_ADD_TC(tp, s_negative_udp_socket_v4); 1208 ATF_TP_ADD_TC(tp, s_negative_udp_socket_v6); 1209 1210 return (atf_no_error()); 1211 } 1212