#include <assert.h> #include <string.h> #include <stdlib.h> #include <unistd.h> #include <errno.h> #include <fcntl.h> #include <sys/socket.h> #include <sys/wait.h> int main(int ac, const char **av) { int i; int data = 0; struct iovec iov = { .iov_base = &data, .iov_len = sizeof(iov) }; while ((i = open("/dev/null", O_RDWR)) < 3) assert(i >= 0); (void) close(3); int sv[2]; assert(socketpair(AF_UNIX, SOCK_STREAM, 0, sv) == 0); int one = 1; assert(setsockopt(sv[0], SOL_SOCKET, SO_PASSCRED, &one, sizeof(one)) == 0); pid_t pid = fork(); assert(pid >= 0); if (pid) { assert(close(sv[0]) == 0); assert(dup2(sv[1], 1) == 1); assert(close(sv[1]) == 0); int fds[ac]; assert((fds[0] = open("/dev/null", O_RDWR)) == 3); for (i = 1; i < ac; ++i) assert((fds[i] = open(av[i], O_RDONLY)) == i + 3); union { struct cmsghdr cmsg; char buf[CMSG_LEN(sizeof(fds))]; } control; control.cmsg.cmsg_level = SOL_SOCKET; control.cmsg.cmsg_type = SCM_RIGHTS; control.cmsg.cmsg_len = CMSG_LEN(sizeof(fds)); memcpy(CMSG_DATA(&control.cmsg), fds, sizeof(fds)); struct msghdr mh = { .msg_iov = &iov, .msg_iovlen = 1, .msg_control = &control, .msg_controllen = sizeof(control) }; assert(sendmsg(1, &mh, 0) == sizeof(iov)); assert(close(1) == 0); int status; assert(waitpid(pid, &status, 0) == pid); assert(status == 0); } else { assert(close(sv[1]) == 0); assert(dup2(sv[0], 0) == 0); assert(close(sv[0]) == 0); struct cmsghdr control[4 + ac * sizeof(int) / sizeof(struct cmsghdr)]; struct msghdr mh = { .msg_iov = &iov, .msg_iovlen = 1, .msg_control = control, .msg_controllen = sizeof(control) }; assert(recvmsg(0, &mh, 0) == sizeof(iov)); assert(close(0) == 0); } return 0; }