2012-02-01 13:00:49 +03:00
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/un.h>
|
|
|
|
|
2012-03-28 17:36:00 +04:00
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "compiler.h"
|
2013-01-09 17:02:47 +04:00
|
|
|
#include "asm/memcpy_64.h"
|
|
|
|
#include "asm/types.h"
|
2012-02-01 13:00:49 +03:00
|
|
|
#include "syscall.h"
|
|
|
|
|
2012-03-28 17:36:00 +04:00
|
|
|
#include "util-net.h"
|
|
|
|
|
|
|
|
static void scm_fdset_init_chunk(struct scm_fdset *fdset, int nr_fds)
|
2012-02-01 13:00:49 +03:00
|
|
|
{
|
2012-03-28 17:36:00 +04:00
|
|
|
struct cmsghdr *cmsg;
|
|
|
|
|
|
|
|
fdset->hdr.msg_controllen = CMSG_LEN(sizeof(int) * nr_fds);
|
|
|
|
|
|
|
|
cmsg = CMSG_FIRSTHDR(&fdset->hdr);
|
|
|
|
cmsg->cmsg_len = fdset->hdr.msg_controllen;
|
|
|
|
}
|
|
|
|
|
2012-04-10 18:36:13 +04:00
|
|
|
static int *scm_fdset_init(struct scm_fdset *fdset, struct sockaddr_un *saddr,
|
|
|
|
int saddr_len, bool with_flags)
|
2012-03-28 17:36:00 +04:00
|
|
|
{
|
|
|
|
struct cmsghdr *cmsg;
|
|
|
|
|
|
|
|
BUILD_BUG_ON(CR_SCM_MAX_FD > SCM_MAX_FD);
|
|
|
|
BUILD_BUG_ON(sizeof(fdset->msg_buf) < (CMSG_SPACE(sizeof(int) * CR_SCM_MAX_FD)));
|
|
|
|
|
2012-09-05 16:41:14 +04:00
|
|
|
fdset->iov.iov_base = fdset->opts;
|
|
|
|
fdset->iov.iov_len = with_flags ? sizeof(fdset->opts) : 1;
|
2012-03-28 17:36:00 +04:00
|
|
|
|
|
|
|
fdset->hdr.msg_iov = &fdset->iov;
|
|
|
|
fdset->hdr.msg_iovlen = 1;
|
|
|
|
fdset->hdr.msg_name = (struct sockaddr *)saddr;
|
|
|
|
fdset->hdr.msg_namelen = saddr_len;
|
2012-02-01 13:00:49 +03:00
|
|
|
|
2012-03-28 17:36:00 +04:00
|
|
|
fdset->hdr.msg_control = &fdset->msg_buf;
|
|
|
|
fdset->hdr.msg_controllen = CMSG_LEN(sizeof(int) * CR_SCM_MAX_FD);
|
2012-02-01 13:00:49 +03:00
|
|
|
|
2012-03-28 17:36:00 +04:00
|
|
|
cmsg = CMSG_FIRSTHDR(&fdset->hdr);
|
|
|
|
cmsg->cmsg_len = fdset->hdr.msg_controllen;
|
|
|
|
cmsg->cmsg_level = SOL_SOCKET;
|
|
|
|
cmsg->cmsg_type = SCM_RIGHTS;
|
2012-02-01 13:00:49 +03:00
|
|
|
|
2012-03-28 17:36:00 +04:00
|
|
|
return (int *)CMSG_DATA(cmsg);
|
|
|
|
}
|
2012-02-01 13:00:49 +03:00
|
|
|
|
2012-04-10 18:36:13 +04:00
|
|
|
int send_fds(int sock, struct sockaddr_un *saddr, int len,
|
|
|
|
int *fds, int nr_fds, bool with_flags)
|
2012-03-28 17:36:00 +04:00
|
|
|
{
|
|
|
|
struct scm_fdset fdset;
|
|
|
|
int *cmsg_data;
|
2012-03-28 23:44:00 +04:00
|
|
|
int i, min_fd, ret;
|
2012-02-01 13:00:49 +03:00
|
|
|
|
2012-04-10 18:36:13 +04:00
|
|
|
cmsg_data = scm_fdset_init(&fdset, saddr, len, with_flags);
|
2012-03-28 23:44:00 +04:00
|
|
|
for (i = 0; i < nr_fds; i += min_fd) {
|
|
|
|
min_fd = min(CR_SCM_MAX_FD, nr_fds - i);
|
|
|
|
scm_fdset_init_chunk(&fdset, min_fd);
|
|
|
|
builtin_memcpy(cmsg_data, &fds[i], sizeof(int) * min_fd);
|
2012-02-01 13:00:49 +03:00
|
|
|
|
2012-04-10 18:36:13 +04:00
|
|
|
if (with_flags) {
|
|
|
|
int j;
|
|
|
|
|
|
|
|
for (j = 0; j < min_fd; j++) {
|
2012-09-05 16:41:14 +04:00
|
|
|
int flags, fd = fds[i + j];
|
|
|
|
struct fd_opts *p = fdset.opts + j;
|
|
|
|
struct f_owner_ex owner_ex;
|
|
|
|
u32 v[2];
|
2012-04-10 18:36:13 +04:00
|
|
|
|
2012-09-05 16:41:14 +04:00
|
|
|
flags = sys_fcntl(fd, F_GETFD, 0);
|
2012-04-10 18:36:13 +04:00
|
|
|
if (flags < 0)
|
|
|
|
return -1;
|
|
|
|
|
2012-09-05 16:41:14 +04:00
|
|
|
p->flags = (char)flags;
|
|
|
|
|
|
|
|
if (sys_fcntl(fd, F_GETOWN_EX, (long)&owner_ex))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Simple case -- nothing is changed.
|
|
|
|
*/
|
|
|
|
if (owner_ex.pid == 0) {
|
|
|
|
p->fown.pid = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sys_fcntl(fd, F_GETOWNER_UIDS, (long)&v))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
p->fown.uid = v[0];
|
|
|
|
p->fown.euid = v[1];
|
|
|
|
p->fown.pid_type = owner_ex.type;
|
|
|
|
p->fown.pid = owner_ex.pid;
|
2012-04-10 18:36:13 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-28 23:44:00 +04:00
|
|
|
ret = sys_sendmsg(sock, &fdset.hdr, 0);
|
|
|
|
if (ret <= 0)
|
|
|
|
return ret ? : -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2012-02-01 13:00:49 +03:00
|
|
|
}
|
|
|
|
|
2012-09-05 16:41:14 +04:00
|
|
|
int recv_fds(int sock, int *fds, int nr_fds, struct fd_opts *opts)
|
2012-02-01 13:00:49 +03:00
|
|
|
{
|
2012-03-28 17:36:00 +04:00
|
|
|
struct scm_fdset fdset;
|
2012-02-01 13:00:49 +03:00
|
|
|
struct cmsghdr *cmsg;
|
|
|
|
int *cmsg_data;
|
|
|
|
int ret;
|
2012-03-28 23:44:00 +04:00
|
|
|
int i, min_fd;
|
2012-02-01 13:00:49 +03:00
|
|
|
|
2012-09-05 16:41:14 +04:00
|
|
|
cmsg_data = scm_fdset_init(&fdset, NULL, 0, opts != NULL);
|
2012-03-28 23:44:00 +04:00
|
|
|
for (i = 0; i < nr_fds; i += min_fd) {
|
|
|
|
min_fd = min(CR_SCM_MAX_FD, nr_fds - i);
|
|
|
|
scm_fdset_init_chunk(&fdset, min_fd);
|
|
|
|
|
|
|
|
ret = sys_recvmsg(sock, &fdset.hdr, 0);
|
|
|
|
if (ret <= 0)
|
|
|
|
return ret ? : -1;
|
|
|
|
|
|
|
|
cmsg = CMSG_FIRSTHDR(&fdset.hdr);
|
|
|
|
if (!cmsg || cmsg->cmsg_type != SCM_RIGHTS)
|
|
|
|
return -EINVAL;
|
2012-08-21 19:40:10 +04:00
|
|
|
if (fdset.hdr.msg_flags & MSG_CTRUNC)
|
|
|
|
return -ENFILE;
|
2012-03-28 23:44:00 +04:00
|
|
|
|
|
|
|
min_fd = (cmsg->cmsg_len - sizeof(struct cmsghdr)) / sizeof(int);
|
|
|
|
/*
|
|
|
|
* In case if kernel screwed the recepient, most probably
|
|
|
|
* the caller stack frame will be overwriten, just scream
|
|
|
|
* and exit.
|
|
|
|
*
|
|
|
|
* FIXME Need to sanitize util.h to be able to include it
|
|
|
|
* into files which do not have glibc and a couple of
|
|
|
|
* sys_write_ helpers. Meawhile opencoded BUG_ON here.
|
|
|
|
*/
|
|
|
|
if (unlikely(min_fd > CR_SCM_MAX_FD))
|
|
|
|
*(volatile unsigned long *)NULL = 0xdead0000 + __LINE__;
|
|
|
|
if (unlikely(min_fd <= 0))
|
|
|
|
return -1;
|
|
|
|
builtin_memcpy(&fds[i], cmsg_data, sizeof(int) * min_fd);
|
2012-09-05 16:41:14 +04:00
|
|
|
if (opts)
|
|
|
|
builtin_memcpy(opts + i, fdset.opts, sizeof(struct fd_opts) * min_fd);
|
2012-03-28 23:44:00 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2012-02-01 13:00:49 +03:00
|
|
|
}
|
|
|
|
|