1*43a90889SApple OSS Distributions #include <stdio.h> 2*43a90889SApple OSS Distributions #include <assert.h> 3*43a90889SApple OSS Distributions #include <stdlib.h> 4*43a90889SApple OSS Distributions #include <unistd.h> 5*43a90889SApple OSS Distributions #include <mach/mach.h> 6*43a90889SApple OSS Distributions #include <pthread.h> 7*43a90889SApple OSS Distributions #include <sys/event.h> 8*43a90889SApple OSS Distributions #include <errno.h> 9*43a90889SApple OSS Distributions #include <string.h> 10*43a90889SApple OSS Distributions #include <libproc.h> 11*43a90889SApple OSS Distributions 12*43a90889SApple OSS Distributions #include <darwintest.h> 13*43a90889SApple OSS Distributions 14*43a90889SApple OSS Distributions T_GLOBAL_META(T_META_NAMESPACE("xnu.kevent"), 15*43a90889SApple OSS Distributions T_META_RADAR_COMPONENT_NAME("xnu"), 16*43a90889SApple OSS Distributions T_META_RADAR_COMPONENT_VERSION("kevent")); 17*43a90889SApple OSS Distributions 18*43a90889SApple OSS Distributions extern int __proc_info(int32_t callnum, int32_t pid, uint32_t flavor, uint64_t arg, user_addr_t buffer, int32_t buffersize); 19*43a90889SApple OSS Distributions 20*43a90889SApple OSS Distributions T_DECL(avoid_leaking_KASLR, "rdar://101248992", T_META_TAG_VM_PREFERRED) { 21*43a90889SApple OSS Distributions int kq = kqueue(); 22*43a90889SApple OSS Distributions T_ASSERT_GE(kq, 0, "Valid kqueue"); 23*43a90889SApple OSS Distributions 24*43a90889SApple OSS Distributions mach_port_t sync_port = MACH_PORT_NULL, mq_port = MACH_PORT_NULL; 25*43a90889SApple OSS Distributions kern_return_t kr = KERN_SUCCESS; 26*43a90889SApple OSS Distributions 27*43a90889SApple OSS Distributions kr = mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, &sync_port); 28*43a90889SApple OSS Distributions T_ASSERT_MACH_SUCCESS(kr, "allocated sync port"); 29*43a90889SApple OSS Distributions kr = mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, &mq_port); 30*43a90889SApple OSS Distributions T_ASSERT_MACH_SUCCESS(kr, "allocated mq port"); 31*43a90889SApple OSS Distributions 32*43a90889SApple OSS Distributions 33*43a90889SApple OSS Distributions /* Create a kmsg which has the receive right of mq port in it for later copy 34*43a90889SApple OSS Distributions * out */ 35*43a90889SApple OSS Distributions typedef struct msg_request_s { 36*43a90889SApple OSS Distributions mach_msg_header_t header; 37*43a90889SApple OSS Distributions mach_msg_body_t body; 38*43a90889SApple OSS Distributions mach_msg_port_descriptor_t port; 39*43a90889SApple OSS Distributions }* msg_request_t; 40*43a90889SApple OSS Distributions 41*43a90889SApple OSS Distributions typedef struct msg_reply_s { 42*43a90889SApple OSS Distributions mach_msg_header_t header; 43*43a90889SApple OSS Distributions mach_msg_body_t body; 44*43a90889SApple OSS Distributions mach_msg_port_descriptor_t port; 45*43a90889SApple OSS Distributions mach_msg_trailer_t trailer; 46*43a90889SApple OSS Distributions }* msg_reply_t; 47*43a90889SApple OSS Distributions 48*43a90889SApple OSS Distributions union { 49*43a90889SApple OSS Distributions struct msg_request_s request; 50*43a90889SApple OSS Distributions struct msg_reply_s reply; 51*43a90889SApple OSS Distributions } message; 52*43a90889SApple OSS Distributions memset(&message, 0, sizeof(message)); 53*43a90889SApple OSS Distributions 54*43a90889SApple OSS Distributions msg_request_t requestp = &message.request; 55*43a90889SApple OSS Distributions msg_reply_t replyp = &message.reply; 56*43a90889SApple OSS Distributions 57*43a90889SApple OSS Distributions *requestp = (struct msg_request_s) { 58*43a90889SApple OSS Distributions .header = { 59*43a90889SApple OSS Distributions .msgh_bits = MACH_MSGH_BITS_SET(MACH_MSG_TYPE_MAKE_SEND_ONCE, 0, 0, MACH_MSGH_BITS_COMPLEX), 60*43a90889SApple OSS Distributions .msgh_remote_port = sync_port, 61*43a90889SApple OSS Distributions .msgh_local_port = MACH_PORT_NULL, 62*43a90889SApple OSS Distributions .msgh_voucher_port = MACH_PORT_NULL, 63*43a90889SApple OSS Distributions .msgh_size = sizeof(*requestp), 64*43a90889SApple OSS Distributions .msgh_id = 0x88888888, 65*43a90889SApple OSS Distributions }, 66*43a90889SApple OSS Distributions .body = { 67*43a90889SApple OSS Distributions .msgh_descriptor_count = 1, 68*43a90889SApple OSS Distributions }, 69*43a90889SApple OSS Distributions .port = { 70*43a90889SApple OSS Distributions .name = mq_port, 71*43a90889SApple OSS Distributions .type = MACH_MSG_PORT_DESCRIPTOR, 72*43a90889SApple OSS Distributions .disposition = MACH_MSG_TYPE_MOVE_RECEIVE, 73*43a90889SApple OSS Distributions }, 74*43a90889SApple OSS Distributions }; 75*43a90889SApple OSS Distributions 76*43a90889SApple OSS Distributions /* 77*43a90889SApple OSS Distributions * Send the receive right of mq_port to sync_port for later copyout. 78*43a90889SApple OSS Distributions */ 79*43a90889SApple OSS Distributions kr = mach_msg(&requestp->header, MACH_SEND_MSG, sizeof(*requestp), 0, MACH_PORT_NULL, MACH_MSG_TIMEOUT_NONE, MACH_PORT_NULL); 80*43a90889SApple OSS Distributions T_ASSERT_MACH_SUCCESS(kr, "sending message to sync port"); 81*43a90889SApple OSS Distributions 82*43a90889SApple OSS Distributions /* 83*43a90889SApple OSS Distributions * The EV_DISPATCH is required that can pass filt_machport_kqueue_has_turnstile(). 84*43a90889SApple OSS Distributions * The received message will be copied out to replyp. 85*43a90889SApple OSS Distributions * In filt_machport_stash_port(), the value in ext[3] will be set to the mq_port object. 86*43a90889SApple OSS Distributions */ 87*43a90889SApple OSS Distributions struct kevent_qos_s req_event = { 88*43a90889SApple OSS Distributions .ident = sync_port, 89*43a90889SApple OSS Distributions .filter = EVFILT_MACHPORT, 90*43a90889SApple OSS Distributions .flags = EV_ADD | EV_DISPATCH, 91*43a90889SApple OSS Distributions .fflags = MACH_RCV_MSG, 92*43a90889SApple OSS Distributions .ext = { 93*43a90889SApple OSS Distributions [0] = (uint64_t)replyp, 94*43a90889SApple OSS Distributions [1] = sizeof(*replyp), 95*43a90889SApple OSS Distributions }, 96*43a90889SApple OSS Distributions }; 97*43a90889SApple OSS Distributions struct kevent_qos_s reply_event = {}; 98*43a90889SApple OSS Distributions 99*43a90889SApple OSS Distributions int nevents = kevent_qos(kq, &req_event, 1, &reply_event, 1, NULL, NULL, 0); 100*43a90889SApple OSS Distributions T_ASSERT_EQ(nevents, 1, NULL); 101*43a90889SApple OSS Distributions T_ASSERT_EQ(replyp->body.msgh_descriptor_count, 1, NULL); 102*43a90889SApple OSS Distributions assert(MACH_PORT_VALID(replyp->port.name) && replyp->port.disposition == MACH_MSG_TYPE_MOVE_RECEIVE); 103*43a90889SApple OSS Distributions 104*43a90889SApple OSS Distributions struct kevent_extinfo extinfo; 105*43a90889SApple OSS Distributions int knotes = __proc_info(PROC_INFO_CALL_PIDFDINFO, getpid(), PROC_PIDFDKQUEUE_EXTINFO, kq, (user_addr_t)&extinfo, sizeof(extinfo)); 106*43a90889SApple OSS Distributions T_ASSERT_EQ(knotes, 1, NULL); 107*43a90889SApple OSS Distributions T_ASSERT_EQ(extinfo.kqext_kev.ident, sync_port, NULL); 108*43a90889SApple OSS Distributions 109*43a90889SApple OSS Distributions uint64_t leaked_addr = extinfo.kqext_kev.ext[3]; 110*43a90889SApple OSS Distributions T_ASSERT_EQ(leaked_addr, NULL, "Leaked kernel address"); 111*43a90889SApple OSS Distributions } 112