2020-06-01 20:32:43 +00:00
|
|
|
// Copyright 2018-2020, Collabora, Ltd.
|
2019-03-18 05:52:32 +00:00
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief Holds system related entrypoints.
|
|
|
|
* @author Jakob Bornecrantz <jakob@collabora.com>
|
|
|
|
* @ingroup oxr_main
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2020-03-02 23:35:57 +00:00
|
|
|
#include <inttypes.h>
|
2019-03-18 05:52:32 +00:00
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
#include "xrt/xrt_device.h"
|
2019-04-03 16:55:20 +00:00
|
|
|
#include "util/u_debug.h"
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
#include "oxr_objects.h"
|
|
|
|
#include "oxr_logger.h"
|
|
|
|
#include "oxr_two_call.h"
|
|
|
|
|
2019-04-03 16:55:20 +00:00
|
|
|
// clang-format off
|
|
|
|
DEBUG_GET_ONCE_NUM_OPTION(scale_percentage, "OXR_VIEWPORT_SCALE_PERCENTAGE", 140)
|
|
|
|
// clang-format on
|
|
|
|
|
2019-09-09 19:47:19 +00:00
|
|
|
static inline size_t
|
|
|
|
min_size_t(size_t a, size_t b)
|
|
|
|
{
|
|
|
|
return a < b ? a : b;
|
|
|
|
}
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
static bool
|
|
|
|
oxr_system_matches(struct oxr_logger *log,
|
|
|
|
struct oxr_system *sys,
|
|
|
|
XrFormFactor form_factor)
|
|
|
|
{
|
|
|
|
return form_factor == sys->form_factor;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_system_select(struct oxr_logger *log,
|
|
|
|
struct oxr_system **systems,
|
|
|
|
uint32_t num_systems,
|
|
|
|
XrFormFactor form_factor,
|
|
|
|
struct oxr_system **out_selected)
|
|
|
|
{
|
2019-10-19 21:17:57 +00:00
|
|
|
if (num_systems == 0) {
|
|
|
|
return oxr_error(
|
|
|
|
log, XR_ERROR_FORM_FACTOR_UNSUPPORTED,
|
|
|
|
"(getInfo->formFactor) no system available (given: %i)",
|
|
|
|
form_factor);
|
|
|
|
}
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
struct oxr_system *selected = NULL;
|
|
|
|
for (uint32_t i = 0; i < num_systems; i++) {
|
|
|
|
if (oxr_system_matches(log, systems[i], form_factor)) {
|
|
|
|
selected = systems[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (selected == NULL) {
|
|
|
|
return oxr_error(log, XR_ERROR_FORM_FACTOR_UNSUPPORTED,
|
2019-10-19 21:17:57 +00:00
|
|
|
"(getInfo->formFactor) no matching system "
|
|
|
|
"(given: %i, first: %i)",
|
|
|
|
form_factor, systems[0]->form_factor);
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
*out_selected = selected;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2019-06-05 19:21:25 +00:00
|
|
|
XrResult
|
|
|
|
oxr_system_verify_id(struct oxr_logger *log,
|
|
|
|
const struct oxr_instance *inst,
|
|
|
|
XrSystemId systemId)
|
|
|
|
{
|
|
|
|
if (systemId != 1) {
|
|
|
|
return oxr_error(log, XR_ERROR_SYSTEM_INVALID,
|
2020-05-31 15:49:25 +00:00
|
|
|
"Invalid system %" PRIu64, systemId);
|
2019-06-05 19:21:25 +00:00
|
|
|
}
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_system_get_by_id(struct oxr_logger *log,
|
|
|
|
struct oxr_instance *inst,
|
|
|
|
XrSystemId systemId,
|
|
|
|
struct oxr_system **system)
|
|
|
|
{
|
|
|
|
XrResult result = oxr_system_verify_id(log, inst, systemId);
|
|
|
|
if (result != XR_SUCCESS) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* right now only have one system. */
|
|
|
|
*system = &inst->system;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
XrResult
|
|
|
|
oxr_system_fill_in(struct oxr_logger *log,
|
|
|
|
struct oxr_instance *inst,
|
|
|
|
XrSystemId systemId,
|
|
|
|
struct oxr_system *sys,
|
2019-07-24 16:15:15 +00:00
|
|
|
struct xrt_device **xdevs,
|
|
|
|
size_t num_xdevs)
|
2019-03-18 05:52:32 +00:00
|
|
|
{
|
2019-09-09 20:56:42 +00:00
|
|
|
sys->num_xdevs = min_size_t(ARRAY_SIZE(sys->xdevs), num_xdevs);
|
2019-07-24 16:15:15 +00:00
|
|
|
|
2019-09-09 20:56:42 +00:00
|
|
|
for (uint32_t i = 0; i < sys->num_xdevs; i++) {
|
2019-07-24 16:15:15 +00:00
|
|
|
sys->xdevs[i] = xdevs[i];
|
|
|
|
}
|
2019-09-09 20:56:42 +00:00
|
|
|
for (size_t i = sys->num_xdevs; i < num_xdevs; i++) {
|
2019-07-24 16:15:15 +00:00
|
|
|
oxr_xdev_destroy(&xdevs[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct xrt_device *head = sys->head;
|
|
|
|
struct xrt_device *left = sys->left;
|
|
|
|
struct xrt_device *right = sys->right;
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
if (head == NULL) {
|
2019-03-18 05:52:32 +00:00
|
|
|
return oxr_error(log, XR_ERROR_INITIALIZATION_FAILED,
|
2020-05-31 15:49:25 +00:00
|
|
|
"Failed to probe device");
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
|
2019-06-24 18:06:45 +00:00
|
|
|
if (head->tracking_origin->type == XRT_TRACKING_TYPE_NONE) {
|
2019-05-07 12:47:18 +00:00
|
|
|
// "nominal height" 1.6m
|
2019-06-24 18:06:45 +00:00
|
|
|
head->tracking_origin->offset.position.x = 0.0f;
|
|
|
|
head->tracking_origin->offset.position.y = 1.6f;
|
|
|
|
head->tracking_origin->offset.position.z = 0.0f;
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
|
|
|
|
2019-06-24 18:06:45 +00:00
|
|
|
if (left != NULL &&
|
|
|
|
left->tracking_origin->type == XRT_TRACKING_TYPE_NONE) {
|
|
|
|
left->tracking_origin->offset.position.x = -0.2f;
|
|
|
|
left->tracking_origin->offset.position.y = 1.3f;
|
|
|
|
left->tracking_origin->offset.position.z = -0.5f;
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
|
|
|
|
2019-06-24 18:06:45 +00:00
|
|
|
if (right != NULL &&
|
|
|
|
right->tracking_origin->type == XRT_TRACKING_TYPE_NONE) {
|
|
|
|
right->tracking_origin->offset.position.x = 0.2f;
|
|
|
|
right->tracking_origin->offset.position.y = 1.3f;
|
|
|
|
right->tracking_origin->offset.position.z = -0.5f;
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
// clang-format off
|
2019-04-03 16:55:20 +00:00
|
|
|
sys->inst = inst;
|
|
|
|
sys->systemId = systemId;
|
|
|
|
sys->form_factor = XR_FORM_FACTOR_HEAD_MOUNTED_DISPLAY;
|
2019-03-18 05:52:32 +00:00
|
|
|
sys->view_config_type = XR_VIEW_CONFIGURATION_TYPE_PRIMARY_STEREO;
|
|
|
|
|
2019-04-03 16:55:20 +00:00
|
|
|
double scale = debug_get_num_option_scale_percentage() / 100.0;
|
2019-11-05 10:35:53 +00:00
|
|
|
if (scale > 2.0) {
|
|
|
|
scale = 2.0;
|
|
|
|
oxr_log(log, "Clamped scale to 200%%\n");
|
|
|
|
}
|
2019-04-03 16:55:20 +00:00
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
uint32_t w0 = (uint32_t)(head->hmd->views[0].display.w_pixels * scale);
|
2019-11-05 10:30:46 +00:00
|
|
|
uint32_t h0 = (uint32_t)(head->hmd->views[0].display.h_pixels * scale);
|
2019-05-07 12:47:18 +00:00
|
|
|
uint32_t w1 = (uint32_t)(head->hmd->views[1].display.w_pixels * scale);
|
2019-11-05 10:30:46 +00:00
|
|
|
uint32_t h1 = (uint32_t)(head->hmd->views[1].display.h_pixels * scale);
|
2019-04-03 16:55:20 +00:00
|
|
|
|
2019-11-05 10:35:53 +00:00
|
|
|
uint32_t w0_2 = head->hmd->views[0].display.w_pixels * 2;
|
|
|
|
uint32_t h0_2 = head->hmd->views[0].display.h_pixels * 2;
|
|
|
|
uint32_t w1_2 = head->hmd->views[1].display.w_pixels * 2;
|
|
|
|
uint32_t h1_2 = head->hmd->views[1].display.h_pixels * 2;
|
|
|
|
|
2019-04-03 16:55:20 +00:00
|
|
|
sys->views[0].recommendedImageRectWidth = w0;
|
2019-11-05 10:35:53 +00:00
|
|
|
sys->views[0].maxImageRectWidth = w0_2;
|
2019-04-03 16:55:20 +00:00
|
|
|
sys->views[0].recommendedImageRectHeight = h0;
|
2019-11-05 10:35:53 +00:00
|
|
|
sys->views[0].maxImageRectHeight = h0_2;
|
2019-03-18 05:52:32 +00:00
|
|
|
sys->views[0].recommendedSwapchainSampleCount = 1;
|
|
|
|
sys->views[0].maxSwapchainSampleCount = 1;
|
|
|
|
|
2019-04-03 16:55:20 +00:00
|
|
|
sys->views[1].recommendedImageRectWidth = w1;
|
2019-11-05 10:35:53 +00:00
|
|
|
sys->views[1].maxImageRectWidth = w1_2;
|
2019-04-03 16:55:20 +00:00
|
|
|
sys->views[1].recommendedImageRectHeight = h1;
|
2019-11-05 10:35:53 +00:00
|
|
|
sys->views[1].maxImageRectHeight = h1_2;
|
2019-03-18 05:52:32 +00:00
|
|
|
sys->views[1].recommendedSwapchainSampleCount = 1;
|
|
|
|
sys->views[1].maxSwapchainSampleCount = 1;
|
2019-04-03 16:55:20 +00:00
|
|
|
// clang-format on
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
uint32_t i = 0;
|
2019-05-07 12:47:18 +00:00
|
|
|
if (head->hmd->blend_mode & XRT_BLEND_MODE_OPAQUE) {
|
2019-03-18 05:52:32 +00:00
|
|
|
sys->blend_modes[i++] = XR_ENVIRONMENT_BLEND_MODE_OPAQUE;
|
|
|
|
}
|
2019-05-07 12:47:18 +00:00
|
|
|
if (head->hmd->blend_mode & XRT_BLEND_MODE_ADDITIVE) {
|
2019-03-18 05:52:32 +00:00
|
|
|
sys->blend_modes[i++] = XR_ENVIRONMENT_BLEND_MODE_ADDITIVE;
|
|
|
|
}
|
2019-05-07 12:47:18 +00:00
|
|
|
if (head->hmd->blend_mode & XRT_BLEND_MODE_ALPHA_BLEND) {
|
2019-03-18 05:52:32 +00:00
|
|
|
sys->blend_modes[i++] = XR_ENVIRONMENT_BLEND_MODE_ALPHA_BLEND;
|
|
|
|
}
|
|
|
|
sys->num_blend_modes = i;
|
|
|
|
|
|
|
|
assert(i < ARRAY_SIZE(sys->blend_modes));
|
|
|
|
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_system_get_properties(struct oxr_logger *log,
|
|
|
|
struct oxr_system *sys,
|
|
|
|
XrSystemProperties *properties)
|
|
|
|
{
|
|
|
|
properties->vendorId = 42;
|
|
|
|
properties->systemId = sys->systemId;
|
2019-04-04 21:01:28 +00:00
|
|
|
|
2019-09-06 14:45:48 +00:00
|
|
|
// The magical 247 number, is to silence warnings.
|
|
|
|
snprintf(properties->systemName, XR_MAX_SYSTEM_NAME_SIZE,
|
|
|
|
"Monado: %.*s", 247, sys->head->str);
|
2019-04-04 21:01:28 +00:00
|
|
|
|
2020-05-28 20:44:26 +00:00
|
|
|
//! Get from compositor.
|
|
|
|
uint32_t max = XR_MIN_COMPOSITION_LAYERS_SUPPORTED;
|
|
|
|
|
|
|
|
properties->graphicsProperties.maxLayerCount = max;
|
2019-03-18 05:52:32 +00:00
|
|
|
properties->graphicsProperties.maxSwapchainImageWidth = 1024 * 16;
|
|
|
|
properties->graphicsProperties.maxSwapchainImageHeight = 1024 * 16;
|
|
|
|
properties->trackingProperties.orientationTracking = XR_TRUE;
|
|
|
|
properties->trackingProperties.positionTracking = XR_FALSE;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_system_enumerate_view_confs(struct oxr_logger *log,
|
|
|
|
struct oxr_system *sys,
|
|
|
|
uint32_t viewConfigurationTypeCapacityInput,
|
|
|
|
uint32_t *viewConfigurationTypeCountOutput,
|
|
|
|
XrViewConfigurationType *viewConfigurationTypes)
|
|
|
|
{
|
|
|
|
OXR_TWO_CALL_HELPER(log, viewConfigurationTypeCapacityInput,
|
|
|
|
viewConfigurationTypeCountOutput,
|
2019-09-26 20:25:05 +00:00
|
|
|
viewConfigurationTypes, 1, &sys->view_config_type,
|
|
|
|
XR_SUCCESS);
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_system_enumerate_blend_modes(struct oxr_logger *log,
|
|
|
|
struct oxr_system *sys,
|
2019-07-13 16:17:57 +00:00
|
|
|
XrViewConfigurationType viewConfigurationType,
|
2019-03-18 05:52:32 +00:00
|
|
|
uint32_t environmentBlendModeCapacityInput,
|
|
|
|
uint32_t *environmentBlendModeCountOutput,
|
|
|
|
XrEnvironmentBlendMode *environmentBlendModes)
|
|
|
|
{
|
2019-07-13 16:17:57 +00:00
|
|
|
//! @todo Take into account viewConfigurationType
|
2019-03-18 05:52:32 +00:00
|
|
|
OXR_TWO_CALL_HELPER(log, environmentBlendModeCapacityInput,
|
|
|
|
environmentBlendModeCountOutput,
|
|
|
|
environmentBlendModes, sys->num_blend_modes,
|
2019-09-26 20:25:05 +00:00
|
|
|
sys->blend_modes, XR_SUCCESS);
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_system_get_view_conf_properties(
|
|
|
|
struct oxr_logger *log,
|
|
|
|
struct oxr_system *sys,
|
|
|
|
XrViewConfigurationType viewConfigurationType,
|
|
|
|
XrViewConfigurationProperties *configurationProperties)
|
|
|
|
{
|
|
|
|
if (viewConfigurationType != sys->view_config_type) {
|
|
|
|
return oxr_error(log,
|
|
|
|
XR_ERROR_VIEW_CONFIGURATION_TYPE_UNSUPPORTED,
|
2020-05-31 15:49:25 +00:00
|
|
|
"Invalid view configuration type");
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// clang-format off
|
|
|
|
configurationProperties->viewConfigurationType = sys->view_config_type;
|
2019-08-16 22:02:18 +00:00
|
|
|
configurationProperties->fovMutable = XR_FALSE;
|
2019-03-18 05:52:32 +00:00
|
|
|
// clang-format on
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2020-03-04 14:50:48 +00:00
|
|
|
static void
|
|
|
|
view_configuration_view_fill_in(XrViewConfigurationView *target_view,
|
|
|
|
XrViewConfigurationView *source_view)
|
|
|
|
{
|
|
|
|
// clang-format off
|
|
|
|
target_view->recommendedImageRectWidth = source_view->recommendedImageRectWidth;
|
|
|
|
target_view->maxImageRectWidth = source_view->maxImageRectWidth;
|
|
|
|
target_view->recommendedImageRectHeight = source_view->recommendedImageRectHeight;
|
|
|
|
target_view->maxImageRectHeight = source_view->maxImageRectHeight;
|
|
|
|
target_view->recommendedSwapchainSampleCount = source_view->recommendedSwapchainSampleCount;
|
|
|
|
target_view->maxSwapchainSampleCount = source_view->maxSwapchainSampleCount;
|
|
|
|
// clang-format on
|
|
|
|
}
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
XrResult
|
|
|
|
oxr_system_enumerate_view_conf_views(
|
|
|
|
struct oxr_logger *log,
|
|
|
|
struct oxr_system *sys,
|
|
|
|
XrViewConfigurationType viewConfigurationType,
|
|
|
|
uint32_t viewCapacityInput,
|
|
|
|
uint32_t *viewCountOutput,
|
|
|
|
XrViewConfigurationView *views)
|
|
|
|
{
|
|
|
|
if (viewConfigurationType != sys->view_config_type) {
|
|
|
|
return oxr_error(log,
|
|
|
|
XR_ERROR_VIEW_CONFIGURATION_TYPE_UNSUPPORTED,
|
2020-05-31 15:49:25 +00:00
|
|
|
"Invalid view configuration type");
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
|
2020-03-04 14:50:48 +00:00
|
|
|
OXR_TWO_CALL_FILL_IN_HELPER(log, viewCapacityInput, viewCountOutput,
|
|
|
|
views, 2, view_configuration_view_fill_in,
|
|
|
|
sys->views, XR_SUCCESS);
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|