1*bbb1b6f9SApple OSS Distributions #include <darwintest.h> 2*bbb1b6f9SApple OSS Distributions 3*bbb1b6f9SApple OSS Distributions #include <errno.h> 4*bbb1b6f9SApple OSS Distributions #include <fcntl.h> 5*bbb1b6f9SApple OSS Distributions #include <signal.h> 6*bbb1b6f9SApple OSS Distributions #include <spawn.h> 7*bbb1b6f9SApple OSS Distributions #include <spawn_private.h> 8*bbb1b6f9SApple OSS Distributions #include <stdbool.h> 9*bbb1b6f9SApple OSS Distributions #include <stdint.h> 10*bbb1b6f9SApple OSS Distributions #include <stdio.h> 11*bbb1b6f9SApple OSS Distributions #include <stdlib.h> 12*bbb1b6f9SApple OSS Distributions #include <string.h> 13*bbb1b6f9SApple OSS Distributions #include <sys/spawn_internal.h> 14*bbb1b6f9SApple OSS Distributions #include <sys/sysctl.h> 15*bbb1b6f9SApple OSS Distributions #include <sys/syslimits.h> 16*bbb1b6f9SApple OSS Distributions #include <sysexits.h> 17*bbb1b6f9SApple OSS Distributions #include <unistd.h> 18*bbb1b6f9SApple OSS Distributions 19*bbb1b6f9SApple OSS Distributions T_GLOBAL_META(T_META_RUN_CONCURRENTLY(true)); 20*bbb1b6f9SApple OSS Distributions 21*bbb1b6f9SApple OSS Distributions /* TEST_PATH needs to be something that exists, but is not the cwd */ 22*bbb1b6f9SApple OSS Distributions #define TEST_PATH "/System/Library/Caches" 23*bbb1b6f9SApple OSS Distributions 24*bbb1b6f9SApple OSS Distributions T_DECL(posix_spawn_file_actions_addchdir, "Check posix_spawn_file_actions_addchdir", 25*bbb1b6f9SApple OSS Distributions T_META_ASROOT(true), T_META_TAG_VM_PREFERRED) 26*bbb1b6f9SApple OSS Distributions { 27*bbb1b6f9SApple OSS Distributions posix_spawn_file_actions_t file_actions; 28*bbb1b6f9SApple OSS Distributions int ret; 29*bbb1b6f9SApple OSS Distributions 30*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_init(&file_actions); 31*bbb1b6f9SApple OSS Distributions T_QUIET; 32*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "posix_spawn_file_actions_init"); 33*bbb1b6f9SApple OSS Distributions 34*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_addchdir(&file_actions, TEST_PATH); 35*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "posix_spawn_file_actions_addchdir"); 36*bbb1b6f9SApple OSS Distributions 37*bbb1b6f9SApple OSS Distributions char * const prog = "/bin/sh"; 38*bbb1b6f9SApple OSS Distributions char * const argv_child[] = { prog, 39*bbb1b6f9SApple OSS Distributions "-c", 40*bbb1b6f9SApple OSS Distributions "test $(pwd) = \"" TEST_PATH "\"", 41*bbb1b6f9SApple OSS Distributions NULL, }; 42*bbb1b6f9SApple OSS Distributions pid_t child_pid; 43*bbb1b6f9SApple OSS Distributions extern char **environ; 44*bbb1b6f9SApple OSS Distributions 45*bbb1b6f9SApple OSS Distributions ret = posix_spawn(&child_pid, prog, &file_actions, NULL, argv_child, environ); 46*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "posix_spawn"); 47*bbb1b6f9SApple OSS Distributions 48*bbb1b6f9SApple OSS Distributions T_LOG("parent: spawned child with pid %d\n", child_pid); 49*bbb1b6f9SApple OSS Distributions 50*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_destroy(&file_actions); 51*bbb1b6f9SApple OSS Distributions T_QUIET; 52*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "posix_spawn_file_actions_destroy"); 53*bbb1b6f9SApple OSS Distributions 54*bbb1b6f9SApple OSS Distributions T_LOG("parent: waiting for child process\n"); 55*bbb1b6f9SApple OSS Distributions 56*bbb1b6f9SApple OSS Distributions int status = 0; 57*bbb1b6f9SApple OSS Distributions int waitpid_result = waitpid(child_pid, &status, 0); 58*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(waitpid_result, "waitpid"); 59*bbb1b6f9SApple OSS Distributions T_ASSERT_EQ(waitpid_result, child_pid, "waitpid should return child we spawned"); 60*bbb1b6f9SApple OSS Distributions T_ASSERT_EQ(WIFEXITED(status), 1, "child should have exited normally"); 61*bbb1b6f9SApple OSS Distributions T_ASSERT_EQ(WEXITSTATUS(status), EX_OK, "child should have exited with success"); 62*bbb1b6f9SApple OSS Distributions } 63*bbb1b6f9SApple OSS Distributions 64*bbb1b6f9SApple OSS Distributions T_DECL(posix_spawn_file_actions_addchdir_errors, "Check posix_spawn_file_actions_addchdir errors", 65*bbb1b6f9SApple OSS Distributions T_META_ASROOT(true), T_META_TAG_VM_PREFERRED) 66*bbb1b6f9SApple OSS Distributions { 67*bbb1b6f9SApple OSS Distributions char longpath[PATH_MAX + 1]; 68*bbb1b6f9SApple OSS Distributions posix_spawn_file_actions_t file_actions; 69*bbb1b6f9SApple OSS Distributions int ret; 70*bbb1b6f9SApple OSS Distributions 71*bbb1b6f9SApple OSS Distributions memset(longpath, 'a', PATH_MAX); 72*bbb1b6f9SApple OSS Distributions longpath[PATH_MAX] = '\0'; 73*bbb1b6f9SApple OSS Distributions 74*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_init(&file_actions); 75*bbb1b6f9SApple OSS Distributions T_QUIET; 76*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "posix_spawn_file_actions_init"); 77*bbb1b6f9SApple OSS Distributions 78*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_addchdir(NULL, "/"); 79*bbb1b6f9SApple OSS Distributions T_ASSERT_EQ(ret, EINVAL, "NULL *file_actions returns EINVAL"); 80*bbb1b6f9SApple OSS Distributions 81*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_addchdir(&file_actions, longpath); 82*bbb1b6f9SApple OSS Distributions T_ASSERT_EQ(ret, ENAMETOOLONG, "Path longer than PATH_MAX returns ENAMETOOLONG"); 83*bbb1b6f9SApple OSS Distributions 84*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_destroy(&file_actions); 85*bbb1b6f9SApple OSS Distributions T_QUIET; 86*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "posix_spawn_file_actions_destroy"); 87*bbb1b6f9SApple OSS Distributions } 88*bbb1b6f9SApple OSS Distributions 89*bbb1b6f9SApple OSS Distributions T_DECL(posix_spawn_file_actions_addfchdir, "Check posix_spawn_file_actions_addfchdir", 90*bbb1b6f9SApple OSS Distributions T_META_ASROOT(true), T_META_TAG_VM_PREFERRED) 91*bbb1b6f9SApple OSS Distributions { 92*bbb1b6f9SApple OSS Distributions posix_spawn_file_actions_t file_actions; 93*bbb1b6f9SApple OSS Distributions int ret; 94*bbb1b6f9SApple OSS Distributions int test_fd; 95*bbb1b6f9SApple OSS Distributions 96*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_init(&file_actions); 97*bbb1b6f9SApple OSS Distributions T_QUIET; 98*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "posix_spawn_file_actions_init"); 99*bbb1b6f9SApple OSS Distributions 100*bbb1b6f9SApple OSS Distributions test_fd = open(TEST_PATH, O_RDONLY | O_CLOEXEC); 101*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(test_fd, "open " TEST_PATH); 102*bbb1b6f9SApple OSS Distributions 103*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_addfchdir(&file_actions, test_fd); 104*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "posix_spawn_file_actions_addfchdir"); 105*bbb1b6f9SApple OSS Distributions 106*bbb1b6f9SApple OSS Distributions char * const prog = "/bin/sh"; 107*bbb1b6f9SApple OSS Distributions char * const argv_child[] = { prog, 108*bbb1b6f9SApple OSS Distributions "-c", 109*bbb1b6f9SApple OSS Distributions "test $(pwd) = \"" TEST_PATH "\"", 110*bbb1b6f9SApple OSS Distributions NULL, }; 111*bbb1b6f9SApple OSS Distributions pid_t child_pid; 112*bbb1b6f9SApple OSS Distributions extern char **environ; 113*bbb1b6f9SApple OSS Distributions 114*bbb1b6f9SApple OSS Distributions ret = posix_spawn(&child_pid, prog, &file_actions, NULL, argv_child, environ); 115*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "posix_spawn"); 116*bbb1b6f9SApple OSS Distributions 117*bbb1b6f9SApple OSS Distributions T_LOG("parent: spawned child with pid %d\n", child_pid); 118*bbb1b6f9SApple OSS Distributions 119*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_destroy(&file_actions); 120*bbb1b6f9SApple OSS Distributions T_QUIET; 121*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "posix_spawn_file_actions_destroy"); 122*bbb1b6f9SApple OSS Distributions 123*bbb1b6f9SApple OSS Distributions T_LOG("parent: waiting for child process\n"); 124*bbb1b6f9SApple OSS Distributions 125*bbb1b6f9SApple OSS Distributions int status = 0; 126*bbb1b6f9SApple OSS Distributions int waitpid_result = waitpid(child_pid, &status, 0); 127*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(waitpid_result, "waitpid"); 128*bbb1b6f9SApple OSS Distributions T_ASSERT_EQ(waitpid_result, child_pid, "waitpid should return child we spawned"); 129*bbb1b6f9SApple OSS Distributions T_ASSERT_EQ(WIFEXITED(status), 1, "child should have exited normally"); 130*bbb1b6f9SApple OSS Distributions T_ASSERT_EQ(WEXITSTATUS(status), EX_OK, "child should have exited with success"); 131*bbb1b6f9SApple OSS Distributions 132*bbb1b6f9SApple OSS Distributions ret = close(test_fd); 133*bbb1b6f9SApple OSS Distributions T_QUIET; 134*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "close test fd"); 135*bbb1b6f9SApple OSS Distributions } 136*bbb1b6f9SApple OSS Distributions 137*bbb1b6f9SApple OSS Distributions T_DECL(posix_spawn_file_actions_addfchdir_errors, "Check posix_spawn_file_actions_addfchdir errors", 138*bbb1b6f9SApple OSS Distributions T_META_ASROOT(true), T_META_TAG_VM_PREFERRED) 139*bbb1b6f9SApple OSS Distributions { 140*bbb1b6f9SApple OSS Distributions posix_spawn_file_actions_t file_actions; 141*bbb1b6f9SApple OSS Distributions int ret; 142*bbb1b6f9SApple OSS Distributions 143*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_init(&file_actions); 144*bbb1b6f9SApple OSS Distributions T_QUIET; 145*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "posix_spawn_file_actions_init"); 146*bbb1b6f9SApple OSS Distributions 147*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_addfchdir(NULL, 0); 148*bbb1b6f9SApple OSS Distributions T_ASSERT_EQ(ret, EINVAL, "NULL *file_actions returns EINVAL"); 149*bbb1b6f9SApple OSS Distributions 150*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_addfchdir(&file_actions, -1); 151*bbb1b6f9SApple OSS Distributions T_ASSERT_EQ(ret, EBADF, "-1 file descriptor returns EBADF"); 152*bbb1b6f9SApple OSS Distributions 153*bbb1b6f9SApple OSS Distributions ret = posix_spawn_file_actions_destroy(&file_actions); 154*bbb1b6f9SApple OSS Distributions T_QUIET; 155*bbb1b6f9SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "posix_spawn_file_actions_destroy"); 156*bbb1b6f9SApple OSS Distributions } 157