1*d4514f0bSApple OSS Distributions #include <darwintest.h>
2*d4514f0bSApple OSS Distributions #include <darwintest_utils.h>
3*d4514f0bSApple OSS Distributions
4*d4514f0bSApple OSS Distributions #include <TargetConditionals.h>
5*d4514f0bSApple OSS Distributions
6*d4514f0bSApple OSS Distributions #include <mach/mach.h>
7*d4514f0bSApple OSS Distributions #include <mach/mach_types.h>
8*d4514f0bSApple OSS Distributions #include <mach/mach_vm.h>
9*d4514f0bSApple OSS Distributions #include <mach/mach_error.h>
10*d4514f0bSApple OSS Distributions #include <mach/task.h>
11*d4514f0bSApple OSS Distributions
12*d4514f0bSApple OSS Distributions #include <servers/bootstrap.h>
13*d4514f0bSApple OSS Distributions
14*d4514f0bSApple OSS Distributions #include <sys/resource.h>
15*d4514f0bSApple OSS Distributions
16*d4514f0bSApple OSS Distributions #include <kern/kcdata.h>
17*d4514f0bSApple OSS Distributions
18*d4514f0bSApple OSS Distributions #include <os/reason_private.h>
19*d4514f0bSApple OSS Distributions
20*d4514f0bSApple OSS Distributions #include <System/uuid/uuid.h>
21*d4514f0bSApple OSS Distributions #include "exc_helpers.h"
22*d4514f0bSApple OSS Distributions
23*d4514f0bSApple OSS Distributions #include <unistd.h>
24*d4514f0bSApple OSS Distributions #include <errno.h>
25*d4514f0bSApple OSS Distributions
26*d4514f0bSApple OSS Distributions T_GLOBAL_META(
27*d4514f0bSApple OSS Distributions T_META_NAMESPACE("xnu.ipc"),
28*d4514f0bSApple OSS Distributions T_META_RUN_CONCURRENTLY(TRUE),
29*d4514f0bSApple OSS Distributions T_META_RADAR_COMPONENT_NAME("xnu"),
30*d4514f0bSApple OSS Distributions T_META_RADAR_COMPONENT_VERSION("IPC"));
31*d4514f0bSApple OSS Distributions
32*d4514f0bSApple OSS Distributions
33*d4514f0bSApple OSS Distributions bool expect_backtrace = TRUE;
34*d4514f0bSApple OSS Distributions
35*d4514f0bSApple OSS Distributions static kern_return_t
exc_handler_backtrace(mach_port_t kcdata_object,exception_type_t exception,mach_exception_data_t codes)36*d4514f0bSApple OSS Distributions exc_handler_backtrace(
37*d4514f0bSApple OSS Distributions mach_port_t kcdata_object,
38*d4514f0bSApple OSS Distributions exception_type_t exception,
39*d4514f0bSApple OSS Distributions mach_exception_data_t codes)
40*d4514f0bSApple OSS Distributions {
41*d4514f0bSApple OSS Distributions kern_return_t kr;
42*d4514f0bSApple OSS Distributions mach_vm_address_t btinfo_begin;
43*d4514f0bSApple OSS Distributions mach_vm_size_t btinfo_size;
44*d4514f0bSApple OSS Distributions
45*d4514f0bSApple OSS Distributions if (expect_backtrace == FALSE) {
46*d4514f0bSApple OSS Distributions T_FAIL("Does not expect backtrace for this test case");
47*d4514f0bSApple OSS Distributions }
48*d4514f0bSApple OSS Distributions
49*d4514f0bSApple OSS Distributions T_LOG("Received backtrace exception.");
50*d4514f0bSApple OSS Distributions T_ASSERT_EQ(exception, EXC_CORPSE_NOTIFY, "Exception should be corpse notify");
51*d4514f0bSApple OSS Distributions T_ASSERT_EQ(codes[0], EXC_GUARD, "Effective exception should be EXC_GUARD");
52*d4514f0bSApple OSS Distributions
53*d4514f0bSApple OSS Distributions kr = task_map_kcdata_object_64(mach_task_self(), kcdata_object, &btinfo_begin, &btinfo_size);
54*d4514f0bSApple OSS Distributions T_ASSERT_MACH_SUCCESS(kr, "task_map_kcdata_object_64() should succeed");
55*d4514f0bSApple OSS Distributions
56*d4514f0bSApple OSS Distributions kcdata_iter_t btdata = kcdata_iter((void *)btinfo_begin, (unsigned long)btinfo_size);
57*d4514f0bSApple OSS Distributions if (kcdata_iter_valid(btdata) && kcdata_iter_type(btdata) == TASK_BTINFO_BEGIN) {
58*d4514f0bSApple OSS Distributions /* loop through data provided by kernel */
59*d4514f0bSApple OSS Distributions KCDATA_ITER_FOREACH(btdata) {
60*d4514f0bSApple OSS Distributions switch (kcdata_iter_type(btdata)) {
61*d4514f0bSApple OSS Distributions case TASK_BTINFO_GID: {
62*d4514f0bSApple OSS Distributions int gid = *(int *)(kcdata_iter_payload(btdata));
63*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_GID: %d", gid);
64*d4514f0bSApple OSS Distributions break;
65*d4514f0bSApple OSS Distributions }
66*d4514f0bSApple OSS Distributions case TASK_BTINFO_CPUTYPE: {
67*d4514f0bSApple OSS Distributions cpu_type_t type = *(cpu_type_t *)(kcdata_iter_payload(btdata));
68*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_CPUTYPE: %d", type);
69*d4514f0bSApple OSS Distributions break;
70*d4514f0bSApple OSS Distributions }
71*d4514f0bSApple OSS Distributions case TASK_BTINFO_THREAD_STATE: {
72*d4514f0bSApple OSS Distributions struct btinfo_thread_state_data_t data = *(struct btinfo_thread_state_data_t *)(kcdata_iter_payload(btdata));
73*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_THREAD_STATE: Flavor %d, Count %d", data.flavor, data.count);
74*d4514f0bSApple OSS Distributions break;
75*d4514f0bSApple OSS Distributions }
76*d4514f0bSApple OSS Distributions case TASK_BTINFO_THREAD_EXCEPTION_STATE: {
77*d4514f0bSApple OSS Distributions struct btinfo_thread_state_data_t data = *(struct btinfo_thread_state_data_t *)(kcdata_iter_payload(btdata));
78*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_THREAD_EXCEPTION_STATE: Flavor %d, Count %d", data.flavor, data.count);
79*d4514f0bSApple OSS Distributions break;
80*d4514f0bSApple OSS Distributions }
81*d4514f0bSApple OSS Distributions case TASK_BTINFO_PROC_NAME: {
82*d4514f0bSApple OSS Distributions #define MAXCOMLEN 16
83*d4514f0bSApple OSS Distributions char process_name[MAXCOMLEN + 1];
84*d4514f0bSApple OSS Distributions memcpy(process_name, kcdata_iter_payload(btdata), MAXCOMLEN); // limited to 16 chars
85*d4514f0bSApple OSS Distributions process_name[MAXCOMLEN] = '\0'; // ensure string is null terminated
86*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_PROC_NAME: %s", process_name);
87*d4514f0bSApple OSS Distributions break;
88*d4514f0bSApple OSS Distributions }
89*d4514f0bSApple OSS Distributions case TASK_BTINFO_PROC_PATH: {
90*d4514f0bSApple OSS Distributions const char *c_str = (const char *)kcdata_iter_payload(btdata);
91*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_PROC_PATH: %s", c_str);
92*d4514f0bSApple OSS Distributions break;
93*d4514f0bSApple OSS Distributions }
94*d4514f0bSApple OSS Distributions case TASK_BTINFO_PLATFORM: {
95*d4514f0bSApple OSS Distributions uint32_t platform = *(uint32_t *)(kcdata_iter_payload(btdata));
96*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_PLATFORM: %d", platform);
97*d4514f0bSApple OSS Distributions break;
98*d4514f0bSApple OSS Distributions }
99*d4514f0bSApple OSS Distributions case TASK_BTINFO_RUSAGE_INFO: {
100*d4514f0bSApple OSS Distributions struct rusage_info_v0 rui = *(struct rusage_info_v0 *)kcdata_iter_payload(btdata);
101*d4514f0bSApple OSS Distributions uuid_string_t uuid;
102*d4514f0bSApple OSS Distributions uint64_t _proc_start_abstime = rui.ri_proc_start_abstime;
103*d4514f0bSApple OSS Distributions uint64_t _proc_exit_abstime = rui.ri_proc_exit_abstime;
104*d4514f0bSApple OSS Distributions uuid_unparse(rui.ri_uuid, uuid);
105*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_RUSAGE_INFO: uuid: %s, start time: %llu, \
106*d4514f0bSApple OSS Distributions exit time: %llu", uuid, _proc_start_abstime, _proc_exit_abstime);
107*d4514f0bSApple OSS Distributions break;
108*d4514f0bSApple OSS Distributions }
109*d4514f0bSApple OSS Distributions case TASK_BTINFO_SC_LOADINFO64: {
110*d4514f0bSApple OSS Distributions struct btinfo_sc_load_info64 info = *(struct btinfo_sc_load_info64 *)kcdata_iter_payload(btdata);
111*d4514f0bSApple OSS Distributions uuid_string_t uuid;
112*d4514f0bSApple OSS Distributions uuid_unparse(info.sharedCacheUUID, uuid);
113*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_SC_LOADINFO64: uuid: %s, slide: %llu, base: %llu",
114*d4514f0bSApple OSS Distributions uuid, info.sharedCacheSlide, info.sharedCacheBaseAddress);
115*d4514f0bSApple OSS Distributions break;
116*d4514f0bSApple OSS Distributions }
117*d4514f0bSApple OSS Distributions case TASK_BTINFO_SC_LOADINFO: {
118*d4514f0bSApple OSS Distributions struct btinfo_sc_load_info info = *(struct btinfo_sc_load_info *)kcdata_iter_payload(btdata);
119*d4514f0bSApple OSS Distributions uuid_string_t uuid;
120*d4514f0bSApple OSS Distributions uuid_unparse(info.sharedCacheUUID, uuid);
121*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_SC_LOADINFO: uuid: %s, slide: %d, base: %d",
122*d4514f0bSApple OSS Distributions uuid, info.sharedCacheSlide, info.sharedCacheBaseAddress);
123*d4514f0bSApple OSS Distributions break;
124*d4514f0bSApple OSS Distributions }
125*d4514f0bSApple OSS Distributions case EXIT_REASON_SNAPSHOT: {
126*d4514f0bSApple OSS Distributions struct exit_reason_snapshot *snapshot = (struct exit_reason_snapshot *)kcdata_iter_payload(btdata);
127*d4514f0bSApple OSS Distributions T_LOG("Found EXIT_REASON_SNAPSHOT with namespace %x code %x", snapshot->ers_namespace, snapshot->ers_code);
128*d4514f0bSApple OSS Distributions break;
129*d4514f0bSApple OSS Distributions }
130*d4514f0bSApple OSS Distributions case KCDATA_TYPE_ARRAY: {
131*d4514f0bSApple OSS Distributions int count = kcdata_iter_array_elem_count(btdata);
132*d4514f0bSApple OSS Distributions uint32_t type = kcdata_iter_array_elem_type(btdata);
133*d4514f0bSApple OSS Distributions uint32_t size = kcdata_iter_array_elem_size(btdata);
134*d4514f0bSApple OSS Distributions
135*d4514f0bSApple OSS Distributions if (type == TASK_BTINFO_BACKTRACE64) {
136*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_BACKTRACE64, with %d backtrace frames", count);
137*d4514f0bSApple OSS Distributions T_QUIET; T_ASSERT_EQ(size, sizeof(uint64_t), "Address size should be 64 bits");
138*d4514f0bSApple OSS Distributions
139*d4514f0bSApple OSS Distributions for (int i = 0; i < count; i++) {
140*d4514f0bSApple OSS Distributions T_LOG("Frame %d: %p", i, ((uint64_t *)kcdata_iter_payload(btdata))[i]);
141*d4514f0bSApple OSS Distributions }
142*d4514f0bSApple OSS Distributions } else if (type == TASK_BTINFO_DYLD_LOADINFO64) {
143*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_DYLD_LOADINFO64, with %d image infos", count);
144*d4514f0bSApple OSS Distributions T_QUIET; T_ASSERT_EQ(size, sizeof(struct dyld_uuid_info_64), "Struct size should match");
145*d4514f0bSApple OSS Distributions
146*d4514f0bSApple OSS Distributions for (int i = 0; i < count; i++) {
147*d4514f0bSApple OSS Distributions uuid_string_t uuid_str;
148*d4514f0bSApple OSS Distributions uuid_unparse(((struct dyld_uuid_info_64 *)kcdata_iter_payload(btdata))[i].imageUUID, uuid_str);
149*d4514f0bSApple OSS Distributions
150*d4514f0bSApple OSS Distributions T_LOG("Image %d: <%s, %p>", i, uuid_str,
151*d4514f0bSApple OSS Distributions ((struct dyld_uuid_info_64 *)kcdata_iter_payload(btdata))[i].imageLoadAddress);
152*d4514f0bSApple OSS Distributions }
153*d4514f0bSApple OSS Distributions } else if (type == TASK_BTINFO_BACKTRACE) {
154*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_BACKTRACE, with %d backtrace frames", count);
155*d4514f0bSApple OSS Distributions T_QUIET; T_ASSERT_EQ(size, sizeof(uint64_t), "Address size on arm64_32 should be 64 bits");
156*d4514f0bSApple OSS Distributions
157*d4514f0bSApple OSS Distributions for (int i = 0; i < count; i++) {
158*d4514f0bSApple OSS Distributions T_LOG("Frame %d: %p", i, ((uint32_t *)kcdata_iter_payload(btdata))[i]);
159*d4514f0bSApple OSS Distributions }
160*d4514f0bSApple OSS Distributions } else if (type == TASK_BTINFO_DYLD_LOADINFO) {
161*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_DYLD_LOADINFO, with %d image infos", count);
162*d4514f0bSApple OSS Distributions T_QUIET; T_ASSERT_EQ(size, sizeof(struct dyld_uuid_info_32), "Struct size should match");
163*d4514f0bSApple OSS Distributions
164*d4514f0bSApple OSS Distributions for (int i = 0; i < count; i++) {
165*d4514f0bSApple OSS Distributions uuid_string_t uuid_str;
166*d4514f0bSApple OSS Distributions uuid_unparse(((struct dyld_uuid_info_32 *)kcdata_iter_payload(btdata))[i].imageUUID, uuid_str);
167*d4514f0bSApple OSS Distributions
168*d4514f0bSApple OSS Distributions T_LOG("Image %d: <%s, %p>", i, uuid_str,
169*d4514f0bSApple OSS Distributions ((struct dyld_uuid_info_32 *)kcdata_iter_payload(btdata))[i].imageLoadAddress);
170*d4514f0bSApple OSS Distributions }
171*d4514f0bSApple OSS Distributions }
172*d4514f0bSApple OSS Distributions break;
173*d4514f0bSApple OSS Distributions }
174*d4514f0bSApple OSS Distributions case TASK_BTINFO_THREAD_ID: {
175*d4514f0bSApple OSS Distributions uint64_t thread_id = *(uint64_t *)(kcdata_iter_payload(btdata));
176*d4514f0bSApple OSS Distributions T_LOG("Found TASK_BTINFO_THREAD_ID: 0x%lx", thread_id);
177*d4514f0bSApple OSS Distributions break;
178*d4514f0bSApple OSS Distributions }
179*d4514f0bSApple OSS Distributions default:
180*d4514f0bSApple OSS Distributions break;
181*d4514f0bSApple OSS Distributions }
182*d4514f0bSApple OSS Distributions }
183*d4514f0bSApple OSS Distributions } else {
184*d4514f0bSApple OSS Distributions T_FAIL("Unexpected kcdata object type");
185*d4514f0bSApple OSS Distributions }
186*d4514f0bSApple OSS Distributions
187*d4514f0bSApple OSS Distributions mach_vm_deallocate(mach_task_self(), btinfo_begin, btinfo_size);
188*d4514f0bSApple OSS Distributions mach_port_deallocate(mach_task_self(), kcdata_object);
189*d4514f0bSApple OSS Distributions
190*d4514f0bSApple OSS Distributions T_END;
191*d4514f0bSApple OSS Distributions }
192*d4514f0bSApple OSS Distributions
193*d4514f0bSApple OSS Distributions static size_t
exc_handler_identity_protected(task_id_token_t token,__unused uint64_t thread_id,__unused exception_type_t type,__unused exception_data_t codes)194*d4514f0bSApple OSS Distributions exc_handler_identity_protected(
195*d4514f0bSApple OSS Distributions task_id_token_t token,
196*d4514f0bSApple OSS Distributions __unused uint64_t thread_id,
197*d4514f0bSApple OSS Distributions __unused exception_type_t type,
198*d4514f0bSApple OSS Distributions __unused exception_data_t codes)
199*d4514f0bSApple OSS Distributions {
200*d4514f0bSApple OSS Distributions mach_port_t port1, port2;
201*d4514f0bSApple OSS Distributions kern_return_t kr;
202*d4514f0bSApple OSS Distributions
203*d4514f0bSApple OSS Distributions if (expect_backtrace) {
204*d4514f0bSApple OSS Distributions T_FAIL("Expect backtrace for this test case");
205*d4514f0bSApple OSS Distributions }
206*d4514f0bSApple OSS Distributions
207*d4514f0bSApple OSS Distributions T_LOG("Got protected exception!");
208*d4514f0bSApple OSS Distributions
209*d4514f0bSApple OSS Distributions port1 = mach_task_self();
210*d4514f0bSApple OSS Distributions kr = task_identity_token_get_task_port(token, TASK_FLAVOR_CONTROL, &port2); /* Immovable control port for self */
211*d4514f0bSApple OSS Distributions T_ASSERT_MACH_SUCCESS(kr, "task_identity_token_get_task_port() - CONTROL");
212*d4514f0bSApple OSS Distributions T_EXPECT_EQ(port1, port2, "Control port matches!");
213*d4514f0bSApple OSS Distributions
214*d4514f0bSApple OSS Distributions T_END;
215*d4514f0bSApple OSS Distributions }
216*d4514f0bSApple OSS Distributions
217*d4514f0bSApple OSS Distributions /* Lightweight corpse not enabled on macOS yet */
218*d4514f0bSApple OSS Distributions #if !TARGET_OS_OSX
219*d4514f0bSApple OSS Distributions T_DECL(corpse_backtrace_os_log_lightweight,
220*d4514f0bSApple OSS Distributions "Test os_log_fault() fast backtracing with lightweight corpse",
221*d4514f0bSApple OSS Distributions T_META_CHECK_LEAKS(false), T_META_TAG_VM_PREFERRED) /* Test may otherwise time out after T_END */
222*d4514f0bSApple OSS Distributions {
223*d4514f0bSApple OSS Distributions mach_port_t exc_port = MACH_PORT_NULL;
224*d4514f0bSApple OSS Distributions
225*d4514f0bSApple OSS Distributions expect_backtrace = TRUE;
226*d4514f0bSApple OSS Distributions
227*d4514f0bSApple OSS Distributions exc_port = create_exception_port_behavior64(EXC_MASK_CORPSE_NOTIFY,
228*d4514f0bSApple OSS Distributions EXCEPTION_IDENTITY_PROTECTED | MACH_EXCEPTION_BACKTRACE_PREFERRED);
229*d4514f0bSApple OSS Distributions
230*d4514f0bSApple OSS Distributions T_ASSERT_NE(exc_port, MACH_PORT_NULL, "Exception port should be valid.");
231*d4514f0bSApple OSS Distributions
232*d4514f0bSApple OSS Distributions T_LOG("Exception port: %d\n", exc_port);
233*d4514f0bSApple OSS Distributions
234*d4514f0bSApple OSS Distributions run_exception_handler_behavior64(exc_port, exc_handler_backtrace, NULL,
235*d4514f0bSApple OSS Distributions EXCEPTION_IDENTITY_PROTECTED | MACH_EXCEPTION_BACKTRACE_PREFERRED, true);
236*d4514f0bSApple OSS Distributions
237*d4514f0bSApple OSS Distributions /* Generate a non-fatal EXC_GUARD */
238*d4514f0bSApple OSS Distributions uint64_t payload = 0xDEADBEEF;
239*d4514f0bSApple OSS Distributions int ret = os_fault_with_payload(OS_REASON_LIBSYSTEM, OS_REASON_LIBSYSTEM_CODE_FAULT,
240*d4514f0bSApple OSS Distributions &payload, sizeof(payload), "Generating a user fault", 0);
241*d4514f0bSApple OSS Distributions T_QUIET; T_ASSERT_EQ(ret, 0, "os_fault_with_payload should succeed");
242*d4514f0bSApple OSS Distributions
243*d4514f0bSApple OSS Distributions T_LOG("Wait for exception on main thread..");
244*d4514f0bSApple OSS Distributions for (int i = 0; i < 10; i++) {
245*d4514f0bSApple OSS Distributions sleep(2);
246*d4514f0bSApple OSS Distributions }
247*d4514f0bSApple OSS Distributions
248*d4514f0bSApple OSS Distributions T_FAIL("Did not receive exception within timeout");
249*d4514f0bSApple OSS Distributions }
250*d4514f0bSApple OSS Distributions #endif
251*d4514f0bSApple OSS Distributions
252*d4514f0bSApple OSS Distributions T_DECL(corpse_backtrace_bad_access,
253*d4514f0bSApple OSS Distributions "Test os_bad_access fast backtracing with lightweight corpse",
254*d4514f0bSApple OSS Distributions T_META_CHECK_LEAKS(false), T_META_TAG_VM_PREFERRED)
255*d4514f0bSApple OSS Distributions {
256*d4514f0bSApple OSS Distributions mach_port_t exc_port = MACH_PORT_NULL;
257*d4514f0bSApple OSS Distributions
258*d4514f0bSApple OSS Distributions expect_backtrace = FALSE;
259*d4514f0bSApple OSS Distributions /* Prefer backtrace on EXC_BAD_ACCESS, but should be ignored by kernel */
260*d4514f0bSApple OSS Distributions exc_port = create_exception_port_behavior64(EXC_MASK_BAD_ACCESS,
261*d4514f0bSApple OSS Distributions EXCEPTION_IDENTITY_PROTECTED | MACH_EXCEPTION_BACKTRACE_PREFERRED);
262*d4514f0bSApple OSS Distributions
263*d4514f0bSApple OSS Distributions T_ASSERT_NE(exc_port, MACH_PORT_NULL, "Exception port should be valid.");
264*d4514f0bSApple OSS Distributions
265*d4514f0bSApple OSS Distributions T_LOG("Exception port 2: %d\n", exc_port);
266*d4514f0bSApple OSS Distributions
267*d4514f0bSApple OSS Distributions run_exception_handler_behavior64(exc_port, exc_handler_backtrace, exc_handler_identity_protected,
268*d4514f0bSApple OSS Distributions EXCEPTION_IDENTITY_PROTECTED | MACH_EXCEPTION_BACKTRACE_PREFERRED, true);
269*d4514f0bSApple OSS Distributions
270*d4514f0bSApple OSS Distributions /* Generate an EXC_BAD_ACCESS */
271*d4514f0bSApple OSS Distributions *(void *volatile*)0 = 0;
272*d4514f0bSApple OSS Distributions
273*d4514f0bSApple OSS Distributions T_FAIL("Should not reach here");
274*d4514f0bSApple OSS Distributions }
275*d4514f0bSApple OSS Distributions
276*d4514f0bSApple OSS Distributions T_DECL(corpse_backtrace_os_log_lightweight_reportcrash,
277*d4514f0bSApple OSS Distributions "Test os_log_fault() fast backtracing with lightweight corpse and report crash",
278*d4514f0bSApple OSS Distributions T_META_ENABLED(false), T_META_TAG_VM_PREFERRED)
279*d4514f0bSApple OSS Distributions {
280*d4514f0bSApple OSS Distributions mach_port_t rc_port = MACH_PORT_NULL, bootstrap = MACH_PORT_NULL;
281*d4514f0bSApple OSS Distributions kern_return_t kr;
282*d4514f0bSApple OSS Distributions
283*d4514f0bSApple OSS Distributions expect_backtrace = TRUE;
284*d4514f0bSApple OSS Distributions
285*d4514f0bSApple OSS Distributions kr = task_get_bootstrap_port(mach_task_self(), &bootstrap);
286*d4514f0bSApple OSS Distributions
287*d4514f0bSApple OSS Distributions kr = bootstrap_look_up(bootstrap, "com.apple.ReportCrash", &rc_port);
288*d4514f0bSApple OSS Distributions
289*d4514f0bSApple OSS Distributions T_ASSERT_NE(rc_port, MACH_PORT_NULL, "Exception port should be valid.");
290*d4514f0bSApple OSS Distributions
291*d4514f0bSApple OSS Distributions T_LOG("ReportCrash exception port: %d\n", rc_port);
292*d4514f0bSApple OSS Distributions
293*d4514f0bSApple OSS Distributions #ifndef ARM_THREAD_STATE
294*d4514f0bSApple OSS Distributions #define ARM_THREAD_STATE 1
295*d4514f0bSApple OSS Distributions #endif
296*d4514f0bSApple OSS Distributions
297*d4514f0bSApple OSS Distributions kr = task_set_exception_ports(mach_task_self(),
298*d4514f0bSApple OSS Distributions EXC_MASK_CORPSE_NOTIFY,
299*d4514f0bSApple OSS Distributions rc_port,
300*d4514f0bSApple OSS Distributions EXCEPTION_IDENTITY_PROTECTED | MACH_EXCEPTION_BACKTRACE_PREFERRED | MACH_EXCEPTION_CODES,
301*d4514f0bSApple OSS Distributions ARM_THREAD_STATE);
302*d4514f0bSApple OSS Distributions
303*d4514f0bSApple OSS Distributions T_QUIET; T_ASSERT_EQ(kr, KERN_SUCCESS, "Registration with ReportCrash should succeed");
304*d4514f0bSApple OSS Distributions
305*d4514f0bSApple OSS Distributions /* Generate a non-fatal EXC_GUARD */
306*d4514f0bSApple OSS Distributions uint64_t payload = 0xDEADBEEF;
307*d4514f0bSApple OSS Distributions int ret = os_fault_with_payload(OS_REASON_LIBSYSTEM, OS_REASON_LIBSYSTEM_CODE_FAULT,
308*d4514f0bSApple OSS Distributions &payload, sizeof(payload), "Generating a user fault", 0);
309*d4514f0bSApple OSS Distributions T_QUIET; T_ASSERT_EQ(ret, 0, "os_fault_with_payload should succeed");
310*d4514f0bSApple OSS Distributions
311*d4514f0bSApple OSS Distributions T_LOG("Kernel should have delivered a message to ReportCrash. Exit now.");
312*d4514f0bSApple OSS Distributions T_END;
313*d4514f0bSApple OSS Distributions }
314