monado/src/xrt/state_trackers/oxr/oxr_swapchain_gl.c
2021-03-24 17:50:08 +00:00

125 lines
3.7 KiB
C

// Copyright 2019-2021, Collabora, Ltd.
// SPDX-License-Identifier: BSL-1.0
/*!
* @file
* @brief Holds OpenGL swapchain related functions.
* @author Jakob Bornecrantz <jakob@collabora.com>
* @ingroup oxr_main
* @ingroup comp_client
*/
#include <assert.h>
#include <stdlib.h>
#include "xrt/xrt_gfx_xlib.h"
#include "util/u_debug.h"
#include "oxr_objects.h"
#include "oxr_logger.h"
#if defined(XR_USE_GRAPHICS_API_OPENGL) || defined(XR_USE_GRAPHICS_API_OPENGL_ES)
static XrResult
oxr_swapchain_gl_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;
}
#if defined(XR_USE_GRAPHICS_API_OPENGL)
static XrResult
oxr_swapchain_gl_enumerate_images_gl(struct oxr_logger *log,
struct oxr_swapchain *sc,
uint32_t count,
XrSwapchainImageOpenGLKHR *images)
{
struct xrt_swapchain_gl *xsc = (struct xrt_swapchain_gl *)sc->swapchain;
for (uint32_t i = 0; i < count; i++) {
if (images[i].type != images[0].type) {
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE, "Images array contains mixed types");
}
images[i].image = xsc->images[i];
}
return oxr_session_success_result(sc->sess);
}
#endif // XR_USE_GRAPHICS_API_OPENGL
#if defined(XR_USE_GRAPHICS_API_OPENGL_ES)
static XrResult
oxr_swapchain_gl_enumerate_images_gles(struct oxr_logger *log,
struct oxr_swapchain *sc,
uint32_t count,
XrSwapchainImageOpenGLESKHR *images)
{
struct xrt_swapchain_gl *xsc = (struct xrt_swapchain_gl *)sc->swapchain;
for (uint32_t i = 0; i < count; i++) {
if (images[i].type != images[0].type) {
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE, "Images array contains mixed types");
}
images[i].image = xsc->images[i];
}
return oxr_session_success_result(sc->sess);
}
#endif // XR_USE_GRAPHICS_API_OPENGL_ES
static XrResult
oxr_swapchain_gl_enumerate_images(struct oxr_logger *log,
struct oxr_swapchain *sc,
uint32_t count,
XrSwapchainImageBaseHeader *images)
{
assert(count > 0);
switch (images[0].type) {
#if defined(XR_USE_GRAPHICS_API_OPENGL)
case XR_TYPE_SWAPCHAIN_IMAGE_OPENGL_KHR:
return oxr_swapchain_gl_enumerate_images_gl(log, sc, count, (XrSwapchainImageOpenGLKHR *)images);
#endif // XR_USE_GRAPHICS_API_OPENGL
#if defined(XR_USE_GRAPHICS_API_OPENGL_ES)
case XR_TYPE_SWAPCHAIN_IMAGE_OPENGL_ES_KHR:
return oxr_swapchain_gl_enumerate_images_gles(log, sc, count, (XrSwapchainImageOpenGLESKHR *)images);
#endif // XR_USE_GRAPHICS_API_OPENGL_ES
default: return oxr_error(log, XR_ERROR_VALIDATION_FAILURE, "Unsupported XrSwapchainImageBaseHeader type");
}
}
XrResult
oxr_swapchain_gl_create(struct oxr_logger *log,
struct oxr_session *sess,
const XrSwapchainCreateInfo *createInfo,
struct oxr_swapchain **out_swapchain)
{
struct oxr_swapchain *sc;
XrResult ret;
ret = oxr_create_swapchain(log, sess, createInfo, &sc);
if (ret != XR_SUCCESS) {
return ret;
}
sc->destroy = oxr_swapchain_gl_destroy;
sc->enumerate_images = oxr_swapchain_gl_enumerate_images;
*out_swapchain = sc;
return XR_SUCCESS;
}
#endif // XR_USE_GRAPHICS_API_OPENGL || XR_USE_GRAPHICS_API_OPENGL_ES