move set_nonblock, create_socket and path_exists to libi3

This commit is contained in:
Michael Stapelberg 2021-01-15 14:48:10 +01:00 committed by Michael Stapelberg
parent e4f12ac349
commit 131a6158c8
8 changed files with 139 additions and 76 deletions

View File

@ -638,3 +638,27 @@ void draw_util_clear_surface(surface_t *surface, color_t color);
*/ */
void draw_util_copy_surface(surface_t *src, surface_t *dest, double src_x, double src_y, void draw_util_copy_surface(surface_t *src, surface_t *dest, double src_x, double src_y,
double dest_x, double dest_y, double width, double height); double dest_x, double dest_y, double width, double height);
/**
* Puts the given socket file descriptor into non-blocking mode or dies if
* setting O_NONBLOCK failed. Non-blocking sockets are a good idea for our
* IPC model because we should by no means block the window manager.
*
*/
void set_nonblock(int sockfd);
/**
* Creates the UNIX domain socket at the given path, sets it to non-blocking
* mode, bind()s and listen()s on it.
*
* The full path to the socket is stored in the char* that out_socketpath points
* to.
*
*/
int create_socket(const char *filename, char **out_socketpath);
/**
* Checks if the given path exists by calling stat().
*
*/
bool path_exists(const char *path);

70
libi3/create_socket.c Normal file
View File

@ -0,0 +1,70 @@
/*
* vim:ts=4:sw=4:expandtab
*
* i3 - an improved dynamic tiling window manager
* © 2009 Michael Stapelberg and contributors (see also: LICENSE)
*
*/
#include "libi3.h"
#include <unistd.h>
#include <libgen.h>
#include <err.h>
#include <fcntl.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <sys/un.h>
/*
* Creates the UNIX domain socket at the given path, sets it to non-blocking
* mode, bind()s and listen()s on it.
*
* The full path to the socket is stored in the char* that out_socketpath points
* to.
*
*/
int create_socket(const char *filename, char **out_socketpath) {
int sockfd;
char *resolved = resolve_tilde(filename);
DLOG("Creating UNIX socket at %s\n", resolved);
char *copy = sstrdup(resolved);
const char *dir = dirname(copy);
if (!path_exists(dir))
mkdirp(dir, DEFAULT_DIR_MODE);
free(copy);
/* Unlink the unix domain socket before */
unlink(resolved);
if ((sockfd = socket(AF_LOCAL, SOCK_STREAM, 0)) < 0) {
perror("socket()");
free(resolved);
return -1;
}
(void)fcntl(sockfd, F_SETFD, FD_CLOEXEC);
struct sockaddr_un addr;
memset(&addr, 0, sizeof(struct sockaddr_un));
addr.sun_family = AF_LOCAL;
strncpy(addr.sun_path, resolved, sizeof(addr.sun_path) - 1);
if (bind(sockfd, (struct sockaddr *)&addr, sizeof(struct sockaddr_un)) < 0) {
perror("bind()");
free(resolved);
return -1;
}
set_nonblock(sockfd);
if (listen(sockfd, 5) < 0) {
perror("listen()");
free(resolved);
return -1;
}
free(*out_socketpath);
*out_socketpath = resolved;
return sockfd;
}

View File

@ -11,15 +11,6 @@
#include <string.h> #include <string.h>
#include <sys/stat.h> #include <sys/stat.h>
/*
* Checks if the given path exists by calling stat().
*
*/
static bool path_exists(const char *path) {
struct stat buf;
return (stat(path, &buf) == 0);
}
/* /*
* Get the path of the first configuration file found. If override_configpath is * Get the path of the first configuration file found. If override_configpath is
* specified, that path is returned and saved for further calls. Otherwise, * specified, that path is returned and saved for further calls. Otherwise,

20
libi3/nonblock.c Normal file
View File

@ -0,0 +1,20 @@
#include "libi3.h"
#include <err.h>
#include <fcntl.h>
/*
* Puts the given socket file descriptor into non-blocking mode or dies if
* setting O_NONBLOCK failed. Non-blocking sockets are a good idea for our
* IPC model because we should by no means block the window manager.
*
*/
void set_nonblock(int sockfd) {
int flags = fcntl(sockfd, F_GETFL, 0);
if (flags & O_NONBLOCK) {
return;
}
flags |= O_NONBLOCK;
if (fcntl(sockfd, F_SETFL, flags) < 0)
err(-1, "Could not set O_NONBLOCK");
}

21
libi3/path_exists.c Normal file
View File

@ -0,0 +1,21 @@
/*
* vim:ts=4:sw=4:expandtab
*
* i3 - an improved dynamic tiling window manager
* © 2009 Michael Stapelberg and contributors (see also: LICENSE)
*
*/
#include "libi3.h"
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
/*
* Checks if the given path exists by calling stat().
*
*/
bool path_exists(const char *path) {
struct stat buf;
return (stat(path, &buf) == 0);
}

View File

@ -325,6 +325,7 @@ ev_dep = cc.find_library('ev')
inc = include_directories('include') inc = include_directories('include')
libi3srcs = [ libi3srcs = [
'libi3/create_socket.c',
'libi3/dpi.c', 'libi3/dpi.c',
'libi3/draw_util.c', 'libi3/draw_util.c',
'libi3/fake_configure_notify.c', 'libi3/fake_configure_notify.c',
@ -341,11 +342,13 @@ libi3srcs = [
'libi3/ipc_recv_message.c', 'libi3/ipc_recv_message.c',
'libi3/ipc_send_message.c', 'libi3/ipc_send_message.c',
'libi3/is_debug_build.c', 'libi3/is_debug_build.c',
'libi3/path_exists.c',
'libi3/resolve_tilde.c', 'libi3/resolve_tilde.c',
'libi3/root_atom_contents.c', 'libi3/root_atom_contents.c',
'libi3/safewrappers.c', 'libi3/safewrappers.c',
'libi3/string.c', 'libi3/string.c',
'libi3/ucs2_conversion.c', 'libi3/ucs2_conversion.c',
'libi3/nonblock.c',
] ]
if not cdata.get('HAVE_STRNDUP') if not cdata.get('HAVE_STRNDUP')

View File

@ -27,22 +27,6 @@ char *current_socketpath = NULL;
TAILQ_HEAD(ipc_client_head, ipc_client) all_clients = TAILQ_HEAD_INITIALIZER(all_clients); TAILQ_HEAD(ipc_client_head, ipc_client) all_clients = TAILQ_HEAD_INITIALIZER(all_clients);
/*
* Puts the given socket file descriptor into non-blocking mode or dies if
* setting O_NONBLOCK failed. Non-blocking sockets are a good idea for our
* IPC model because we should by no means block the window manager.
*
*/
static void set_nonblock(int sockfd) {
int flags = fcntl(sockfd, F_GETFL, 0);
if (flags & O_NONBLOCK) {
return;
}
flags |= O_NONBLOCK;
if (fcntl(sockfd, F_SETFL, flags) < 0)
err(-1, "Could not set O_NONBLOCK");
}
static void ipc_client_timeout(EV_P_ ev_timer *w, int revents); static void ipc_client_timeout(EV_P_ ev_timer *w, int revents);
static void ipc_socket_writeable_cb(EV_P_ struct ev_io *w, int revents); static void ipc_socket_writeable_cb(EV_P_ struct ev_io *w, int revents);
@ -1529,49 +1513,7 @@ ipc_client *ipc_new_client_on_fd(EV_P_ int fd) {
* *
*/ */
int ipc_create_socket(const char *filename) { int ipc_create_socket(const char *filename) {
int sockfd; return create_socket(filename, &current_socketpath);
FREE(current_socketpath);
char *resolved = resolve_tilde(filename);
DLOG("Creating IPC-socket at %s\n", resolved);
char *copy = sstrdup(resolved);
const char *dir = dirname(copy);
if (!path_exists(dir))
mkdirp(dir, DEFAULT_DIR_MODE);
free(copy);
/* Unlink the unix domain socket before */
unlink(resolved);
if ((sockfd = socket(AF_LOCAL, SOCK_STREAM, 0)) < 0) {
perror("socket()");
free(resolved);
return -1;
}
(void)fcntl(sockfd, F_SETFD, FD_CLOEXEC);
struct sockaddr_un addr;
memset(&addr, 0, sizeof(struct sockaddr_un));
addr.sun_family = AF_LOCAL;
strncpy(addr.sun_path, resolved, sizeof(addr.sun_path) - 1);
if (bind(sockfd, (struct sockaddr *)&addr, sizeof(struct sockaddr_un)) < 0) {
perror("bind()");
free(resolved);
return -1;
}
set_nonblock(sockfd);
if (listen(sockfd, 5) < 0) {
perror("listen()");
free(resolved);
return -1;
}
current_socketpath = resolved;
return sockfd;
} }
/* /*

View File

@ -176,14 +176,6 @@ void exec_i3_utility(char *name, char *argv[]) {
_exit(2); _exit(2);
} }
/*
* Checks if the given path exists by calling stat().
*
*/
bool path_exists(const char *path) {
struct stat buf;
return (stat(path, &buf) == 0);
}
/* /*
* Goes through the list of arguments (for exec()) and add/replace the given option, * Goes through the list of arguments (for exec()) and add/replace the given option,