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