1*5e3eaea3SApple OSS Distributions /*
2*5e3eaea3SApple OSS Distributions * Test based on POC attached to rdar://96567281 (Knote Use-after-Free in XNU)
3*5e3eaea3SApple OSS Distributions *
4*5e3eaea3SApple OSS Distributions */
5*5e3eaea3SApple OSS Distributions #include <darwintest.h>
6*5e3eaea3SApple OSS Distributions #include <mach/mach.h>
7*5e3eaea3SApple OSS Distributions #include <pthread.h>
8*5e3eaea3SApple OSS Distributions #include <sys/event.h>
9*5e3eaea3SApple OSS Distributions #include <stdlib.h>
10*5e3eaea3SApple OSS Distributions
11*5e3eaea3SApple OSS Distributions T_GLOBAL_META(
12*5e3eaea3SApple OSS Distributions T_META_NAMESPACE("xnu.ipc"),
13*5e3eaea3SApple OSS Distributions T_META_RADAR_COMPONENT_NAME("xnu"),
14*5e3eaea3SApple OSS Distributions T_META_RADAR_COMPONENT_VERSION("IPC"),
15*5e3eaea3SApple OSS Distributions T_META_RUN_CONCURRENTLY(TRUE));
16*5e3eaea3SApple OSS Distributions
17*5e3eaea3SApple OSS Distributions typedef struct knote_context_s knote_context_t;
18*5e3eaea3SApple OSS Distributions struct knote_context_s {
19*5e3eaea3SApple OSS Distributions volatile int initialized;
20*5e3eaea3SApple OSS Distributions volatile int start;
21*5e3eaea3SApple OSS Distributions };
22*5e3eaea3SApple OSS Distributions
23*5e3eaea3SApple OSS Distributions static void *
th_allocate_knotes(void * arg)24*5e3eaea3SApple OSS Distributions th_allocate_knotes(void *arg)
25*5e3eaea3SApple OSS Distributions {
26*5e3eaea3SApple OSS Distributions knote_context_t *context = (knote_context_t *)arg;
27*5e3eaea3SApple OSS Distributions kern_return_t kr = KERN_SUCCESS;
28*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(context->initialized, (int)0, "th_allocate_knotes context is initialized.");
29*5e3eaea3SApple OSS Distributions
30*5e3eaea3SApple OSS Distributions mach_port_t sync_port = MACH_PORT_NULL;
31*5e3eaea3SApple OSS Distributions kr = mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, &sync_port);
32*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "mach_port_allocate sync_port");
33*5e3eaea3SApple OSS Distributions
34*5e3eaea3SApple OSS Distributions mach_port_t kq_port = MACH_PORT_NULL;
35*5e3eaea3SApple OSS Distributions kr = mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, &kq_port);
36*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "mach_port_allocate kq_port");
37*5e3eaea3SApple OSS Distributions
38*5e3eaea3SApple OSS Distributions int kq = kqueue();
39*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(kq, "kqueue");
40*5e3eaea3SApple OSS Distributions
41*5e3eaea3SApple OSS Distributions #define PORTS_COUNT 0x1000
42*5e3eaea3SApple OSS Distributions
43*5e3eaea3SApple OSS Distributions mach_port_t *ports = calloc(PORTS_COUNT, sizeof(mach_port_t));
44*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_NOTNULL(ports, "calloc");
45*5e3eaea3SApple OSS Distributions
46*5e3eaea3SApple OSS Distributions for (size_t i = 0; i < PORTS_COUNT; i++) {
47*5e3eaea3SApple OSS Distributions mach_port_t port = MACH_PORT_NULL;
48*5e3eaea3SApple OSS Distributions kr = mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, &port);
49*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "mach_port_allocate");
50*5e3eaea3SApple OSS Distributions
51*5e3eaea3SApple OSS Distributions typedef struct move_receive_request_s move_receive_request_t;
52*5e3eaea3SApple OSS Distributions typedef struct move_receive_reply_s move_receive_reply_t;
53*5e3eaea3SApple OSS Distributions
54*5e3eaea3SApple OSS Distributions struct move_receive_request_s {
55*5e3eaea3SApple OSS Distributions mach_msg_header_t header;
56*5e3eaea3SApple OSS Distributions mach_msg_body_t body;
57*5e3eaea3SApple OSS Distributions mach_msg_port_descriptor_t port;
58*5e3eaea3SApple OSS Distributions mach_msg_port_descriptor_t kq_port;
59*5e3eaea3SApple OSS Distributions };
60*5e3eaea3SApple OSS Distributions
61*5e3eaea3SApple OSS Distributions struct move_receive_reply_s {
62*5e3eaea3SApple OSS Distributions mach_msg_header_t header;
63*5e3eaea3SApple OSS Distributions mach_msg_body_t body;
64*5e3eaea3SApple OSS Distributions mach_msg_port_descriptor_t port;
65*5e3eaea3SApple OSS Distributions mach_msg_port_descriptor_t kq_port;
66*5e3eaea3SApple OSS Distributions mach_msg_trailer_t trailer;
67*5e3eaea3SApple OSS Distributions };
68*5e3eaea3SApple OSS Distributions
69*5e3eaea3SApple OSS Distributions union {
70*5e3eaea3SApple OSS Distributions move_receive_request_t request;
71*5e3eaea3SApple OSS Distributions move_receive_reply_t reply;
72*5e3eaea3SApple OSS Distributions } message;
73*5e3eaea3SApple OSS Distributions
74*5e3eaea3SApple OSS Distributions move_receive_request_t *request = &message.request;
75*5e3eaea3SApple OSS Distributions move_receive_reply_t *reply = &message.reply;
76*5e3eaea3SApple OSS Distributions
77*5e3eaea3SApple OSS Distributions request->header = (mach_msg_header_t){
78*5e3eaea3SApple OSS Distributions .msgh_remote_port = sync_port,
79*5e3eaea3SApple OSS Distributions .msgh_local_port = MACH_PORT_NULL,
80*5e3eaea3SApple OSS Distributions .msgh_voucher_port = MACH_PORT_NULL,
81*5e3eaea3SApple OSS Distributions .msgh_id = (mach_msg_id_t)0x88888888,
82*5e3eaea3SApple OSS Distributions .msgh_size = sizeof(*request),
83*5e3eaea3SApple OSS Distributions .msgh_bits = MACH_MSGH_BITS_SET(MACH_MSG_TYPE_MAKE_SEND, 0, 0, MACH_MSGH_BITS_COMPLEX),
84*5e3eaea3SApple OSS Distributions };
85*5e3eaea3SApple OSS Distributions
86*5e3eaea3SApple OSS Distributions request->body = (mach_msg_body_t){
87*5e3eaea3SApple OSS Distributions .msgh_descriptor_count = 2,
88*5e3eaea3SApple OSS Distributions };
89*5e3eaea3SApple OSS Distributions
90*5e3eaea3SApple OSS Distributions request->port = (mach_msg_port_descriptor_t){
91*5e3eaea3SApple OSS Distributions .name = port,
92*5e3eaea3SApple OSS Distributions .disposition = MACH_MSG_TYPE_MOVE_RECEIVE,
93*5e3eaea3SApple OSS Distributions .type = MACH_MSG_PORT_DESCRIPTOR,
94*5e3eaea3SApple OSS Distributions };
95*5e3eaea3SApple OSS Distributions request->kq_port = (mach_msg_port_descriptor_t){
96*5e3eaea3SApple OSS Distributions .name = kq_port,
97*5e3eaea3SApple OSS Distributions .disposition = MACH_MSG_TYPE_MOVE_RECEIVE,
98*5e3eaea3SApple OSS Distributions .type = MACH_MSG_PORT_DESCRIPTOR,
99*5e3eaea3SApple OSS Distributions };
100*5e3eaea3SApple OSS Distributions
101*5e3eaea3SApple OSS Distributions kr = mach_msg(&request->header, MACH_SEND_MSG, sizeof(*request), 0, MACH_PORT_NULL, MACH_MSG_TIMEOUT_NONE, MACH_PORT_NULL);
102*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(kr, MACH_MSG_SUCCESS, "mach_msg");
103*5e3eaea3SApple OSS Distributions
104*5e3eaea3SApple OSS Distributions struct kevent_qos_s event = {
105*5e3eaea3SApple OSS Distributions .ident = sync_port,
106*5e3eaea3SApple OSS Distributions .filter = EVFILT_MACHPORT,
107*5e3eaea3SApple OSS Distributions .flags = EV_ADD | EV_ENABLE | EV_DISPATCH,
108*5e3eaea3SApple OSS Distributions .qos = 0xA00,
109*5e3eaea3SApple OSS Distributions .udata = 0x42424242,
110*5e3eaea3SApple OSS Distributions .fflags = MACH_RCV_MSG,
111*5e3eaea3SApple OSS Distributions .xflags = 0x00,
112*5e3eaea3SApple OSS Distributions .data = 0x00,
113*5e3eaea3SApple OSS Distributions .ext = {(uint64_t)reply, sizeof(*reply), 0, 0},
114*5e3eaea3SApple OSS Distributions };
115*5e3eaea3SApple OSS Distributions
116*5e3eaea3SApple OSS Distributions struct kevent_qos_s out_events[1];
117*5e3eaea3SApple OSS Distributions
118*5e3eaea3SApple OSS Distributions int nevents = kevent_qos(kq, &event, 1, out_events, 1, NULL, NULL, 0);
119*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(nevents, (int)1, "kevent_qos");
120*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(out_events[0].udata, (uint64_t)0x42424242, "kevent_qos");
121*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_BITS_SET(reply->header.msgh_bits, MACH_MSGH_BITS_COMPLEX, "message is complex");
122*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(reply->body.msgh_descriptor_count, (mach_msg_size_t)2, "mach_msg");
123*5e3eaea3SApple OSS Distributions
124*5e3eaea3SApple OSS Distributions ports[i] = reply->port.name;
125*5e3eaea3SApple OSS Distributions kq_port = reply->kq_port.name;
126*5e3eaea3SApple OSS Distributions }
127*5e3eaea3SApple OSS Distributions
128*5e3eaea3SApple OSS Distributions context->initialized = 1;
129*5e3eaea3SApple OSS Distributions while (!context->start) {
130*5e3eaea3SApple OSS Distributions }
131*5e3eaea3SApple OSS Distributions
132*5e3eaea3SApple OSS Distributions for (size_t i = 0; i < PORTS_COUNT; i++) {
133*5e3eaea3SApple OSS Distributions uint32_t wl_id = (uint32_t)0x99999999;
134*5e3eaea3SApple OSS Distributions
135*5e3eaea3SApple OSS Distributions struct kevent_qos_s event = {
136*5e3eaea3SApple OSS Distributions .ident = ports[i],
137*5e3eaea3SApple OSS Distributions .filter = EVFILT_WORKLOOP,
138*5e3eaea3SApple OSS Distributions .flags = EV_ADD | EV_DISABLE,
139*5e3eaea3SApple OSS Distributions .qos = 0x00,
140*5e3eaea3SApple OSS Distributions .udata = 0x88888888,
141*5e3eaea3SApple OSS Distributions .fflags = NOTE_WL_SYNC_IPC,
142*5e3eaea3SApple OSS Distributions .xflags = 0x00,
143*5e3eaea3SApple OSS Distributions .data = 0x66666666,
144*5e3eaea3SApple OSS Distributions .ext = {0x00, 0x00, 0x00, 0x00},
145*5e3eaea3SApple OSS Distributions };
146*5e3eaea3SApple OSS Distributions struct kevent_qos_s output = { };
147*5e3eaea3SApple OSS Distributions int ret = kevent_id(wl_id, &event, 1, &output, 1, NULL, NULL,
148*5e3eaea3SApple OSS Distributions KEVENT_FLAG_WORKLOOP | KEVENT_FLAG_ERROR_EVENTS);
149*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "kevent_id");
150*5e3eaea3SApple OSS Distributions }
151*5e3eaea3SApple OSS Distributions return NULL;
152*5e3eaea3SApple OSS Distributions }
153*5e3eaea3SApple OSS Distributions
154*5e3eaea3SApple OSS Distributions T_DECL(test_knote_use_after_free,
155*5e3eaea3SApple OSS Distributions "Verify knote use-after-free issue does not reproduce - rdar://96567281 (Knote Use-after-Free in XNU)",
156*5e3eaea3SApple OSS Distributions T_META_CHECK_LEAKS(false))
157*5e3eaea3SApple OSS Distributions {
158*5e3eaea3SApple OSS Distributions mach_port_t task = mach_task_self();
159*5e3eaea3SApple OSS Distributions
160*5e3eaea3SApple OSS Distributions knote_context_t context = {
161*5e3eaea3SApple OSS Distributions .initialized = 0,
162*5e3eaea3SApple OSS Distributions .start = 0,
163*5e3eaea3SApple OSS Distributions };
164*5e3eaea3SApple OSS Distributions
165*5e3eaea3SApple OSS Distributions pthread_t thknote;
166*5e3eaea3SApple OSS Distributions T_ASSERT_POSIX_ZERO(pthread_create(&thknote, NULL, th_allocate_knotes, &context), "pthread_create");
167*5e3eaea3SApple OSS Distributions
168*5e3eaea3SApple OSS Distributions int kq = kqueue();
169*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(kq, "kqueue");
170*5e3eaea3SApple OSS Distributions
171*5e3eaea3SApple OSS Distributions #define KNOTE_PORT_COUNT 2
172*5e3eaea3SApple OSS Distributions
173*5e3eaea3SApple OSS Distributions kern_return_t kr = KERN_SUCCESS;
174*5e3eaea3SApple OSS Distributions mach_port_t sync_port = MACH_PORT_NULL, knote_port[KNOTE_PORT_COUNT];
175*5e3eaea3SApple OSS Distributions kr = mach_port_allocate(task, MACH_PORT_RIGHT_RECEIVE, &sync_port);
176*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "mach_port_allocate sync_port");
177*5e3eaea3SApple OSS Distributions
178*5e3eaea3SApple OSS Distributions for (size_t i = 0; i < KNOTE_PORT_COUNT; i++) {
179*5e3eaea3SApple OSS Distributions knote_port[i] = MACH_PORT_NULL;
180*5e3eaea3SApple OSS Distributions kr = mach_port_allocate(task, MACH_PORT_RIGHT_RECEIVE, &knote_port[i]);
181*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "mach_port_allocate knote_port");
182*5e3eaea3SApple OSS Distributions }
183*5e3eaea3SApple OSS Distributions
184*5e3eaea3SApple OSS Distributions typedef struct sync_knote_msg_local_s sync_knote_msg_local_t;
185*5e3eaea3SApple OSS Distributions typedef struct sync_knote_msg_remote_s sync_knote_msg_remote_t;
186*5e3eaea3SApple OSS Distributions
187*5e3eaea3SApple OSS Distributions #pragma pack(4)
188*5e3eaea3SApple OSS Distributions struct sync_knote_msg_local_s {
189*5e3eaea3SApple OSS Distributions mach_msg_header_t header;
190*5e3eaea3SApple OSS Distributions mach_msg_body_t body;
191*5e3eaea3SApple OSS Distributions mach_msg_port_descriptor_t port[KNOTE_PORT_COUNT];
192*5e3eaea3SApple OSS Distributions uint64_t sequence;
193*5e3eaea3SApple OSS Distributions };
194*5e3eaea3SApple OSS Distributions #pragma pack(0)
195*5e3eaea3SApple OSS Distributions
196*5e3eaea3SApple OSS Distributions #pragma pack(4)
197*5e3eaea3SApple OSS Distributions struct sync_knote_msg_remote_s {
198*5e3eaea3SApple OSS Distributions mach_msg_header_t header;
199*5e3eaea3SApple OSS Distributions mach_msg_body_t body;
200*5e3eaea3SApple OSS Distributions mach_msg_port_descriptor_t port[KNOTE_PORT_COUNT];
201*5e3eaea3SApple OSS Distributions uint64_t sequence;
202*5e3eaea3SApple OSS Distributions mach_msg_trailer_t trailer;
203*5e3eaea3SApple OSS Distributions };
204*5e3eaea3SApple OSS Distributions #pragma pack(0)
205*5e3eaea3SApple OSS Distributions
206*5e3eaea3SApple OSS Distributions union {
207*5e3eaea3SApple OSS Distributions sync_knote_msg_local_t local;
208*5e3eaea3SApple OSS Distributions sync_knote_msg_remote_t remote;
209*5e3eaea3SApple OSS Distributions } message;
210*5e3eaea3SApple OSS Distributions
211*5e3eaea3SApple OSS Distributions sync_knote_msg_local_t *local = &message.local;
212*5e3eaea3SApple OSS Distributions sync_knote_msg_remote_t *remote = &message.remote;
213*5e3eaea3SApple OSS Distributions
214*5e3eaea3SApple OSS Distributions local->header = (mach_msg_header_t){
215*5e3eaea3SApple OSS Distributions .msgh_bits = MACH_MSGH_BITS_SET(MACH_MSG_TYPE_MAKE_SEND, 0, 0, MACH_MSGH_BITS_COMPLEX),
216*5e3eaea3SApple OSS Distributions .msgh_remote_port = sync_port,
217*5e3eaea3SApple OSS Distributions .msgh_local_port = MACH_PORT_NULL,
218*5e3eaea3SApple OSS Distributions .msgh_voucher_port = MACH_PORT_NULL,
219*5e3eaea3SApple OSS Distributions .msgh_size = sizeof(sync_knote_msg_local_t),
220*5e3eaea3SApple OSS Distributions .msgh_id = (mach_msg_id_t)0x88888888,
221*5e3eaea3SApple OSS Distributions };
222*5e3eaea3SApple OSS Distributions local->body.msgh_descriptor_count = KNOTE_PORT_COUNT;
223*5e3eaea3SApple OSS Distributions for (size_t i = 0; i < KNOTE_PORT_COUNT; i++) {
224*5e3eaea3SApple OSS Distributions local->port[i] = (mach_msg_port_descriptor_t){
225*5e3eaea3SApple OSS Distributions .name = knote_port[i],
226*5e3eaea3SApple OSS Distributions .disposition = MACH_MSG_TYPE_MOVE_RECEIVE,
227*5e3eaea3SApple OSS Distributions .type = MACH_MSG_PORT_DESCRIPTOR,
228*5e3eaea3SApple OSS Distributions };
229*5e3eaea3SApple OSS Distributions }
230*5e3eaea3SApple OSS Distributions local->sequence = 0x6666666666666666;
231*5e3eaea3SApple OSS Distributions kr = mach_msg(&local->header, MACH_SEND_MSG, sizeof(sync_knote_msg_local_t), 0, MACH_PORT_NULL,
232*5e3eaea3SApple OSS Distributions 0, MACH_PORT_NULL);
233*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(kr, MACH_MSG_SUCCESS, "mach_msg");
234*5e3eaea3SApple OSS Distributions
235*5e3eaea3SApple OSS Distributions struct kevent_qos_s event = {
236*5e3eaea3SApple OSS Distributions .ident = sync_port,
237*5e3eaea3SApple OSS Distributions .filter = EVFILT_MACHPORT,
238*5e3eaea3SApple OSS Distributions .flags = EV_ADD | EV_ENABLE | EV_DISPATCH,
239*5e3eaea3SApple OSS Distributions .qos = 0xA00,
240*5e3eaea3SApple OSS Distributions .udata = 42424242,
241*5e3eaea3SApple OSS Distributions .fflags = MACH_RCV_MSG,
242*5e3eaea3SApple OSS Distributions .xflags = 0x00,
243*5e3eaea3SApple OSS Distributions .data = 0x00,
244*5e3eaea3SApple OSS Distributions .ext = {(uint64_t)remote, sizeof(*remote), 0, 0},
245*5e3eaea3SApple OSS Distributions };
246*5e3eaea3SApple OSS Distributions
247*5e3eaea3SApple OSS Distributions struct kevent_qos_s out_events[1];
248*5e3eaea3SApple OSS Distributions
249*5e3eaea3SApple OSS Distributions int nevents = kevent_qos(kq, &event, 1, out_events, 1, NULL, NULL, 0);
250*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(nevents, (int)1, "kevent_qos nevents");
251*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(remote->sequence, (uint64_t)0x6666666666666666, "kevent_qos remote->sequence");
252*5e3eaea3SApple OSS Distributions
253*5e3eaea3SApple OSS Distributions int ret = 0;
254*5e3eaea3SApple OSS Distributions struct kevent_qos_s del_event = {
255*5e3eaea3SApple OSS Distributions .ident = sync_port,
256*5e3eaea3SApple OSS Distributions .filter = EVFILT_MACHPORT,
257*5e3eaea3SApple OSS Distributions .flags = EV_DELETE,
258*5e3eaea3SApple OSS Distributions .qos = 0xA00,
259*5e3eaea3SApple OSS Distributions .udata = 0x00,
260*5e3eaea3SApple OSS Distributions .fflags = MACH_RCV_MSG,
261*5e3eaea3SApple OSS Distributions .xflags = 0x00,
262*5e3eaea3SApple OSS Distributions .data = 0x00,
263*5e3eaea3SApple OSS Distributions .ext = {0, 0, 0, 0},
264*5e3eaea3SApple OSS Distributions };
265*5e3eaea3SApple OSS Distributions
266*5e3eaea3SApple OSS Distributions ret = kevent_qos(kq, &del_event, 1, NULL, 0, NULL, NULL, 0);
267*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_EQ(ret, (int)0, "kevent_qos return");
268*5e3eaea3SApple OSS Distributions
269*5e3eaea3SApple OSS Distributions while (!context.initialized) {
270*5e3eaea3SApple OSS Distributions }
271*5e3eaea3SApple OSS Distributions
272*5e3eaea3SApple OSS Distributions context.start = 1;
273*5e3eaea3SApple OSS Distributions T_ASSERT_POSIX_ZERO(pthread_join(thknote, NULL), "pthread_join");
274*5e3eaea3SApple OSS Distributions
275*5e3eaea3SApple OSS Distributions kr = _kernelrpc_mach_port_insert_right_trap(task, sync_port, sync_port, MACH_MSG_TYPE_MOVE_RECEIVE);
276*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "_kernelrpc_mach_port_insert_right_trap");
277*5e3eaea3SApple OSS Distributions }
278