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

612 lines
16 KiB
C
Raw Normal View History

// Copyright 2018-2020, Collabora, Ltd.
2019-03-18 05:52:32 +00:00
// SPDX-License-Identifier: BSL-1.0
/*!
* @file
2020-03-31 16:24:58 +00:00
* @brief File for verifying app input into api functions.
2019-03-18 05:52:32 +00:00
* @author Ryan Pavlik <ryan.pavlik@collabora.com>
* @author Jakob Bornecrantz <jakob@collabora.com>
* @ingroup oxr_main
* @ingroup oxr_api
*/
2019-10-08 14:40:40 +00:00
#include <assert.h>
#include <stdio.h>
#include <string.h>
2019-03-18 05:52:32 +00:00
#include "xrt/xrt_compiler.h"
#include "util/u_debug.h"
#include "oxr_objects.h"
#include "oxr_logger.h"
#include "oxr_api_verify.h"
#include "oxr_chain.h"
2019-03-18 05:52:32 +00:00
/*
*
* Path verification.
*
*/
static bool
valid_path_char(const char c)
{
if ('a' <= c && c <= 'z') {
return true;
}
if ('0' <= c && c <= '9') {
return true;
}
if (c == '-' || c == '_' || c == '.' || c == '/') {
return true;
}
return false;
}
static bool
contains_zero(const char *path, uint32_t size)
2019-03-18 05:52:32 +00:00
{
for (uint32_t i = 0; i < size; i++) {
if (path[i] == '\0') {
return true;
}
}
return false;
}
2019-10-08 14:40:40 +00:00
XrResult
oxr_verify_fixed_size_single_level_path(struct oxr_logger *log,
const char *path,
uint32_t array_size,
const char *name)
2019-03-18 05:52:32 +00:00
{
if (array_size == 0) {
2019-04-08 18:41:19 +00:00
return oxr_error(log, XR_ERROR_RUNTIME_FAILURE,
2019-03-18 05:52:32 +00:00
"(%s) internal runtime error", name);
}
if (path[0] == '\0') {
return oxr_error(log, XR_ERROR_NAME_INVALID,
2019-03-18 05:52:32 +00:00
"(%s) can not be empty", name);
}
if (!contains_zero(path, array_size)) {
return oxr_error(log, XR_ERROR_PATH_FORMAT_INVALID,
2019-03-18 05:52:32 +00:00
"(%s) must include zero termination '\\0'.",
name);
}
size_t length = strlen(path);
for (size_t i = 0; i < length; i++) {
const char c = path[i];
// Slashes are not valid in single level paths.
if (valid_path_char(c) && c != '/') {
continue;
}
return oxr_error(
log, XR_ERROR_PATH_FORMAT_INVALID,
"(%s) 0x%02x is not a valid character at position %u", name,
c, (uint32_t)i);
}
2019-03-18 05:52:32 +00:00
return XR_SUCCESS;
}
2019-10-08 14:40:40 +00:00
XrResult
oxr_verify_localized_name(struct oxr_logger *log,
const char *string,
2019-04-13 15:11:47 +00:00
uint32_t array_size,
const char *name)
2019-04-13 15:11:47 +00:00
{
if (array_size == 0) {
return oxr_error(log, XR_ERROR_RUNTIME_FAILURE,
"(%s) internal runtime error", name);
}
if (string[0] == '\0') {
return oxr_error(log, XR_ERROR_LOCALIZED_NAME_INVALID,
2019-04-13 15:11:47 +00:00
"(%s) can not be empty", name);
}
if (!contains_zero(string, array_size)) {
return oxr_error(log, XR_ERROR_LOCALIZED_NAME_INVALID,
2019-04-13 15:11:47 +00:00
"(%s) must include zero termination '\\0'.",
name);
}
// Future work: validate well-formed UTF-8?
return XR_SUCCESS;
}
2019-10-08 14:40:40 +00:00
enum verify_state
2019-04-05 11:54:37 +00:00
{
2019-10-08 14:40:40 +00:00
VERIFY_START,
VERIFY_MIDDLE,
VERIFY_SLASH,
VERIFY_SLASHDOTS,
2019-04-05 11:54:37 +00:00
};
2019-10-08 14:40:40 +00:00
XrResult
oxr_verify_full_path_c(struct oxr_logger *log,
const char *path,
const char *name)
2019-04-05 11:54:37 +00:00
{
// XR_MAX_PATH_LENGTH is max including null terminator,
// length will not include null terminator
size_t length = XR_MAX_PATH_LENGTH;
for (size_t i = 0; i < XR_MAX_PATH_LENGTH; i++) {
if (path[i] == '\0') {
length = i;
break;
}
2019-04-05 11:54:37 +00:00
}
return oxr_verify_full_path(log, path, (uint32_t)length, name);
}
2019-10-08 14:40:40 +00:00
XrResult
oxr_verify_full_path(struct oxr_logger *log,
const char *path,
2019-04-05 11:54:37 +00:00
size_t length,
const char *name)
2019-04-05 11:54:37 +00:00
{
2019-10-08 14:40:40 +00:00
enum verify_state state = VERIFY_START;
2019-04-05 11:54:37 +00:00
bool valid = true;
if (length >= XR_MAX_PATH_LENGTH) {
char formatted_path[XR_MAX_PATH_LENGTH + 6];
snprintf(formatted_path, XR_MAX_PATH_LENGTH + 6, "%s[...]",
path);
return oxr_error(log, XR_ERROR_PATH_FORMAT_INVALID,
"(%s) is too long for a path, must be shorter "
"than %u characters",
name, XR_MAX_PATH_LENGTH);
2019-04-05 11:54:37 +00:00
}
for (uint32_t i = 0; i < length; i++) {
const char c = path[i];
switch (state) {
2019-10-08 14:40:40 +00:00
case VERIFY_START:
2019-04-05 11:54:37 +00:00
if (c != '/') {
return oxr_error(log,
XR_ERROR_PATH_FORMAT_INVALID,
"(%s) does not start with a "
"fowrward slash",
name);
}
2019-10-08 14:40:40 +00:00
state = VERIFY_SLASH;
2019-04-05 11:54:37 +00:00
break;
2019-10-08 14:40:40 +00:00
case VERIFY_SLASH:
2019-04-05 11:54:37 +00:00
switch (c) {
case '.':
// Is valid and starts the SlashDot(s) state.
2019-10-08 14:40:40 +00:00
state = VERIFY_SLASHDOTS;
2019-04-05 11:54:37 +00:00
break;
case '/':
return oxr_error(
log, XR_ERROR_PATH_FORMAT_INVALID,
"(%s) '//' is not a valid in a path", name);
default:
valid = valid_path_char(c);
2019-10-08 14:40:40 +00:00
state = VERIFY_MIDDLE;
2019-04-05 11:54:37 +00:00
}
break;
2019-10-08 14:40:40 +00:00
case VERIFY_MIDDLE:
2019-04-05 11:54:37 +00:00
switch (c) {
2019-10-08 14:40:40 +00:00
case '/': state = VERIFY_SLASH; break;
2019-04-05 11:54:37 +00:00
default:
valid = valid_path_char(c);
2019-10-08 14:40:40 +00:00
state = VERIFY_MIDDLE;
2019-04-05 11:54:37 +00:00
}
break;
2019-10-08 14:40:40 +00:00
case VERIFY_SLASHDOTS:
2019-04-05 11:54:37 +00:00
switch (c) {
case '/':
return oxr_error(
log, XR_ERROR_PATH_FORMAT_INVALID,
"(%s) '/.[.]*/' is not a valid in a path",
name);
case '.':
// It's valid, more ShashDot(s).
break;
default:
valid = valid_path_char(c);
2019-10-08 14:40:40 +00:00
state = VERIFY_MIDDLE;
2019-04-05 11:54:37 +00:00
}
break;
}
2019-09-24 15:19:52 +00:00
if (valid) {
// Can't end with slash
valid = (path[length - 1] != '/');
}
2019-04-05 11:54:37 +00:00
if (!valid) {
return oxr_error(log, XR_ERROR_PATH_FORMAT_INVALID,
"(%s) 0x%02x is not a valid character "
"at position %u",
name, c, (uint32_t)length);
}
}
switch (state) {
2019-10-08 14:40:40 +00:00
case VERIFY_START:
2019-04-05 11:54:37 +00:00
// Empty string
return oxr_error(log, XR_ERROR_PATH_FORMAT_INVALID,
"(%s) a empty string is not a valid path",
name);
2019-10-08 14:40:40 +00:00
case VERIFY_SLASH:
2019-04-05 11:54:37 +00:00
// Is this '/foo/' or '/'
if (length > 1) {
// It was '/foo/'
return XR_SUCCESS;
}
// It was '/'
return oxr_error(log, XR_ERROR_PATH_FORMAT_INVALID,
"(%s) the string '%s' is not a valid path",
name, path);
2019-10-08 14:40:40 +00:00
case VERIFY_SLASHDOTS:
2019-04-05 11:54:37 +00:00
// Does the path ends with '/..'
return oxr_error(
log, XR_ERROR_PATH_FORMAT_INVALID,
"(%s) strings ending with '/.[.]*' is not a valid", name);
2019-10-08 14:40:40 +00:00
case VERIFY_MIDDLE:
2019-04-05 11:54:37 +00:00
// '/foo/bar' okay!
return XR_SUCCESS;
default:
// We should not end up here.
return oxr_error(
log, XR_ERROR_RUNTIME_FAILURE,
"(%s) internal runtime error validating path (%s)", name,
path);
}
}
2019-03-18 05:52:32 +00:00
/*
*
* Subaction path functions.
*
*/
static XrResult
subaction_path_no_dups(struct oxr_logger *log,
struct oxr_instance *inst,
2019-10-08 14:40:40 +00:00
struct oxr_sub_paths *sub_paths,
XrPath path,
const char *variable,
uint32_t index)
{
2019-10-08 14:40:40 +00:00
assert(sub_paths);
bool duplicate = false;
if (path == XR_NULL_PATH) {
return oxr_error(log, XR_ERROR_PATH_INVALID,
"(%s[%u] == XR_NULL_PATH) not a "
"valid subaction path.",
variable, index);
2019-08-16 22:02:18 +00:00
}
if (path == inst->path_cache.user) {
2019-10-08 14:40:40 +00:00
if (sub_paths->user) {
duplicate = true;
} else {
2019-10-08 14:40:40 +00:00
sub_paths->user = true;
}
} else if (path == inst->path_cache.head) {
2019-10-08 14:40:40 +00:00
if (sub_paths->head) {
duplicate = true;
} else {
2019-10-08 14:40:40 +00:00
sub_paths->head = true;
}
} else if (path == inst->path_cache.left) {
2019-10-08 14:40:40 +00:00
if (sub_paths->left) {
duplicate = true;
} else {
2019-10-08 14:40:40 +00:00
sub_paths->left = true;
}
} else if (path == inst->path_cache.right) {
2019-10-08 14:40:40 +00:00
if (sub_paths->right) {
duplicate = true;
} else {
2019-10-08 14:40:40 +00:00
sub_paths->right = true;
}
} else if (path == inst->path_cache.gamepad) {
2019-10-08 14:40:40 +00:00
if (sub_paths->gamepad) {
duplicate = true;
} else {
2019-10-08 14:40:40 +00:00
sub_paths->gamepad = true;
}
} else {
const char *str = NULL;
size_t length = 0;
oxr_path_get_string(log, inst, path, &str, &length);
return oxr_error(log, XR_ERROR_PATH_UNSUPPORTED,
"(%s[%u] == '%s') path is not a "
"valid subaction path.",
variable, index, str);
}
if (duplicate) {
const char *str = NULL;
size_t length = 0;
oxr_path_get_string(log, inst, path, &str, &length);
return oxr_error(log, XR_ERROR_PATH_UNSUPPORTED,
"(%s[%u] == '%s') duplicate paths", variable,
index, str);
}
return XR_SUCCESS;
}
2019-10-08 14:40:40 +00:00
XrResult
oxr_verify_subaction_paths_create(struct oxr_logger *log,
struct oxr_instance *inst,
uint32_t countSubactionPaths,
const XrPath *subactionPaths,
const char *variable)
{
2019-10-08 14:40:40 +00:00
struct oxr_sub_paths sub_paths = {0};
for (uint32_t i = 0; i < countSubactionPaths; i++) {
XrPath path = subactionPaths[i];
2019-10-08 14:40:40 +00:00
XrResult ret = subaction_path_no_dups(log, inst, &sub_paths,
path, variable, i);
if (ret != XR_SUCCESS) {
return ret;
}
}
return XR_SUCCESS;
}
2019-10-08 14:40:40 +00:00
XrResult
oxr_verify_subaction_path_sync(struct oxr_logger *log,
struct oxr_instance *inst,
XrPath path,
uint32_t index)
{
if (path == XR_NULL_PATH || path == inst->path_cache.user ||
path == inst->path_cache.head || path == inst->path_cache.left ||
path == inst->path_cache.right ||
path == inst->path_cache.gamepad) {
return XR_SUCCESS;
}
const char *str = NULL;
2019-08-16 22:02:18 +00:00
size_t length = 0;
oxr_path_get_string(log, inst, path, &str, &length);
return oxr_error(log, XR_ERROR_PATH_INVALID,
"(actionSets[%i].subactionPath == '%s') path "
"is not a valid subaction path.",
index, str);
}
2019-10-08 14:40:40 +00:00
XrResult
oxr_verify_subaction_path_get(struct oxr_logger *log,
struct oxr_instance *inst,
XrPath path,
const struct oxr_sub_paths *act_sub_paths,
struct oxr_sub_paths *out_sub_paths,
const char *variable)
{
2019-10-08 14:40:40 +00:00
struct oxr_sub_paths sub_paths = {0};
if (path == XR_NULL_PATH) {
sub_paths.any = true;
} else if (path == inst->path_cache.user) {
sub_paths.user = true;
} else if (path == inst->path_cache.head) {
sub_paths.head = true;
} else if (path == inst->path_cache.left) {
sub_paths.left = true;
} else if (path == inst->path_cache.right) {
sub_paths.right = true;
} else if (path == inst->path_cache.gamepad) {
sub_paths.gamepad = true;
} else {
const char *str = NULL;
size_t length = 0;
oxr_path_get_string(log, inst, path, &str, &length);
return oxr_error(log, XR_ERROR_PATH_INVALID,
"(%s == '%s') path is not "
"a valid subaction path.",
variable, str);
}
if ((sub_paths.user && !act_sub_paths->user) ||
(sub_paths.head && !act_sub_paths->head) ||
(sub_paths.left && !act_sub_paths->left) ||
(sub_paths.right && !act_sub_paths->right) ||
(sub_paths.gamepad && !act_sub_paths->gamepad)) {
const char *str = NULL;
size_t length = 0;
oxr_path_get_string(log, inst, path, &str, &length);
return oxr_error(log, XR_ERROR_PATH_UNSUPPORTED,
"(%s == '%s') the subaction path was "
"not specified at action creation",
variable, str);
}
*out_sub_paths = sub_paths;
return XR_SUCCESS;
}
2019-03-18 05:52:32 +00:00
/*
*
* Other verification.
*
*/
XrResult
oxr_verify_view_config_type(struct oxr_logger *log,
struct oxr_instance *inst,
XrViewConfigurationType view_conf,
const char *view_conf_name)
{
// These are always valid.
if (view_conf == XR_VIEW_CONFIGURATION_TYPE_PRIMARY_MONO ||
view_conf == XR_VIEW_CONFIGURATION_TYPE_PRIMARY_STEREO) {
return XR_SUCCESS;
}
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
"(%s == 0x%08x) invalid view configuration type",
view_conf_name, view_conf);
}
2019-10-08 14:40:40 +00:00
XrResult
oxr_verify_XrSessionCreateInfo(struct oxr_logger *log,
const struct oxr_instance *inst,
const XrSessionCreateInfo *createInfo)
2019-03-18 05:52:32 +00:00
{
if (createInfo->type != XR_TYPE_SESSION_CREATE_INFO) {
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
"(createInfo->type)");
2019-03-18 05:52:32 +00:00
}
2019-08-14 22:14:59 +00:00
if (createInfo->createFlags != 0) {
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
"Non-zero session create flags");
2019-08-14 22:14:59 +00:00
}
XrResult result = oxr_system_verify_id(log, inst, createInfo->systemId);
if (result != XR_SUCCESS) {
return result;
}
2019-08-19 17:37:50 +00:00
#if defined(OXR_HAVE_KHR_opengl_enable) && defined(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) {
2019-08-19 17:37:50 +00:00
OXR_VERIFY_EXTENSION(log, inst, KHR_opengl_enable);
return oxr_verify_XrGraphicsBindingOpenGLXlibKHR(log,
opengl_xlib);
}
2019-08-19 17:37:50 +00:00
#endif // defined(OXR_HAVE_KHR_opengl_enable) && defined(XR_USE_PLATFORM_XLIB)
2019-08-19 17:37:50 +00:00
#ifdef OXR_HAVE_KHR_vulkan_enable
XrGraphicsBindingVulkanKHR const *vulkan = OXR_GET_INPUT_FROM_CHAIN(
createInfo, XR_TYPE_GRAPHICS_BINDING_VULKAN_KHR,
XrGraphicsBindingVulkanKHR);
if (vulkan != NULL) {
2019-08-19 17:37:50 +00:00
OXR_VERIFY_EXTENSION(log, inst, KHR_vulkan_enable);
return oxr_verify_XrGraphicsBindingVulkanKHR(log, vulkan);
}
2019-08-19 17:37:50 +00:00
#endif // OXR_HAVE_KHR_vulkan_enable
#if defined(OXR_HAVE_MNDX_egl_enable) && defined(XR_USE_PLATFORM_EGL)
XrGraphicsBindingEGLMNDX const *egl = OXR_GET_INPUT_FROM_CHAIN(
createInfo, XR_TYPE_GRAPHICS_BINDING_EGL_MNDX,
XrGraphicsBindingEGLMNDX);
if (egl != NULL) {
OXR_VERIFY_EXTENSION(log, inst, MNDX_egl_enable);
return oxr_verify_XrGraphicsBindingEGLMNDX(log, egl);
}
#endif // defined(OXR_HAVE_MNDX_egl_enable) && defined(XR_USE_PLATFORM_EGL_KHR)
/*
* Add any new graphics binding structs here - before the headless
* check. (order for non-headless checks not specified in standard.)
* Add a new verify function below.
* Any new addition will also need to be added to
* oxr_session_create_impl.
*/
/* We didn't recognize any graphics binding structs in the chain - our
* last hope is headless. */
2019-08-19 17:37:50 +00:00
if (inst->extensions.MND_headless) {
return XR_SUCCESS;
2019-03-18 05:52:32 +00:00
}
return oxr_error(log, XR_ERROR_GRAPHICS_DEVICE_INVALID,
"(createInfo->next) Argument chain does not contain "
"any known graphics bindings");
2019-03-18 05:52:32 +00:00
}
2019-12-03 19:08:10 +00:00
#if defined(XR_USE_PLATFORM_XLIB) && defined(XR_USE_GRAPHICS_API_OPENGL)
2019-10-08 14:40:40 +00:00
XrResult
2019-03-18 05:52:32 +00:00
oxr_verify_XrGraphicsBindingOpenGLXlibKHR(
struct oxr_logger *log, const XrGraphicsBindingOpenGLXlibKHR *next)
2019-03-18 05:52:32 +00:00
{
if (next->type != XR_TYPE_GRAPHICS_BINDING_OPENGL_XLIB_KHR) {
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
"Graphics binding has invalid type");
}
if (next->xDisplay == NULL) {
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
"xDisplay is NULL");
2019-03-18 05:52:32 +00:00
}
if (next->glxContext == NULL) {
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
"glxContext is NULL");
}
if (next->glxDrawable == NULL) {
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
"glxDrawable is NULL");
}
2019-03-18 05:52:32 +00:00
return XR_SUCCESS;
}
2019-12-03 19:08:10 +00:00
#endif // defined(XR_USE_PLATFORM_XLIB) && defined(XR_USE_GRAPHICS_API_OPENGL)
2019-03-18 05:52:32 +00:00
2019-03-18 05:52:32 +00:00
#ifdef XR_USE_GRAPHICS_API_VULKAN
2019-10-08 14:40:40 +00:00
XrResult
oxr_verify_XrGraphicsBindingVulkanKHR(struct oxr_logger *log,
const XrGraphicsBindingVulkanKHR *next)
2019-03-18 05:52:32 +00:00
{
if (next->type != XR_TYPE_GRAPHICS_BINDING_VULKAN_KHR) {
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
"Graphics binding has invalid type");
2019-03-18 05:52:32 +00:00
}
return XR_SUCCESS;
}
2019-03-18 05:52:32 +00:00
#endif
#ifdef XR_USE_PLATFORM_EGL
XrResult
oxr_verify_XrGraphicsBindingEGLMNDX(struct oxr_logger *log,
const XrGraphicsBindingEGLMNDX *next)
{
if (next->type != XR_TYPE_GRAPHICS_BINDING_EGL_MNDX) {
return oxr_error(log, XR_ERROR_VALIDATION_FAILURE,
"Graphics binding has invalid type");
}
return XR_SUCCESS;
}
#endif