SketchyBar/src/sketchybar.c
2021-09-08 17:53:17 +02:00

228 lines
6.4 KiB
C

#include "misc/socket.h"
#define SOCKET_PATH_FMT "/tmp/sketchybar_%s.socket"
#define LCFILE_PATH_FMT "/tmp/sketchybar_%s.lock"
#define CLIENT_OPT_LONG "--message"
#define CLIENT_OPT_SHRT "-m"
#define DEBUG_VERBOSE_OPT_LONG "--verbose"
#define DEBUG_VERBOSE_OPT_SHRT "-V"
#define VERSION_OPT_LONG "--version"
#define VERSION_OPT_SHRT "-v"
#define CONFIG_OPT_LONG "--config"
#define CONFIG_OPT_SHRT "-c"
#define MAJOR 1
#define MINOR 1
#define PATCH 0
extern int SLSMainConnectionID(void);
extern int RunApplicationEventLoop(void);
#define CONNECTION_CALLBACK(name) void name(uint32_t type, void *data, size_t data_length, void *context, int cid)
typedef CONNECTION_CALLBACK(connection_callback);
struct event_loop g_event_loop;
void *g_workspace_context;
struct daemon g_daemon;
struct bar_manager g_bar_manager;
int g_connection;
char g_socket_file[MAXLEN];
char g_config_file[4096];
char g_lock_file[MAXLEN];
bool g_verbose;
static int client_send_message(int argc, char **argv) {
if (argc <= 1) {
return EXIT_SUCCESS;
}
char *user = getenv("USER");
if (!user) {
error("sketchybar-msg: 'env USER' not set! abort..\n");
}
int sockfd;
char socket_file[MAXLEN];
snprintf(socket_file, sizeof(socket_file), SOCKET_PATH_FMT, user);
int count = 0;
while (!socket_connect_un(&sockfd, socket_file)) {
count++;
if (count > 100) error("sketchybar-msg: failed to connect to socket..\n");
}
int message_length = argc;
int argl[argc];
for (int i = 1; i < argc; ++i) {
argl[i] = strlen(argv[i]);
message_length += argl[i];
}
char message[message_length];
char *temp = message;
for (int i = 1; i < argc; ++i) {
memcpy(temp, argv[i], argl[i]);
temp += argl[i];
*temp++ = '\0';
}
*temp++ = '\0';
count = 0;
while (!socket_write_bytes(sockfd, message, message_length)) {
count++;
if (count > 100) {
shutdown(sockfd, SHUT_WR);
socket_close(sockfd);
error("sketchybar-msg: failed to send data..\n");
}
}
shutdown(sockfd, SHUT_WR);
socket_close(sockfd);
return EXIT_SUCCESS;
}
static void acquire_lockfile(void) {
int handle = open(g_lock_file, O_CREAT | O_WRONLY, 0600);
if (handle == -1) {
error("sketchybar: could not create lock-file! abort..\n");
}
struct flock lockfd = {
.l_start = 0,
.l_len = 0,
.l_pid = getpid(),
.l_type = F_WRLCK,
.l_whence = SEEK_SET
};
if (fcntl(handle, F_SETLK, &lockfd) == -1) {
error("sketchybar: could not acquire lock-file! abort..\n");
}
}
static bool get_config_file(char *restrict filename, char *restrict buffer, int buffer_size) {
char *xdg_home = getenv("XDG_CONFIG_HOME");
if (xdg_home && *xdg_home) {
snprintf(buffer, buffer_size, "%s/sketchybar/%s", xdg_home, filename);
if (file_exists(buffer)) return true;
}
char *home = getenv("HOME");
if (!home) return false;
snprintf(buffer, buffer_size, "%s/.config/sketchybar/%s", home, filename);
if (file_exists(buffer)) return true;
snprintf(buffer, buffer_size, "%s/.%s", home, filename);
return file_exists(buffer);
}
static void exec_config_file(void) {
if (!*g_config_file && !get_config_file("sketchybarrc", g_config_file, sizeof(g_config_file))) {
printf("could not locate config file..");
return;
}
if (!file_exists(g_config_file)) {
printf("file '%s' does not exist..", g_config_file);
return;
}
if (!ensure_executable_permission(g_config_file)) {
printf("could not set the executable permission bit for '%s'", g_config_file);
return;
}
if (!fork_exec(g_config_file, NULL)) {
printf("failed to execute file '%s'", g_config_file);
return;
}
}
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
static inline void init_misc_settings(void) {
char *user = getenv("USER");
if (!user) {
error("sketchybar: 'env USER' not set! abort..\n");
}
snprintf(g_socket_file, sizeof(g_socket_file), SOCKET_PATH_FMT, user);
snprintf(g_lock_file, sizeof(g_lock_file), LCFILE_PATH_FMT, user);
NSApplicationLoad();
signal(SIGCHLD, SIG_IGN);
signal(SIGPIPE, SIG_IGN);
CGSetLocalEventsSuppressionInterval(0.0f);
CGEnableEventStateCombining(false);
g_connection = SLSMainConnectionID();
}
#pragma clang diagnostic pop
static void parse_arguments(int argc, char **argv) {
if ((string_equals(argv[1], VERSION_OPT_LONG)) ||
(string_equals(argv[1], VERSION_OPT_SHRT))) {
fprintf(stdout, "sketchybar-v%d.%d.%d\n", MAJOR, MINOR, PATCH);
exit(EXIT_SUCCESS);
}
if ((string_equals(argv[1], CLIENT_OPT_LONG)) ||
(string_equals(argv[1], CLIENT_OPT_SHRT))) {
exit(client_send_message(argc-1, argv+1));
}
for (int i = 1; i < argc; ++i) {
char *opt = argv[i];
if ((string_equals(opt, DEBUG_VERBOSE_OPT_LONG)) ||
(string_equals(opt, DEBUG_VERBOSE_OPT_SHRT))) {
g_verbose = true;
} else if ((string_equals(opt, CONFIG_OPT_LONG)) ||
(string_equals(opt, CONFIG_OPT_SHRT))) {
char *val = i < argc - 1 ? argv[++i] : NULL;
if (!val) error("sketchybar: option '%s|%s' requires an argument!\n", CONFIG_OPT_LONG, CONFIG_OPT_SHRT);
snprintf(g_config_file, sizeof(g_config_file), "%s", val);
} else {
error("sketchybar: '%s' is not a valid option!\n", opt);
}
}
}
int main(int argc, char **argv) {
if (argc > 1) {
parse_arguments(argc, argv);
}
if (is_root()) {
error("sketchybar: running as root is not allowed! abort..\n");
}
init_misc_settings();
acquire_lockfile();
if (!event_loop_init(&g_event_loop)) {
error("sketchybar: could not initialize event_loop! abort..\n");
}
workspace_event_handler_init(&g_workspace_context);
bar_manager_init(&g_bar_manager);
event_loop_begin(&g_event_loop);
mouse_begin();
display_begin();
workspace_event_handler_begin(&g_workspace_context);
bar_manager_begin(&g_bar_manager);
if (!socket_daemon_begin_un(&g_daemon, g_socket_file, message_handler)) {
error("sketchybar: could not initialize daemon! abort..\n");
}
exec_config_file();
RunApplicationEventLoop();
return 0;
}