1 /* Copyright (C) 2011 Free Software Foundation, Inc.
3 This program is free software: you can redistribute it and/or modify
4 it under the terms of the GNU General Public License as published by
5 the Free Software Foundation; either version 3 of the License, or
6 (at your option) any later version.
8 This program is distributed in the hope that it will be useful,
9 but WITHOUT ANY WARRANTY; without even the implied warranty of
10 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 GNU General Public License for more details.
13 You should have received a copy of the GNU General Public License
14 along with this program. If not, see <http://www.gnu.org/licenses/>. */
26 #include <sys/types.h>
30 #include <sys/socket.h>
37 #ifndef MSG_CMSG_CLOEXEC
38 # define MSG_CMSG_CLOEXEC 0
41 /* sendfd sends the file descriptor fd along the socket
42 to a process calling recvfd on the other end.
44 Return 0 on success, or -1 with errno set in case of error.
47 sendfd (int sock, int fd)
53 /* send at least one char */
54 memset (&msg, 0, sizeof msg);
63 #if HAVE_UNIXSOCKET_SCM_RIGHTS_BSD44_WAY
65 char buf[CMSG_SPACE (sizeof fd)];
67 msg.msg_control = buf;
68 msg.msg_controllen = sizeof buf;
69 cmsg = CMSG_FIRSTHDR (&msg);
70 cmsg->cmsg_level = SOL_SOCKET;
71 cmsg->cmsg_type = SCM_RIGHTS;
72 cmsg->cmsg_len = CMSG_LEN (sizeof fd);
73 /* Initialize the payload: */
74 memcpy (CMSG_DATA (cmsg), &fd, sizeof fd);
75 #elif HAVE_UNIXSOCKET_SCM_RIGHTS_BSD43_WAY
76 msg.msg_accrights = &fd;
77 msg.msg_accrightslen = sizeof fd;
84 if (sendmsg (sock, &msg, 0) != iov.iov_len)
89 /* recvfd receives a file descriptor through the socket.
90 The flags are a bitmask, possibly including O_CLOEXEC (defined in <fcntl.h>).
92 Return 0 on success, or -1 with errno set in case of error.
95 recvfd (int sock, int flags)
101 if ((flags & ~O_CLOEXEC) != 0)
107 /* send at least one char */
108 iov.iov_base = &recv;
116 #if HAVE_UNIXSOCKET_SCM_RIGHTS_BSD44_WAY
118 struct cmsghdr *cmsg;
119 char buf[CMSG_SPACE (sizeof fd)];
121 int flags_recvmsg = flags & O_CLOEXEC ? MSG_CMSG_CLOEXEC : 0;
123 msg.msg_control = buf;
124 msg.msg_controllen = sizeof buf;
125 cmsg = CMSG_FIRSTHDR (&msg);
126 cmsg->cmsg_level = SOL_SOCKET;
127 cmsg->cmsg_type = SCM_RIGHTS;
128 cmsg->cmsg_len = CMSG_LEN (sizeof mone);
129 /* Initialize the payload: */
130 memcpy (CMSG_DATA (cmsg), &mone, sizeof mone);
131 msg.msg_controllen = cmsg->cmsg_len;
133 if (recvmsg (sock, &msg, flags_recvmsg) < 0)
136 cmsg = CMSG_FIRSTHDR (&msg);
138 if (cmsg == NULL || cmsg->cmsg_len != CMSG_LEN (sizeof fd)
139 || cmsg->cmsg_level != SOL_SOCKET || cmsg->cmsg_type != SCM_RIGHTS)
141 /* fake errno: at end the file is not available */
146 memcpy (&fd, CMSG_DATA (cmsg), sizeof fd);
148 /* set close-on-exec flag */
149 if (!MSG_CMSG_CLOEXEC && (flags & O_CLOEXEC))
151 if (set_cloexec_flag (fd, true) < 0)
153 int saved_errno = errno;
162 #elif HAVE_UNIXSOCKET_SCM_RIGHTS_BSD43_WAY
165 msg.msg_accrights = &fd;
166 msg.msg_accrightslen = sizeof fd;
167 if (recvmsg (sock, &msg, 0) < 0)
170 /* set close-on-exec flag */
171 if (flags & O_CLOEXEC)
173 if (set_cloexec_flag (fd, true) < 0)
175 int saved_errno = errno;