2019-03-18 05:52:32 +00:00
|
|
|
// Copyright 2018-2019, Collabora, Ltd.
|
|
|
|
// 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 "xrt/xrt_gfx_xlib.h"
|
2019-03-21 20:19:52 +00:00
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
#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"
|
2019-04-05 19:18:03 +00:00
|
|
|
#include "oxr_handle.h"
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
static XrResult
|
|
|
|
oxr_swapchain_acquire_image(struct oxr_logger *log,
|
|
|
|
struct oxr_swapchain *sc,
|
|
|
|
const XrSwapchainImageAcquireInfo *acquireInfo,
|
|
|
|
uint32_t *out_index)
|
|
|
|
{
|
|
|
|
uint32_t index;
|
|
|
|
if (sc->acquired_index >= 0) {
|
|
|
|
return oxr_error(log, XR_ERROR_CALL_ORDER_INVALID,
|
|
|
|
" image already acquired");
|
|
|
|
}
|
|
|
|
|
|
|
|
struct xrt_swapchain *xsc = (struct xrt_swapchain *)sc->swapchain;
|
|
|
|
if (!xsc->acquire_image(xsc, &index)) {
|
|
|
|
return oxr_error(log, XR_ERROR_RUNTIME_FAILURE,
|
|
|
|
" call to xsc->acquire_image failed");
|
|
|
|
}
|
|
|
|
|
|
|
|
sc->acquired_index = (int)index;
|
2019-06-28 15:03:31 +00:00
|
|
|
|
|
|
|
// If the compositor is resuing the image,
|
|
|
|
// mark it as invalid to use in xrEndFrame.
|
|
|
|
if (sc->released_index == (int)index) {
|
|
|
|
sc->released_index = -1;
|
|
|
|
}
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
*out_index = index;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static XrResult
|
|
|
|
oxr_swapchain_wait_image(struct oxr_logger *log,
|
|
|
|
struct oxr_swapchain *sc,
|
|
|
|
const XrSwapchainImageWaitInfo *waitInfo)
|
|
|
|
{
|
|
|
|
if (sc->acquired_index < 0) {
|
|
|
|
return oxr_error(log, XR_ERROR_CALL_ORDER_INVALID,
|
|
|
|
" no image acquired");
|
|
|
|
}
|
|
|
|
|
|
|
|
struct xrt_swapchain *xsc = (struct xrt_swapchain *)sc->swapchain;
|
|
|
|
if (!xsc->wait_image(xsc, waitInfo->timeout,
|
|
|
|
(uint32_t)sc->acquired_index)) {
|
|
|
|
return oxr_error(log, XR_ERROR_RUNTIME_FAILURE,
|
|
|
|
" call to xsc->wait_image failed");
|
|
|
|
}
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static XrResult
|
|
|
|
oxr_swapchain_release_image(struct oxr_logger *log,
|
|
|
|
struct oxr_swapchain *sc,
|
|
|
|
const XrSwapchainImageReleaseInfo *releaseInfo)
|
|
|
|
{
|
|
|
|
if (sc->acquired_index < 0) {
|
|
|
|
return oxr_error(log, XR_ERROR_CALL_ORDER_INVALID,
|
|
|
|
" no image acquired");
|
|
|
|
}
|
|
|
|
|
|
|
|
struct xrt_swapchain *xsc = (struct xrt_swapchain *)sc->swapchain;
|
|
|
|
if (!xsc->release_image(xsc, (uint32_t)sc->acquired_index)) {
|
|
|
|
return oxr_error(log, XR_ERROR_RUNTIME_FAILURE,
|
|
|
|
" call to xsc->release_image failed");
|
|
|
|
}
|
2019-06-28 15:03:31 +00:00
|
|
|
|
|
|
|
// Overwrite the old released image, with new.
|
2019-04-30 15:33:56 +00:00
|
|
|
sc->released_index = sc->acquired_index;
|
2019-03-18 05:52:32 +00:00
|
|
|
sc->acquired_index = -1;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2019-04-05 19:18:03 +00:00
|
|
|
static XrResult
|
|
|
|
oxr_swapchain_destroy(struct oxr_logger *log, struct oxr_handle_base *hb)
|
|
|
|
{
|
|
|
|
struct oxr_swapchain *sc = (struct oxr_swapchain *)hb;
|
|
|
|
|
|
|
|
return sc->destroy(log, sc);
|
|
|
|
}
|
|
|
|
|
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)
|
|
|
|
{
|
|
|
|
struct xrt_swapchain *xsc = sess->compositor->create_swapchain(
|
|
|
|
sess->compositor,
|
|
|
|
(enum xrt_swapchain_create_flags)createInfo->createFlags,
|
|
|
|
(enum xrt_swapchain_usage_bits)createInfo->usageFlags,
|
|
|
|
createInfo->format, createInfo->sampleCount, createInfo->width,
|
|
|
|
createInfo->height, createInfo->faceCount, createInfo->arraySize,
|
|
|
|
createInfo->mipCount);
|
|
|
|
|
|
|
|
if (xsc == NULL) {
|
|
|
|
return oxr_error(log, XR_ERROR_RUNTIME_FAILURE,
|
|
|
|
" failed to create swapchain");
|
|
|
|
}
|
|
|
|
|
2019-04-05 19:18:03 +00:00
|
|
|
struct oxr_swapchain *sc = NULL;
|
|
|
|
OXR_ALLOCATE_HANDLE_OR_RETURN(log, sc, OXR_XR_DEBUG_SWAPCHAIN,
|
|
|
|
oxr_swapchain_destroy, &sess->handle);
|
2019-03-18 05:52:32 +00:00
|
|
|
sc->sess = sess;
|
|
|
|
sc->swapchain = xsc;
|
|
|
|
sc->acquire_image = oxr_swapchain_acquire_image;
|
|
|
|
sc->wait_image = oxr_swapchain_wait_image;
|
|
|
|
sc->release_image = oxr_swapchain_release_image;
|
|
|
|
sc->acquired_index = -1;
|
2019-04-30 15:33:56 +00:00
|
|
|
sc->released_index = -1;
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
*out_swapchain = sc;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|