#include <sys/socket.h>
#include <sys/un.h>
+#include <unistd.h>
#include <common/macros.h>
}
#endif
+#ifdef __sun__
+
+# ifndef CMSG_ALIGN
+# ifdef _CMSG_DATA_ALIGN
+# define CMSG_ALIGN(len) _CMSG_DATA_ALIGN(len)
+# else
+ /* aligning to sizeof (long) is assumed to be portable (fd.o#40235) */
+# define CMSG_ALIGN(len) (((len) + sizeof (long) - 1) & ~(sizeof (long) - 1))
+# endif
+# ifndef CMSG_SPACE
+# define CMSG_SPACE(len) (CMSG_ALIGN (sizeof (struct cmsghdr)) + CMSG_ALIGN (len))
+# endif
+# ifndef CMSG_LEN
+# define CMSG_LEN(len) (CMSG_ALIGN (sizeof (struct cmsghdr)) + (len))
+# endif
+# endif
+
+#include <ucred.h>
-#ifdef __linux__
+static inline
+int getpeereid(int s, uid_t *euid, gid_t *gid)
+{
+ int ret = 0;
+ ucred_t *ucred = NULL;
+
+ ret = getpeerucred(s, &ucred);
+ if (ret == -1) {
+ goto end;
+ }
+
+ ret = ucred_geteuid(ucred);
+ if (ret == -1) {
+ goto free;
+ }
+ *euid = ret;
+
+ ret = ucred_getrgid(ucred);
+ if (ret == -1) {
+ goto free;
+ }
+ *gid = ret;
+
+ ret = 0;
+free:
+ ucred_free(ucred);
+end:
+ return ret;
+}
+#endif /* __sun__ */
+
+
+#if defined(__linux__) || defined(__CYGWIN__)
#define LTTNG_SOCK_CREDS SCM_CREDENTIALS
#define LTTNG_SOCK_GET_GID_CRED(c) LTTNG_REF(c)->gid
#define LTTNG_SOCK_GET_PID_CRED(c) LTTNG_REF(c)->pid
-#elif (defined(__FreeBSD__) || defined(__CYGWIN__) || defined(__sun__) || defined(__APPLE__))
+#elif (defined(__FreeBSD__) || defined(__sun__) || defined(__APPLE__))
struct lttng_sock_cred {
uid_t uid;
#define LTTNG_SOCK_SET_UID_CRED(c, u) LTTNG_REF(c)->uid = u
#define LTTNG_SOCK_SET_GID_CRED(c, g) LTTNG_REF(c)->gid = g
-#define LTTNG_SOCK_SET_PID_CRED(c, p)
+#define LTTNG_SOCK_SET_PID_CRED(c, p) LTTNG_REF(c)->pid = p
#define LTTNG_SOCK_GET_UID_CRED(c) LTTNG_REF(c)->uid
#define LTTNG_SOCK_GET_GID_CRED(c) LTTNG_REF(c)->gid
-#define LTTNG_SOCK_GET_PID_CRED(c) -1
-
-#else
-#error "Please add support for your OS."
-#endif /* __linux__ , __FreeBSD__ */
-
-
-#ifdef __sun__
+#define LTTNG_SOCK_GET_PID_CRED(c) LTTNG_REF(c)->pid
-# ifndef CMSG_ALIGN
-# ifdef _CMSG_DATA_ALIGN
-# define CMSG_ALIGN(len) _CMSG_DATA_ALIGN(len)
-# else
- /* aligning to sizeof (long) is assumed to be portable (fd.o#40235) */
-# define CMSG_ALIGN(len) (((len) + sizeof (long) - 1) & ~(sizeof (long) - 1))
-# endif
-# ifndef CMSG_SPACE
-# define CMSG_SPACE(len) (CMSG_ALIGN (sizeof (struct cmsghdr)) + CMSG_ALIGN (len))
-# endif
-# ifndef CMSG_LEN
-# define CMSG_LEN(len) (CMSG_ALIGN (sizeof (struct cmsghdr)) + (len))
-# endif
-# endif
+#ifdef __APPLE__
+static inline
+int lttng_get_unix_socket_peer_pid(int socket_fd, pid_t *pid)
+{
+ /* The getsockopt LOCAL_PEERPID option is available since macOS 10.8. */
+ return getsockopt(socket_fd, SOL_LOCAL, LOCAL_PEERPID, pid,
+ &((socklen_t) {sizeof(*pid)}));
+}
-#include <ucred.h>
+#elif defined(__sun__)
+/* Use the getpeerucreds interface on Solaris. */
static inline
-int getpeereid(int s, uid_t *euid, gid_t *gid, pid_t *pid)
+int lttng_get_unix_socket_peer_pid(int socket_fd, pid_t *pid)
{
int ret = 0;
ucred_t *ucred = NULL;
goto end;
}
- ret = ucred_geteuid(ucred);
- if (ret == -1) {
- goto free;
- }
- *euid = ret;
-
- ret = ucred_getrgid(ucred);
- if (ret == -1) {
- goto free;
- }
- *gid = ret;
-
ret = ucred_getpid(ucred);
if (ret == -1) {
goto free;
}
- *pid = ret;
+ *pid = ret;
ret = 0;
free:
ucred_free(ucred);
return ret;
}
-#endif /* __sun__ */
+#elif defined(__FreeBSD__)
+
+#include <sys/ucred.h>
+
+static inline
+int lttng_get_unix_socket_peer_pid(int socket_fd, pid_t *pid)
+{
+ int ret;
+ struct xucred sock_creds = {};
+
+ /* Only available in FreeBSD 13.0 and up. */
+ ret = getsockopt(socket_fd, SOL_LOCAL, LOCAL_PEERCRED, &sock_creds,
+ &((socklen_t) {sizeof(sock_creds)}));
+ if (ret) {
+ goto end;
+ }
+
+ *pid = sock_creds.cr_pid;
+end:
+ return ret;
+}
+
+#endif /* __APPLE__ */
+
+
+static inline
+int lttng_get_unix_socket_peer_creds(int socket_fd, struct lttng_sock_cred *creds)
+{
+ int ret;
+
+ /* This is a BSD extension that is supported by Cygwin. */
+ ret = getpeereid(socket_fd, &creds->uid, &creds->gid);
+ if (ret) {
+ goto end;
+ }
+
+ /*
+ * Getting a peer's PID is a bit more troublesome as it is platform
+ * specific.
+ */
+ ret = lttng_get_unix_socket_peer_pid(socket_fd, &creds->pid);
+end:
+ return ret;
+}
+
+#else
+#error "Please add support for your OS."
+#endif /* __linux__ , __FreeBSD__, __APPLE__ */
#endif /* _COMPAT_SOCKET_H */
struct msghdr msg;
struct iovec iov[1];
ssize_t ret = -1;
-#ifdef __linux__
+#if defined(__linux__) || defined(__CYGWIN__)
struct cmsghdr *cmptr;
size_t sizeof_cred = sizeof(lttng_sock_cred);
char anc_buf[CMSG_SPACE(sizeof_cred)];
lttng_sock_cred *creds;
memset(anc_buf, 0, CMSG_SPACE(sizeof_cred) * sizeof(char));
-#endif /* __linux__ */
+#endif /* __linux__, __CYGWIN__ */
memset(&msg, 0, sizeof(msg));
msg.msg_iov = iov;
msg.msg_iovlen = 1;
-#ifdef __linux__
+#if defined(__linux__) || defined(__CYGWIN__)
msg.msg_control = (caddr_t) anc_buf;
msg.msg_controllen = CMSG_LEN(sizeof_cred);
LTTNG_SOCK_SET_UID_CRED(creds, geteuid());
LTTNG_SOCK_SET_GID_CRED(creds, getegid());
LTTNG_SOCK_SET_PID_CRED(creds, getpid());
-#endif /* __linux__ */
+#endif /* __linux__, __CYGWIN__ */
do {
ret = sendmsg(sock, &msg, 0);
struct iovec iov[1];
ssize_t ret;
size_t len_last;
-#ifdef __linux__
+#if defined(__linux__) || defined(__CYGWIN__)
struct cmsghdr *cmptr;
size_t sizeof_cred = sizeof(lttng_sock_cred);
char anc_buf[CMSG_SPACE(sizeof_cred)];
-#endif /* __linux__ */
+#endif /* __linux__, __CYGWIN__ */
assert(sock);
assert(buf);
msg.msg_iov = iov;
msg.msg_iovlen = 1;
-#ifdef __linux__
+#if defined(__linux__) || defined(__CYGWIN__)
msg.msg_control = anc_buf;
msg.msg_controllen = sizeof(anc_buf);
-#endif /* __linux__ */
+#endif /* __linux__, __CYGWIN__ */
do {
len_last = iov[0].iov_len;
}
/* Else ret = 0 meaning an orderly shutdown. */
-#ifdef __linux__
+#if defined(__linux__) || defined(__CYGWIN__)
if (msg.msg_flags & MSG_CTRUNC) {
fprintf(stderr, "Error: Control message truncated.\n");
ret = -1;
}
memcpy(creds, CMSG_DATA(cmptr), sizeof_cred);
-#elif (defined(__FreeBSD__) || defined(__CYGWIN__) || defined(__sun__) || defined(__APPLE__))
- {
- int peer_ret;
-
- peer_ret = getpeereid(sock, &creds->uid, &creds->gid, &creds->pid);
- if (peer_ret != 0) {
- return peer_ret;
- }
+#elif (defined(__FreeBSD__) || defined(__sun__) || defined(__APPLE__))
+ if (lttng_get_unix_socket_peer_creds(sock, creds)) {
+ fprintf(stderr, "ARG\n");
+ ret = -1;
+ goto end;
}
#else
#error "Please implement credential support for your OS."
-#endif /* __linux__ */
+#endif /* __linux__, __CYGWIN__ */
end:
return ret;
/*
* Set socket option to use credentials passing.
*/
-#ifdef __linux__
+#if defined(__linux__) || defined(__CYGWIN__)
LTTNG_HIDDEN
int lttcomm_setsockopt_creds_unix_sock(int sock)
{
}
return ret;
}
-#elif (defined(__FreeBSD__) || defined(__CYGWIN__) || defined(__sun__) || defined(__APPLE__))
+#elif (defined(__FreeBSD__) || defined(__sun__) || defined(__APPLE__))
LTTNG_HIDDEN
int lttcomm_setsockopt_creds_unix_sock(int sock)
{