]> git.saurik.com Git - apple/xnu.git/blob - tests/exception_ports_info.c
xnu-7195.101.1.tar.gz
[apple/xnu.git] / tests / exception_ports_info.c
1 #include <darwintest.h>
2 #include <mach/mach.h>
3 #include <mach/mach_types.h>
4 #include <mach/task.h>
5 #include <mach/thread_act.h>
6 #include <mach_debug/ipc_info.h>
7
8 T_GLOBAL_META(
9 T_META_NAMESPACE("xnu.ipc"),
10 T_META_RUN_CONCURRENTLY(true));
11
12 T_DECL(exception_ports_info, "Test {task, thread}_get_exception_ports_info")
13 {
14 kern_return_t kr;
15 mach_port_t exc_port1, exc_port2, exc_port3;
16
17 mach_msg_type_number_t count = EXC_TYPES_COUNT;
18 exception_mask_t masks[EXC_TYPES_COUNT];
19 ipc_info_port_t ports_info[EXC_TYPES_COUNT];
20 exception_behavior_t behaviors[EXC_TYPES_COUNT];
21 thread_state_flavor_t flavors[EXC_TYPES_COUNT];
22
23 mach_msg_type_number_t count2 = EXC_TYPES_COUNT;
24 exception_mask_t masks2[EXC_TYPES_COUNT];
25 mach_port_t ports[EXC_TYPES_COUNT];
26 exception_behavior_t behaviors2[EXC_TYPES_COUNT];
27 thread_state_flavor_t flavors2[EXC_TYPES_COUNT];
28
29 unsigned int exc_port1_kotype = 0, exc_port1_kaddr = 0;
30 unsigned int exc_port2_kotype = 0, exc_port2_kaddr = 0;
31 unsigned int kotype = 0, kobject = 0, exc_port3_kotype = 0, exc_port3_kaddr = 0;
32 boolean_t found_exc_port1 = false;
33 boolean_t found_exc_port2 = false;
34 boolean_t found_exc_port3 = false;
35
36 ipc_info_space_t info_space;
37 ipc_info_name_array_t table;
38 ipc_info_tree_name_array_t tree;
39 mach_msg_type_number_t tblcnt = 0, treecnt = 0;
40
41 /* Create the mach port the exception messages will be sent to. */
42 kr = mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, &exc_port1);
43 T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "Allocated mach exception port");
44 kr = mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, &exc_port2);
45 T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "Allocated mach exception port");
46 kr = mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, &exc_port3);
47 T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "Allocated mach exception port");
48
49 /*
50 * Insert a send right into the exception port that the kernel will use to
51 * send the exception thread the exception messages.
52 */
53 kr = mach_port_insert_right(mach_task_self(), exc_port1, exc_port1, MACH_MSG_TYPE_MAKE_SEND);
54 T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "Inserted a SEND right into the exception port");
55 kr = mach_port_insert_right(mach_task_self(), exc_port2, exc_port2, MACH_MSG_TYPE_MAKE_SEND);
56 T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "Inserted a SEND right into the exception port");
57 kr = mach_port_insert_right(mach_task_self(), exc_port3, exc_port3, MACH_MSG_TYPE_MAKE_SEND);
58 T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "Inserted a SEND right into the exception port");
59
60 T_LOG("exc_port1: 0x%x", exc_port1);
61 T_LOG("exc_port2: 0x%x", exc_port2);
62 T_LOG("exc_port3: 0x%x", exc_port3);
63
64 /* Tell the kernel what port to send exceptions to. */
65 kr = task_set_exception_ports(
66 mach_task_self(),
67 EXC_MASK_GUARD,
68 exc_port1,
69 (exception_behavior_t)(EXCEPTION_STATE_IDENTITY | MACH_EXCEPTION_CODES),
70 THREAD_STATE_NONE);
71 T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "Set the exception port to my custom handler");
72
73 kr = task_set_exception_ports(
74 mach_task_self(),
75 EXC_MASK_RPC_ALERT, /* why can't be EXC_CRASH or EXC_MASK_CORPSE_NOTIFY ? */
76 exc_port2,
77 (exception_behavior_t)(EXCEPTION_STATE_IDENTITY | MACH_EXCEPTION_CODES),
78 THREAD_STATE_NONE);
79 T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "Set the exception port to my custom handler");
80
81 kr = task_set_exception_ports(
82 mach_task_self(),
83 EXC_MASK_RESOURCE | EXC_MASK_BREAKPOINT | EXC_MASK_SYSCALL,
84 exc_port3,
85 (exception_behavior_t)(EXCEPTION_STATE_IDENTITY | MACH_EXCEPTION_CODES),
86 THREAD_STATE_NONE);
87 T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "Set the exception port to my custom handler");
88
89 /* now, get exception ports info */
90 kr = thread_get_exception_ports(mach_thread_self(), EXC_MASK_ALL, masks2, &count2, ports, behaviors2, flavors2);
91 T_EXPECT_MACH_SUCCESS(kr, "thread_get_exception_ports(): 0x%x", kr);
92 T_EXPECT_EQ(count2, 0, "should have 0 exception ports");
93
94 kr = thread_get_exception_ports_info(mach_thread_self(), EXC_MASK_ALL, masks, &count, ports_info, behaviors, flavors);
95 T_EXPECT_MACH_SUCCESS(kr, "thread_get_exception_ports_info(): 0x%x", kr);
96 T_EXPECT_EQ(count, 0, "should have 0 exception ports");
97
98 count = EXC_TYPES_COUNT;
99 count2 = EXC_TYPES_COUNT;
100
101 kr = task_get_exception_ports_info(mach_task_self(), EXC_MASK_ALL, masks, &count, ports_info, behaviors, flavors);
102 T_EXPECT_MACH_SUCCESS(kr, "task_get_exception_ports_info(): 0x%x", kr);
103 T_EXPECT_EQ(count, 4, "should have 4 masks"); /* Returns 3 if one exc_port registers for EXC_CRASH */
104
105 /* get exception ports */
106 kr = task_get_exception_ports(mach_task_self(), EXC_MASK_ALL, masks2, &count2, ports, behaviors2, flavors2);
107 T_EXPECT_MACH_SUCCESS(kr, "task_get_exception_ports(): 0x%x", kr);
108
109 for (int i = 0; i < count2; i++) {
110 T_LOG("exception port name: 0x%x", ports[i]);
111 }
112 T_EXPECT_EQ(count, count2, "should return same mask count");
113
114 kr = memcmp(masks, masks2, count * sizeof(exception_mask_t));
115 T_EXPECT_EQ(kr, 0, "masks should be the same");
116
117 kr = memcmp(behaviors, behaviors2, count * sizeof(exception_behavior_t));
118 T_EXPECT_EQ(kr, 0, "behaviors should be the same");
119
120 kr = memcmp(flavors, flavors, count * sizeof(thread_state_flavor_t));
121 T_EXPECT_EQ(kr, 0, "flavors should be the same");
122
123 kr = mach_port_kernel_object(mach_task_self(), mach_task_self(), &kotype, &kobject);
124 T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "mach_port_kernel_object(): 0x%x", kr);
125 T_LOG("task_self kobject: 0x%x", kobject);
126
127 T_QUIET; T_EXPECT_MACH_SUCCESS(mach_port_space_info(mach_task_self(), &info_space, &table,
128 &tblcnt, &tree, &treecnt), "mach_port_space_info(): 0x%x", kr);
129
130 for (int i = 0; i < tblcnt; i++) {
131 if (table[i].iin_name == exc_port1) {
132 exc_port1_kaddr = table[i].iin_object;
133 }
134 if (table[i].iin_name == exc_port2) {
135 exc_port2_kaddr = table[i].iin_object;
136 }
137 if (table[i].iin_name == exc_port3) {
138 exc_port3_kaddr = table[i].iin_object;
139 }
140 }
141
142 T_LOG("exc_port_1_kaddr: 0x%x", exc_port1_kaddr);
143 T_LOG("exc_port_2_kaddr: 0x%x", exc_port2_kaddr);
144 T_LOG("exc_port_3_kaddr: 0x%x", exc_port3_kaddr);
145
146 for (int i = 0; i < count; i++) {
147 T_LOG("ports_info[%d].iip_port_object: 0x%x", i, ports_info[i].iip_port_object);
148
149 if (ports_info[i].iip_port_object == exc_port1_kaddr) {
150 T_EXPECT_NE(ports_info[i].iip_port_object, 0,
151 "on debug/kernel, port object should be non-zero: 0x%x", ports_info[i].iip_port_object);
152 T_EXPECT_EQ(ports_info[i].iip_receiver_object, kobject,
153 "receiver object should match task self kobject: 0x%x", ports_info[i].iip_receiver_object);
154 T_EXPECT_EQ(masks[i], EXC_MASK_GUARD, "check if mask for exc_port1 is correct");
155 found_exc_port1 = true;
156 }
157 if (ports_info[i].iip_port_object == exc_port2_kaddr) {
158 T_EXPECT_NE(ports_info[i].iip_port_object, 0,
159 "on debug/kernel, port object should be non-zero: 0x%x", ports_info[i].iip_port_object);
160 T_EXPECT_EQ(ports_info[i].iip_receiver_object, kobject,
161 "receiver object should match task self kobject: 0x%x", ports_info[i].iip_receiver_object);
162 T_EXPECT_EQ(masks[i], EXC_MASK_RPC_ALERT, "check if mask for exc_port2 is correct");
163 found_exc_port2 = true;
164 }
165 if (ports_info[i].iip_port_object == exc_port3_kaddr) {
166 T_EXPECT_NE(ports_info[i].iip_port_object, 0,
167 "on debug/kernel, port object should be non-zero: 0x%x", ports_info[i].iip_port_object);
168 T_EXPECT_EQ(ports_info[i].iip_receiver_object, kobject,
169 "receiver object should match task self kobject: 0x%x", ports_info[i].iip_receiver_object);
170 T_EXPECT_EQ(masks[i], EXC_MASK_RESOURCE | EXC_MASK_BREAKPOINT | EXC_MASK_SYSCALL, "check if mask for exc_port3 is correct");
171 found_exc_port3 = true;
172 }
173 }
174
175 T_EXPECT_TRUE(found_exc_port1, "should find exc_port1");
176 T_EXPECT_TRUE(found_exc_port2, "should find exc_port2");
177 T_EXPECT_TRUE(found_exc_port3, "should find exc_port3");
178 }