monado/src/xrt/state_trackers/oxr/oxr_swapchain.c

274 lines
7.4 KiB
C
Raw Normal View History

// Copyright 2018-2023, Collabora, Ltd.
2019-03-18 05:52:32 +00:00
// SPDX-License-Identifier: BSL-1.0
/*!
* @file
* @brief Holds swapchain related functions.
* @author Jakob Bornecrantz <jakob@collabora.com>
* @ingroup oxr_main
*/
#include <stdlib.h>
#include "util/u_debug.h"
2019-03-21 20:19:52 +00:00
#include "util/u_misc.h"
2019-03-18 05:52:32 +00:00
#include "oxr_objects.h"
#include "oxr_logger.h"
#include "oxr_handle.h"
#include "oxr_xret.h"
/*
*
* Conversion functions.
*
*/
static enum xrt_swapchain_create_flags
convert_create_flags(XrSwapchainCreateFlags xr_flags)
{
enum xrt_swapchain_create_flags flags = 0;
2019-03-18 05:52:32 +00:00
if ((xr_flags & XR_SWAPCHAIN_CREATE_PROTECTED_CONTENT_BIT) != 0) {
flags |= XRT_SWAPCHAIN_CREATE_PROTECTED_CONTENT;
}
if ((xr_flags & XR_SWAPCHAIN_CREATE_STATIC_IMAGE_BIT) != 0) {
flags |= XRT_SWAPCHAIN_CREATE_STATIC_IMAGE;
}
return flags;
}
static enum xrt_swapchain_usage_bits
convert_usage_bits(XrSwapchainUsageFlags xr_usage)
{
enum xrt_swapchain_usage_bits usage = 0;
if ((xr_usage & XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT) != 0) {
usage |= XRT_SWAPCHAIN_USAGE_COLOR;
}
if ((xr_usage & XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT) != 0) {
usage |= XRT_SWAPCHAIN_USAGE_DEPTH_STENCIL;
}
if ((xr_usage & XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT) != 0) {
usage |= XRT_SWAPCHAIN_USAGE_UNORDERED_ACCESS;
}
if ((xr_usage & XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT) != 0) {
usage |= XRT_SWAPCHAIN_USAGE_TRANSFER_SRC;
}
if ((xr_usage & XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT) != 0) {
usage |= XRT_SWAPCHAIN_USAGE_TRANSFER_DST;
}
if ((xr_usage & XR_SWAPCHAIN_USAGE_SAMPLED_BIT) != 0) {
usage |= XRT_SWAPCHAIN_USAGE_SAMPLED;
}
if ((xr_usage & XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT) != 0) {
usage |= XRT_SWAPCHAIN_USAGE_MUTABLE_FORMAT;
}
// aliased to XR_SWAPCHAIN_USAGE_INPUT_ATTACHMENT_BIT_MND
if ((xr_usage & XR_SWAPCHAIN_USAGE_INPUT_ATTACHMENT_BIT_KHR) != 0) {
usage |= XRT_SWAPCHAIN_USAGE_INPUT_ATTACHMENT;
}
return usage;
}
/*
*
* Internal API functions.
*
*/
2019-03-18 05:52:32 +00:00
static XrResult
acquire_image(struct oxr_logger *log,
struct oxr_swapchain *sc,
const XrSwapchainImageAcquireInfo *acquireInfo,
uint32_t *out_index)
2019-03-18 05:52:32 +00:00
{
uint32_t index;
if (sc->acquired.num >= sc->swapchain->image_count) {
2021-01-14 14:13:48 +00:00
return oxr_error(log, XR_ERROR_CALL_ORDER_INVALID, "All images have been acquired");
2019-03-18 05:52:32 +00:00
}
if (sc->is_static && (sc->released.yes || sc->waited.yes)) {
2021-01-14 14:13:48 +00:00
return oxr_error(log, XR_ERROR_CALL_ORDER_INVALID, "Can only acquire once on a static swapchain");
}
2019-03-18 05:52:32 +00:00
struct xrt_swapchain *xsc = (struct xrt_swapchain *)sc->swapchain;
xrt_result_t xret = xrt_swapchain_acquire_image(xsc, &index);
OXR_CHECK_XRET(log, sc->sess, xret, xrt_swapchain_acquire_image);
2019-03-18 05:52:32 +00:00
if (sc->images[index].state != OXR_IMAGE_STATE_READY) {
2022-05-17 18:19:00 +00:00
return oxr_error(log, XR_ERROR_RUNTIME_FAILURE,
"Internal xrt_swapchain_acquire_image call returned non-ready image.");
}
sc->acquired.num++;
u_index_fifo_push(&sc->acquired.fifo, index);
sc->images[index].state = OXR_IMAGE_STATE_ACQUIRED;
// If the compositor is resuing the image,
// mark it as invalid to use in xrEndFrame.
if (sc->released.index == (int)index) {
sc->released.yes = false;
sc->released.index = -1;
}
2019-03-18 05:52:32 +00:00
*out_index = index;
return oxr_session_success_result(sc->sess);
2019-03-18 05:52:32 +00:00
}
static XrResult
wait_image(struct oxr_logger *log, struct oxr_swapchain *sc, const XrSwapchainImageWaitInfo *waitInfo)
2019-03-18 05:52:32 +00:00
{
if (sc->waited.yes) {
2021-01-14 14:13:48 +00:00
return oxr_error(log, XR_ERROR_CALL_ORDER_INVALID, "Swapchain has already been waited, call release");
}
if (u_index_fifo_is_empty(&sc->acquired.fifo)) {
2021-01-14 14:13:48 +00:00
return oxr_error(log, XR_ERROR_CALL_ORDER_INVALID, "No image acquired");
2019-03-18 05:52:32 +00:00
}
uint32_t index;
u_index_fifo_pop(&sc->acquired.fifo, &index);
2019-03-18 05:52:32 +00:00
struct xrt_swapchain *xsc = (struct xrt_swapchain *)sc->swapchain;
xrt_result_t xret = xrt_swapchain_wait_image(xsc, waitInfo->timeout, index);
OXR_CHECK_XRET(log, sc->sess, xret, xrt_swapchain_wait_image);
2019-03-18 05:52:32 +00:00
// The app can only wait on one image.
sc->waited.yes = true;
sc->waited.index = index;
sc->images[index].state = OXR_IMAGE_STATE_WAITED;
return oxr_session_success_result(sc->sess);
2019-03-18 05:52:32 +00:00
}
static XrResult
release_image(struct oxr_logger *log, struct oxr_swapchain *sc, const XrSwapchainImageReleaseInfo *releaseInfo)
2019-03-18 05:52:32 +00:00
{
if (!sc->waited.yes) {
2021-01-14 14:13:48 +00:00
return oxr_error(log, XR_ERROR_CALL_ORDER_INVALID, "No swapchain images waited on");
2019-03-18 05:52:32 +00:00
}
sc->waited.yes = false;
uint32_t index = sc->waited.index;
2019-03-18 05:52:32 +00:00
struct xrt_swapchain *xsc = (struct xrt_swapchain *)sc->swapchain;
xrt_result_t xret = xrt_swapchain_release_image(xsc, index);
OXR_CHECK_XRET(log, sc->sess, xret, xrt_swapchain_release_image);
// Only decerement here.
sc->acquired.num--;
// Overwrite the old released image, with new.
sc->released.yes = true;
sc->released.index = index;
sc->images[index].state = OXR_IMAGE_STATE_READY;
2019-03-18 05:52:32 +00:00
return oxr_session_success_result(sc->sess);
2019-03-18 05:52:32 +00:00
}
static XrResult
destroy(struct oxr_logger *log, struct oxr_swapchain *sc)
{
// Release any waited image.
if (sc->waited.yes) {
sc->release_image(log, sc, NULL);
}
// Release any acquired images.
XrSwapchainImageWaitInfo waitInfo = {0};
while (!u_index_fifo_is_empty(&sc->acquired.fifo)) {
sc->wait_image(log, sc, &waitInfo);
sc->release_image(log, sc, NULL);
}
// Drop our reference, does NULL checking.
xrt_swapchain_reference(&sc->swapchain, NULL);
return XR_SUCCESS;
}
/*
*
* Handle function.
*
*/
static XrResult
destroy_handle(struct oxr_logger *log, struct oxr_handle_base *hb)
{
struct oxr_swapchain *sc = (struct oxr_swapchain *)hb;
XrResult ret = sc->destroy(log, sc);
free(sc);
return ret;
}
/*
*
* 'Exported' functions.
*
*/
2019-03-18 05:52:32 +00:00
XrResult
oxr_create_swapchain(struct oxr_logger *log,
struct oxr_session *sess,
const XrSwapchainCreateInfo *createInfo,
struct oxr_swapchain **out_swapchain)
{
xrt_result_t xret = XRT_SUCCESS;
struct xrt_swapchain_create_info info;
info.create = convert_create_flags(createInfo->createFlags);
info.bits = convert_usage_bits(createInfo->usageFlags);
info.format = createInfo->format;
info.sample_count = createInfo->sampleCount;
info.width = createInfo->width;
info.height = createInfo->height;
info.face_count = createInfo->faceCount;
info.array_size = createInfo->arraySize;
info.mip_count = createInfo->mipCount;
struct xrt_swapchain *xsc = NULL; // Has to be NULL.
xret = xrt_comp_create_swapchain(sess->compositor, &info, &xsc);
if (xret == XRT_ERROR_SWAPCHAIN_FLAG_VALID_BUT_UNSUPPORTED) {
return oxr_error(log, XR_ERROR_FEATURE_UNSUPPORTED,
"Specified swapchain creation flag is valid, "
"but not supported");
}
if (xret == XRT_ERROR_SWAPCHAIN_FORMAT_UNSUPPORTED) {
return oxr_error(log, XR_ERROR_SWAPCHAIN_FORMAT_UNSUPPORTED,
"Specified swapchain format is not supported");
}
if (xret != XRT_SUCCESS) {
return oxr_error(log, XR_ERROR_RUNTIME_FAILURE, "Failed to create swapchain");
}
assert(xsc != NULL);
2019-03-18 05:52:32 +00:00
struct oxr_swapchain *sc = NULL;
OXR_ALLOCATE_HANDLE_OR_RETURN(log, sc, OXR_XR_DEBUG_SWAPCHAIN, destroy_handle, &sess->handle);
2019-03-18 05:52:32 +00:00
sc->sess = sess;
sc->swapchain = xsc;
sc->width = createInfo->width;
sc->height = createInfo->height;
sc->array_layer_count = createInfo->arraySize;
2022-07-01 19:23:53 +00:00
sc->face_count = createInfo->faceCount;
sc->acquire_image = acquire_image;
sc->wait_image = wait_image;
sc->release_image = release_image;
sc->destroy = destroy;
2021-01-14 14:13:48 +00:00
sc->is_static = (createInfo->createFlags & XR_SWAPCHAIN_CREATE_STATIC_IMAGE_BIT) != 0;
2019-03-18 05:52:32 +00:00
*out_swapchain = sc;
return XR_SUCCESS;
}