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