mirror of
				https://gitlab.freedesktop.org/wlroots/wlroots.git
				synced 2025-11-03 09:01:40 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			168 lines
		
	
	
	
		
			3.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			168 lines
		
	
	
	
		
			3.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
#define _XOPEN_SOURCE 700
 | 
						|
#ifdef __FreeBSD__
 | 
						|
// for SOCK_CLOEXEC
 | 
						|
#define __BSD_VISIBLE 1
 | 
						|
#endif
 | 
						|
#include <errno.h>
 | 
						|
#include <fcntl.h>
 | 
						|
#include <signal.h>
 | 
						|
#include <stddef.h>
 | 
						|
#include <stdint.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <sys/socket.h>
 | 
						|
#include <sys/stat.h>
 | 
						|
#include <sys/types.h>
 | 
						|
#include <sys/un.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <wlr/util/log.h>
 | 
						|
#include "sockets.h"
 | 
						|
 | 
						|
static const char *lock_fmt = "/tmp/.X%d-lock";
 | 
						|
static const char *socket_dir = "/tmp/.X11-unix";
 | 
						|
static const char *socket_fmt = "/tmp/.X11-unix/X%d";
 | 
						|
#ifndef __linux__
 | 
						|
static const char *socket_fmt2 = "/tmp/.X11-unix/X%d_";
 | 
						|
#endif
 | 
						|
 | 
						|
static int open_socket(struct sockaddr_un *addr, size_t path_size) {
 | 
						|
	int fd, rc;
 | 
						|
	socklen_t size = offsetof(struct sockaddr_un, sun_path) + path_size + 1;
 | 
						|
 | 
						|
	fd = socket(PF_LOCAL, SOCK_STREAM | SOCK_CLOEXEC, 0);
 | 
						|
	if (fd < 0) {
 | 
						|
		wlr_log_errno(L_DEBUG, "Failed to create socket %c%s",
 | 
						|
			addr->sun_path[0] ? addr->sun_path[0] : '@',
 | 
						|
			addr->sun_path + 1);
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
 | 
						|
	if (addr->sun_path[0]) {
 | 
						|
		unlink(addr->sun_path);
 | 
						|
	}
 | 
						|
	if (bind(fd, (struct sockaddr*)addr, size) < 0) {
 | 
						|
		rc = errno;
 | 
						|
		wlr_log_errno(L_DEBUG, "Failed to bind socket %c%s",
 | 
						|
			addr->sun_path[0] ? addr->sun_path[0] : '@',
 | 
						|
			addr->sun_path + 1);
 | 
						|
		goto cleanup;
 | 
						|
	}
 | 
						|
	if (listen(fd, 1) < 0) {
 | 
						|
		rc = errno;
 | 
						|
		wlr_log_errno(L_DEBUG, "Failed to listen to socket %c%s",
 | 
						|
			addr->sun_path[0] ? addr->sun_path[0] : '@',
 | 
						|
			addr->sun_path + 1);
 | 
						|
		goto cleanup;
 | 
						|
	}
 | 
						|
 | 
						|
	return fd;
 | 
						|
 | 
						|
cleanup:
 | 
						|
	close(fd);
 | 
						|
	if (addr->sun_path[0]) {
 | 
						|
		unlink(addr->sun_path);
 | 
						|
	}
 | 
						|
	errno = rc;
 | 
						|
	return -1;
 | 
						|
}
 | 
						|
 | 
						|
static bool open_sockets(int socks[2], int display) {
 | 
						|
	struct sockaddr_un addr = { .sun_family = AF_LOCAL };
 | 
						|
	size_t path_size;
 | 
						|
 | 
						|
	mkdir(socket_dir, 0777);
 | 
						|
 | 
						|
#ifdef __linux__
 | 
						|
	addr.sun_path[0] = 0;
 | 
						|
	path_size = snprintf(addr.sun_path + 1, sizeof(addr.sun_path) - 1, socket_fmt, display);
 | 
						|
#else
 | 
						|
	path_size = snprintf(addr.sun_path, sizeof(addr.sun_path), socket_fmt2, display);
 | 
						|
#endif
 | 
						|
	socks[0] = open_socket(&addr, path_size);
 | 
						|
	if (socks[0] < 0) {
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	path_size = snprintf(addr.sun_path, sizeof(addr.sun_path), socket_fmt, display);
 | 
						|
	socks[1] = open_socket(&addr, path_size);
 | 
						|
	if (socks[1] < 0) {
 | 
						|
		close(socks[0]);
 | 
						|
		socks[0] = -1;
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
void unlink_display_sockets(int display) {
 | 
						|
	char sun_path[64];
 | 
						|
 | 
						|
	snprintf(sun_path, sizeof(sun_path), socket_fmt, display);
 | 
						|
	unlink(sun_path);
 | 
						|
 | 
						|
#ifndef __linux__
 | 
						|
	snprintf(sun_path, sizeof(sun_path), socket_fmt2, display);
 | 
						|
	unlink(sun_path);
 | 
						|
#endif
 | 
						|
 | 
						|
	snprintf(sun_path, sizeof(sun_path), lock_fmt, display);
 | 
						|
	unlink(sun_path);
 | 
						|
}
 | 
						|
 | 
						|
int open_display_sockets(int socks[2]) {
 | 
						|
	int lock_fd, display;
 | 
						|
	char lock_name[64];
 | 
						|
 | 
						|
	for (display = 0; display <= 32; display++) {
 | 
						|
		snprintf(lock_name, sizeof(lock_name), lock_fmt, display);
 | 
						|
		if ((lock_fd = open(lock_name, O_WRONLY | O_CREAT | O_EXCL | O_CLOEXEC, 0444)) >= 0) {
 | 
						|
			if (!open_sockets(socks, display)) {
 | 
						|
				unlink(lock_name);
 | 
						|
				close(lock_fd);
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
			char pid[12];
 | 
						|
			snprintf(pid, sizeof(pid), "%10d", getpid());
 | 
						|
			if (write(lock_fd, pid, sizeof(pid) - 1) != sizeof(pid) - 1) {
 | 
						|
				unlink(lock_name);
 | 
						|
				close(lock_fd);
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
			close(lock_fd);
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		if ((lock_fd = open(lock_name, O_RDONLY | O_CLOEXEC)) < 0) {
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
 | 
						|
		char pid[12] = { 0 }, *end_pid;
 | 
						|
		ssize_t bytes = read(lock_fd, pid, sizeof(pid) - 1);
 | 
						|
		close(lock_fd);
 | 
						|
 | 
						|
		if (bytes != sizeof(pid) - 1) {
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		long int read_pid;
 | 
						|
		read_pid = strtol(pid, &end_pid, 10);
 | 
						|
		if (read_pid < 0 || read_pid > INT32_MAX || end_pid != pid + sizeof(pid) - 2) {
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		errno = 0;
 | 
						|
		if (kill((pid_t)read_pid, 0) != 0 && errno == ESRCH) {
 | 
						|
			if (unlink(lock_name) != 0) {
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
			// retry
 | 
						|
			display--;
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (display > 32) {
 | 
						|
		wlr_log(L_ERROR, "No display available in the first 33");
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
 | 
						|
	return display;
 | 
						|
}
 |