#include #include #include #include #include #include #include "wlr-layer-shell-unstable-v1-client-protocol.h" #include "shm.h" struct wb_color { uint32_t r, g, b; }; static bool parse_color(const char *str, struct wb_color *color) { if (str[0] == '#') { str++; } uint32_t r, g, b; char dummy[2]; int res = sscanf(str, "%02x%02x%02x%s", &r, &g, &b, dummy); if (res != 3 || dummy[0] != '\0') { return false; } color->r = r; color->g = g; color->b = b; return true; } struct wb_config { struct wb_color *bg_color; }; static void print_config(struct wb_config *config) { struct wb_color *color = config->bg_color; printf("config:\n" " - color: [%d, %d, %d]\n", color->r, color->g, color->b); } struct wb_output { uint32_t wl_name; struct wl_output *wl_output; char *make, *model; uint32_t width, height; int32_t scale; bool configured; struct wl_list link; }; struct wb_state { struct wl_display *display; struct wl_registry *registry; struct wl_compositor *compositor; struct wl_shm *shm; struct zwlr_layer_shell_v1 *layer_shell; struct wl_list outputs; }; static void destroy_output(struct wb_output *output) { if (output == NULL) { return; } wl_list_remove(&output->link); wl_output_destroy(output->wl_output); free(output->make); free(output->model); free(output); } static void output_geometry(void *data, struct wl_output *wl_output, int32_t x, int32_t y, int32_t width_mm, int32_t height_mm, int32_t subpixel, const char *make, const char *model, int32_t transform) { (void)wl_output; (void)x; (void)y; (void)width_mm; (void)height_mm; (void)subpixel; (void)transform; struct wb_output *output = data; free(output->make); free(output->model); if (make != NULL) { uint32_t len = strlen(make) + 1; char *new_make = malloc(len); if (new_make != NULL) { memcpy(new_make, make, len); } output->make = new_make; } if (model != NULL) { uint32_t len = strlen(model) + 1; char *new_model = malloc(len); if (new_model != NULL) { memcpy(new_model, model, len); } output->model = new_model; } } static void output_mode(void *data, struct wl_output *wl_output, uint32_t flags, int32_t width, int32_t height, int32_t refresh) { (void)wl_output; (void)refresh; /* is this necessary? */ if ((flags & WL_OUTPUT_MODE_CURRENT) == 0) { return; } struct wb_output *output = data; output->width = width; output->height = height; } static void output_done(void *data, struct wl_output *wl_output) { (void)wl_output; struct wb_output *output = data; printf("output: %s %s (%dx%d, scale=%d)\n", output->make, output->model, output->width, output->height, output->scale); } static void output_scale(void *data, struct wl_output *wl_output, int32_t factor) { (void)data; (void)wl_output; struct wb_output *output = data; output->scale = factor; } static const struct wl_output_listener output_listener = { .geometry = output_geometry, .mode = output_mode, .done = output_done, .scale = output_scale, }; static void handle_global(void *data, struct wl_registry *registry, uint32_t name, const char *interface, uint32_t version) { (void)version; struct wb_state *state = data; if (strcmp(interface, wl_compositor_interface.name) == 0) { state->compositor = wl_registry_bind(registry, name, &wl_compositor_interface, 4); } else if (strcmp(interface, wl_shm_interface.name) == 0) { state->shm = wl_registry_bind(registry, name, &wl_shm_interface, 1); } else if (strcmp(interface, wl_output_interface.name) == 0) { struct wb_output *output = calloc(1, sizeof(struct wb_output)); output->wl_name = name; output->wl_output = wl_registry_bind(registry, name, &wl_output_interface, 3); wl_output_add_listener(output->wl_output, &output_listener, output); wl_list_insert(&state->outputs, &output->link); } else if (strcmp(interface, zwlr_layer_shell_v1_interface.name) == 0) { state->layer_shell = wl_registry_bind(registry, name, &zwlr_layer_shell_v1_interface, 1); } } static void handle_global_remove(void *data, struct wl_registry *registry, uint32_t name) { (void)registry; struct wb_state *state = data; struct wb_output *output, *tmp; wl_list_for_each_safe(output, tmp, &state->outputs, link) { if (output->wl_name == name) { destroy_output(output); break; } } } static const struct wl_registry_listener registry_listener = { .global = handle_global, .global_remove = handle_global_remove, }; static bool init_wayland(struct wb_state *state) { state->display = wl_display_connect(NULL); if (state->display == NULL) { fprintf(stderr, "err: failed to connect to wayland display\n"); return false; } state->registry = wl_display_get_registry(state->display); if (state->registry == NULL) { fprintf(stderr, "err: failed to get display registry\n"); return false; } wl_list_init(&state->outputs); wl_registry_add_listener(state->registry, ®istry_listener, state); wl_display_roundtrip(state->display); if (state->compositor == NULL || state->shm == NULL || state->layer_shell == NULL) { fprintf( stderr, "err: one or multiple not found: compositor, shm, layer shell\n"); return false; } return true; } static void finish_wayland(struct wb_state *state) { struct wb_output *output, *tmp; wl_list_for_each_safe(output, tmp, &state->outputs, link) { destroy_output(output); } if (state->compositor != NULL) { wl_compositor_destroy(state->compositor); } if (state->shm != NULL) { wl_shm_destroy(state->shm); } if (state->layer_shell != NULL) { zwlr_layer_shell_v1_destroy(state->layer_shell); } if (state->registry != NULL) { wl_registry_destroy(state->registry); } if (state->display != NULL) { wl_display_disconnect(state->display); } } static void usage(const char *progname) { fprintf(stderr, "usage: %s [-hV] [-c color]\n" " -h, --help show usage message and exit\n" " -V, --version show version information and exit\n" " -c, --color set the background color option\n", progname); } /* NOTE: color configuration is kept, but no rendering is done yet */ int main(int argc, char *argv[]) { const char *progname = argv[0]; const struct option longopts[] = { {"help", no_argument, 0, 'h'}, {"version", no_argument, 0, 'V'}, {"color", required_argument, 0, 'c'}, {NULL, no_argument, 0, 0}, }; /* app config data + defaults */ struct wb_color color = {18, 18, 18}; struct wb_config config = {.bg_color = &color}; int c; do { c = getopt_long(argc, argv, ":hVc:", longopts, NULL); switch (c) { case 'h': usage(progname); return EXIT_SUCCESS; case 'V': fprintf(stderr, "%s version " WAYBACK_VERSION "\n", progname); return EXIT_SUCCESS; case 'c': if (!parse_color(optarg, &color)) { fprintf(stderr, "%s: invalid color argument -- '%s'\n", progname, optarg); return EXIT_FAILURE; } break; case ':': fprintf(stderr, "%s: option requires an argument -- '%c'\n", progname, optopt); usage(progname); return EXIT_FAILURE; case '?': fprintf(stderr, "%s: invalid option -- '%c'\n", progname, optopt); usage(progname); return EXIT_FAILURE; } } while (c != -1); print_config(&config); putchar('\n'); struct wb_state state = {0}; if (!init_wayland(&state)) { finish_wayland(&state); return EXIT_FAILURE; } while (wl_display_dispatch(state.display) != -1) { /* err... what goes here? */ } finish_wayland(&state); return EXIT_SUCCESS; }