diff options
author | Daniel Eklöf <daniel@ekloef.se> | 2020-08-01 15:32:48 +0200 |
---|---|---|
committer | Daniel Eklöf <daniel@ekloef.se> | 2020-08-01 15:32:48 +0200 |
commit | b947448795d604f169246b0c3191028ae6a0ab83 (patch) | |
tree | 6d99443c8f46d6e2730344efcca6b9d972fdce00 /main.c | |
download | wbg-b947448795d604f169246b0c3191028ae6a0ab83.tar.gz |
Initial commit
Can display a single PNG image scaled-to-fit on all outputs.
Diffstat (limited to 'main.c')
-rw-r--r-- | main.c | 421 |
1 files changed, 421 insertions, 0 deletions
@@ -0,0 +1,421 @@ +#include <stdlib.h> +#include <stdbool.h> +#include <string.h> +#include <poll.h> +#include <errno.h> +#include <signal.h> +#include <unistd.h> +#include <locale.h> +#include <assert.h> + +#include <sys/signalfd.h> + +#include <wayland-client.h> +#include <wayland-cursor.h> + +#include <wlr-layer-shell-unstable-v1.h> +#include <pixman.h> +#include <tllist.h> + +#define LOG_MODULE "wbg" +#define LOG_ENABLE_DBG 1 +#include "log.h" +#include "png-wbg.h" +#include "shm.h" + +/* Top-level globals */ +static struct wl_display *display; +static struct wl_registry *registry; +static struct wl_compositor *compositor; +static struct wl_shm *shm; +static struct zwlr_layer_shell_v1 *layer_shell; + +static bool have_argb8888 = false; + +/* TODO: one per output */ +static pixman_image_t *image; + +struct output { + struct wl_output *wl_output; + uint32_t wl_name; + + char *make; + char *model; + + int scale; + int width; + int height; + + int render_width; + int render_height; + + struct wl_surface *surf; + struct zwlr_layer_surface_v1 *layer; +}; +static tll(struct output) outputs; + +static void +render(struct output *output) +{ + const int width = output->render_width; + const int height = output->render_height; + const int scale = output->scale; + + struct buffer *buf = shm_get_buffer( + shm, width * scale, height * scale, (uintptr_t)(void *)output); + assert(buf != NULL); + + LOG_INFO("render: w=%d, h=%d", width * scale, height * scale); + + pixman_color_t color = {.red = 0xffff, .green = 0, .blue = 0, .alpha = 0xffff}; + pixman_image_fill_rectangles( + PIXMAN_OP_SRC, buf->pix, &color, 1, + &(pixman_rectangle16_t){0, 0, width * scale, height * scale}); + + uint32_t *data = pixman_image_get_data(image); + int img_width = pixman_image_get_width(image); + int img_height = pixman_image_get_height(image); + int img_stride = pixman_image_get_stride(image); + pixman_format_code_t img_fmt = pixman_image_get_format(image); + + double sx = (double)img_width / width; + double sy = (double)img_height / height; + + pixman_image_t *pix = pixman_image_create_bits_no_clear( + img_fmt, img_width, img_height, data, img_stride); + pixman_f_transform_t t; + pixman_transform_t t2; + pixman_f_transform_init_scale(&t, sx, sy); + pixman_transform_from_pixman_f_transform(&t2, &t); + pixman_image_set_transform(pix, &t2); + + pixman_image_composite32( + PIXMAN_OP_SRC, + pix, NULL, buf->pix, 0, 0, 0, 0, 0, 0, + width, height); + + pixman_image_unref(pix); + + wl_surface_set_buffer_scale(output->surf, scale); + wl_surface_attach(output->surf, buf->wl_buf, 0, 0); + wl_surface_damage_buffer(output->surf, 0, 0, width * scale, height * scale); + wl_surface_commit(output->surf); +} + +static void +layer_surface_configure(void *data, struct zwlr_layer_surface_v1 *surface, + uint32_t serial, uint32_t w, uint32_t h) +{ + struct output *output = data; + output->render_width = w; + output->render_height = h; + zwlr_layer_surface_v1_ack_configure(surface, serial); + render(output); +} + +static void +layer_surface_closed(void *data, struct zwlr_layer_surface_v1 *surface) +{ +} + +static const struct zwlr_layer_surface_v1_listener layer_surface_listener = { + .configure = &layer_surface_configure, + .closed = &layer_surface_closed, +}; + +static void +output_destroy(struct output *output) +{ + free(output->make); + free(output->model); + if (output->layer != NULL) + zwlr_layer_surface_v1_destroy(output->layer); + if (output->surf != NULL) + wl_surface_destroy(output->surf); + if (output->wl_output != NULL) + wl_output_destroy(output->wl_output); +} + +static void +output_geometry(void *data, struct wl_output *wl_output, int32_t x, int32_t y, + int32_t physical_width, int32_t physical_height, + int32_t subpixel, const char *make, const char *model, + int32_t transform) +{ + struct output *output = data; + + free(output->make); + free(output->model); + + output->make = make != NULL ? strdup(make) : NULL; + output->model = model != NULL ? strdup(model) : NULL; +} + +static void +output_mode(void *data, struct wl_output *wl_output, uint32_t flags, + int32_t width, int32_t height, int32_t refresh) +{ + if ((flags & WL_OUTPUT_MODE_CURRENT) == 0) + return; + + struct output *output = data; + output->width = width; + output->height = height; +} + +static void +output_done(void *data, struct wl_output *wl_output) +{ + struct output *output = data; + const int width = output->width; + const int height = output->height; + const int scale = output->scale; + + LOG_INFO("output: %s %s (%dx%d, scale=%d)", + output->make, output->model, width, height, scale); + + assert(output->surf != NULL); + assert(output->layer != NULL); +} + +static void +output_scale(void *data, struct wl_output *wl_output, int32_t factor) +{ + struct 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 +shm_format(void *data, struct wl_shm *wl_shm, uint32_t format) +{ + if (format == WL_SHM_FORMAT_ARGB8888) + have_argb8888 = true; +} + +static const struct wl_shm_listener shm_listener = { + .format = &shm_format, +}; + +static bool +verify_iface_version(const char *iface, uint32_t version, uint32_t wanted) +{ + if (version >= wanted) + return true; + + LOG_ERR("%s: need interface version %u, but compositor only implements %u", + iface, wanted, version); + return false; +} + +static void +handle_global(void *data, struct wl_registry *registry, + uint32_t name, const char *interface, uint32_t version) +{ + if (strcmp(interface, wl_compositor_interface.name) == 0) { + const uint32_t required = 4; + if (!verify_iface_version(interface, version, required)) + return; + + compositor = wl_registry_bind( + registry, name, &wl_compositor_interface, required); + } + + else if (strcmp(interface, wl_shm_interface.name) == 0) { + const uint32_t required = 1; + if (!verify_iface_version(interface, version, required)) + return; + + shm = wl_registry_bind( + registry, name, &wl_shm_interface, required); + wl_shm_add_listener(shm, &shm_listener, NULL); + } + + else if (strcmp(interface, wl_output_interface.name) == 0) { + const uint32_t required = 3; + if (!verify_iface_version(interface, version, required)) + return; + + struct wl_output *wl_output = wl_registry_bind( + registry, name, &wl_output_interface, required); + + struct wl_surface *surf = wl_compositor_create_surface(compositor); + + /* Default input region is 'infinite', while we want it to be empty */ + struct wl_region *empty_region =wl_compositor_create_region(compositor); + wl_surface_set_input_region(surf, empty_region); + wl_region_destroy(empty_region); + + struct zwlr_layer_surface_v1 *layer = zwlr_layer_shell_v1_get_layer_surface( + layer_shell, surf, wl_output, ZWLR_LAYER_SHELL_V1_LAYER_BACKGROUND, "wallpaper"); + + zwlr_layer_surface_v1_set_exclusive_zone(layer, -1); + zwlr_layer_surface_v1_set_anchor(layer, + ZWLR_LAYER_SURFACE_V1_ANCHOR_TOP | + ZWLR_LAYER_SURFACE_V1_ANCHOR_RIGHT | + ZWLR_LAYER_SURFACE_V1_ANCHOR_BOTTOM | + ZWLR_LAYER_SURFACE_V1_ANCHOR_LEFT); + + tll_push_back(outputs, ((struct output){ + .wl_output = wl_output, .wl_name = name, .surf = surf, .layer = layer})); + struct output *output = &tll_back(outputs); + + wl_output_add_listener(wl_output, &output_listener, output); + zwlr_layer_surface_v1_add_listener(layer, &layer_surface_listener, output); + wl_surface_commit(surf); + } + + else if (strcmp(interface, zwlr_layer_shell_v1_interface.name) == 0) { + const uint32_t required = 2; + if (!verify_iface_version(interface, version, required)) + return; + + layer_shell = wl_registry_bind( + registry, name, &zwlr_layer_shell_v1_interface, required); + } + +} + +static void +handle_global_remove(void *data, struct wl_registry *registry, uint32_t name) +{ + tll_foreach(outputs, it) { + if (it->item.wl_name == name) { + LOG_DBG("destroyed: %s %s", it->item.make, it->item.model); + output_destroy(&it->item); + tll_remove(outputs, it); + return; + } + } +} + +static const struct wl_registry_listener registry_listener = { + .global = &handle_global, + .global_remove = &handle_global_remove, +}; + +int +main(int argc, const char *const *argv) +{ + if (argc < 2) { + LOG_ERR("missing required argument: image path"); + return EXIT_FAILURE; + } + + const char *image_path = argv[1]; + image = png_load(image_path); + if (image == NULL) { + LOG_ERR("%s: failed to load", image_path); + return EXIT_FAILURE; + } + + setlocale(LC_CTYPE, ""); + log_init(LOG_COLORIZE_AUTO, false, LOG_FACILITY_DAEMON, LOG_CLASS_WARNING); + + display = wl_display_connect(NULL); + assert(display != NULL); + + registry = wl_display_get_registry(display); + assert(registry != NULL); + + wl_registry_add_listener(registry, ®istry_listener, NULL); + wl_display_roundtrip(display); + + assert(compositor != NULL); + assert(shm != NULL); + assert(layer_shell != NULL); + + wl_display_roundtrip(display); + assert(have_argb8888); + + sigset_t mask; + sigemptyset(&mask); + sigaddset(&mask, SIGINT); + sigaddset(&mask, SIGQUIT); + + sigprocmask(SIG_BLOCK, &mask, NULL); + + int sig_fd = signalfd(-1, &mask, 0); + assert(sig_fd >= 0); + + int exit_code = EXIT_FAILURE; + + while (true) { + wl_display_flush(display); + + struct pollfd fds[] = { + {.fd = wl_display_get_fd(display), .events = POLLIN}, + {.fd = sig_fd, .events = POLLIN}, + }; + int ret = poll(fds, sizeof(fds) / sizeof(fds[0]), -1); + + if (ret < 0) { + if (ret == EINTR) + continue; + + LOG_ERRNO("failed to poll"); + break; + } + + if (fds[0].revents & POLLHUP) { + LOG_WARN("disconnected by compositor"); + break; + } + + if (fds[0].revents & POLLIN) + wl_display_dispatch(display); + + if (fds[1].revents & POLLHUP) + abort(); + + if (fds[1].revents & POLLIN) { + struct signalfd_siginfo info; + ssize_t count = read(sig_fd, &info, sizeof(info)); + if (count < 0) { + if (errno == EINTR) + continue; + + LOG_ERRNO("failed to read from signal FD"); + break; + } + + assert(count == sizeof(info)); + assert(info.ssi_signo == SIGINT || info.ssi_signo == SIGQUIT); + + LOG_INFO("goodbye"); + exit_code = EXIT_SUCCESS; + break; + } + } + + if (sig_fd >= 0) + close(sig_fd); + + tll_foreach(outputs, it) + output_destroy(&it->item); + tll_free(outputs); + + shm_fini(); + if (layer_shell != NULL) + zwlr_layer_shell_v1_destroy(layer_shell); + if (shm != NULL) + wl_shm_destroy(shm); + if (compositor != NULL) + wl_compositor_destroy(compositor); + if (registry != NULL) + wl_registry_destroy(registry); + if (display != NULL) + wl_display_disconnect(display); + if (image != NULL) { + free(pixman_image_get_data(image)); + pixman_image_unref(image); + } + log_deinit(); + return exit_code; +} |