| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | #define _POSIX_C_SOURCE 200809L
 | 
					
						
							| 
									
										
										
										
											2017-10-10 01:23:43 +03:00
										 |  |  | #ifdef __FreeBSD__
 | 
					
						
							|  |  |  | #define __BSD_VISIBLE 1
 | 
					
						
							|  |  |  | #define INPUT_MAJOR 0
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | #include <errno.h>
 | 
					
						
							| 
									
										
										
										
											2018-02-12 21:29:23 +01:00
										 |  |  | #include <fcntl.h>
 | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2017-07-11 00:14:55 +12:00
										 |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | #include <sys/socket.h>
 | 
					
						
							|  |  |  | #include <sys/stat.h>
 | 
					
						
							|  |  |  | #include <sys/types.h>
 | 
					
						
							|  |  |  | #include <sys/wait.h>
 | 
					
						
							| 
									
										
										
										
											2018-02-12 21:29:23 +01:00
										 |  |  | #include <unistd.h>
 | 
					
						
							|  |  |  | #include <wlr/config.h>
 | 
					
						
							|  |  |  | #include <wlr/util/log.h>
 | 
					
						
							|  |  |  | #include <xf86drm.h>
 | 
					
						
							| 
									
										
										
										
											2017-10-10 01:23:43 +03:00
										 |  |  | #ifdef __linux__
 | 
					
						
							|  |  |  | #include <sys/sysmacros.h>
 | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | #include <linux/major.h>
 | 
					
						
							| 
									
										
										
										
											2017-10-10 01:23:43 +03:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-07-11 19:18:34 +12:00
										 |  |  | #include "backend/session/direct-ipc.h"
 | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 
 | 
					
						
							|  |  |  | enum { DRM_MAJOR = 226 }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-26 18:51:27 +01:00
										 |  |  | #ifdef WLR_HAS_LIBCAP
 | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | #include <sys/capability.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool have_permissions(void) { | 
					
						
							|  |  |  | 	cap_t cap = cap_get_proc(); | 
					
						
							|  |  |  | 	cap_flag_value_t val; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!cap || cap_get_flag(cap, CAP_SYS_ADMIN, CAP_PERMITTED, &val) || val != CAP_SET) { | 
					
						
							| 
									
										
										
										
											2018-07-09 22:49:54 +01:00
										 |  |  | 		wlr_log(WLR_ERROR, "Do not have CAP_SYS_ADMIN; cannot become DRM master"); | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 		cap_free(cap); | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cap_free(cap); | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | static bool have_permissions(void) { | 
					
						
							| 
									
										
										
										
											2017-10-10 01:23:43 +03:00
										 |  |  | #ifdef __linux__
 | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 	if (geteuid() != 0) { | 
					
						
							| 
									
										
										
										
											2018-07-09 22:49:54 +01:00
										 |  |  | 		wlr_log(WLR_ERROR, "Do not have root privileges; cannot become DRM master"); | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-10-10 01:23:43 +03:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void send_msg(int sock, int fd, void *buf, size_t buf_len) { | 
					
						
							|  |  |  | 	char control[CMSG_SPACE(sizeof(fd))] = {0}; | 
					
						
							|  |  |  | 	struct iovec iovec = { .iov_base = buf, .iov_len = buf_len }; | 
					
						
							|  |  |  | 	struct msghdr msghdr = {0}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (buf) { | 
					
						
							|  |  |  | 		msghdr.msg_iov = &iovec; | 
					
						
							|  |  |  | 		msghdr.msg_iovlen = 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (fd >= 0) { | 
					
						
							|  |  |  | 		msghdr.msg_control = &control; | 
					
						
							|  |  |  | 		msghdr.msg_controllen = sizeof(control); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		struct cmsghdr *cmsg = CMSG_FIRSTHDR(&msghdr); | 
					
						
							|  |  |  | 		*cmsg = (struct cmsghdr) { | 
					
						
							|  |  |  | 			.cmsg_level = SOL_SOCKET, | 
					
						
							|  |  |  | 			.cmsg_type = SCM_RIGHTS, | 
					
						
							|  |  |  | 			.cmsg_len = CMSG_LEN(sizeof(fd)), | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2017-07-11 00:14:55 +12:00
										 |  |  | 		memcpy(CMSG_DATA(cmsg), &fd, sizeof(fd)); | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ssize_t ret; | 
					
						
							|  |  |  | 	do { | 
					
						
							|  |  |  | 		ret = sendmsg(sock, &msghdr, 0); | 
					
						
							|  |  |  | 	} while (ret < 0 && errno == EINTR); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static ssize_t recv_msg(int sock, int *fd_out, void *buf, size_t buf_len) { | 
					
						
							|  |  |  | 	char control[CMSG_SPACE(sizeof(*fd_out))] = {0}; | 
					
						
							|  |  |  | 	struct iovec iovec = { .iov_base = buf, .iov_len = buf_len }; | 
					
						
							|  |  |  | 	struct msghdr msghdr = {0}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (buf) { | 
					
						
							|  |  |  | 		msghdr.msg_iov = &iovec; | 
					
						
							|  |  |  | 		msghdr.msg_iovlen = 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (fd_out) { | 
					
						
							|  |  |  | 		msghdr.msg_control = &control; | 
					
						
							|  |  |  | 		msghdr.msg_controllen = sizeof(control); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ssize_t ret; | 
					
						
							|  |  |  | 	do { | 
					
						
							|  |  |  | 		ret = recvmsg(sock, &msghdr, MSG_CMSG_CLOEXEC); | 
					
						
							|  |  |  | 	} while (ret < 0 && errno == EINTR); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (fd_out) { | 
					
						
							|  |  |  | 		struct cmsghdr *cmsg = CMSG_FIRSTHDR(&msghdr); | 
					
						
							| 
									
										
										
										
											2017-07-11 00:14:55 +12:00
										 |  |  | 		if (cmsg) { | 
					
						
							|  |  |  | 			memcpy(fd_out, CMSG_DATA(cmsg), sizeof(*fd_out)); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			*fd_out = -1; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum msg_type { | 
					
						
							|  |  |  | 	MSG_OPEN, | 
					
						
							|  |  |  | 	MSG_SETMASTER, | 
					
						
							|  |  |  | 	MSG_DROPMASTER, | 
					
						
							|  |  |  | 	MSG_END, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct msg { | 
					
						
							|  |  |  | 	enum msg_type type; | 
					
						
							|  |  |  | 	char path[256]; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void communicate(int sock) { | 
					
						
							|  |  |  | 	struct msg msg; | 
					
						
							|  |  |  | 	int drm_fd = -1; | 
					
						
							|  |  |  | 	bool running = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-20 21:35:15 +02:00
										 |  |  | 	while (running && recv_msg(sock, &drm_fd, &msg, sizeof(msg)) > 0) { | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 		switch (msg.type) { | 
					
						
							|  |  |  | 		case MSG_OPEN: | 
					
						
							|  |  |  | 			errno = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// These are the same flags that logind opens files with
 | 
					
						
							|  |  |  | 			int fd = open(msg.path, O_RDWR|O_CLOEXEC|O_NOCTTY|O_NONBLOCK); | 
					
						
							|  |  |  | 			int ret = errno; | 
					
						
							|  |  |  | 			if (fd == -1) { | 
					
						
							|  |  |  | 				goto error; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			struct stat st; | 
					
						
							|  |  |  | 			if (fstat(fd, &st) < 0) { | 
					
						
							|  |  |  | 				ret = errno; | 
					
						
							|  |  |  | 				goto error; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			uint32_t maj = major(st.st_rdev); | 
					
						
							|  |  |  | 			if (maj != INPUT_MAJOR && maj != DRM_MAJOR) { | 
					
						
							|  |  |  | 				ret = ENOTSUP; | 
					
						
							|  |  |  | 				goto error; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-09 22:23:54 +12:00
										 |  |  | 			if (maj == DRM_MAJOR && drmSetMaster(fd)) { | 
					
						
							|  |  |  | 				ret = errno; | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 			} | 
					
						
							|  |  |  | error: | 
					
						
							|  |  |  | 			send_msg(sock, ret ? -1 : fd, &ret, sizeof(ret)); | 
					
						
							| 
									
										
										
										
											2018-06-30 10:28:41 +09:00
										 |  |  | 			if (fd >= 0) { | 
					
						
							|  |  |  | 				close(fd); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case MSG_SETMASTER: | 
					
						
							|  |  |  | 			drmSetMaster(drm_fd); | 
					
						
							| 
									
										
										
										
											2017-07-09 22:23:54 +12:00
										 |  |  | 			close(drm_fd); | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 			send_msg(sock, -1, NULL, 0); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case MSG_DROPMASTER: | 
					
						
							|  |  |  | 			drmDropMaster(drm_fd); | 
					
						
							| 
									
										
										
										
											2017-07-09 22:23:54 +12:00
										 |  |  | 			close(drm_fd); | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 			send_msg(sock, -1, NULL, 0); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case MSG_END: | 
					
						
							|  |  |  | 			running = false; | 
					
						
							|  |  |  | 			send_msg(sock, -1, NULL, 0); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	close(sock); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int direct_ipc_open(int sock, const char *path) { | 
					
						
							|  |  |  | 	struct msg msg = { .type = MSG_OPEN }; | 
					
						
							|  |  |  | 	snprintf(msg.path, sizeof(msg.path), "%s", path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	send_msg(sock, -1, &msg, sizeof(msg)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int fd, err; | 
					
						
							|  |  |  | 	recv_msg(sock, &fd, &err, sizeof(err)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return err ? -err : fd; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-09 22:23:54 +12:00
										 |  |  | void direct_ipc_setmaster(int sock, int fd) { | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 	struct msg msg = { .type = MSG_SETMASTER }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-09 22:23:54 +12:00
										 |  |  | 	send_msg(sock, fd, &msg, sizeof(msg)); | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 	recv_msg(sock, NULL, NULL, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-09 22:23:54 +12:00
										 |  |  | void direct_ipc_dropmaster(int sock, int fd) { | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 	struct msg msg = { .type = MSG_DROPMASTER }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-09 22:23:54 +12:00
										 |  |  | 	send_msg(sock, fd, &msg, sizeof(msg)); | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 	recv_msg(sock, NULL, NULL, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void direct_ipc_finish(int sock, pid_t pid) { | 
					
						
							|  |  |  | 	struct msg msg = { .type = MSG_END }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	send_msg(sock, -1, &msg, sizeof(msg)); | 
					
						
							|  |  |  | 	recv_msg(sock, NULL, NULL, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	waitpid(pid, NULL, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-26 11:56:43 +12:00
										 |  |  | int direct_ipc_init(pid_t *pid_out) { | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 	if (!have_permissions()) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int sock[2]; | 
					
						
							|  |  |  | 	if (socketpair(AF_UNIX, SOCK_SEQPACKET, 0, sock) < 0) { | 
					
						
							| 
									
										
										
										
											2018-07-09 22:49:54 +01:00
										 |  |  | 		wlr_log_errno(WLR_ERROR, "Failed to create socket pair"); | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pid_t pid = fork(); | 
					
						
							|  |  |  | 	if (pid < 0) { | 
					
						
							| 
									
										
										
										
											2018-07-09 22:49:54 +01:00
										 |  |  | 		wlr_log_errno(WLR_ERROR, "Fork failed"); | 
					
						
							| 
									
										
										
										
											2017-07-09 22:12:50 +12:00
										 |  |  | 		close(sock[0]); | 
					
						
							|  |  |  | 		close(sock[1]); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} else if (pid == 0) { | 
					
						
							|  |  |  | 		close(sock[0]); | 
					
						
							|  |  |  | 		communicate(sock[1]); | 
					
						
							|  |  |  | 		_Exit(0); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	close(sock[1]); | 
					
						
							|  |  |  | 	*pid_out = pid; | 
					
						
							|  |  |  | 	return sock[0]; | 
					
						
							|  |  |  | } |