1*0f4c859eSApple OSS Distributions #include <stdint.h> 2*0f4c859eSApple OSS Distributions #include <stdio.h> 3*0f4c859eSApple OSS Distributions #include <unistd.h> 4*0f4c859eSApple OSS Distributions 5*0f4c859eSApple OSS Distributions #include <mach/mach.h> 6*0f4c859eSApple OSS Distributions #include <mach/mk_timer.h> 7*0f4c859eSApple OSS Distributions 8*0f4c859eSApple OSS Distributions #include <darwintest.h> 9*0f4c859eSApple OSS Distributions 10*0f4c859eSApple OSS Distributions T_GLOBAL_META(T_META_RUN_CONCURRENTLY(true), 11*0f4c859eSApple OSS Distributions T_META_NAMESPACE("xnu.ipc"), 12*0f4c859eSApple OSS Distributions T_META_RADAR_COMPONENT_NAME("xnu"), 13*0f4c859eSApple OSS Distributions T_META_RADAR_COMPONENT_VERSION("IPC")); 14*0f4c859eSApple OSS Distributions 15*0f4c859eSApple OSS Distributions T_DECL(mktimer_kobject, "mktimer_kobject()", T_META_ALL_VALID_ARCHS(true), T_META_IGNORECRASHES(".*mktimer_kobject.*")) 16*0f4c859eSApple OSS Distributions { 17*0f4c859eSApple OSS Distributions mach_port_t timer_port = MACH_PORT_NULL; 18*0f4c859eSApple OSS Distributions mach_port_t notify_port = MACH_PORT_NULL; 19*0f4c859eSApple OSS Distributions 20*0f4c859eSApple OSS Distributions kern_return_t kr = KERN_SUCCESS; 21*0f4c859eSApple OSS Distributions task_exc_guard_behavior_t old, new; 22*0f4c859eSApple OSS Distributions 23*0f4c859eSApple OSS Distributions /* 24*0f4c859eSApple OSS Distributions * Disable [optional] Mach port guard exceptions to avoid fatal crash 25*0f4c859eSApple OSS Distributions */ 26*0f4c859eSApple OSS Distributions kr = task_get_exc_guard_behavior(mach_task_self(), &old); 27*0f4c859eSApple OSS Distributions T_ASSERT_MACH_SUCCESS(kr, "task_get_exc_guard_behavior"); 28*0f4c859eSApple OSS Distributions new = (old & ~TASK_EXC_GUARD_MP_DELIVER); 29*0f4c859eSApple OSS Distributions kr = task_set_exc_guard_behavior(mach_task_self(), new); 30*0f4c859eSApple OSS Distributions T_ASSERT_MACH_SUCCESS(kr, "task_set_exc_guard_behavior new"); 31*0f4c859eSApple OSS Distributions 32*0f4c859eSApple OSS Distributions /* 33*0f4c859eSApple OSS Distributions * timer port 34*0f4c859eSApple OSS Distributions * This is a receive right which is also a kobject 35*0f4c859eSApple OSS Distributions */ 36*0f4c859eSApple OSS Distributions timer_port = mk_timer_create(); 37*0f4c859eSApple OSS Distributions T_ASSERT_NE(timer_port, (mach_port_t)MACH_PORT_NULL, "mk_timer_create: %s", mach_error_string(kr)); 38*0f4c859eSApple OSS Distributions 39*0f4c859eSApple OSS Distributions mach_port_set_context(mach_task_self(), timer_port, (mach_port_context_t) 0x1); 40*0f4c859eSApple OSS Distributions T_ASSERT_EQ(kr, KERN_SUCCESS, "mach_port_set_context(timer_port): %s", mach_error_string(kr)); 41*0f4c859eSApple OSS Distributions 42*0f4c859eSApple OSS Distributions /* notification port for the mk_timer port to come back on */ 43*0f4c859eSApple OSS Distributions kr = mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, ¬ify_port); 44*0f4c859eSApple OSS Distributions T_ASSERT_EQ(kr, KERN_SUCCESS, "mach_port_allocate(notify_port): %s", mach_error_string(kr)); 45*0f4c859eSApple OSS Distributions 46*0f4c859eSApple OSS Distributions kr = mach_port_set_context(mach_task_self(), notify_port, (mach_port_context_t) 0x2); 47*0f4c859eSApple OSS Distributions T_ASSERT_EQ(kr, KERN_SUCCESS, "mach_port_set_context(notify_port): %s", mach_error_string(kr)); 48*0f4c859eSApple OSS Distributions 49*0f4c859eSApple OSS Distributions T_LOG("timer: 0x%x, notify: 0x%x", timer_port, notify_port); 50*0f4c859eSApple OSS Distributions 51*0f4c859eSApple OSS Distributions /* 52*0f4c859eSApple OSS Distributions * This code generates a mach port guard exception and should be tested with an exception catcher. 53*0f4c859eSApple OSS Distributions * Will be updated in <rdar://problem/70971318> 54*0f4c859eSApple OSS Distributions */ 55*0f4c859eSApple OSS Distributions mach_port_t previous = MACH_PORT_NULL; 56*0f4c859eSApple OSS Distributions 57*0f4c859eSApple OSS Distributions /* request a port-destroyed notification on the timer port */ 58*0f4c859eSApple OSS Distributions kr = mach_port_request_notification(mach_task_self(), timer_port, MACH_NOTIFY_PORT_DESTROYED, 59*0f4c859eSApple OSS Distributions 0, notify_port, MACH_MSG_TYPE_MAKE_SEND_ONCE, &previous); 60*0f4c859eSApple OSS Distributions /* this will ordinarily fail with a guard exception! */ 61*0f4c859eSApple OSS Distributions T_ASSERT_MACH_ERROR(kr, KERN_INVALID_RIGHT, "notifications should NOT work on mk_timer ports!"); 62*0f4c859eSApple OSS Distributions 63*0f4c859eSApple OSS Distributions /* restore the old guard behavior */ 64*0f4c859eSApple OSS Distributions kr = task_set_exc_guard_behavior(mach_task_self(), old); 65*0f4c859eSApple OSS Distributions T_ASSERT_MACH_SUCCESS(kr, "task_set_exc_guard_behavior old"); 66*0f4c859eSApple OSS Distributions 67*0f4c859eSApple OSS Distributions T_LOG("done"); 68*0f4c859eSApple OSS Distributions } 69