#define _POSIX_C_SOURCE 200809L #include <stdio.h> #include <stdint.h> #include <stdlib.h> #include <string.h> #include <sys/socket.h> #include <sys/un.h> #include <unistd.h> #include "ipc-client.h" #include "readline.h" #include "log.h" static const char ipc_magic[] = {'i', '3', '-', 'i', 'p', 'c'}; static const size_t ipc_header_size = sizeof(ipc_magic)+8; char *get_socketpath(void) { const char *swaysock = getenv("SWAYSOCK"); if (swaysock) { return strdup(swaysock); } FILE *fp = popen("sway --get-socketpath 2>/dev/null", "r"); if (fp) { char *line = read_line(fp); pclose(fp); if (line && *line) { return line; } free(line); } const char *i3sock = getenv("I3SOCK"); if (i3sock) { return strdup(i3sock); } fp = popen("i3 --get-socketpath 2>/dev/null", "r"); if (fp) { char *line = read_line(fp); pclose(fp); if (line && *line) { return line; } free(line); } return NULL; } int ipc_open_socket(const char *socket_path) { struct sockaddr_un addr; int socketfd; if ((socketfd = socket(AF_UNIX, SOCK_STREAM, 0)) == -1) { sway_abort("Unable to open Unix socket"); } addr.sun_family = AF_UNIX; strncpy(addr.sun_path, socket_path, sizeof(addr.sun_path) - 1); addr.sun_path[sizeof(addr.sun_path) - 1] = 0; int l = sizeof(struct sockaddr_un); if (connect(socketfd, (struct sockaddr *)&addr, l) == -1) { sway_abort("Unable to connect to %s", socket_path); } return socketfd; } struct ipc_response *ipc_recv_response(int socketfd) { char data[ipc_header_size]; uint32_t *data32 = (uint32_t *)(data + sizeof(ipc_magic)); size_t total = 0; while (total < ipc_header_size) { ssize_t received = recv(socketfd, data + total, ipc_header_size - total, 0); if (received <= 0) { sway_abort("Unable to receive IPC response"); } total += received; } struct ipc_response *response = malloc(sizeof(struct ipc_response)); if (!response) { goto error_1; } total = 0; response->size = data32[0]; response->type = data32[1]; char *payload = malloc(response->size + 1); if (!payload) { goto error_2; } while (total < response->size) { ssize_t received = recv(socketfd, payload + total, response->size - total, 0); if (received < 0) { sway_abort("Unable to receive IPC response"); } total += received; } payload[response->size] = '\0'; response->payload = payload; return response; error_2: free(response); error_1: wlr_log(WLR_ERROR, "Unable to allocate memory for IPC response"); return NULL; } void free_ipc_response(struct ipc_response *response) { free(response->payload); free(response); } char *ipc_single_command(int socketfd, uint32_t type, const char *payload, uint32_t *len) { char data[ipc_header_size]; uint32_t *data32 = (uint32_t *)(data + sizeof(ipc_magic)); memcpy(data, ipc_magic, sizeof(ipc_magic)); data32[0] = *len; data32[1] = type; if (write(socketfd, data, ipc_header_size) == -1) { sway_abort("Unable to send IPC header"); } if (write(socketfd, payload, *len) == -1) { sway_abort("Unable to send IPC payload"); } struct ipc_response *resp = ipc_recv_response(socketfd); char *response = resp->payload; *len = resp->size; free(resp); return response; }