2017-06-05 13:44:29 +02:00
|
|
|
#define _XOPEN_SOURCE 700
|
2017-03-11 05:41:24 +01:00
|
|
|
#define _POSIX_C_SOURCE 200112L
|
2017-11-18 17:22:02 +01:00
|
|
|
#include <getopt.h>
|
2018-07-05 01:01:09 +02:00
|
|
|
#include <pango/pangocairo.h>
|
2017-11-18 17:22:02 +01:00
|
|
|
#include <signal.h>
|
2015-08-05 03:30:40 +02:00
|
|
|
#include <stdbool.h>
|
2017-11-18 17:22:02 +01:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2016-12-02 01:27:35 +01:00
|
|
|
#include <sys/stat.h>
|
2017-11-18 17:22:02 +01:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/wait.h>
|
2015-10-18 19:53:56 +02:00
|
|
|
#include <sys/un.h>
|
2015-12-12 19:01:00 +01:00
|
|
|
#include <unistd.h>
|
2017-01-12 03:25:03 +01:00
|
|
|
#ifdef __linux__
|
2017-04-06 01:42:25 +02:00
|
|
|
#include <sys/capability.h>
|
2017-01-12 03:25:03 +01:00
|
|
|
#include <sys/prctl.h>
|
|
|
|
#endif
|
2017-12-31 22:32:28 +01:00
|
|
|
#include <wlr/util/log.h>
|
2018-05-16 05:29:30 +02:00
|
|
|
#include "sway/commands.h"
|
2017-12-05 10:40:55 +01:00
|
|
|
#include "sway/config.h"
|
2018-04-06 17:49:27 +02:00
|
|
|
#include "sway/debug.h"
|
2018-07-15 07:20:21 +02:00
|
|
|
#include "sway/desktop/transaction.h"
|
2017-10-22 16:37:30 +02:00
|
|
|
#include "sway/server.h"
|
2018-03-30 05:41:33 +02:00
|
|
|
#include "sway/tree/layout.h"
|
2017-11-23 03:37:07 +01:00
|
|
|
#include "sway/ipc-server.h"
|
2015-12-12 19:01:00 +01:00
|
|
|
#include "ipc-client.h"
|
2016-09-01 14:18:37 +02:00
|
|
|
#include "readline.h"
|
|
|
|
#include "stringop.h"
|
2017-05-10 01:51:28 +02:00
|
|
|
#include "util.h"
|
2015-08-20 15:12:34 +02:00
|
|
|
|
|
|
|
static bool terminate_request = false;
|
2016-02-26 09:08:05 +01:00
|
|
|
static int exit_value = 0;
|
2017-10-22 16:37:30 +02:00
|
|
|
struct sway_server server;
|
2015-08-20 15:12:34 +02:00
|
|
|
|
2016-02-26 09:08:05 +01:00
|
|
|
void sway_terminate(int exit_code) {
|
2015-08-20 15:12:34 +02:00
|
|
|
terminate_request = true;
|
2016-02-26 09:08:05 +01:00
|
|
|
exit_value = exit_code;
|
2017-10-22 16:37:30 +02:00
|
|
|
wl_display_terminate(server.wl_display);
|
2015-08-20 15:12:34 +02:00
|
|
|
}
|
2015-08-05 03:02:46 +02:00
|
|
|
|
2015-12-29 13:00:35 +01:00
|
|
|
void sig_handler(int signal) {
|
2017-11-18 17:22:02 +01:00
|
|
|
//close_views(&root_container);
|
2016-02-26 09:08:05 +01:00
|
|
|
sway_terminate(EXIT_SUCCESS);
|
2015-12-29 13:00:35 +01:00
|
|
|
}
|
|
|
|
|
2017-08-14 19:38:43 +02:00
|
|
|
void detect_raspi() {
|
|
|
|
bool raspi = false;
|
|
|
|
FILE *f = fopen("/sys/firmware/devicetree/base/model", "r");
|
|
|
|
if (!f) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
char *line;
|
|
|
|
while(!feof(f)) {
|
|
|
|
if (!(line = read_line(f))) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (strstr(line, "Raspberry Pi")) {
|
|
|
|
raspi = true;
|
|
|
|
}
|
|
|
|
free(line);
|
|
|
|
}
|
|
|
|
fclose(f);
|
|
|
|
FILE *g = fopen("/proc/modules", "r");
|
|
|
|
if (!g) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
bool vc4 = false;
|
|
|
|
while (!feof(g)) {
|
|
|
|
if (!(line = read_line(g))) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (strstr(line, "vc4")) {
|
|
|
|
vc4 = true;
|
|
|
|
}
|
|
|
|
free(line);
|
|
|
|
}
|
|
|
|
fclose(g);
|
|
|
|
if (!vc4 && raspi) {
|
|
|
|
fprintf(stderr, "\x1B[1;31mWarning: You have a "
|
|
|
|
"Raspberry Pi, but the vc4 Module is "
|
|
|
|
"not loaded! Set 'dtoverlay=vc4-kms-v3d'"
|
|
|
|
"in /boot/config.txt and reboot.\x1B[0m\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-14 17:13:44 +01:00
|
|
|
void detect_proprietary() {
|
2015-09-02 17:46:21 +02:00
|
|
|
FILE *f = fopen("/proc/modules", "r");
|
|
|
|
if (!f) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
while (!feof(f)) {
|
2016-12-15 23:08:56 +01:00
|
|
|
char *line;
|
|
|
|
if (!(line = read_line(f))) {
|
|
|
|
break;
|
|
|
|
}
|
2015-09-02 17:46:21 +02:00
|
|
|
if (strstr(line, "nvidia")) {
|
2017-11-28 11:40:00 +01:00
|
|
|
fprintf(stderr, "\x1B[1;31mWarning: Proprietary Nvidia drivers are "
|
|
|
|
"NOT supported. Use Nouveau.\x1B[0m\n");
|
|
|
|
free(line);
|
|
|
|
break;
|
2015-09-02 17:46:21 +02:00
|
|
|
}
|
2015-12-14 17:13:44 +01:00
|
|
|
if (strstr(line, "fglrx")) {
|
2017-04-12 16:04:28 +02:00
|
|
|
fprintf(stderr, "\x1B[1;31mWarning: Proprietary AMD drivers do "
|
|
|
|
"NOT support Wayland. Use radeon.\x1B[0m\n");
|
2015-12-14 17:13:44 +01:00
|
|
|
free(line);
|
|
|
|
break;
|
|
|
|
}
|
2015-09-02 17:46:21 +02:00
|
|
|
free(line);
|
|
|
|
}
|
2015-09-02 17:47:15 +02:00
|
|
|
fclose(f);
|
2015-09-02 17:46:21 +02:00
|
|
|
}
|
|
|
|
|
2015-12-12 19:01:00 +01:00
|
|
|
void run_as_ipc_client(char *command, char *socket_path) {
|
|
|
|
int socketfd = ipc_open_socket(socket_path);
|
|
|
|
uint32_t len = strlen(command);
|
|
|
|
char *resp = ipc_single_command(socketfd, IPC_COMMAND, command, &len);
|
|
|
|
printf("%s\n", resp);
|
|
|
|
close(socketfd);
|
|
|
|
}
|
|
|
|
|
2016-10-27 16:37:16 +02:00
|
|
|
static void log_env() {
|
|
|
|
const char *log_vars[] = {
|
|
|
|
"PATH",
|
2018-03-31 16:43:46 +02:00
|
|
|
"LD_LIBRARY_PATH",
|
2016-10-27 16:37:16 +02:00
|
|
|
"LD_PRELOAD_PATH",
|
2016-10-27 17:05:04 +02:00
|
|
|
"LD_LIBRARY_PATH",
|
2016-10-27 16:37:16 +02:00
|
|
|
"SWAY_CURSOR_THEME",
|
|
|
|
"SWAY_CURSOR_SIZE",
|
2017-10-22 16:37:30 +02:00
|
|
|
"SWAYSOCK"
|
2016-10-27 16:37:16 +02:00
|
|
|
};
|
|
|
|
for (size_t i = 0; i < sizeof(log_vars) / sizeof(char *); ++i) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_INFO, "%s=%s", log_vars[i], getenv(log_vars[i]));
|
2016-10-27 16:37:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-27 16:48:46 +02:00
|
|
|
static void log_distro() {
|
|
|
|
const char *paths[] = {
|
|
|
|
"/etc/lsb-release",
|
|
|
|
"/etc/os-release",
|
|
|
|
"/etc/debian_version",
|
|
|
|
"/etc/redhat-release",
|
|
|
|
"/etc/gentoo-release",
|
|
|
|
};
|
|
|
|
for (size_t i = 0; i < sizeof(paths) / sizeof(char *); ++i) {
|
|
|
|
FILE *f = fopen(paths[i], "r");
|
|
|
|
if (f) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_INFO, "Contents of %s:", paths[i]);
|
2016-10-27 16:48:46 +02:00
|
|
|
while (!feof(f)) {
|
2016-12-15 23:08:56 +01:00
|
|
|
char *line;
|
|
|
|
if (!(line = read_line(f))) {
|
|
|
|
break;
|
|
|
|
}
|
2016-10-27 16:48:46 +02:00
|
|
|
if (*line) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_INFO, "%s", line);
|
2016-10-27 16:48:46 +02:00
|
|
|
}
|
|
|
|
free(line);
|
|
|
|
}
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-27 16:50:22 +02:00
|
|
|
static void log_kernel() {
|
|
|
|
FILE *f = popen("uname -a", "r");
|
|
|
|
if (!f) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_INFO, "Unable to determine kernel version");
|
2016-10-27 16:50:22 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
while (!feof(f)) {
|
2016-12-15 23:08:56 +01:00
|
|
|
char *line;
|
|
|
|
if (!(line = read_line(f))) {
|
|
|
|
break;
|
|
|
|
}
|
2016-10-27 16:50:22 +02:00
|
|
|
if (*line) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_INFO, "%s", line);
|
2016-10-27 16:50:22 +02:00
|
|
|
}
|
|
|
|
free(line);
|
|
|
|
}
|
2018-06-30 09:44:36 +02:00
|
|
|
pclose(f);
|
2016-10-27 16:50:22 +02:00
|
|
|
}
|
|
|
|
|
2016-12-02 01:27:35 +01:00
|
|
|
static void security_sanity_check() {
|
|
|
|
// TODO: Notify users visually if this has issues
|
2016-12-02 03:58:38 +01:00
|
|
|
struct stat s;
|
2016-12-02 01:27:35 +01:00
|
|
|
if (stat("/proc", &s)) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR,
|
2016-12-02 01:27:35 +01:00
|
|
|
"!! DANGER !! /proc is not available - sway CANNOT enforce security rules!");
|
|
|
|
}
|
2016-12-08 13:34:08 +01:00
|
|
|
#ifdef __linux__
|
2016-12-03 00:37:01 +01:00
|
|
|
cap_flag_value_t v;
|
|
|
|
cap_t cap = cap_get_proc();
|
|
|
|
if (!cap || cap_get_flag(cap, CAP_SYS_PTRACE, CAP_PERMITTED, &v) != 0 || v != CAP_SET) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR,
|
2016-12-03 00:37:01 +01:00
|
|
|
"!! DANGER !! Sway does not have CAP_SYS_PTRACE and cannot enforce security rules for processes running as other users.");
|
|
|
|
}
|
|
|
|
if (cap) {
|
|
|
|
cap_free(cap);
|
|
|
|
}
|
2016-12-08 13:34:08 +01:00
|
|
|
#endif
|
2016-12-02 01:27:35 +01:00
|
|
|
}
|
|
|
|
|
2017-05-10 01:51:28 +02:00
|
|
|
static void executable_sanity_check() {
|
|
|
|
#ifdef __linux__
|
|
|
|
struct stat sb;
|
|
|
|
char *exe = realpath("/proc/self/exe", NULL);
|
|
|
|
stat(exe, &sb);
|
|
|
|
// We assume that cap_get_file returning NULL implies ENODATA
|
|
|
|
if (sb.st_mode & (S_ISUID|S_ISGID) && cap_get_file(exe)) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR,
|
2017-05-10 01:51:28 +02:00
|
|
|
"sway executable has both the s(g)uid bit AND file caps set.");
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR,
|
2017-05-10 01:51:28 +02:00
|
|
|
"This is strongly discouraged (and completely broken).");
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR,
|
2017-05-10 01:51:28 +02:00
|
|
|
"Please clear one of them (either the suid bit, or the file caps).");
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR,
|
2017-05-10 01:51:28 +02:00
|
|
|
"If unsure, strip the file caps.");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
free(exe);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-10-22 16:37:30 +02:00
|
|
|
static void drop_permissions(bool keep_caps) {
|
|
|
|
if (getuid() != geteuid() || getgid() != getegid()) {
|
|
|
|
if (setgid(getgid()) != 0) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR, "Unable to drop root");
|
2017-10-22 16:37:30 +02:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
if (setuid(getuid()) != 0) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR, "Unable to drop root");
|
2017-10-22 16:37:30 +02:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (setuid(0) != -1) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR, "Root privileges can be restored.");
|
2017-10-22 16:37:30 +02:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
#ifdef __linux__
|
|
|
|
if (keep_caps) {
|
|
|
|
// Drop every cap except CAP_SYS_PTRACE
|
|
|
|
cap_t caps = cap_init();
|
|
|
|
cap_value_t keep = CAP_SYS_PTRACE;
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_INFO, "Dropping extra capabilities");
|
2017-10-22 16:37:30 +02:00
|
|
|
if (cap_set_flag(caps, CAP_PERMITTED, 1, &keep, CAP_SET) ||
|
|
|
|
cap_set_flag(caps, CAP_EFFECTIVE, 1, &keep, CAP_SET) ||
|
|
|
|
cap_set_proc(caps)) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR, "Failed to drop extra capabilities");
|
2017-10-22 16:37:30 +02:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2018-07-14 17:24:22 +02:00
|
|
|
void enable_debug_flag(const char *flag) {
|
|
|
|
if (strcmp(flag, "render-tree") == 0) {
|
|
|
|
enable_debug_tree = true;
|
|
|
|
} else if (strncmp(flag, "damage=", 7) == 0) {
|
|
|
|
damage_debug = &flag[7];
|
|
|
|
} else if (strcmp(flag, "txn-debug") == 0) {
|
|
|
|
txn_debug = true;
|
|
|
|
} else if (strncmp(flag, "txn-timeout=", 12) == 0) {
|
|
|
|
txn_timeout_ms = atoi(&flag[12]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-05 03:30:40 +02:00
|
|
|
int main(int argc, char **argv) {
|
2015-08-20 14:37:09 +02:00
|
|
|
static int verbose = 0, debug = 0, validate = 0;
|
|
|
|
|
|
|
|
static struct option long_options[] = {
|
2015-11-26 20:03:21 +01:00
|
|
|
{"help", no_argument, NULL, 'h'},
|
2015-08-20 14:37:09 +02:00
|
|
|
{"config", required_argument, NULL, 'c'},
|
2015-11-28 15:35:44 +01:00
|
|
|
{"validate", no_argument, NULL, 'C'},
|
|
|
|
{"debug", no_argument, NULL, 'd'},
|
2015-08-20 14:37:09 +02:00
|
|
|
{"version", no_argument, NULL, 'v'},
|
2015-11-28 15:35:44 +01:00
|
|
|
{"verbose", no_argument, NULL, 'V'},
|
2015-08-20 14:37:09 +02:00
|
|
|
{"get-socketpath", no_argument, NULL, 'p'},
|
2015-08-26 20:01:26 +02:00
|
|
|
{0, 0, 0, 0}
|
2015-08-20 14:37:09 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
char *config_path = NULL;
|
2015-11-26 20:01:37 +01:00
|
|
|
|
|
|
|
const char* usage =
|
|
|
|
"Usage: sway [options] [command]\n"
|
|
|
|
"\n"
|
2015-11-26 20:03:21 +01:00
|
|
|
" -h, --help Show help message and quit.\n"
|
2015-11-26 20:01:37 +01:00
|
|
|
" -c, --config <config> Specify a config file.\n"
|
|
|
|
" -C, --validate Check the validity of the config file, then exit.\n"
|
|
|
|
" -d, --debug Enables full logging, including debug information.\n"
|
|
|
|
" -v, --version Show the version number and quit.\n"
|
|
|
|
" -V, --verbose Enables more verbose logging.\n"
|
|
|
|
" --get-socketpath Gets the IPC socket path and prints it, then exits.\n"
|
|
|
|
"\n";
|
|
|
|
|
2016-12-02 16:23:30 +01:00
|
|
|
// Security:
|
|
|
|
unsetenv("LD_PRELOAD");
|
2017-01-16 00:05:05 +01:00
|
|
|
#ifdef _LD_LIBRARY_PATH
|
2016-12-02 16:23:30 +01:00
|
|
|
setenv("LD_LIBRARY_PATH", _LD_LIBRARY_PATH, 1);
|
2017-01-16 00:05:05 +01:00
|
|
|
#else
|
|
|
|
unsetenv("LD_LIBRARY_PATH");
|
|
|
|
#endif
|
2016-12-02 14:47:03 +01:00
|
|
|
|
2015-08-20 14:37:09 +02:00
|
|
|
int c;
|
|
|
|
while (1) {
|
|
|
|
int option_index = 0;
|
2018-07-14 17:24:22 +02:00
|
|
|
c = getopt_long(argc, argv, "hCdD:vVc:", long_options, &option_index);
|
2015-08-20 14:37:09 +02:00
|
|
|
if (c == -1) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
switch (c) {
|
2015-11-26 20:03:21 +01:00
|
|
|
case 'h': // help
|
|
|
|
fprintf(stdout, "%s", usage);
|
|
|
|
exit(EXIT_SUCCESS);
|
|
|
|
break;
|
2015-08-20 14:37:09 +02:00
|
|
|
case 'c': // config
|
|
|
|
config_path = strdup(optarg);
|
|
|
|
break;
|
|
|
|
case 'C': // validate
|
|
|
|
validate = 1;
|
|
|
|
break;
|
|
|
|
case 'd': // debug
|
|
|
|
debug = 1;
|
|
|
|
break;
|
2018-04-06 17:49:27 +02:00
|
|
|
case 'D': // extended debug options
|
2018-07-14 17:24:22 +02:00
|
|
|
enable_debug_flag(optarg);
|
2018-04-06 17:49:27 +02:00
|
|
|
break;
|
2015-08-20 14:37:09 +02:00
|
|
|
case 'v': // version
|
2017-04-26 11:29:30 +02:00
|
|
|
fprintf(stdout, "sway version " SWAY_VERSION "\n");
|
2015-11-28 14:47:44 +01:00
|
|
|
exit(EXIT_SUCCESS);
|
2015-08-20 14:37:09 +02:00
|
|
|
break;
|
|
|
|
case 'V': // verbose
|
|
|
|
verbose = 1;
|
|
|
|
break;
|
2015-10-18 19:53:56 +02:00
|
|
|
case 'p': ; // --get-socketpath
|
2015-11-13 17:53:46 +01:00
|
|
|
if (getenv("SWAYSOCK")) {
|
|
|
|
fprintf(stdout, "%s\n", getenv("SWAYSOCK"));
|
2015-11-28 14:47:44 +01:00
|
|
|
exit(EXIT_SUCCESS);
|
2015-11-13 17:53:46 +01:00
|
|
|
} else {
|
|
|
|
fprintf(stderr, "sway socket not detected.\n");
|
2015-11-28 14:47:44 +01:00
|
|
|
exit(EXIT_FAILURE);
|
2015-11-13 17:53:46 +01:00
|
|
|
}
|
2015-08-20 14:37:09 +02:00
|
|
|
break;
|
2015-11-26 20:01:37 +01:00
|
|
|
default:
|
|
|
|
fprintf(stderr, "%s", usage);
|
|
|
|
exit(EXIT_FAILURE);
|
2015-08-20 14:37:09 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-22 16:37:30 +02:00
|
|
|
// TODO: switch logging over to wlroots?
|
2017-05-11 18:29:10 +02:00
|
|
|
if (debug) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log_init(WLR_DEBUG, NULL);
|
2017-05-11 18:29:10 +02:00
|
|
|
} else if (verbose || validate) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log_init(WLR_INFO, NULL);
|
2017-05-11 18:29:10 +02:00
|
|
|
} else {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log_init(WLR_ERROR, NULL);
|
2017-05-11 18:29:10 +02:00
|
|
|
}
|
|
|
|
|
2015-12-12 19:01:00 +01:00
|
|
|
if (optind < argc) { // Behave as IPC client
|
2016-02-25 22:19:33 +01:00
|
|
|
if(optind != 1) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR, "Don't use options with the IPC client");
|
2016-02-26 09:08:05 +01:00
|
|
|
exit(EXIT_FAILURE);
|
2016-02-25 22:19:33 +01:00
|
|
|
}
|
2017-10-22 16:37:30 +02:00
|
|
|
drop_permissions(false);
|
2015-12-12 19:01:00 +01:00
|
|
|
char *socket_path = getenv("SWAYSOCK");
|
|
|
|
if (!socket_path) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR, "Unable to retrieve socket path");
|
2016-02-26 09:08:05 +01:00
|
|
|
exit(EXIT_FAILURE);
|
2015-12-12 19:01:00 +01:00
|
|
|
}
|
|
|
|
char *command = join_args(argv + optind, argc - optind);
|
|
|
|
run_as_ipc_client(command, socket_path);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-05-10 01:51:28 +02:00
|
|
|
executable_sanity_check();
|
2017-01-12 03:25:03 +01:00
|
|
|
bool suid = false;
|
2018-06-13 00:39:24 +02:00
|
|
|
|
|
|
|
if (!server_privileged_prepare(&server)) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-10-22 16:37:30 +02:00
|
|
|
#ifdef __linux__
|
2017-01-12 03:25:03 +01:00
|
|
|
if (getuid() != geteuid() || getgid() != getegid()) {
|
|
|
|
// Retain capabilities after setuid()
|
|
|
|
if (prctl(PR_SET_KEEPCAPS, 1, 0, 0, 0)) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR, "Cannot keep caps after setuid()");
|
2017-01-12 03:25:03 +01:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
suid = true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-02-23 14:32:11 +01:00
|
|
|
log_kernel();
|
|
|
|
log_distro();
|
2015-12-14 17:13:44 +01:00
|
|
|
detect_proprietary();
|
2017-08-14 19:38:43 +02:00
|
|
|
detect_raspi();
|
2015-11-28 19:52:28 +01:00
|
|
|
|
2017-01-12 03:25:03 +01:00
|
|
|
#ifdef __linux__
|
2017-10-22 16:37:30 +02:00
|
|
|
drop_permissions(suid);
|
2017-01-12 03:25:03 +01:00
|
|
|
#endif
|
2015-12-29 13:00:35 +01:00
|
|
|
// handle SIGTERM signals
|
|
|
|
signal(SIGTERM, sig_handler);
|
|
|
|
|
2016-01-22 02:29:18 +01:00
|
|
|
// prevent ipc from crashing sway
|
|
|
|
signal(SIGPIPE, SIG_IGN);
|
|
|
|
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_INFO, "Starting sway version " SWAY_VERSION);
|
2015-09-02 15:42:27 +02:00
|
|
|
|
2018-03-30 05:41:33 +02:00
|
|
|
layout_init();
|
2017-12-09 20:06:00 +01:00
|
|
|
|
2017-10-22 16:37:30 +02:00
|
|
|
if (!server_init(&server)) {
|
|
|
|
return 1;
|
|
|
|
}
|
2017-11-11 17:58:43 +01:00
|
|
|
|
2017-11-23 03:37:07 +01:00
|
|
|
ipc_init(&server);
|
2017-12-02 13:49:23 +01:00
|
|
|
log_env();
|
2016-10-03 00:29:40 +02:00
|
|
|
|
2017-12-05 10:40:55 +01:00
|
|
|
if (validate) {
|
|
|
|
bool valid = load_main_config(config_path, false);
|
|
|
|
return valid ? 0 : 1;
|
|
|
|
}
|
2015-08-20 14:37:09 +02:00
|
|
|
|
2017-12-05 10:40:55 +01:00
|
|
|
if (!load_main_config(config_path, false)) {
|
|
|
|
sway_terminate(EXIT_FAILURE);
|
|
|
|
}
|
2016-03-24 22:13:42 +01:00
|
|
|
|
2015-08-20 14:37:09 +02:00
|
|
|
if (config_path) {
|
|
|
|
free(config_path);
|
|
|
|
}
|
2015-08-18 00:15:05 +02:00
|
|
|
|
2016-12-02 14:42:26 +01:00
|
|
|
security_sanity_check();
|
|
|
|
|
2018-07-04 06:50:23 +02:00
|
|
|
setenv("WAYLAND_DISPLAY", server.socket, true);
|
2018-07-19 07:39:58 +02:00
|
|
|
if (!terminate_request) {
|
|
|
|
if (!server_start_backend(&server)) {
|
|
|
|
sway_terminate(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
config->active = true;
|
2018-05-16 05:29:30 +02:00
|
|
|
// Execute commands until there are none left
|
2018-07-19 07:39:58 +02:00
|
|
|
wlr_log(WLR_DEBUG, "Running deferred commands");
|
2018-05-16 05:29:30 +02:00
|
|
|
while (config->cmd_queue->length) {
|
|
|
|
char *line = config->cmd_queue->items[0];
|
|
|
|
struct cmd_results *res = execute_command(line, NULL);
|
|
|
|
if (res->status != CMD_SUCCESS) {
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_ERROR, "Error on line '%s': %s", line, res->error);
|
2018-05-16 05:29:30 +02:00
|
|
|
}
|
|
|
|
free_cmd_results(res);
|
|
|
|
free(line);
|
|
|
|
list_del(config->cmd_queue, 0);
|
|
|
|
}
|
2018-07-15 07:20:21 +02:00
|
|
|
transaction_commit_dirty();
|
2017-12-06 12:36:06 +01:00
|
|
|
|
2015-08-20 15:12:34 +02:00
|
|
|
if (!terminate_request) {
|
2017-11-11 17:58:43 +01:00
|
|
|
server_run(&server);
|
2015-08-20 15:12:34 +02:00
|
|
|
}
|
|
|
|
|
2018-07-09 23:54:30 +02:00
|
|
|
wlr_log(WLR_INFO, "Shutting down sway");
|
2017-12-29 11:29:13 +01:00
|
|
|
|
2017-10-22 16:37:30 +02:00
|
|
|
server_fini(&server);
|
2016-01-17 11:53:37 +01:00
|
|
|
|
2017-12-05 10:40:55 +01:00
|
|
|
if (config) {
|
|
|
|
free_config(config);
|
|
|
|
}
|
2016-02-24 18:53:09 +01:00
|
|
|
|
2018-07-05 01:01:09 +02:00
|
|
|
pango_cairo_font_map_set_default(NULL);
|
|
|
|
|
2016-02-26 09:08:05 +01:00
|
|
|
return exit_value;
|
2015-08-05 03:02:46 +02:00
|
|
|
}
|