2019-03-18 05:52:32 +00:00
|
|
|
// Copyright 2018-2019, Collabora, Ltd.
|
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief Holds session related functions.
|
|
|
|
* @author Jakob Bornecrantz <jakob@collabora.com>
|
|
|
|
* @ingroup oxr_main
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2019-04-05 22:13:58 +00:00
|
|
|
#include <assert.h>
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
#include "util/u_debug.h"
|
2019-03-23 01:11:14 +00:00
|
|
|
#include "util/u_misc.h"
|
2019-03-18 05:52:32 +00:00
|
|
|
#include "math/m_api.h"
|
|
|
|
#include "util/u_time.h"
|
|
|
|
|
|
|
|
#include "xrt/xrt_device.h"
|
|
|
|
#include "xrt/xrt_gfx_xlib.h"
|
|
|
|
#include "xrt/xrt_gfx_vk.h"
|
|
|
|
|
|
|
|
#include "oxr_objects.h"
|
|
|
|
#include "oxr_logger.h"
|
|
|
|
#include "oxr_two_call.h"
|
2019-04-05 19:18:03 +00:00
|
|
|
#include "oxr_handle.h"
|
2019-06-03 19:17:45 +00:00
|
|
|
#include "oxr_chain.h"
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
DEBUG_GET_ONCE_BOOL_OPTION(views, "OXR_DEBUG_VIEWS", false)
|
2019-03-25 22:26:57 +00:00
|
|
|
DEBUG_GET_ONCE_BOOL_OPTION(dynamic_prediction, "OXR_DYNAMIC_PREDICTION", true)
|
2019-03-18 05:52:32 +00:00
|
|
|
DEBUG_GET_ONCE_NUM_OPTION(ipd, "OXR_DEBUG_IPD_MM", 63)
|
|
|
|
DEBUG_GET_ONCE_NUM_OPTION(prediction_ms, "OXR_DEBUG_PREDICTION_MS", 11)
|
|
|
|
|
|
|
|
static bool
|
|
|
|
is_running(XrSessionState state)
|
|
|
|
{
|
|
|
|
switch (state) {
|
2019-07-13 16:17:57 +00:00
|
|
|
case XR_SESSION_STATE_SYNCHRONIZED: return true;
|
|
|
|
case XR_SESSION_STATE_VISIBLE: return true;
|
|
|
|
case XR_SESSION_STATE_FOCUSED: return true;
|
|
|
|
default: return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
should_render(XrSessionState state)
|
|
|
|
{
|
|
|
|
switch (state) {
|
2019-03-18 05:52:32 +00:00
|
|
|
case XR_SESSION_STATE_VISIBLE: return true;
|
|
|
|
case XR_SESSION_STATE_FOCUSED: return true;
|
|
|
|
default: return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_session_enumerate_formats(struct oxr_logger *log,
|
|
|
|
struct oxr_session *sess,
|
|
|
|
uint32_t formatCapacityInput,
|
|
|
|
uint32_t *formatCountOutput,
|
|
|
|
int64_t *formats)
|
|
|
|
{
|
|
|
|
struct xrt_compositor *xc = sess->compositor;
|
2019-06-05 19:14:18 +00:00
|
|
|
if (formatCountOutput == NULL) {
|
|
|
|
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
|
|
|
|
"(formatCountOutput)");
|
|
|
|
}
|
2019-03-23 01:11:14 +00:00
|
|
|
if (xc == NULL) {
|
2019-06-05 19:14:18 +00:00
|
|
|
if (formatCountOutput != NULL) {
|
|
|
|
*formatCountOutput = 0;
|
|
|
|
}
|
2019-03-23 01:11:14 +00:00
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
OXR_TWO_CALL_HELPER(log, formatCapacityInput, formatCountOutput,
|
|
|
|
formats, xc->num_formats, xc->formats);
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_session_begin(struct oxr_logger *log,
|
|
|
|
struct oxr_session *sess,
|
|
|
|
const XrSessionBeginInfo *beginInfo)
|
|
|
|
{
|
|
|
|
if (is_running(sess->state)) {
|
|
|
|
return oxr_error(log, XR_ERROR_SESSION_RUNNING,
|
|
|
|
" session is already running");
|
|
|
|
}
|
|
|
|
struct xrt_compositor *xc = sess->compositor;
|
2019-03-23 01:11:14 +00:00
|
|
|
if (xc != NULL) {
|
2019-08-14 22:15:15 +00:00
|
|
|
XrViewConfigurationType view_type =
|
|
|
|
beginInfo->primaryViewConfigurationType;
|
|
|
|
|
|
|
|
if (view_type != sess->sys->view_config_type) {
|
|
|
|
/*! @todo we only support a single view config type per
|
|
|
|
* system right now */
|
|
|
|
return oxr_error(
|
|
|
|
log, XR_ERROR_VIEW_CONFIGURATION_TYPE_UNSUPPORTED,
|
|
|
|
" view configuration type not supported");
|
|
|
|
}
|
|
|
|
|
2019-03-23 01:11:14 +00:00
|
|
|
xc->begin_session(xc, (enum xrt_view_type)beginInfo
|
|
|
|
->primaryViewConfigurationType);
|
|
|
|
}
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
oxr_event_push_XrEventDataSessionStateChanged(
|
2019-07-13 16:17:57 +00:00
|
|
|
log, sess, XR_SESSION_STATE_SYNCHRONIZED, 0);
|
2019-03-18 05:52:32 +00:00
|
|
|
oxr_event_push_XrEventDataSessionStateChanged(
|
|
|
|
log, sess, XR_SESSION_STATE_VISIBLE, 0);
|
|
|
|
oxr_event_push_XrEventDataSessionStateChanged(
|
|
|
|
log, sess, XR_SESSION_STATE_FOCUSED, 0);
|
|
|
|
|
|
|
|
sess->state = XR_SESSION_STATE_FOCUSED;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_session_end(struct oxr_logger *log, struct oxr_session *sess)
|
|
|
|
{
|
|
|
|
struct xrt_compositor *xc = sess->compositor;
|
|
|
|
|
|
|
|
if (!is_running(sess->state)) {
|
|
|
|
return oxr_error(log, XR_ERROR_SESSION_NOT_RUNNING,
|
|
|
|
" session is not running");
|
|
|
|
}
|
|
|
|
|
2019-03-23 01:11:14 +00:00
|
|
|
if (xc != NULL) {
|
|
|
|
if (sess->frame_started) {
|
|
|
|
xc->discard_frame(xc);
|
|
|
|
sess->frame_started = false;
|
|
|
|
}
|
2019-03-18 05:52:32 +00:00
|
|
|
|
2019-03-23 01:11:14 +00:00
|
|
|
xc->end_session(xc);
|
|
|
|
}
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
oxr_event_push_XrEventDataSessionStateChanged(
|
|
|
|
log, sess, XR_SESSION_STATE_STOPPING, 0);
|
|
|
|
oxr_event_push_XrEventDataSessionStateChanged(log, sess,
|
|
|
|
XR_SESSION_STATE_IDLE, 0);
|
|
|
|
oxr_event_push_XrEventDataSessionStateChanged(
|
|
|
|
log, sess, XR_SESSION_STATE_READY, 0);
|
|
|
|
|
|
|
|
sess->state = XR_SESSION_STATE_READY;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
XrResult
|
|
|
|
oxr_session_get_view_pose_at(struct oxr_logger *log,
|
|
|
|
struct oxr_session *sess,
|
|
|
|
XrTime at_time,
|
|
|
|
struct xrt_pose *pose)
|
|
|
|
{
|
|
|
|
// @todo This function needs to be massively expanded to support all
|
|
|
|
// use cases this drive. The main use of this function is to get
|
|
|
|
// either the predicted position of the headset device. Right now
|
|
|
|
// it only returns the current position. But it must also deal
|
|
|
|
// with past values are allowed by the spec. See displayTime
|
|
|
|
// argument on the xrLocateViews function. It will also drive
|
|
|
|
// the function xrLocateSpace view using the view space.
|
|
|
|
// @todo If using orientation tracking only implement a neck model to
|
|
|
|
// get at least a slightly better position.
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
struct xrt_device *xdev = sess->sys->head;
|
|
|
|
struct xrt_pose *offset = &xdev->tracking->offset;
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
struct xrt_space_relation relation;
|
2019-03-22 15:39:35 +00:00
|
|
|
int64_t timestamp;
|
2019-05-09 10:54:48 +00:00
|
|
|
xdev->get_tracked_pose(xdev, XRT_INPUT_GENERIC_HEAD_RELATION,
|
|
|
|
sess->sys->inst->timekeeping, ×tamp,
|
2019-03-22 15:39:35 +00:00
|
|
|
&relation);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
// Add in the offset from the tracking system.
|
|
|
|
math_relation_accumulate_transform(offset, &relation);
|
|
|
|
|
|
|
|
// clang-format off
|
|
|
|
bool valid_pos = (relation.relation_flags & XRT_SPACE_RELATION_POSITION_VALID_BIT) != 0;
|
|
|
|
bool valid_ori = (relation.relation_flags & XRT_SPACE_RELATION_ORIENTATION_VALID_BIT) != 0;
|
|
|
|
bool valid_vel = (relation.relation_flags & XRT_SPACE_RELATION_ANGULAR_VELOCITY_VALID_BIT) != 0;
|
|
|
|
// clang-format on
|
|
|
|
|
|
|
|
if (valid_ori) {
|
2019-03-18 05:52:32 +00:00
|
|
|
pose->orientation = relation.pose.orientation;
|
|
|
|
} else {
|
2019-05-07 12:47:18 +00:00
|
|
|
// If the orientation is not valid just use the offset.
|
|
|
|
pose->orientation = offset->orientation;
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
if (valid_pos) {
|
2019-03-18 05:52:32 +00:00
|
|
|
pose->position = relation.pose.position;
|
|
|
|
} else {
|
2019-05-07 12:47:18 +00:00
|
|
|
// If the position is not valid just use the offset.
|
|
|
|
pose->position = offset->position;
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
if (valid_vel) {
|
2019-03-18 05:52:32 +00:00
|
|
|
//! @todo Forcing a fixed amount of prediction for now since
|
|
|
|
//! devices don't tell us timestamps yet.
|
2019-03-22 16:07:02 +00:00
|
|
|
int64_t ns_diff = at_time - timestamp;
|
|
|
|
float interval;
|
|
|
|
if (debug_get_bool_option_dynamic_prediction()) {
|
|
|
|
interval =
|
|
|
|
time_ns_to_s(ns_diff) + sess->static_prediction_s;
|
|
|
|
} else {
|
|
|
|
interval = sess->static_prediction_s;
|
|
|
|
}
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
struct xrt_quat predicted;
|
2019-03-22 16:07:02 +00:00
|
|
|
math_quat_integrate_velocity(&pose->orientation,
|
|
|
|
&relation.angular_velocity,
|
|
|
|
interval, &predicted);
|
2019-03-18 05:52:32 +00:00
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
if (debug_get_bool_option_views()) {
|
2019-03-22 16:07:02 +00:00
|
|
|
fprintf(stderr,
|
|
|
|
"\toriginal quat = {%f, %f, %f, %f} "
|
|
|
|
"(time requested: %li, Interval %li nsec, with "
|
|
|
|
"static interval %f s)\n",
|
|
|
|
pose->orientation.x, pose->orientation.y,
|
|
|
|
pose->orientation.z, pose->orientation.w,
|
|
|
|
at_time, ns_diff, interval);
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
pose->orientation = predicted;
|
|
|
|
}
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
print_view_fov(uint32_t index, const struct xrt_fov *fov)
|
|
|
|
{
|
|
|
|
if (!debug_get_bool_option_views()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(stderr, "\tviews[%i].fov = {%f, %f, %f, %f}\n", index,
|
|
|
|
fov->angle_left, fov->angle_right, fov->angle_up,
|
|
|
|
fov->angle_down);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
print_view_pose(uint32_t index, const struct xrt_pose *pose)
|
|
|
|
{
|
|
|
|
if (!debug_get_bool_option_views()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(stderr, "\tviews[%i].pose = {{%f, %f, %f, %f}, {%f, %f, %f}}\n",
|
|
|
|
index, pose->orientation.x, pose->orientation.y,
|
|
|
|
pose->orientation.z, pose->orientation.w, pose->position.x,
|
|
|
|
pose->position.y, pose->position.z);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_session_views(struct oxr_logger *log,
|
|
|
|
struct oxr_session *sess,
|
|
|
|
const XrViewLocateInfo *viewLocateInfo,
|
|
|
|
XrViewState *viewState,
|
|
|
|
uint32_t viewCapacityInput,
|
|
|
|
uint32_t *viewCountOutput,
|
|
|
|
XrView *views)
|
|
|
|
{
|
2019-05-07 12:47:18 +00:00
|
|
|
struct xrt_device *xdev = sess->sys->head;
|
2019-03-18 05:52:32 +00:00
|
|
|
struct oxr_space *baseSpc = (struct oxr_space *)viewLocateInfo->space;
|
|
|
|
uint32_t num_views = 2;
|
|
|
|
|
|
|
|
// Does this apply for all calls?
|
|
|
|
if (!baseSpc->is_reference) {
|
|
|
|
viewState->viewStateFlags = 0;
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start two call handling.
|
|
|
|
if (viewCountOutput != NULL) {
|
|
|
|
*viewCountOutput = num_views;
|
|
|
|
}
|
|
|
|
if (viewCapacityInput == 0) {
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
if (viewCapacityInput < num_views) {
|
|
|
|
return oxr_error(log, XR_ERROR_SIZE_INSUFFICIENT,
|
|
|
|
"(viewCapacityInput == %u) need %u",
|
|
|
|
viewCapacityInput, num_views);
|
|
|
|
}
|
|
|
|
// End two call handling.
|
|
|
|
|
|
|
|
if (debug_get_bool_option_views()) {
|
|
|
|
fprintf(stderr, "%s\n", __func__);
|
|
|
|
fprintf(stderr, "\tviewLocateInfo->displayTime %lu\n",
|
|
|
|
viewLocateInfo->displayTime);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the viewLocateInfo->space to view space relation.
|
|
|
|
struct xrt_space_relation pure_relation;
|
|
|
|
oxr_space_ref_relation(log, sess, XR_REFERENCE_SPACE_TYPE_VIEW,
|
|
|
|
baseSpc->type, viewLocateInfo->displayTime,
|
|
|
|
&pure_relation);
|
|
|
|
|
|
|
|
struct xrt_pose pure = pure_relation.pose;
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
// @todo the fov information that we get from xdev->hmd->views[i].fov is
|
|
|
|
// not properly filled out in oh_device.c, fix before wasting time
|
|
|
|
// on debugging weird rendering when adding stuff here.
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
for (uint32_t i = 0; i < num_views; i++) {
|
|
|
|
//! @todo Do not hardcode IPD.
|
|
|
|
struct xrt_vec3 eye_relation = {
|
|
|
|
sess->ipd_meters,
|
|
|
|
0.0f,
|
|
|
|
0.0f,
|
|
|
|
};
|
|
|
|
struct xrt_pose view_pose;
|
|
|
|
|
|
|
|
// Get the per view pose from the device.
|
|
|
|
xdev->get_view_pose(xdev, &eye_relation, i, &view_pose);
|
|
|
|
|
|
|
|
// Do the magical space relation dance here.
|
|
|
|
math_pose_openxr_locate(&view_pose, &pure, &baseSpc->pose,
|
|
|
|
(struct xrt_pose *)&views[i].pose);
|
|
|
|
|
|
|
|
// Copy the fov information directly from the device.
|
2019-05-09 10:54:48 +00:00
|
|
|
views[i].fov = *(XrFovf *)&xdev->hmd->views[i].fov;
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
print_view_fov(i, (struct xrt_fov *)&views[i].fov);
|
|
|
|
print_view_pose(i, (struct xrt_pose *)&views[i].pose);
|
|
|
|
}
|
|
|
|
|
|
|
|
// @todo Add tracking bit once we have them.
|
|
|
|
viewState->viewStateFlags = 0;
|
|
|
|
viewState->viewStateFlags |= XR_VIEW_STATE_POSITION_VALID_BIT;
|
|
|
|
viewState->viewStateFlags |= XR_VIEW_STATE_ORIENTATION_VALID_BIT;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_session_frame_wait(struct oxr_logger *log,
|
|
|
|
struct oxr_session *sess,
|
|
|
|
XrFrameState *frameState)
|
|
|
|
{
|
|
|
|
if (!is_running(sess->state)) {
|
|
|
|
return oxr_error(log, XR_ERROR_SESSION_NOT_RUNNING,
|
|
|
|
" session is not running");
|
|
|
|
}
|
|
|
|
|
2019-03-18 03:34:07 +00:00
|
|
|
//! @todo this should be carefully synchronized, because there may be
|
|
|
|
//! more than one session per instance.
|
2019-04-01 16:07:23 +00:00
|
|
|
XRT_MAYBE_UNUSED timepoint_ns now =
|
2019-03-18 05:52:32 +00:00
|
|
|
time_state_get_now_and_update(sess->sys->inst->timekeeping);
|
|
|
|
|
|
|
|
struct xrt_compositor *xc = sess->compositor;
|
|
|
|
xc->wait_frame(xc, &frameState->predictedDisplayTime,
|
|
|
|
&frameState->predictedDisplayPeriod);
|
|
|
|
|
2019-07-13 16:17:57 +00:00
|
|
|
frameState->shouldRender = should_render(sess->state);
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_session_frame_begin(struct oxr_logger *log, struct oxr_session *sess)
|
|
|
|
{
|
|
|
|
if (!is_running(sess->state)) {
|
|
|
|
return oxr_error(log, XR_ERROR_SESSION_NOT_RUNNING,
|
|
|
|
" session is not running");
|
|
|
|
}
|
|
|
|
|
|
|
|
struct xrt_compositor *xc = sess->compositor;
|
|
|
|
|
|
|
|
XrResult ret;
|
|
|
|
if (sess->frame_started) {
|
|
|
|
ret = XR_FRAME_DISCARDED;
|
2019-03-26 20:43:15 +00:00
|
|
|
if (xc != NULL) {
|
|
|
|
xc->discard_frame(xc);
|
|
|
|
}
|
2019-03-18 05:52:32 +00:00
|
|
|
} else {
|
|
|
|
ret = XR_SUCCESS;
|
|
|
|
sess->frame_started = true;
|
|
|
|
}
|
2019-03-26 20:43:15 +00:00
|
|
|
if (xc != NULL) {
|
|
|
|
xc->begin_frame(xc);
|
|
|
|
}
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static enum xrt_blend_mode
|
|
|
|
oxr_blend_mode_to_xrt(XrEnvironmentBlendMode blend_mode)
|
|
|
|
{
|
|
|
|
// clang-format off
|
|
|
|
switch (blend_mode) {
|
|
|
|
case XR_ENVIRONMENT_BLEND_MODE_OPAQUE: return XRT_BLEND_MODE_OPAQUE;
|
|
|
|
case XR_ENVIRONMENT_BLEND_MODE_ADDITIVE: return XRT_BLEND_MODE_ADDITIVE;
|
|
|
|
case XR_ENVIRONMENT_BLEND_MODE_ALPHA_BLEND: return XRT_BLEND_MODE_ALPHA_BLEND;
|
|
|
|
default: return (enum xrt_blend_mode)0;
|
|
|
|
}
|
|
|
|
// clang-format on
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_session_frame_end(struct oxr_logger *log,
|
|
|
|
struct oxr_session *sess,
|
|
|
|
const XrFrameEndInfo *frameEndInfo)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Session state and call order.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (!is_running(sess->state)) {
|
|
|
|
return oxr_error(log, XR_ERROR_SESSION_NOT_RUNNING,
|
|
|
|
" session is not running");
|
|
|
|
}
|
|
|
|
if (!sess->frame_started) {
|
|
|
|
return oxr_error(log, XR_ERROR_CALL_ORDER_INVALID,
|
|
|
|
" frame not begun with xrBeginFrame");
|
|
|
|
}
|
|
|
|
|
|
|
|
struct xrt_compositor *xc = sess->compositor;
|
|
|
|
|
2019-03-26 20:43:15 +00:00
|
|
|
/*
|
|
|
|
* early out for headless sessions.
|
|
|
|
*/
|
|
|
|
if (xc == NULL) {
|
|
|
|
sess->frame_started = false;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
/*
|
|
|
|
* Early out for discarded frame if layer count is 0,
|
|
|
|
* since then blend mode, etc. doesn't matter.
|
|
|
|
*/
|
|
|
|
if (frameEndInfo->layerCount == 0) {
|
|
|
|
xc->discard_frame(xc);
|
|
|
|
sess->frame_started = false;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Blend mode.
|
|
|
|
*/
|
|
|
|
|
|
|
|
enum xrt_blend_mode blend_mode =
|
|
|
|
oxr_blend_mode_to_xrt(frameEndInfo->environmentBlendMode);
|
|
|
|
|
|
|
|
if (blend_mode == 0) {
|
|
|
|
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
|
|
|
|
"(frameEndInfo->environmentBlendMode) "
|
|
|
|
"unknown environment blend mode");
|
|
|
|
}
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
if ((blend_mode & sess->sys->head->hmd->blend_mode) == 0) {
|
2019-03-18 05:52:32 +00:00
|
|
|
return oxr_error(log,
|
|
|
|
XR_ERROR_ENVIRONMENT_BLEND_MODE_UNSUPPORTED,
|
|
|
|
"(frameEndInfo->environmentBlendMode) "
|
|
|
|
"is not supported");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Layers.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (frameEndInfo->layers == NULL) {
|
|
|
|
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
|
|
|
|
"(frameEndInfo->layers == NULL)");
|
|
|
|
}
|
|
|
|
if (frameEndInfo->layers[0]->type !=
|
|
|
|
XR_TYPE_COMPOSITION_LAYER_PROJECTION) {
|
|
|
|
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
|
|
|
|
"(frameEndInfo->layers[0]->type)");
|
|
|
|
}
|
|
|
|
|
|
|
|
XrCompositionLayerProjection *proj =
|
|
|
|
(XrCompositionLayerProjection *)frameEndInfo->layers[0];
|
|
|
|
|
|
|
|
if (proj->viewCount != 2) {
|
|
|
|
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
|
|
|
|
"(frameEndInfo->layers[0]->viewCount == %u)"
|
|
|
|
" must be 2",
|
|
|
|
proj->viewCount);
|
|
|
|
}
|
|
|
|
|
2019-06-28 14:55:45 +00:00
|
|
|
// Check for valid swapchain states.
|
|
|
|
for (uint32_t i = 0; i < proj->viewCount; i++) {
|
|
|
|
//! @todo More validation?
|
|
|
|
struct oxr_swapchain *sc =
|
|
|
|
(struct oxr_swapchain *)proj->views[i].subImage.swapchain;
|
|
|
|
|
|
|
|
if (sc->released_index == -1) {
|
|
|
|
return oxr_error(
|
|
|
|
log, XR_ERROR_CALL_ORDER_INVALID,
|
|
|
|
"(frameEndInfo->layers[0]->views[%i].subImage."
|
|
|
|
"swapchain) Swapchain has not been released!",
|
|
|
|
i);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sc->released_index >= (int)sc->swapchain->num_images) {
|
|
|
|
return oxr_error(
|
|
|
|
log, XR_ERROR_RUNTIME_FAILURE,
|
|
|
|
"(frameEndInfo->layers[0]->views[%i].subImage."
|
|
|
|
"swapchain) Internal image index out of bounds!",
|
|
|
|
i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Doing the real work.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct xrt_swapchain *chains[2];
|
2019-04-30 15:33:56 +00:00
|
|
|
uint32_t image_index[2];
|
2019-05-01 15:17:17 +00:00
|
|
|
uint32_t layers[2];
|
2019-03-18 05:52:32 +00:00
|
|
|
uint32_t num_chains = ARRAY_SIZE(chains);
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < num_chains; i++) {
|
|
|
|
struct oxr_swapchain *sc =
|
|
|
|
(struct oxr_swapchain *)proj->views[i].subImage.swapchain;
|
|
|
|
chains[i] = sc->swapchain;
|
2019-05-01 15:17:17 +00:00
|
|
|
layers[i] = proj->views[i].subImage.imageArrayIndex;
|
2019-04-30 15:33:56 +00:00
|
|
|
image_index[i] = sc->released_index;
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
|
2019-05-01 15:17:17 +00:00
|
|
|
xc->end_frame(xc, blend_mode, chains, image_index, layers, num_chains);
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
sess->frame_started = false;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2019-04-05 22:13:58 +00:00
|
|
|
static XrResult
|
|
|
|
oxr_session_destroy(struct oxr_logger *log, struct oxr_handle_base *hb)
|
|
|
|
{
|
|
|
|
struct oxr_session *sess = (struct oxr_session *)hb;
|
|
|
|
if (sess->compositor != NULL) {
|
|
|
|
sess->compositor->destroy(sess->compositor);
|
|
|
|
}
|
|
|
|
free(sess);
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define OXR_SESSION_ALLOCATE(LOG, SYS, OUT) \
|
|
|
|
do { \
|
|
|
|
OXR_ALLOCATE_HANDLE_OR_RETURN(LOG, OUT, OXR_XR_DEBUG_SESSION, \
|
|
|
|
oxr_session_destroy, \
|
2019-06-03 19:11:31 +00:00
|
|
|
&(SYS)->inst->handle); \
|
|
|
|
(OUT)->sys = (SYS); \
|
2019-04-05 22:13:58 +00:00
|
|
|
} while (0)
|
|
|
|
|
2019-06-03 19:17:45 +00:00
|
|
|
/* Just the allocation and populate part, so we can use early-returns to
|
|
|
|
* simplify code flow and avoid weird if/else */
|
|
|
|
static XrResult
|
|
|
|
oxr_session_create_impl(struct oxr_logger *log,
|
|
|
|
struct oxr_system *sys,
|
|
|
|
const XrSessionCreateInfo *createInfo,
|
|
|
|
struct oxr_session **out_session)
|
|
|
|
{
|
|
|
|
#ifdef XR_USE_PLATFORM_XLIB
|
|
|
|
XrGraphicsBindingOpenGLXlibKHR const *opengl_xlib =
|
|
|
|
OXR_GET_INPUT_FROM_CHAIN(createInfo,
|
|
|
|
XR_TYPE_GRAPHICS_BINDING_OPENGL_XLIB_KHR,
|
|
|
|
XrGraphicsBindingOpenGLXlibKHR);
|
|
|
|
if (opengl_xlib != NULL) {
|
|
|
|
OXR_SESSION_ALLOCATE(log, sys, *out_session);
|
|
|
|
return oxr_session_populate_gl_xlib(log, sys, opengl_xlib,
|
|
|
|
*out_session);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef XR_USE_GRAPHICS_API_VULKAN
|
|
|
|
XrGraphicsBindingVulkanKHR const *vulkan = OXR_GET_INPUT_FROM_CHAIN(
|
|
|
|
createInfo, XR_TYPE_GRAPHICS_BINDING_VULKAN_KHR,
|
|
|
|
XrGraphicsBindingVulkanKHR);
|
|
|
|
if (vulkan != NULL) {
|
|
|
|
OXR_SESSION_ALLOCATE(log, sys, *out_session);
|
|
|
|
return oxr_session_populate_vk(log, sys, vulkan, *out_session);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Add any new graphics binding structs here - before the headless
|
|
|
|
* check. (order for non-headless checks not specified in standard.)
|
|
|
|
* Any new addition will also need to be added to
|
|
|
|
* oxr_verify_XrSessionCreateInfo and have its own associated verify
|
|
|
|
* function added.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (sys->inst->headless) {
|
|
|
|
OXR_SESSION_ALLOCATE(log, sys, *out_session);
|
|
|
|
(*out_session)->compositor = NULL;
|
|
|
|
(*out_session)->create_swapchain = NULL;
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
|
|
|
|
"(createInfo->next->type)");
|
|
|
|
}
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
XrResult
|
|
|
|
oxr_session_create(struct oxr_logger *log,
|
|
|
|
struct oxr_system *sys,
|
2019-06-03 19:17:45 +00:00
|
|
|
const XrSessionCreateInfo *createInfo,
|
2019-03-18 05:52:32 +00:00
|
|
|
struct oxr_session **out_session)
|
|
|
|
{
|
|
|
|
struct oxr_session *sess;
|
|
|
|
|
2019-06-03 19:17:45 +00:00
|
|
|
/* Try allocating and populating. */
|
|
|
|
XrResult ret = oxr_session_create_impl(log, sys, createInfo, &sess);
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
if (ret != XR_SUCCESS) {
|
2019-04-05 22:13:58 +00:00
|
|
|
/* clean up allocation first */
|
|
|
|
XrResult cleanup_result =
|
|
|
|
oxr_handle_destroy(log, &sess->handle);
|
|
|
|
assert(cleanup_result == XR_SUCCESS);
|
2019-07-24 23:52:07 +00:00
|
|
|
(void)cleanup_result;
|
2019-03-18 05:52:32 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
sess->ipd_meters = debug_get_num_option_ipd() / 1000.0f;
|
|
|
|
sess->static_prediction_s =
|
|
|
|
debug_get_num_option_prediction_ms() / 1000.0f;
|
|
|
|
|
|
|
|
oxr_event_push_XrEventDataSessionStateChanged(log, sess,
|
|
|
|
XR_SESSION_STATE_IDLE, 0);
|
|
|
|
oxr_event_push_XrEventDataSessionStateChanged(
|
|
|
|
log, sess, XR_SESSION_STATE_READY, 0);
|
|
|
|
sess->state = XR_SESSION_STATE_READY;
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
u_hashmap_int_create(&sess->act_sets);
|
|
|
|
u_hashmap_int_create(&sess->sources);
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
*out_session = sess;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|