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