xref: /xnu-8796.101.5/tests/stackshot_spawn_exit_stress.c (revision aca3beaa3dfbd42498b42c5e5ce20a938e6554e5)
1*aca3beaaSApple OSS Distributions #include <stdio.h>
2*aca3beaaSApple OSS Distributions #include <stdlib.h>
3*aca3beaaSApple OSS Distributions 
4*aca3beaaSApple OSS Distributions #include <darwintest.h>
5*aca3beaaSApple OSS Distributions #include <darwintest_utils.h>
6*aca3beaaSApple OSS Distributions 
7*aca3beaaSApple OSS Distributions #include <dispatch/dispatch.h>
8*aca3beaaSApple OSS Distributions #include <kern/debug.h>
9*aca3beaaSApple OSS Distributions #include <libproc.h>
10*aca3beaaSApple OSS Distributions #include <mach-o/dyld.h>
11*aca3beaaSApple OSS Distributions #include <sys/syscall.h>
12*aca3beaaSApple OSS Distributions #include <sys/stackshot.h>
13*aca3beaaSApple OSS Distributions #include <spawn.h>
14*aca3beaaSApple OSS Distributions 
15*aca3beaaSApple OSS Distributions T_GLOBAL_META(
16*aca3beaaSApple OSS Distributions 	T_META_NAMESPACE("xnu.stackshot"),
17*aca3beaaSApple OSS Distributions 	T_META_RADAR_COMPONENT_NAME("xnu"),
18*aca3beaaSApple OSS Distributions 	T_META_RADAR_COMPONENT_VERSION("stackshot"),
19*aca3beaaSApple OSS Distributions 	T_META_OWNER("jonathan_w_adams"),
20*aca3beaaSApple OSS Distributions 	T_META_CHECK_LEAKS(false),
21*aca3beaaSApple OSS Distributions 	T_META_ASROOT(true)
22*aca3beaaSApple OSS Distributions 	);
23*aca3beaaSApple OSS Distributions 
24*aca3beaaSApple OSS Distributions #define TEST_DURATION_NS (60 * NSEC_PER_SEC)
25*aca3beaaSApple OSS Distributions 
26*aca3beaaSApple OSS Distributions #define REAP_INTERVAL 10
27*aca3beaaSApple OSS Distributions 
28*aca3beaaSApple OSS Distributions static void*
loop(void * arg)29*aca3beaaSApple OSS Distributions loop(__attribute__ ((unused)) void *arg)
30*aca3beaaSApple OSS Distributions {
31*aca3beaaSApple OSS Distributions 	exit(0);
32*aca3beaaSApple OSS Distributions }
33*aca3beaaSApple OSS Distributions 
34*aca3beaaSApple OSS Distributions T_HELPER_DECL(spawn_children_helper, "spawn_children helper")
35*aca3beaaSApple OSS Distributions {
36*aca3beaaSApple OSS Distributions 	pthread_t pthread;
37*aca3beaaSApple OSS Distributions 
38*aca3beaaSApple OSS Distributions 	T_QUIET; T_ASSERT_POSIX_ZERO(pthread_create(&pthread, NULL, loop, NULL), "pthread_create");
39*aca3beaaSApple OSS Distributions 
40*aca3beaaSApple OSS Distributions 	while (1) {
41*aca3beaaSApple OSS Distributions 		;
42*aca3beaaSApple OSS Distributions 	}
43*aca3beaaSApple OSS Distributions }
44*aca3beaaSApple OSS Distributions 
45*aca3beaaSApple OSS Distributions static void
take_stackshot(void)46*aca3beaaSApple OSS Distributions take_stackshot(void)
47*aca3beaaSApple OSS Distributions {
48*aca3beaaSApple OSS Distributions 	uint64_t stackshot_flags = (STACKSHOT_SAVE_LOADINFO | STACKSHOT_GET_GLOBAL_MEM_STATS |
49*aca3beaaSApple OSS Distributions 	    STACKSHOT_SAVE_IMP_DONATION_PIDS | STACKSHOT_KCDATA_FORMAT);
50*aca3beaaSApple OSS Distributions 
51*aca3beaaSApple OSS Distributions 	void *config = stackshot_config_create();
52*aca3beaaSApple OSS Distributions 	T_QUIET; T_ASSERT_NOTNULL(config, "created stackshot config");
53*aca3beaaSApple OSS Distributions 
54*aca3beaaSApple OSS Distributions 	int ret = stackshot_config_set_flags(config, stackshot_flags);
55*aca3beaaSApple OSS Distributions 	T_QUIET; T_ASSERT_POSIX_ZERO(ret, "set flags on stackshot config");
56*aca3beaaSApple OSS Distributions 
57*aca3beaaSApple OSS Distributions 	int retries_remaining = 5;
58*aca3beaaSApple OSS Distributions 
59*aca3beaaSApple OSS Distributions retry:
60*aca3beaaSApple OSS Distributions 	ret = stackshot_capture_with_config(config);
61*aca3beaaSApple OSS Distributions 
62*aca3beaaSApple OSS Distributions 	if (ret == EBUSY || ret == ETIMEDOUT) {
63*aca3beaaSApple OSS Distributions 		if (retries_remaining > 0) {
64*aca3beaaSApple OSS Distributions 			retries_remaining--;
65*aca3beaaSApple OSS Distributions 			goto retry;
66*aca3beaaSApple OSS Distributions 		} else {
67*aca3beaaSApple OSS Distributions 			T_QUIET; T_ASSERT_POSIX_ZERO(ret,
68*aca3beaaSApple OSS Distributions 			    "called stackshot_capture_with_config (no retries remaining)");
69*aca3beaaSApple OSS Distributions 		}
70*aca3beaaSApple OSS Distributions 	} else {
71*aca3beaaSApple OSS Distributions 		T_QUIET; T_ASSERT_POSIX_ZERO(ret, "called stackshot_capture_with_config");
72*aca3beaaSApple OSS Distributions 	}
73*aca3beaaSApple OSS Distributions 
74*aca3beaaSApple OSS Distributions 	ret = stackshot_config_dealloc(config);
75*aca3beaaSApple OSS Distributions 	T_QUIET; T_EXPECT_POSIX_ZERO(ret, "deallocated stackshot config");
76*aca3beaaSApple OSS Distributions }
77*aca3beaaSApple OSS Distributions 
78*aca3beaaSApple OSS Distributions T_DECL(stackshot_spawn_exit, "tests taking many stackshots while children processes are spawning+exiting", T_META_TIMEOUT(120))
79*aca3beaaSApple OSS Distributions {
80*aca3beaaSApple OSS Distributions 	char path[PATH_MAX];
81*aca3beaaSApple OSS Distributions 	uint32_t path_size = sizeof(path);
82*aca3beaaSApple OSS Distributions 	T_ASSERT_POSIX_ZERO(_NSGetExecutablePath(path, &path_size), "_NSGetExecutablePath");
83*aca3beaaSApple OSS Distributions 	char *args[] = { path, "-n", "spawn_children_helper", NULL };
84*aca3beaaSApple OSS Distributions 
85*aca3beaaSApple OSS Distributions 	uint64_t stop_time = clock_gettime_nsec_np(CLOCK_UPTIME_RAW) + TEST_DURATION_NS;
86*aca3beaaSApple OSS Distributions 
87*aca3beaaSApple OSS Distributions 	dispatch_queue_t stackshot_queue = dispatch_queue_create("stackshot_queue", NULL);
88*aca3beaaSApple OSS Distributions 	dispatch_async(stackshot_queue, ^(void) {
89*aca3beaaSApple OSS Distributions 		int num_stackshots = 0;
90*aca3beaaSApple OSS Distributions 
91*aca3beaaSApple OSS Distributions 		while (1) {
92*aca3beaaSApple OSS Distributions 		        take_stackshot();
93*aca3beaaSApple OSS Distributions 		        num_stackshots++;
94*aca3beaaSApple OSS Distributions 		        if ((num_stackshots % 100) == 0) {
95*aca3beaaSApple OSS Distributions 		                T_LOG("completed %d stackshots", num_stackshots);
96*aca3beaaSApple OSS Distributions 			}
97*aca3beaaSApple OSS Distributions 
98*aca3beaaSApple OSS Distributions 		        // Sleep between each stackshot
99*aca3beaaSApple OSS Distributions 		        usleep(100);
100*aca3beaaSApple OSS Distributions 		}
101*aca3beaaSApple OSS Distributions 	});
102*aca3beaaSApple OSS Distributions 
103*aca3beaaSApple OSS Distributions 	// <rdar://problem/39739547> META option for T_HELPER_DECL to not output test begin on start
104*aca3beaaSApple OSS Distributions 	posix_spawn_file_actions_t actions;
105*aca3beaaSApple OSS Distributions 	T_QUIET; T_ASSERT_POSIX_SUCCESS(posix_spawn_file_actions_init(&actions), "create spawn actions");
106*aca3beaaSApple OSS Distributions 	T_QUIET; T_ASSERT_POSIX_SUCCESS(posix_spawn_file_actions_addopen(&actions, STDOUT_FILENO, "/dev/null", O_WRONLY, 0),
107*aca3beaaSApple OSS Distributions 	    "set stdout of child to NULL");
108*aca3beaaSApple OSS Distributions 
109*aca3beaaSApple OSS Distributions 	int children_unreaped = 0, status;
110*aca3beaaSApple OSS Distributions 	uint64_t iterations_completed = 0;
111*aca3beaaSApple OSS Distributions 	while (clock_gettime_nsec_np(CLOCK_UPTIME_RAW) < stop_time) {
112*aca3beaaSApple OSS Distributions 		pid_t pid;
113*aca3beaaSApple OSS Distributions 
114*aca3beaaSApple OSS Distributions 		int sp_ret = posix_spawn(&pid, args[0], &actions, NULL, args, NULL);
115*aca3beaaSApple OSS Distributions 		T_QUIET; T_ASSERT_POSIX_ZERO(sp_ret, "spawned process '%s' with PID %d", args[0], pid);
116*aca3beaaSApple OSS Distributions 
117*aca3beaaSApple OSS Distributions 		children_unreaped++;
118*aca3beaaSApple OSS Distributions 
119*aca3beaaSApple OSS Distributions 		if (children_unreaped >= REAP_INTERVAL) {
120*aca3beaaSApple OSS Distributions 			while (children_unreaped) {
121*aca3beaaSApple OSS Distributions 				T_QUIET; T_ASSERT_POSIX_SUCCESS(waitpid(-1, &status, 0), "waitpid returned child pid");
122*aca3beaaSApple OSS Distributions 				children_unreaped--;
123*aca3beaaSApple OSS Distributions 			}
124*aca3beaaSApple OSS Distributions 		}
125*aca3beaaSApple OSS Distributions 
126*aca3beaaSApple OSS Distributions 		if ((iterations_completed % 100) == 0) {
127*aca3beaaSApple OSS Distributions 			T_LOG("spawned %llu children thus far", iterations_completed);
128*aca3beaaSApple OSS Distributions 		}
129*aca3beaaSApple OSS Distributions 		iterations_completed++;
130*aca3beaaSApple OSS Distributions 	}
131*aca3beaaSApple OSS Distributions 
132*aca3beaaSApple OSS Distributions 	while (children_unreaped) {
133*aca3beaaSApple OSS Distributions 		T_QUIET; T_ASSERT_POSIX_SUCCESS(waitpid(-1, &status, 0), "waitpid returned child pid");
134*aca3beaaSApple OSS Distributions 		children_unreaped--;
135*aca3beaaSApple OSS Distributions 	}
136*aca3beaaSApple OSS Distributions }
137