11fab62b1SMaxim Sobolev /*-
21fab62b1SMaxim Sobolev  * Copyright (c) 2005 Andrey Simonenko
31fab62b1SMaxim Sobolev  * All rights reserved.
41fab62b1SMaxim Sobolev  *
51fab62b1SMaxim Sobolev  * Redistribution and use in source and binary forms, with or without
61fab62b1SMaxim Sobolev  * modification, are permitted provided that the following conditions
71fab62b1SMaxim Sobolev  * are met:
81fab62b1SMaxim Sobolev  * 1. Redistributions of source code must retain the above copyright
91fab62b1SMaxim Sobolev  *    notice, this list of conditions and the following disclaimer.
101fab62b1SMaxim Sobolev  * 2. Redistributions in binary form must reproduce the above copyright
111fab62b1SMaxim Sobolev  *    notice, this list of conditions and the following disclaimer in the
121fab62b1SMaxim Sobolev  *    documentation and/or other materials provided with the distribution.
131fab62b1SMaxim Sobolev  *
141fab62b1SMaxim Sobolev  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
151fab62b1SMaxim Sobolev  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
161fab62b1SMaxim Sobolev  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
171fab62b1SMaxim Sobolev  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
181fab62b1SMaxim Sobolev  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
191fab62b1SMaxim Sobolev  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
201fab62b1SMaxim Sobolev  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
211fab62b1SMaxim Sobolev  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
221fab62b1SMaxim Sobolev  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
231fab62b1SMaxim Sobolev  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
241fab62b1SMaxim Sobolev  * SUCH DAMAGE.
251fab62b1SMaxim Sobolev  */
261fab62b1SMaxim Sobolev 
271fab62b1SMaxim Sobolev #include <sys/types.h>
281fab62b1SMaxim Sobolev #include <sys/socket.h>
291fab62b1SMaxim Sobolev #include <sys/un.h>
301fab62b1SMaxim Sobolev #include <inttypes.h>
311fab62b1SMaxim Sobolev #include <stdarg.h>
321fab62b1SMaxim Sobolev #include <stdbool.h>
331fab62b1SMaxim Sobolev #include <stdlib.h>
341fab62b1SMaxim Sobolev 
351fab62b1SMaxim Sobolev #include "uc_common.h"
361fab62b1SMaxim Sobolev #include "t_generic.h"
371fab62b1SMaxim Sobolev #include "t_cmsgcred.h"
381fab62b1SMaxim Sobolev 
391fab62b1SMaxim Sobolev int
t_cmsgcred_client(int fd)401fab62b1SMaxim Sobolev t_cmsgcred_client(int fd)
411fab62b1SMaxim Sobolev {
421fab62b1SMaxim Sobolev 	struct msghdr msghdr;
431fab62b1SMaxim Sobolev 	struct iovec iov[1];
441fab62b1SMaxim Sobolev 	void *cmsg_data;
451fab62b1SMaxim Sobolev 	size_t cmsg_size;
461fab62b1SMaxim Sobolev 	int rv;
471fab62b1SMaxim Sobolev 
481fab62b1SMaxim Sobolev 	if (uc_sync_recv() < 0)
491fab62b1SMaxim Sobolev 		return (-2);
501fab62b1SMaxim Sobolev 
511fab62b1SMaxim Sobolev 	rv = -2;
521fab62b1SMaxim Sobolev 
531fab62b1SMaxim Sobolev 	cmsg_size = CMSG_SPACE(sizeof(struct cmsgcred));
541fab62b1SMaxim Sobolev 	cmsg_data = malloc(cmsg_size);
551fab62b1SMaxim Sobolev 	if (cmsg_data == NULL) {
561fab62b1SMaxim Sobolev 		uc_logmsg("malloc");
571fab62b1SMaxim Sobolev 		goto done;
581fab62b1SMaxim Sobolev 	}
591fab62b1SMaxim Sobolev 	uc_msghdr_init_client(&msghdr, iov, cmsg_data, cmsg_size,
601fab62b1SMaxim Sobolev 	    SCM_CREDS, sizeof(struct cmsgcred));
611fab62b1SMaxim Sobolev 
621fab62b1SMaxim Sobolev 	if (uc_socket_connect(fd) < 0)
631fab62b1SMaxim Sobolev 		goto done;
641fab62b1SMaxim Sobolev 
651fab62b1SMaxim Sobolev 	if (uc_message_sendn(fd, &msghdr) < 0)
661fab62b1SMaxim Sobolev 		goto done;
671fab62b1SMaxim Sobolev 
681fab62b1SMaxim Sobolev 	rv = 0;
691fab62b1SMaxim Sobolev done:
701fab62b1SMaxim Sobolev 	free(cmsg_data);
711fab62b1SMaxim Sobolev 	return (rv);
721fab62b1SMaxim Sobolev }
731fab62b1SMaxim Sobolev 
741fab62b1SMaxim Sobolev static int
t_cmsgcred_server(int fd1)751fab62b1SMaxim Sobolev t_cmsgcred_server(int fd1)
761fab62b1SMaxim Sobolev {
771fab62b1SMaxim Sobolev 	struct msghdr msghdr;
781fab62b1SMaxim Sobolev 	struct iovec iov[1];
791fab62b1SMaxim Sobolev 	struct cmsghdr *cmsghdr;
801fab62b1SMaxim Sobolev 	void *cmsg_data;
811fab62b1SMaxim Sobolev 	size_t cmsg_size;
821fab62b1SMaxim Sobolev 	u_int i;
831fab62b1SMaxim Sobolev 	int fd2, rv;
841fab62b1SMaxim Sobolev 
851fab62b1SMaxim Sobolev 	if (uc_sync_send() < 0)
861fab62b1SMaxim Sobolev 		return (-2);
871fab62b1SMaxim Sobolev 
881fab62b1SMaxim Sobolev 	fd2 = -1;
891fab62b1SMaxim Sobolev 	rv = -2;
901fab62b1SMaxim Sobolev 
911fab62b1SMaxim Sobolev 	cmsg_size = CMSG_SPACE(sizeof(struct cmsgcred));
921fab62b1SMaxim Sobolev 	cmsg_data = malloc(cmsg_size);
931fab62b1SMaxim Sobolev 	if (cmsg_data == NULL) {
941fab62b1SMaxim Sobolev 		uc_logmsg("malloc");
951fab62b1SMaxim Sobolev 		goto done;
961fab62b1SMaxim Sobolev 	}
971fab62b1SMaxim Sobolev 
981fab62b1SMaxim Sobolev 	if (uc_cfg.sock_type == SOCK_STREAM) {
991fab62b1SMaxim Sobolev 		fd2 = uc_socket_accept(fd1);
1001fab62b1SMaxim Sobolev 		if (fd2 < 0)
1011fab62b1SMaxim Sobolev 			goto done;
1021fab62b1SMaxim Sobolev 	} else
1031fab62b1SMaxim Sobolev 		fd2 = fd1;
1041fab62b1SMaxim Sobolev 
1051fab62b1SMaxim Sobolev 	rv = -1;
1061fab62b1SMaxim Sobolev 	for (i = 1; i <= uc_cfg.ipc_msg.msg_num; ++i) {
1071fab62b1SMaxim Sobolev 		uc_dbgmsg("message #%u", i);
1081fab62b1SMaxim Sobolev 
1091fab62b1SMaxim Sobolev 		uc_msghdr_init_server(&msghdr, iov, cmsg_data, cmsg_size);
1101fab62b1SMaxim Sobolev 		if (uc_message_recv(fd2, &msghdr) < 0) {
1111fab62b1SMaxim Sobolev 			rv = -2;
1121fab62b1SMaxim Sobolev 			break;
1131fab62b1SMaxim Sobolev 		}
1141fab62b1SMaxim Sobolev 
1151fab62b1SMaxim Sobolev 		if (uc_check_msghdr(&msghdr, sizeof(*cmsghdr)) < 0)
1161fab62b1SMaxim Sobolev 			break;
1171fab62b1SMaxim Sobolev 
1181fab62b1SMaxim Sobolev 		cmsghdr = CMSG_FIRSTHDR(&msghdr);
1191fab62b1SMaxim Sobolev 		if (uc_check_scm_creds_cmsgcred(cmsghdr) < 0)
1201fab62b1SMaxim Sobolev 			break;
1211fab62b1SMaxim Sobolev 	}
1221fab62b1SMaxim Sobolev 	if (i > uc_cfg.ipc_msg.msg_num)
1231fab62b1SMaxim Sobolev 		rv = 0;
1241fab62b1SMaxim Sobolev done:
1251fab62b1SMaxim Sobolev 	free(cmsg_data);
1261fab62b1SMaxim Sobolev 	if (uc_cfg.sock_type == SOCK_STREAM && fd2 >= 0)
1271fab62b1SMaxim Sobolev 		if (uc_socket_close(fd2) < 0)
1281fab62b1SMaxim Sobolev 			rv = -2;
1291fab62b1SMaxim Sobolev 	return (rv);
1301fab62b1SMaxim Sobolev }
1311fab62b1SMaxim Sobolev 
1321fab62b1SMaxim Sobolev int
t_cmsgcred(void)1331fab62b1SMaxim Sobolev t_cmsgcred(void)
1341fab62b1SMaxim Sobolev {
1351fab62b1SMaxim Sobolev 	return (t_generic(t_cmsgcred_client, t_cmsgcred_server));
1361fab62b1SMaxim Sobolev }
137