diff options
Diffstat (limited to 'render/swapchain.c')
| -rw-r--r-- | render/swapchain.c | 108 | 
1 files changed, 108 insertions, 0 deletions
| diff --git a/render/swapchain.c b/render/swapchain.c new file mode 100644 index 00000000..4145efd2 --- /dev/null +++ b/render/swapchain.c @@ -0,0 +1,108 @@ +#include <assert.h> +#include <stdlib.h> +#include <wlr/util/log.h> +#include <wlr/types/wlr_buffer.h> +#include "render/allocator.h" +#include "render/drm_format_set.h" +#include "render/swapchain.h" + +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; +} + +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; + +	swapchain->format = wlr_drm_format_dup(format); +	if (swapchain->format == NULL) { +		free(swapchain); +		return NULL; +	} + +	swapchain->allocator_destroy.notify = swapchain_handle_allocator_destroy; +	wl_signal_add(&alloc->events.destroy, &swapchain->allocator_destroy); + +	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); +	memset(slot, 0, sizeof(*slot)); +} + +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); +	free(swapchain->format); +	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_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(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; +	} + +	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"); +		return NULL; +	} +	return slot_acquire(free_slot); +} | 
