#include #include #include #include #include #include "render/allocator/allocator.h" #include "render/drm_format_set.h" struct wlr_swapchain_buffer { struct wlr_addon buffer_addon; struct wlr_swapchain *swapchain; }; static void wlr_swapchain_buffer_destroy(struct wlr_addon *addon) { struct wlr_swapchain_buffer *swapchain_buffer = wl_container_of(addon, swapchain_buffer, buffer_addon); wlr_addon_finish(&swapchain_buffer->buffer_addon); swapchain_buffer->swapchain = NULL; free(swapchain_buffer); } static const struct wlr_addon_interface buffer_addon_impl = { .name = "wlr_swapchain_buffer", .destroy = wlr_swapchain_buffer_destroy, }; struct wlr_swapchain *wlr_swapchain_try_from_wlr_buffer(struct wlr_buffer *buffer) { struct wlr_addon *addon = wlr_addon_find(&buffer->addons, NULL, &buffer_addon_impl); if (addon == NULL) { return NULL; } struct wlr_swapchain_buffer *swapchain_buffer = wl_container_of(addon, swapchain_buffer, buffer_addon); return swapchain_buffer->swapchain; } static void swapchain_handle_allocator_destroy(struct wl_listener *listener, void *data) { struct wlr_swapchain *swapchain = wl_container_of(listener, swapchain, allocator_destroy); swapchain->allocator = NULL; wl_list_remove(&swapchain->allocator_destroy.link); wl_list_init(&swapchain->allocator_destroy.link); } struct wlr_swapchain *wlr_swapchain_create( struct wlr_allocator *alloc, int width, int height, const struct wlr_drm_format *format) { struct wlr_swapchain *swapchain = calloc(1, sizeof(*swapchain)); if (swapchain == NULL) { return NULL; } swapchain->allocator = alloc; swapchain->width = width; swapchain->height = height; if (!wlr_drm_format_copy(&swapchain->format, format)) { free(swapchain); return NULL; } swapchain->allocator_destroy.notify = swapchain_handle_allocator_destroy; wl_signal_add(&alloc->events.destroy, &swapchain->allocator_destroy); wlr_addon_set_init(&swapchain->addons); return swapchain; } static void slot_reset(struct wlr_swapchain_slot *slot) { if (slot->acquired) { wl_list_remove(&slot->release.link); } wlr_buffer_drop(slot->buffer); *slot = (struct wlr_swapchain_slot){0}; } void wlr_swapchain_destroy(struct wlr_swapchain *swapchain) { if (swapchain == NULL) { return; } for (size_t i = 0; i < WLR_SWAPCHAIN_CAP; i++) { slot_reset(&swapchain->slots[i]); } wl_list_remove(&swapchain->allocator_destroy.link); wlr_drm_format_finish(&swapchain->format); wlr_addon_set_finish(&swapchain->addons); free(swapchain); } static void slot_handle_release(struct wl_listener *listener, void *data) { struct wlr_swapchain_slot *slot = wl_container_of(listener, slot, release); wl_list_remove(&slot->release.link); slot->acquired = false; } static struct wlr_buffer *slot_acquire(struct wlr_swapchain *swapchain, struct wlr_swapchain_slot *slot) { assert(!slot->acquired); assert(slot->buffer != NULL); slot->acquired = true; slot->release.notify = slot_handle_release; wl_signal_add(&slot->buffer->events.release, &slot->release); return wlr_buffer_lock(slot->buffer); } struct wlr_buffer *wlr_swapchain_acquire(struct wlr_swapchain *swapchain) { struct wlr_swapchain_slot *free_slot = NULL; for (size_t i = 0; i < WLR_SWAPCHAIN_CAP; i++) { struct wlr_swapchain_slot *slot = &swapchain->slots[i]; if (slot->acquired) { continue; } if (slot->buffer != NULL) { return slot_acquire(swapchain, slot); } free_slot = slot; } if (free_slot == NULL) { wlr_log(WLR_ERROR, "No free output buffer slot"); return NULL; } if (swapchain->allocator == NULL) { return NULL; } struct wlr_swapchain_buffer *swapchain_buffer = calloc(1, sizeof(*swapchain_buffer)); if (swapchain_buffer == NULL) { return NULL; } swapchain_buffer->swapchain = swapchain; wlr_log(WLR_DEBUG, "Allocating new swapchain buffer"); free_slot->buffer = wlr_allocator_create_buffer(swapchain->allocator, swapchain->width, swapchain->height, &swapchain->format); if (free_slot->buffer == NULL) { wlr_log(WLR_ERROR, "Failed to allocate buffer"); free(swapchain_buffer); return NULL; } wlr_addon_init(&swapchain_buffer->buffer_addon, &free_slot->buffer->addons, NULL, &buffer_addon_impl); return slot_acquire(swapchain, free_slot); } bool wlr_swapchain_has_buffer(struct wlr_swapchain *swapchain, struct wlr_buffer *buffer) { for (size_t i = 0; i < WLR_SWAPCHAIN_CAP; i++) { struct wlr_swapchain_slot *slot = &swapchain->slots[i]; if (slot->buffer == buffer) { return true; } } return false; }