chroma/src/wayland.c
NotAShelf bc77b887ad
various: log memory events
Signed-off-by: NotAShelf <raf@notashelf.dev>
Change-Id: I6a6a69643b6d00277bb9bcfeb4cd01dc78d7cd3d
2025-10-02 21:52:08 +03:00

442 lines
14 KiB
C

#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include "../include/chroma.h"
// Registry listener
static void registry_global(void *data, struct wl_registry *registry,
uint32_t id, const char *interface,
uint32_t version) {
chroma_state_t *state = (chroma_state_t *)data;
chroma_log("DEBUG", "Registry global: %s (id=%u, version=%u)", interface, id,
version);
chroma_log("TRACE", "Checking interface binding for: %s", interface);
if (strcmp(interface, wl_compositor_interface.name) == 0) {
state->compositor = wl_registry_bind(registry, id, &wl_compositor_interface,
version < 4 ? version : 4);
if (!state->compositor) {
chroma_log("ERROR", "Failed to bind compositor");
} else {
chroma_log("INFO", "Bound compositor (version %u)", version);
chroma_log("TRACE", "Compositor binding successful, interface ready");
}
} else if (strcmp(interface, zwlr_layer_shell_v1_interface.name) == 0) {
state->layer_shell =
wl_registry_bind(registry, id, &zwlr_layer_shell_v1_interface,
version < 4 ? version : 4);
if (!state->layer_shell) {
chroma_log("ERROR", "Failed to bind layer shell");
} else {
chroma_log("INFO", "Bound layer shell (version %u)", version);
chroma_log(
"TRACE",
"wlr-layer-shell protocol available, wallpaper support enabled");
}
} else if (strcmp(interface, wl_output_interface.name) == 0) {
struct wl_output *output = wl_registry_bind(
registry, id, &wl_output_interface, version < 4 ? version : 4);
if (!output) {
chroma_log("ERROR", "Failed to bind output %u", id);
return;
}
if (chroma_output_add(state, id, output) != CHROMA_OK) {
chroma_log("ERROR", "Failed to add output %u", id);
wl_output_destroy(output);
} else {
chroma_log("INFO", "Added output %u", id);
}
}
}
static void registry_global_remove(void *data, struct wl_registry *registry,
uint32_t id) {
chroma_state_t *state = (chroma_state_t *)data;
(void)registry;
chroma_log("DEBUG", "Registry global remove: id=%u", id);
chroma_log("TRACE", "Global object removal initiated for id=%u", id);
chroma_output_remove(state, id);
}
const struct wl_registry_listener chroma_registry_listener_impl = {
.global = registry_global,
.global_remove = registry_global_remove,
};
// Layer surface event handlers
static void layer_surface_configure(void *data,
struct zwlr_layer_surface_v1 *layer_surface,
uint32_t serial, uint32_t width,
uint32_t height) {
chroma_output_t *output = (chroma_output_t *)data;
(void)layer_surface;
chroma_log("DEBUG", "Layer surface configure: %ux%u, serial=%u", width,
height, serial);
chroma_log(
"TRACE",
"Configuring layer surface for output %u: dimensions=%ux%u, serial=%u",
output->id, width, height, serial);
output->configure_serial = serial;
// Acknowledge the configure event
zwlr_layer_surface_v1_ack_configure(output->layer_surface, serial);
chroma_log("TRACE", "Sent configure acknowledgment for output %u serial %u",
output->id, serial);
// Commit the surface to apply the acknowledgment
wl_surface_commit(output->surface);
chroma_log("TRACE", "Surface committed for output %u", output->id);
chroma_log("DEBUG", "Acknowledged layer surface configure for output %u",
output->id);
}
static void layer_surface_closed(void *data,
struct zwlr_layer_surface_v1 *layer_surface) {
chroma_output_t *output = (chroma_output_t *)data;
(void)layer_surface;
chroma_log("INFO", "Layer surface closed for output %u", output->id);
// Clean up the surface
if (output->surface) {
chroma_surface_destroy(output);
}
}
const struct zwlr_layer_surface_v1_listener chroma_layer_surface_listener_impl =
{
.configure = layer_surface_configure,
.closed = layer_surface_closed,
};
// Output event handlers
static void output_geometry(void *data, struct wl_output *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) {
chroma_output_t *chroma_output = (chroma_output_t *)data;
(void)output;
(void)subpixel;
(void)make;
(void)model;
chroma_output->x = x;
chroma_output->y = y;
chroma_output->transform = transform;
chroma_log("DEBUG", "Output %u geometry: %dx%d at (%d,%d), transform=%d",
chroma_output->id, physical_width, physical_height, x, y,
transform);
chroma_log("TRACE",
"Output %u geometry details: physical_size=%dx%dmm, "
"position=(%d,%d), transform=%d",
chroma_output->id, physical_width, physical_height, x, y,
transform);
}
static void output_mode(void *data, struct wl_output *output, uint32_t flags,
int32_t width, int32_t height, int32_t refresh) {
chroma_output_t *chroma_output = (chroma_output_t *)data;
(void)output;
if (flags & WL_OUTPUT_MODE_CURRENT) {
chroma_output->width = width;
chroma_output->height = height;
chroma_log("DEBUG", "Output %u mode: %dx%d@%d (current)", chroma_output->id,
width, height, refresh);
chroma_log("TRACE",
"Current mode set for output %u: %dx%d@%dHz, flags=0x%x",
chroma_output->id, width, height, refresh, flags);
} else {
chroma_log("TRACE",
"Non-current mode for output %u: %dx%d@%dHz, flags=0x%x",
chroma_output->id, width, height, refresh, flags);
}
}
static void output_scale(void *data, struct wl_output *output, int32_t scale) {
chroma_output_t *chroma_output = (chroma_output_t *)data;
(void)output;
chroma_output->scale = scale;
chroma_log("DEBUG", "Output %u scale: %d", chroma_output->id, scale);
}
static void output_name(void *data, struct wl_output *output,
const char *name) {
chroma_output_t *chroma_output = (chroma_output_t *)data;
(void)output;
free(chroma_output->name);
chroma_output->name = strdup(name);
if (!chroma_output->name) {
chroma_log("ERROR", "Failed to allocate memory for output name");
return;
}
chroma_log("DEBUG", "Output %u name: %s", chroma_output->id, name);
}
static void output_description(void *data, struct wl_output *output,
const char *description) {
chroma_output_t *chroma_output = (chroma_output_t *)data;
(void)output;
free(chroma_output->description);
chroma_output->description = strdup(description);
if (!chroma_output->description) {
chroma_log("ERROR", "Failed to allocate memory for output description");
return;
}
chroma_log("DEBUG", "Output %u description: %s", chroma_output->id,
description);
}
static void output_done(void *data, struct wl_output *output) {
chroma_output_t *chroma_output = (chroma_output_t *)data;
(void)output;
chroma_log("DEBUG", "Output %u done - configuration complete",
chroma_output->id);
chroma_log("TRACE",
"Output %u configuration finalized: %dx%d, scale=%d, name='%s'",
chroma_output->id, chroma_output->width, chroma_output->height,
chroma_output->scale,
chroma_output->name ? chroma_output->name : "unknown");
// Mark output as active and ready for wallpaper assignment
chroma_output->active = true;
// Trigger wallpaper assignment for this output
if (chroma_output->state) {
chroma_log("TRACE", "Triggering wallpaper assignment for output %u",
chroma_output->id);
handle_output_done(chroma_output->state, chroma_output);
}
}
const struct wl_output_listener chroma_output_listener_impl = {
.geometry = output_geometry,
.mode = output_mode,
.scale = output_scale,
.name = output_name,
.description = output_description,
.done = output_done,
};
// Wayland connection functions
int chroma_wayland_connect(chroma_state_t *state) {
if (!state) {
return CHROMA_ERROR_INIT;
}
// Connect to Wayland display
state->display = wl_display_connect(NULL);
if (!state->display) {
chroma_log("ERROR", "Failed to connect to Wayland display: %s",
strerror(errno));
return CHROMA_ERROR_WAYLAND;
}
// Get registry
state->registry = wl_display_get_registry(state->display);
if (!state->registry) {
chroma_log("ERROR", "Failed to get Wayland registry");
chroma_wayland_disconnect(state);
return CHROMA_ERROR_WAYLAND;
}
// Add registry listener
wl_registry_add_listener(state->registry, &chroma_registry_listener_impl,
state);
// Roundtrip to get all globals
chroma_log("TRACE", "Starting Wayland display roundtrip to discover globals");
if (wl_display_roundtrip(state->display) == -1) {
chroma_log("ERROR", "Failed to roundtrip Wayland display");
chroma_wayland_disconnect(state);
return CHROMA_ERROR_WAYLAND;
}
chroma_log("TRACE", "Wayland display roundtrip completed");
// Check if we got a compositor
if (!state->compositor) {
chroma_log("ERROR", "No compositor available");
chroma_wayland_disconnect(state);
return CHROMA_ERROR_WAYLAND;
}
// Check if we got layer shell
if (!state->layer_shell) {
chroma_log("ERROR", "No layer shell available - compositor may not support "
"wlr-layer-shell");
chroma_wayland_disconnect(state);
return CHROMA_ERROR_WAYLAND;
}
chroma_log("INFO", "Connected to Wayland display, found %d outputs",
state->output_count);
return CHROMA_OK;
}
void chroma_wayland_disconnect(chroma_state_t *state) {
if (!state) {
return;
}
// Clean up all outputs
for (int i = 0; i < state->output_count; i++) {
chroma_output_t *output = &state->outputs[i];
if (output->surface) {
chroma_surface_destroy(output);
}
if (output->wl_output) {
wl_output_destroy(output->wl_output);
}
free(output->name);
free(output->description);
}
state->output_count = 0;
// Clean up Wayland objects
if (state->layer_shell) {
zwlr_layer_shell_v1_destroy(state->layer_shell);
state->layer_shell = NULL;
}
if (state->compositor) {
wl_compositor_destroy(state->compositor);
state->compositor = NULL;
}
if (state->registry) {
wl_registry_destroy(state->registry);
state->registry = NULL;
}
if (state->display) {
wl_display_disconnect(state->display);
state->display = NULL;
}
chroma_log("INFO", "Disconnected from Wayland display");
}
// Output management functions
int chroma_output_add(chroma_state_t *state, uint32_t id,
struct wl_output *output) {
if (!state || !output) {
return CHROMA_ERROR_INIT;
}
if (state->output_count >= MAX_OUTPUTS) {
chroma_log("ERROR", "Maximum number of outputs (%d) exceeded", MAX_OUTPUTS);
return CHROMA_ERROR_MEMORY;
}
chroma_output_t *chroma_output = &state->outputs[state->output_count];
memset(chroma_output, 0, sizeof(chroma_output_t));
chroma_output->wl_output = output;
chroma_output->id = id;
chroma_output->scale = 1; // default scale
chroma_output->active = false;
chroma_output->state = state;
// Add output listener
wl_output_add_listener(output, &chroma_output_listener_impl, chroma_output);
chroma_log("TRACE", "Output listener attached for output %u", id);
state->output_count++;
chroma_log("INFO", "Added output %u (total: %d)", id, state->output_count);
chroma_log("TRACE",
"Output %u initialized with default scale=1, active=false", id);
return CHROMA_OK;
}
void chroma_output_remove(chroma_state_t *state, uint32_t id) {
if (!state) {
return;
}
chroma_output_t *output = chroma_output_find_by_id(state, id);
if (!output) {
chroma_log("WARN", "Attempted to remove non-existent output %u", id);
return;
}
chroma_log("INFO", "Removing output %u (%s)", id,
output->name ? output->name : "unknown");
// Clean up surface if it exists
if (output->surface) {
chroma_surface_destroy(output);
}
// Clean up Wayland output
if (output->wl_output) {
wl_output_destroy(output->wl_output);
}
// Free allocated strings
free(output->name);
free(output->description);
// Remove from array by shifting remaining elements
int index = output - state->outputs;
int remaining = state->output_count - index - 1;
chroma_log("TRACE", "Removing output %u from array: index=%d, remaining=%d",
id, index, remaining);
if (remaining > 0) {
memmove(output, output + 1, remaining * sizeof(chroma_output_t));
chroma_log("TRACE", "Shifted %d outputs in array after removal", remaining);
}
state->output_count--;
chroma_log("INFO", "Removed output %u (remaining: %d)", id,
state->output_count);
}
chroma_output_t *chroma_output_find_by_id(chroma_state_t *state, uint32_t id) {
if (!state) {
return NULL;
}
for (int i = 0; i < state->output_count; i++) {
if (state->outputs[i].id == id) {
return &state->outputs[i];
}
}
return NULL;
}
chroma_output_t *chroma_output_find_by_name(chroma_state_t *state,
const char *name) {
if (!state || !name) {
return NULL;
}
for (int i = 0; i < state->output_count; i++) {
chroma_output_t *output = &state->outputs[i];
if (output->name && strcmp(output->name, name) == 0) {
return output;
}
}
return NULL;
}