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