1*a1e26a70SApple OSS Distributions #include <darwintest.h>
2*a1e26a70SApple OSS Distributions #include <darwintest_multiprocess.h>
3*a1e26a70SApple OSS Distributions #include <launch.h>
4*a1e26a70SApple OSS Distributions #include <servers/bootstrap.h>
5*a1e26a70SApple OSS Distributions #include <sys/sysctl.h>
6*a1e26a70SApple OSS Distributions #include "exc_helpers.h"
7*a1e26a70SApple OSS Distributions
8*a1e26a70SApple OSS Distributions T_GLOBAL_META(
9*a1e26a70SApple OSS Distributions T_META_NAMESPACE("xnu.ipc"),
10*a1e26a70SApple OSS Distributions T_META_RADAR_COMPONENT_NAME("xnu"),
11*a1e26a70SApple OSS Distributions T_META_RADAR_COMPONENT_VERSION("IPC"),
12*a1e26a70SApple OSS Distributions T_META_RUN_CONCURRENTLY(true),
13*a1e26a70SApple OSS Distributions T_META_TAG_VM_PREFERRED);
14*a1e26a70SApple OSS Distributions
15*a1e26a70SApple OSS Distributions #pragma mark - helpers
16*a1e26a70SApple OSS Distributions
17*a1e26a70SApple OSS Distributions #define SERVICE_NAME "com.apple.xnu.test.mach_port"
18*a1e26a70SApple OSS Distributions
19*a1e26a70SApple OSS Distributions struct one_port_msg {
20*a1e26a70SApple OSS Distributions mach_msg_header_t header;
21*a1e26a70SApple OSS Distributions mach_msg_body_t body;
22*a1e26a70SApple OSS Distributions mach_msg_port_descriptor_t port_descriptor;
23*a1e26a70SApple OSS Distributions mach_msg_trailer_t trailer; // subtract this when sending
24*a1e26a70SApple OSS Distributions };
25*a1e26a70SApple OSS Distributions
26*a1e26a70SApple OSS Distributions static mach_port_t
server_checkin(void)27*a1e26a70SApple OSS Distributions server_checkin(void)
28*a1e26a70SApple OSS Distributions {
29*a1e26a70SApple OSS Distributions mach_port_t mp;
30*a1e26a70SApple OSS Distributions kern_return_t kr;
31*a1e26a70SApple OSS Distributions
32*a1e26a70SApple OSS Distributions kr = bootstrap_check_in(bootstrap_port, SERVICE_NAME, &mp);
33*a1e26a70SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "bootstrap_check_in");
34*a1e26a70SApple OSS Distributions return mp;
35*a1e26a70SApple OSS Distributions }
36*a1e26a70SApple OSS Distributions
37*a1e26a70SApple OSS Distributions static mach_port_t
server_lookup(void)38*a1e26a70SApple OSS Distributions server_lookup(void)
39*a1e26a70SApple OSS Distributions {
40*a1e26a70SApple OSS Distributions mach_port_t mp;
41*a1e26a70SApple OSS Distributions kern_return_t kr;
42*a1e26a70SApple OSS Distributions
43*a1e26a70SApple OSS Distributions kr = bootstrap_look_up(bootstrap_port, SERVICE_NAME, &mp);
44*a1e26a70SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "bootstrap_look_up");
45*a1e26a70SApple OSS Distributions return mp;
46*a1e26a70SApple OSS Distributions }
47*a1e26a70SApple OSS Distributions
48*a1e26a70SApple OSS Distributions static mach_port_t
make_sr_port(void)49*a1e26a70SApple OSS Distributions make_sr_port(void)
50*a1e26a70SApple OSS Distributions {
51*a1e26a70SApple OSS Distributions mach_port_options_t opts = {
52*a1e26a70SApple OSS Distributions .flags = MPO_INSERT_SEND_RIGHT,
53*a1e26a70SApple OSS Distributions };
54*a1e26a70SApple OSS Distributions kern_return_t kr;
55*a1e26a70SApple OSS Distributions mach_port_t port;
56*a1e26a70SApple OSS Distributions
57*a1e26a70SApple OSS Distributions kr = mach_port_construct(mach_task_self(), &opts, 0ull, &port);
58*a1e26a70SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "mach_port_construct");
59*a1e26a70SApple OSS Distributions return port;
60*a1e26a70SApple OSS Distributions }
61*a1e26a70SApple OSS Distributions
62*a1e26a70SApple OSS Distributions static void
destroy_port(mach_port_t port,bool receive,int srights)63*a1e26a70SApple OSS Distributions destroy_port(mach_port_t port, bool receive, int srights)
64*a1e26a70SApple OSS Distributions {
65*a1e26a70SApple OSS Distributions kern_return_t kr;
66*a1e26a70SApple OSS Distributions
67*a1e26a70SApple OSS Distributions if (srights) {
68*a1e26a70SApple OSS Distributions kr = mach_port_mod_refs(mach_task_self(), port,
69*a1e26a70SApple OSS Distributions MACH_PORT_RIGHT_SEND, -srights);
70*a1e26a70SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "srights -= %d", srights);
71*a1e26a70SApple OSS Distributions }
72*a1e26a70SApple OSS Distributions if (receive) {
73*a1e26a70SApple OSS Distributions kr = mach_port_mod_refs(mach_task_self(), port,
74*a1e26a70SApple OSS Distributions MACH_PORT_RIGHT_RECEIVE, -1);
75*a1e26a70SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "receive -= 1");
76*a1e26a70SApple OSS Distributions }
77*a1e26a70SApple OSS Distributions }
78*a1e26a70SApple OSS Distributions
79*a1e26a70SApple OSS Distributions static void
send_port(mach_msg_id_t id,mach_port_t dest,mach_port_t right,mach_msg_type_name_t disp)80*a1e26a70SApple OSS Distributions send_port(
81*a1e26a70SApple OSS Distributions mach_msg_id_t id,
82*a1e26a70SApple OSS Distributions mach_port_t dest,
83*a1e26a70SApple OSS Distributions mach_port_t right,
84*a1e26a70SApple OSS Distributions mach_msg_type_name_t disp)
85*a1e26a70SApple OSS Distributions {
86*a1e26a70SApple OSS Distributions struct one_port_msg msg = {
87*a1e26a70SApple OSS Distributions .header = {
88*a1e26a70SApple OSS Distributions .msgh_remote_port = dest,
89*a1e26a70SApple OSS Distributions .msgh_bits = MACH_MSGH_BITS_SET(MACH_MSG_TYPE_COPY_SEND,
90*a1e26a70SApple OSS Distributions 0, MACH_MSG_TYPE_MOVE_SEND, MACH_MSGH_BITS_COMPLEX),
91*a1e26a70SApple OSS Distributions .msgh_id = id,
92*a1e26a70SApple OSS Distributions .msgh_size = offsetof(struct one_port_msg, trailer),
93*a1e26a70SApple OSS Distributions },
94*a1e26a70SApple OSS Distributions .body = {
95*a1e26a70SApple OSS Distributions .msgh_descriptor_count = 1,
96*a1e26a70SApple OSS Distributions },
97*a1e26a70SApple OSS Distributions .port_descriptor = {
98*a1e26a70SApple OSS Distributions .name = right,
99*a1e26a70SApple OSS Distributions .disposition = disp,
100*a1e26a70SApple OSS Distributions .type = MACH_MSG_PORT_DESCRIPTOR,
101*a1e26a70SApple OSS Distributions },
102*a1e26a70SApple OSS Distributions };
103*a1e26a70SApple OSS Distributions kern_return_t kr;
104*a1e26a70SApple OSS Distributions
105*a1e26a70SApple OSS Distributions kr = mach_msg(&msg.header, MACH_SEND_MSG | MACH_SEND_TIMEOUT,
106*a1e26a70SApple OSS Distributions msg.header.msgh_size, 0, MACH_PORT_NULL, 10000, 0);
107*a1e26a70SApple OSS Distributions
108*a1e26a70SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "send(%d)", id);
109*a1e26a70SApple OSS Distributions }
110*a1e26a70SApple OSS Distributions
111*a1e26a70SApple OSS Distributions #pragma mark - basic test about right deduplication
112*a1e26a70SApple OSS Distributions
113*a1e26a70SApple OSS Distributions static mach_port_t
receive_port(mach_msg_id_t expected_id,mach_port_t rcv_port,mach_msg_type_name_t expected_disp)114*a1e26a70SApple OSS Distributions receive_port(
115*a1e26a70SApple OSS Distributions mach_msg_id_t expected_id,
116*a1e26a70SApple OSS Distributions mach_port_t rcv_port,
117*a1e26a70SApple OSS Distributions mach_msg_type_name_t expected_disp)
118*a1e26a70SApple OSS Distributions {
119*a1e26a70SApple OSS Distributions struct one_port_msg msg = { };
120*a1e26a70SApple OSS Distributions kern_return_t kr;
121*a1e26a70SApple OSS Distributions
122*a1e26a70SApple OSS Distributions T_LOG("waiting for message %d", expected_id);
123*a1e26a70SApple OSS Distributions kr = mach_msg(&msg.header, MACH_RCV_MSG, 0,
124*a1e26a70SApple OSS Distributions sizeof(msg), rcv_port, 0, 0);
125*a1e26a70SApple OSS Distributions T_QUIET; T_ASSERT_MACH_SUCCESS(kr, "receive(%d)", expected_id);
126*a1e26a70SApple OSS Distributions T_QUIET; T_ASSERT_EQ(msg.header.msgh_id, expected_id, "message id matches");
127*a1e26a70SApple OSS Distributions T_QUIET; T_ASSERT_NE(msg.header.msgh_bits & MACH_MSGH_BITS_COMPLEX, 0,
128*a1e26a70SApple OSS Distributions "message is complex");
129*a1e26a70SApple OSS Distributions T_QUIET; T_ASSERT_EQ(msg.body.msgh_descriptor_count, 1, "message has one right");
130*a1e26a70SApple OSS Distributions T_QUIET; T_ASSERT_EQ((mach_msg_type_name_t)msg.port_descriptor.disposition, expected_disp,
131*a1e26a70SApple OSS Distributions "port has right disposition");
132*a1e26a70SApple OSS Distributions return msg.port_descriptor.name;
133*a1e26a70SApple OSS Distributions }
134*a1e26a70SApple OSS Distributions
135*a1e26a70SApple OSS Distributions T_HELPER_DECL(right_dedup_server, "right_dedup_server")
136*a1e26a70SApple OSS Distributions {
137*a1e26a70SApple OSS Distributions mach_port_t svc_port = server_checkin();
138*a1e26a70SApple OSS Distributions mach_port_t ports[3];
139*a1e26a70SApple OSS Distributions
140*a1e26a70SApple OSS Distributions ports[0] = receive_port(1, svc_port, MACH_MSG_TYPE_MOVE_RECEIVE);
141*a1e26a70SApple OSS Distributions ports[1] = receive_port(2, svc_port, MACH_MSG_TYPE_MOVE_SEND);
142*a1e26a70SApple OSS Distributions ports[2] = receive_port(3, svc_port, MACH_MSG_TYPE_MOVE_SEND);
143*a1e26a70SApple OSS Distributions T_ASSERT_EQ(ports[0], ports[1], "receive, send, send");
144*a1e26a70SApple OSS Distributions T_ASSERT_EQ(ports[0], ports[2], "receive, send, send");
145*a1e26a70SApple OSS Distributions destroy_port(ports[0], true, 2);
146*a1e26a70SApple OSS Distributions
147*a1e26a70SApple OSS Distributions ports[0] = receive_port(4, svc_port, MACH_MSG_TYPE_MOVE_SEND);
148*a1e26a70SApple OSS Distributions ports[1] = receive_port(5, svc_port, MACH_MSG_TYPE_MOVE_RECEIVE);
149*a1e26a70SApple OSS Distributions ports[2] = receive_port(6, svc_port, MACH_MSG_TYPE_MOVE_SEND);
150*a1e26a70SApple OSS Distributions T_ASSERT_EQ(ports[0], ports[1], "send, receive, send");
151*a1e26a70SApple OSS Distributions T_ASSERT_EQ(ports[0], ports[2], "send, receive, send");
152*a1e26a70SApple OSS Distributions destroy_port(ports[0], true, 2);
153*a1e26a70SApple OSS Distributions
154*a1e26a70SApple OSS Distributions ports[0] = receive_port(7, svc_port, MACH_MSG_TYPE_MOVE_SEND);
155*a1e26a70SApple OSS Distributions ports[1] = receive_port(8, svc_port, MACH_MSG_TYPE_MOVE_SEND);
156*a1e26a70SApple OSS Distributions ports[2] = receive_port(9, svc_port, MACH_MSG_TYPE_MOVE_RECEIVE);
157*a1e26a70SApple OSS Distributions T_ASSERT_EQ(ports[0], ports[1], "send, send, receive");
158*a1e26a70SApple OSS Distributions T_ASSERT_EQ(ports[0], ports[2], "send, send, receive");
159*a1e26a70SApple OSS Distributions destroy_port(ports[0], true, 2);
160*a1e26a70SApple OSS Distributions
161*a1e26a70SApple OSS Distributions T_END;
162*a1e26a70SApple OSS Distributions }
163*a1e26a70SApple OSS Distributions
164*a1e26a70SApple OSS Distributions T_HELPER_DECL(right_dedup_client, "right_dedup_client")
165*a1e26a70SApple OSS Distributions {
166*a1e26a70SApple OSS Distributions mach_port_t svc_port = server_lookup();
167*a1e26a70SApple OSS Distributions mach_port_t port;
168*a1e26a70SApple OSS Distributions
169*a1e26a70SApple OSS Distributions port = make_sr_port();
170*a1e26a70SApple OSS Distributions send_port(1, svc_port, port, MACH_MSG_TYPE_MOVE_RECEIVE);
171*a1e26a70SApple OSS Distributions send_port(2, svc_port, port, MACH_MSG_TYPE_COPY_SEND);
172*a1e26a70SApple OSS Distributions send_port(3, svc_port, port, MACH_MSG_TYPE_MOVE_SEND);
173*a1e26a70SApple OSS Distributions
174*a1e26a70SApple OSS Distributions port = make_sr_port();
175*a1e26a70SApple OSS Distributions send_port(4, svc_port, port, MACH_MSG_TYPE_COPY_SEND);
176*a1e26a70SApple OSS Distributions send_port(5, svc_port, port, MACH_MSG_TYPE_MOVE_RECEIVE);
177*a1e26a70SApple OSS Distributions send_port(6, svc_port, port, MACH_MSG_TYPE_MOVE_SEND);
178*a1e26a70SApple OSS Distributions
179*a1e26a70SApple OSS Distributions port = make_sr_port();
180*a1e26a70SApple OSS Distributions send_port(7, svc_port, port, MACH_MSG_TYPE_COPY_SEND);
181*a1e26a70SApple OSS Distributions send_port(8, svc_port, port, MACH_MSG_TYPE_MOVE_SEND);
182*a1e26a70SApple OSS Distributions send_port(9, svc_port, port, MACH_MSG_TYPE_MOVE_RECEIVE);
183*a1e26a70SApple OSS Distributions }
184*a1e26a70SApple OSS Distributions
185*a1e26a70SApple OSS Distributions T_DECL(right_dedup, "make sure right deduplication works")
186*a1e26a70SApple OSS Distributions {
187*a1e26a70SApple OSS Distributions dt_helper_t helpers[] = {
188*a1e26a70SApple OSS Distributions dt_launchd_helper_domain("com.apple.xnu.test.mach_port.plist",
189*a1e26a70SApple OSS Distributions "right_dedup_server", NULL, LAUNCH_SYSTEM_DOMAIN),
190*a1e26a70SApple OSS Distributions dt_fork_helper("right_dedup_client"),
191*a1e26a70SApple OSS Distributions };
192*a1e26a70SApple OSS Distributions dt_run_helpers(helpers, 2, 600);
193*a1e26a70SApple OSS Distributions }
194