1*e3723e1fSApple OSS Distributions /* This tests thread_t uaf vulnerability in the XNU kernel due to
2*e3723e1fSApple OSS Distributions * a race condition in unp_connect
3*e3723e1fSApple OSS Distributions */
4*e3723e1fSApple OSS Distributions
5*e3723e1fSApple OSS Distributions #include <sys/un.h>
6*e3723e1fSApple OSS Distributions #include <sys/socket.h>
7*e3723e1fSApple OSS Distributions #include <pthread.h>
8*e3723e1fSApple OSS Distributions #include <sys/proc_info.h>
9*e3723e1fSApple OSS Distributions #include <libproc.h>
10*e3723e1fSApple OSS Distributions #include <darwintest.h>
11*e3723e1fSApple OSS Distributions #include <unistd.h>
12*e3723e1fSApple OSS Distributions
13*e3723e1fSApple OSS Distributions int g_start = 0;
14*e3723e1fSApple OSS Distributions int g_client = 0;
15*e3723e1fSApple OSS Distributions int g_sever1 = 0;
16*e3723e1fSApple OSS Distributions int g_sever2 = 0;
17*e3723e1fSApple OSS Distributions
18*e3723e1fSApple OSS Distributions static void
server_thread1(char * path)19*e3723e1fSApple OSS Distributions server_thread1(char* path)
20*e3723e1fSApple OSS Distributions {
21*e3723e1fSApple OSS Distributions struct sockaddr_un server_sockaddr;
22*e3723e1fSApple OSS Distributions memset(&server_sockaddr, 0, sizeof(struct sockaddr_un));
23*e3723e1fSApple OSS Distributions server_sockaddr.sun_family = AF_UNIX;
24*e3723e1fSApple OSS Distributions strcpy(server_sockaddr.sun_path, path);
25*e3723e1fSApple OSS Distributions unlink(server_sockaddr.sun_path);
26*e3723e1fSApple OSS Distributions
27*e3723e1fSApple OSS Distributions int server_sock = socket(AF_UNIX, SOCK_STREAM, 0);
28*e3723e1fSApple OSS Distributions g_sever1 = server_sock;
29*e3723e1fSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(bind(server_sock,
30*e3723e1fSApple OSS Distributions (struct sockaddr *) &server_sockaddr, sizeof(server_sockaddr)), NULL);
31*e3723e1fSApple OSS Distributions
32*e3723e1fSApple OSS Distributions /*********************************/
33*e3723e1fSApple OSS Distributions /* Listen for any client sockets */
34*e3723e1fSApple OSS Distributions /*********************************/
35*e3723e1fSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(listen(server_sock, -1), NULL);
36*e3723e1fSApple OSS Distributions
37*e3723e1fSApple OSS Distributions return;
38*e3723e1fSApple OSS Distributions }
39*e3723e1fSApple OSS Distributions
40*e3723e1fSApple OSS Distributions static void
server_thread2(char * path)41*e3723e1fSApple OSS Distributions server_thread2(char* path)
42*e3723e1fSApple OSS Distributions {
43*e3723e1fSApple OSS Distributions struct sockaddr_un server_sockaddr;
44*e3723e1fSApple OSS Distributions memset(&server_sockaddr, 0, sizeof(struct sockaddr_un));
45*e3723e1fSApple OSS Distributions server_sockaddr.sun_family = AF_UNIX;
46*e3723e1fSApple OSS Distributions strcpy(server_sockaddr.sun_path, path);
47*e3723e1fSApple OSS Distributions unlink(server_sockaddr.sun_path);
48*e3723e1fSApple OSS Distributions
49*e3723e1fSApple OSS Distributions int server_sock = socket(AF_UNIX, SOCK_STREAM, 0);
50*e3723e1fSApple OSS Distributions g_sever2 = server_sock;
51*e3723e1fSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(bind(server_sock,
52*e3723e1fSApple OSS Distributions (struct sockaddr *) &server_sockaddr, sizeof(server_sockaddr)), NULL);
53*e3723e1fSApple OSS Distributions
54*e3723e1fSApple OSS Distributions /*********************************/
55*e3723e1fSApple OSS Distributions /* Listen for any client sockets */
56*e3723e1fSApple OSS Distributions /*********************************/
57*e3723e1fSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(listen(server_sock, -1), NULL);
58*e3723e1fSApple OSS Distributions
59*e3723e1fSApple OSS Distributions return;
60*e3723e1fSApple OSS Distributions }
61*e3723e1fSApple OSS Distributions
62*e3723e1fSApple OSS Distributions static void
try_to_connect(char * path)63*e3723e1fSApple OSS Distributions try_to_connect(char* path)
64*e3723e1fSApple OSS Distributions {
65*e3723e1fSApple OSS Distributions struct sockaddr_un server_sockaddr;
66*e3723e1fSApple OSS Distributions memset(&server_sockaddr, 0, sizeof(struct sockaddr_un));
67*e3723e1fSApple OSS Distributions server_sockaddr.sun_family = AF_UNIX;
68*e3723e1fSApple OSS Distributions strcpy(server_sockaddr.sun_path, path);
69*e3723e1fSApple OSS Distributions //unlink(server_sockaddr.sun_path);
70*e3723e1fSApple OSS Distributions
71*e3723e1fSApple OSS Distributions while (g_start == 0) {
72*e3723e1fSApple OSS Distributions usleep(100);
73*e3723e1fSApple OSS Distributions }
74*e3723e1fSApple OSS Distributions int ret = connect(g_client, (struct sockaddr *)&server_sockaddr,
75*e3723e1fSApple OSS Distributions sizeof(server_sockaddr));
76*e3723e1fSApple OSS Distributions
77*e3723e1fSApple OSS Distributions T_ASSERT_TRUE(ret == 0 || errno == EALREADY || errno == EISCONN,
78*e3723e1fSApple OSS Distributions "connect with ret: %d(%d)", ret, errno);
79*e3723e1fSApple OSS Distributions }
80*e3723e1fSApple OSS Distributions
81*e3723e1fSApple OSS Distributions
82*e3723e1fSApple OSS Distributions static void
test_unp_connect_multithread()83*e3723e1fSApple OSS Distributions test_unp_connect_multithread()
84*e3723e1fSApple OSS Distributions {
85*e3723e1fSApple OSS Distributions int client_sock;
86*e3723e1fSApple OSS Distributions char path[] = "/tmp/";
87*e3723e1fSApple OSS Distributions char path1[256];
88*e3723e1fSApple OSS Distributions char path2[256];
89*e3723e1fSApple OSS Distributions char path3[256];
90*e3723e1fSApple OSS Distributions
91*e3723e1fSApple OSS Distributions strncpy(path1, path, 255);
92*e3723e1fSApple OSS Distributions strcat(path1, "/1");
93*e3723e1fSApple OSS Distributions strncpy(path2, path, 255);
94*e3723e1fSApple OSS Distributions strcat(path2, "/2");
95*e3723e1fSApple OSS Distributions strncpy(path3, path, 255);
96*e3723e1fSApple OSS Distributions strcat(path3, "/3");
97*e3723e1fSApple OSS Distributions
98*e3723e1fSApple OSS Distributions
99*e3723e1fSApple OSS Distributions for (int i = 0; i < 1024; i++) {
100*e3723e1fSApple OSS Distributions T_SETUPBEGIN;
101*e3723e1fSApple OSS Distributions server_thread1(path1);
102*e3723e1fSApple OSS Distributions server_thread2(path2);
103*e3723e1fSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(client_sock = socket(AF_UNIX, SOCK_STREAM, 0), NULL);
104*e3723e1fSApple OSS Distributions
105*e3723e1fSApple OSS Distributions unlink(path3);
106*e3723e1fSApple OSS Distributions struct sockaddr_un client_sockaddr;
107*e3723e1fSApple OSS Distributions client_sockaddr.sun_family = AF_UNIX;
108*e3723e1fSApple OSS Distributions strcpy(client_sockaddr.sun_path, path3);
109*e3723e1fSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(bind(client_sock, (struct sockaddr *)&client_sockaddr,
110*e3723e1fSApple OSS Distributions sizeof(client_sockaddr)), NULL);
111*e3723e1fSApple OSS Distributions T_SETUPEND;
112*e3723e1fSApple OSS Distributions g_client = client_sock;
113*e3723e1fSApple OSS Distributions g_start = 0;
114*e3723e1fSApple OSS Distributions pthread_t runner1;
115*e3723e1fSApple OSS Distributions pthread_t runner2;
116*e3723e1fSApple OSS Distributions if (pthread_create(&runner1, 0, (void*)try_to_connect, path1)) {
117*e3723e1fSApple OSS Distributions T_ASSERT_FAIL("pthread_create failed");
118*e3723e1fSApple OSS Distributions }
119*e3723e1fSApple OSS Distributions
120*e3723e1fSApple OSS Distributions if (pthread_create(&runner2, 0, (void*)try_to_connect, path2)) {
121*e3723e1fSApple OSS Distributions T_ASSERT_FAIL("pthread_create failed");
122*e3723e1fSApple OSS Distributions }
123*e3723e1fSApple OSS Distributions usleep(300);
124*e3723e1fSApple OSS Distributions g_start = 1;
125*e3723e1fSApple OSS Distributions pthread_join(runner1, 0);
126*e3723e1fSApple OSS Distributions pthread_join(runner2, 0);
127*e3723e1fSApple OSS Distributions
128*e3723e1fSApple OSS Distributions usleep(3000);
129*e3723e1fSApple OSS Distributions
130*e3723e1fSApple OSS Distributions struct socket_fdinfo si_1 = {0};
131*e3723e1fSApple OSS Distributions proc_pidfdinfo(getpid(), g_sever1, PROC_PIDFDSOCKETINFO, &si_1,
132*e3723e1fSApple OSS Distributions sizeof(si_1));
133*e3723e1fSApple OSS Distributions struct socket_fdinfo si_2 = {0};
134*e3723e1fSApple OSS Distributions proc_pidfdinfo(getpid(), g_sever2, PROC_PIDFDSOCKETINFO, &si_2,
135*e3723e1fSApple OSS Distributions sizeof(si_2));
136*e3723e1fSApple OSS Distributions if (si_1.psi.soi_incqlen || si_2.psi.soi_incqlen) {
137*e3723e1fSApple OSS Distributions close(g_sever2);
138*e3723e1fSApple OSS Distributions close(g_sever1);
139*e3723e1fSApple OSS Distributions }
140*e3723e1fSApple OSS Distributions close(client_sock);
141*e3723e1fSApple OSS Distributions close(g_sever2);
142*e3723e1fSApple OSS Distributions close(g_sever1);
143*e3723e1fSApple OSS Distributions }
144*e3723e1fSApple OSS Distributions }
145*e3723e1fSApple OSS Distributions
146*e3723e1fSApple OSS Distributions T_DECL(unp_connect_thread_uaf, "Uaf due to multithreaded unp_connect", T_META_TAG_VM_PREFERRED)
147*e3723e1fSApple OSS Distributions {
148*e3723e1fSApple OSS Distributions test_unp_connect_multithread();
149*e3723e1fSApple OSS Distributions }
150