xref: /freebsd-src/tools/regression/sockets/unix_cmsg/t_sockcred.c (revision 6bfca4dcab07dad45a805879d954876b353c0810)
1*1fab62b1SMaxim Sobolev /*-
2*1fab62b1SMaxim Sobolev  * Copyright (c) 2005 Andrey Simonenko
3*1fab62b1SMaxim Sobolev  * All rights reserved.
4*1fab62b1SMaxim Sobolev  *
5*1fab62b1SMaxim Sobolev  * Redistribution and use in source and binary forms, with or without
6*1fab62b1SMaxim Sobolev  * modification, are permitted provided that the following conditions
7*1fab62b1SMaxim Sobolev  * are met:
8*1fab62b1SMaxim Sobolev  * 1. Redistributions of source code must retain the above copyright
9*1fab62b1SMaxim Sobolev  *    notice, this list of conditions and the following disclaimer.
10*1fab62b1SMaxim Sobolev  * 2. Redistributions in binary form must reproduce the above copyright
11*1fab62b1SMaxim Sobolev  *    notice, this list of conditions and the following disclaimer in the
12*1fab62b1SMaxim Sobolev  *    documentation and/or other materials provided with the distribution.
13*1fab62b1SMaxim Sobolev  *
14*1fab62b1SMaxim Sobolev  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15*1fab62b1SMaxim Sobolev  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16*1fab62b1SMaxim Sobolev  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17*1fab62b1SMaxim Sobolev  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18*1fab62b1SMaxim Sobolev  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19*1fab62b1SMaxim Sobolev  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20*1fab62b1SMaxim Sobolev  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21*1fab62b1SMaxim Sobolev  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22*1fab62b1SMaxim Sobolev  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23*1fab62b1SMaxim Sobolev  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24*1fab62b1SMaxim Sobolev  * SUCH DAMAGE.
25*1fab62b1SMaxim Sobolev  */
26*1fab62b1SMaxim Sobolev 
27*1fab62b1SMaxim Sobolev #include <sys/types.h>
28*1fab62b1SMaxim Sobolev #include <sys/socket.h>
29*1fab62b1SMaxim Sobolev #include <sys/un.h>
30*1fab62b1SMaxim Sobolev #include <inttypes.h>
31*1fab62b1SMaxim Sobolev #include <stdarg.h>
32*1fab62b1SMaxim Sobolev #include <stdbool.h>
33*1fab62b1SMaxim Sobolev #include <stdlib.h>
34*1fab62b1SMaxim Sobolev 
35*1fab62b1SMaxim Sobolev #include "uc_common.h"
36*1fab62b1SMaxim Sobolev #include "t_generic.h"
37*1fab62b1SMaxim Sobolev #include "t_sockcred.h"
38*1fab62b1SMaxim Sobolev 
39*1fab62b1SMaxim Sobolev static int
t_sockcred_client(int type,int fd)40*1fab62b1SMaxim Sobolev t_sockcred_client(int type, int fd)
41*1fab62b1SMaxim Sobolev {
42*1fab62b1SMaxim Sobolev 	struct msghdr msghdr;
43*1fab62b1SMaxim Sobolev 	struct iovec iov[1];
44*1fab62b1SMaxim Sobolev 	int rv;
45*1fab62b1SMaxim Sobolev 
46*1fab62b1SMaxim Sobolev 	if (uc_sync_recv() < 0)
47*1fab62b1SMaxim Sobolev 		return (-2);
48*1fab62b1SMaxim Sobolev 
49*1fab62b1SMaxim Sobolev 	rv = -2;
50*1fab62b1SMaxim Sobolev 
51*1fab62b1SMaxim Sobolev 	uc_msghdr_init_client(&msghdr, iov, NULL, 0, 0, 0);
52*1fab62b1SMaxim Sobolev 
53*1fab62b1SMaxim Sobolev 	if (uc_socket_connect(fd) < 0)
54*1fab62b1SMaxim Sobolev 		goto done;
55*1fab62b1SMaxim Sobolev 
56*1fab62b1SMaxim Sobolev 	if (type == 2)
57*1fab62b1SMaxim Sobolev 		if (uc_sync_recv() < 0)
58*1fab62b1SMaxim Sobolev 			goto done;
59*1fab62b1SMaxim Sobolev 
60*1fab62b1SMaxim Sobolev 	if (uc_message_sendn(fd, &msghdr) < 0)
61*1fab62b1SMaxim Sobolev 		goto done;
62*1fab62b1SMaxim Sobolev 
63*1fab62b1SMaxim Sobolev 	rv = 0;
64*1fab62b1SMaxim Sobolev done:
65*1fab62b1SMaxim Sobolev 	return (rv);
66*1fab62b1SMaxim Sobolev }
67*1fab62b1SMaxim Sobolev 
68*1fab62b1SMaxim Sobolev static int
t_sockcred_server(int type,int fd1)69*1fab62b1SMaxim Sobolev t_sockcred_server(int type, int fd1)
70*1fab62b1SMaxim Sobolev {
71*1fab62b1SMaxim Sobolev 	struct msghdr msghdr;
72*1fab62b1SMaxim Sobolev 	struct iovec iov[1];
73*1fab62b1SMaxim Sobolev 	struct cmsghdr *cmsghdr;
74*1fab62b1SMaxim Sobolev 	void *cmsg_data;
75*1fab62b1SMaxim Sobolev 	size_t cmsg_size;
76*1fab62b1SMaxim Sobolev 	u_int i;
77*1fab62b1SMaxim Sobolev 	int fd2, rv, val;
78*1fab62b1SMaxim Sobolev 
79*1fab62b1SMaxim Sobolev 	fd2 = -1;
80*1fab62b1SMaxim Sobolev 	rv = -2;
81*1fab62b1SMaxim Sobolev 
82*1fab62b1SMaxim Sobolev 	cmsg_size = CMSG_SPACE(SOCKCREDSIZE(uc_cfg.proc_cred.gid_num));
83*1fab62b1SMaxim Sobolev 	cmsg_data = malloc(cmsg_size);
84*1fab62b1SMaxim Sobolev 	if (cmsg_data == NULL) {
85*1fab62b1SMaxim Sobolev 		uc_logmsg("malloc");
86*1fab62b1SMaxim Sobolev 		goto done;
87*1fab62b1SMaxim Sobolev 	}
88*1fab62b1SMaxim Sobolev 
89*1fab62b1SMaxim Sobolev 	if (type == 1) {
90*1fab62b1SMaxim Sobolev 		uc_dbgmsg("setting LOCAL_CREDS");
91*1fab62b1SMaxim Sobolev 		val = 1;
92*1fab62b1SMaxim Sobolev 		if (setsockopt(fd1, 0, LOCAL_CREDS, &val, sizeof(val)) < 0) {
93*1fab62b1SMaxim Sobolev 			uc_logmsg("setsockopt(LOCAL_CREDS)");
94*1fab62b1SMaxim Sobolev 			goto done;
95*1fab62b1SMaxim Sobolev 		}
96*1fab62b1SMaxim Sobolev 	}
97*1fab62b1SMaxim Sobolev 
98*1fab62b1SMaxim Sobolev 	if (uc_sync_send() < 0)
99*1fab62b1SMaxim Sobolev 		goto done;
100*1fab62b1SMaxim Sobolev 
101*1fab62b1SMaxim Sobolev 	if (uc_cfg.sock_type == SOCK_STREAM) {
102*1fab62b1SMaxim Sobolev 		fd2 = uc_socket_accept(fd1);
103*1fab62b1SMaxim Sobolev 		if (fd2 < 0)
104*1fab62b1SMaxim Sobolev 			goto done;
105*1fab62b1SMaxim Sobolev 	} else
106*1fab62b1SMaxim Sobolev 		fd2 = fd1;
107*1fab62b1SMaxim Sobolev 
108*1fab62b1SMaxim Sobolev 	if (type == 2) {
109*1fab62b1SMaxim Sobolev 		uc_dbgmsg("setting LOCAL_CREDS");
110*1fab62b1SMaxim Sobolev 		val = 1;
111*1fab62b1SMaxim Sobolev 		if (setsockopt(fd2, 0, LOCAL_CREDS, &val, sizeof(val)) < 0) {
112*1fab62b1SMaxim Sobolev 			uc_logmsg("setsockopt(LOCAL_CREDS)");
113*1fab62b1SMaxim Sobolev 			goto done;
114*1fab62b1SMaxim Sobolev 		}
115*1fab62b1SMaxim Sobolev 		if (uc_sync_send() < 0)
116*1fab62b1SMaxim Sobolev 			goto done;
117*1fab62b1SMaxim Sobolev 	}
118*1fab62b1SMaxim Sobolev 
119*1fab62b1SMaxim Sobolev 	rv = -1;
120*1fab62b1SMaxim Sobolev 	for (i = 1; i <= uc_cfg.ipc_msg.msg_num; ++i) {
121*1fab62b1SMaxim Sobolev 		uc_dbgmsg("message #%u", i);
122*1fab62b1SMaxim Sobolev 
123*1fab62b1SMaxim Sobolev 		uc_msghdr_init_server(&msghdr, iov, cmsg_data, cmsg_size);
124*1fab62b1SMaxim Sobolev 		if (uc_message_recv(fd2, &msghdr) < 0) {
125*1fab62b1SMaxim Sobolev 			rv = -2;
126*1fab62b1SMaxim Sobolev 			break;
127*1fab62b1SMaxim Sobolev 		}
128*1fab62b1SMaxim Sobolev 
129*1fab62b1SMaxim Sobolev 		if (i > 1 && uc_cfg.sock_type == SOCK_STREAM) {
130*1fab62b1SMaxim Sobolev 			if (uc_check_msghdr(&msghdr, 0) < 0)
131*1fab62b1SMaxim Sobolev 				break;
132*1fab62b1SMaxim Sobolev 		} else {
133*1fab62b1SMaxim Sobolev 			if (uc_check_msghdr(&msghdr, sizeof(*cmsghdr)) < 0)
134*1fab62b1SMaxim Sobolev 				break;
135*1fab62b1SMaxim Sobolev 
136*1fab62b1SMaxim Sobolev 			cmsghdr = CMSG_FIRSTHDR(&msghdr);
137*1fab62b1SMaxim Sobolev 			if (uc_check_scm_creds_sockcred(cmsghdr) < 0)
138*1fab62b1SMaxim Sobolev 				break;
139*1fab62b1SMaxim Sobolev 		}
140*1fab62b1SMaxim Sobolev 	}
141*1fab62b1SMaxim Sobolev 	if (i > uc_cfg.ipc_msg.msg_num)
142*1fab62b1SMaxim Sobolev 		rv = 0;
143*1fab62b1SMaxim Sobolev done:
144*1fab62b1SMaxim Sobolev 	free(cmsg_data);
145*1fab62b1SMaxim Sobolev 	if (uc_cfg.sock_type == SOCK_STREAM && fd2 >= 0)
146*1fab62b1SMaxim Sobolev 		if (uc_socket_close(fd2) < 0)
147*1fab62b1SMaxim Sobolev 			rv = -2;
148*1fab62b1SMaxim Sobolev 	return (rv);
149*1fab62b1SMaxim Sobolev }
150*1fab62b1SMaxim Sobolev 
151*1fab62b1SMaxim Sobolev int
t_sockcred_1(void)152*1fab62b1SMaxim Sobolev t_sockcred_1(void)
153*1fab62b1SMaxim Sobolev {
154*1fab62b1SMaxim Sobolev 	u_int i;
155*1fab62b1SMaxim Sobolev 	int fd, rv, rv_client;
156*1fab62b1SMaxim Sobolev 
157*1fab62b1SMaxim Sobolev 	switch (uc_client_fork()) {
158*1fab62b1SMaxim Sobolev 	case 0:
159*1fab62b1SMaxim Sobolev 		for (i = 1; i <= 2; ++i) {
160*1fab62b1SMaxim Sobolev 			uc_dbgmsg("client #%u", i);
161*1fab62b1SMaxim Sobolev 			fd = uc_socket_create();
162*1fab62b1SMaxim Sobolev 			if (fd < 0)
163*1fab62b1SMaxim Sobolev 				rv = -2;
164*1fab62b1SMaxim Sobolev 			else {
165*1fab62b1SMaxim Sobolev 				rv = t_sockcred_client(1, fd);
166*1fab62b1SMaxim Sobolev 				if (uc_socket_close(fd) < 0)
167*1fab62b1SMaxim Sobolev 					rv = -2;
168*1fab62b1SMaxim Sobolev 			}
169*1fab62b1SMaxim Sobolev 			if (rv != 0)
170*1fab62b1SMaxim Sobolev 				break;
171*1fab62b1SMaxim Sobolev 		}
172*1fab62b1SMaxim Sobolev 		uc_client_exit(rv);
173*1fab62b1SMaxim Sobolev 		break;
174*1fab62b1SMaxim Sobolev 	case 1:
175*1fab62b1SMaxim Sobolev 		fd = uc_socket_create();
176*1fab62b1SMaxim Sobolev 		if (fd < 0)
177*1fab62b1SMaxim Sobolev 			rv = -2;
178*1fab62b1SMaxim Sobolev 		else {
179*1fab62b1SMaxim Sobolev 			rv = t_sockcred_server(1, fd);
180*1fab62b1SMaxim Sobolev 			if (rv == 0)
181*1fab62b1SMaxim Sobolev 				rv = t_sockcred_server(3, fd);
182*1fab62b1SMaxim Sobolev 			rv_client = uc_client_wait();
183*1fab62b1SMaxim Sobolev 			if (rv == 0 || (rv == -2 && rv_client != 0))
184*1fab62b1SMaxim Sobolev 				rv = rv_client;
185*1fab62b1SMaxim Sobolev 			if (uc_socket_close(fd) < 0)
186*1fab62b1SMaxim Sobolev 				rv = -2;
187*1fab62b1SMaxim Sobolev 		}
188*1fab62b1SMaxim Sobolev 		break;
189*1fab62b1SMaxim Sobolev 	default:
190*1fab62b1SMaxim Sobolev 		rv = -2;
191*1fab62b1SMaxim Sobolev 	}
192*1fab62b1SMaxim Sobolev 
193*1fab62b1SMaxim Sobolev 	return (rv);
194*1fab62b1SMaxim Sobolev }
195*1fab62b1SMaxim Sobolev 
196*1fab62b1SMaxim Sobolev static int
t_sockcred_2_client(int fd)197*1fab62b1SMaxim Sobolev t_sockcred_2_client(int fd)
198*1fab62b1SMaxim Sobolev {
199*1fab62b1SMaxim Sobolev 	return (t_sockcred_client(2, fd));
200*1fab62b1SMaxim Sobolev }
201*1fab62b1SMaxim Sobolev 
202*1fab62b1SMaxim Sobolev static int
t_sockcred_2_server(int fd)203*1fab62b1SMaxim Sobolev t_sockcred_2_server(int fd)
204*1fab62b1SMaxim Sobolev {
205*1fab62b1SMaxim Sobolev 	return (t_sockcred_server(2, fd));
206*1fab62b1SMaxim Sobolev }
207*1fab62b1SMaxim Sobolev 
208*1fab62b1SMaxim Sobolev int
t_sockcred_2(void)209*1fab62b1SMaxim Sobolev t_sockcred_2(void)
210*1fab62b1SMaxim Sobolev {
211*1fab62b1SMaxim Sobolev 	return (t_generic(t_sockcred_2_client, t_sockcred_2_server));
212*1fab62b1SMaxim Sobolev }
213