2020-03-05 11:47:39 +00:00
|
|
|
// Copyright 2018-2020, Collabora, Ltd.
|
2019-09-02 21:04:13 +00:00
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief Holds binding related functions.
|
|
|
|
* @author Jakob Bornecrantz <jakob@collabora.com>
|
|
|
|
* @ingroup oxr_main
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "util/u_misc.h"
|
|
|
|
|
|
|
|
#include "xrt/xrt_compiler.h"
|
|
|
|
|
|
|
|
#include "oxr_objects.h"
|
|
|
|
#include "oxr_logger.h"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2019-12-02 23:16:37 +00:00
|
|
|
#include "oxr_binding_data.h"
|
2019-09-02 21:04:13 +00:00
|
|
|
static void
|
|
|
|
setup_paths(struct oxr_logger *log,
|
|
|
|
struct oxr_instance *inst,
|
|
|
|
struct binding_template *templ,
|
|
|
|
struct oxr_binding *binding)
|
|
|
|
{
|
|
|
|
size_t count = 0;
|
|
|
|
while (templ->paths[count] != NULL) {
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
binding->num_paths = count;
|
|
|
|
binding->paths = U_TYPED_ARRAY_CALLOC(XrPath, count);
|
|
|
|
|
|
|
|
for (size_t x = 0; x < binding->num_paths; x++) {
|
|
|
|
const char *str = templ->paths[x];
|
|
|
|
size_t len = strlen(str);
|
|
|
|
oxr_path_get_or_create(log, inst, str, len, &binding->paths[x]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
setup_inputs(struct oxr_logger *log,
|
|
|
|
struct oxr_instance *inst,
|
|
|
|
struct binding_template *templ,
|
|
|
|
struct oxr_binding *binding)
|
|
|
|
{
|
|
|
|
size_t count = 0;
|
|
|
|
while (templ->inputs[count] != 0) {
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
binding->num_inputs = count;
|
|
|
|
binding->inputs = U_TYPED_ARRAY_CALLOC(enum xrt_input_name, count);
|
|
|
|
|
|
|
|
for (size_t x = 0; x < binding->num_inputs; x++) {
|
|
|
|
binding->inputs[x] = templ->inputs[x];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
setup_outputs(struct oxr_logger *log,
|
|
|
|
struct oxr_instance *inst,
|
|
|
|
struct binding_template *templ,
|
|
|
|
struct oxr_binding *binding)
|
|
|
|
{
|
|
|
|
size_t count = 0;
|
|
|
|
while (templ->outputs[count] != 0) {
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
binding->num_outputs = count;
|
|
|
|
binding->outputs = U_TYPED_ARRAY_CALLOC(enum xrt_output_name, count);
|
|
|
|
|
|
|
|
for (size_t x = 0; x < binding->num_outputs; x++) {
|
|
|
|
binding->outputs[x] = templ->outputs[x];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
is_valid_interaction_profile(struct oxr_instance *inst, XrPath path)
|
|
|
|
{
|
|
|
|
return inst->path_cache.khr_simple_controller == path ||
|
|
|
|
inst->path_cache.google_daydream_controller == path ||
|
|
|
|
inst->path_cache.htc_vive_controller == path ||
|
|
|
|
inst->path_cache.htc_vive_pro == path ||
|
|
|
|
inst->path_cache.microsoft_motion_controller == path ||
|
|
|
|
inst->path_cache.microsoft_xbox_controller == path ||
|
|
|
|
inst->path_cache.oculus_go_controller == path ||
|
|
|
|
inst->path_cache.oculus_touch_controller == path ||
|
2019-09-03 14:25:44 +00:00
|
|
|
inst->path_cache.valve_index_controller == path ||
|
|
|
|
inst->path_cache.mnd_ball_on_stick_controller == path;
|
2019-09-02 21:04:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
interaction_profile_find(struct oxr_logger *log,
|
|
|
|
struct oxr_instance *inst,
|
|
|
|
XrPath path,
|
|
|
|
struct oxr_interaction_profile **out_p)
|
|
|
|
{
|
|
|
|
for (size_t x = 0; x < inst->num_profiles; x++) {
|
|
|
|
struct oxr_interaction_profile *p = inst->profiles[x];
|
|
|
|
if (p->path != path) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_p = p;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
interaction_profile_find_or_create(struct oxr_logger *log,
|
|
|
|
struct oxr_instance *inst,
|
|
|
|
XrPath path,
|
|
|
|
struct oxr_interaction_profile **out_p)
|
|
|
|
{
|
|
|
|
if (interaction_profile_find(log, inst, path, out_p)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct profile_template *templ = NULL;
|
|
|
|
for (size_t x = 0; x < ARRAY_SIZE(profiles); x++) {
|
|
|
|
templ = &profiles[x];
|
|
|
|
XrPath t_path = XR_NULL_PATH;
|
|
|
|
|
|
|
|
oxr_path_get_or_create(log, inst, templ->path,
|
|
|
|
strlen(templ->path), &t_path);
|
|
|
|
if (t_path == path) {
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
templ = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (templ == NULL) {
|
|
|
|
*out_p = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct oxr_interaction_profile *p =
|
|
|
|
U_TYPED_CALLOC(struct oxr_interaction_profile);
|
|
|
|
|
|
|
|
p->num_bindings = templ->num_bindings;
|
|
|
|
p->bindings = U_TYPED_ARRAY_CALLOC(struct oxr_binding, p->num_bindings);
|
|
|
|
p->path = path;
|
|
|
|
|
|
|
|
for (size_t x = 0; x < templ->num_bindings; x++) {
|
|
|
|
struct binding_template *t = &templ->bindings[x];
|
|
|
|
struct oxr_binding *b = &p->bindings[x];
|
|
|
|
|
|
|
|
b->sub_path = t->sub_path;
|
|
|
|
setup_paths(log, inst, t, b);
|
|
|
|
setup_inputs(log, inst, t, b);
|
|
|
|
setup_outputs(log, inst, t, b);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add to the list of currently created interaction profiles.
|
2019-12-02 23:15:46 +00:00
|
|
|
U_ARRAY_REALLOC_OR_FREE(inst->profiles,
|
|
|
|
struct oxr_interaction_profile *,
|
|
|
|
(inst->num_profiles + 1));
|
2019-09-02 21:04:13 +00:00
|
|
|
inst->profiles[inst->num_profiles++] = p;
|
|
|
|
|
|
|
|
*out_p = p;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-11-13 20:53:16 +00:00
|
|
|
static void
|
|
|
|
reset_binding_keys(struct oxr_binding *binding)
|
|
|
|
{
|
|
|
|
free(binding->keys);
|
2020-05-04 15:22:15 +00:00
|
|
|
free(binding->preferred_binding_path_index);
|
2019-11-13 20:53:16 +00:00
|
|
|
binding->keys = NULL;
|
2020-05-04 15:22:15 +00:00
|
|
|
binding->preferred_binding_path_index = NULL;
|
2019-11-13 20:53:16 +00:00
|
|
|
binding->num_keys = 0;
|
|
|
|
}
|
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
static void
|
|
|
|
reset_all_keys(struct oxr_binding *bindings, size_t num_bindings)
|
|
|
|
{
|
|
|
|
for (size_t x = 0; x < num_bindings; x++) {
|
2019-11-13 20:53:16 +00:00
|
|
|
reset_binding_keys(&bindings[x]);
|
2019-09-02 21:04:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
add_key_to_matching_bindings(struct oxr_binding *bindings,
|
|
|
|
size_t num_bindings,
|
|
|
|
XrPath path,
|
2020-05-04 15:22:15 +00:00
|
|
|
uint32_t key)
|
2019-09-02 21:04:13 +00:00
|
|
|
{
|
|
|
|
for (size_t x = 0; x < num_bindings; x++) {
|
|
|
|
struct oxr_binding *b = &bindings[x];
|
|
|
|
|
|
|
|
bool found = false;
|
2020-05-04 15:22:15 +00:00
|
|
|
uint32_t preferred_path_index;
|
2019-09-02 21:04:13 +00:00
|
|
|
for (size_t y = 0; y < b->num_paths; y++) {
|
|
|
|
if (b->paths[y] == path) {
|
|
|
|
found = true;
|
2020-05-04 15:22:15 +00:00
|
|
|
preferred_path_index = y;
|
2019-09-02 21:04:13 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!found) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-12-02 23:16:06 +00:00
|
|
|
U_ARRAY_REALLOC_OR_FREE(b->keys, uint32_t, (b->num_keys + 1));
|
2020-05-04 15:22:15 +00:00
|
|
|
U_ARRAY_REALLOC_OR_FREE(b->preferred_binding_path_index,
|
|
|
|
uint32_t, (b->num_keys + 1));
|
|
|
|
b->preferred_binding_path_index[b->num_keys] =
|
|
|
|
preferred_path_index;
|
2019-09-02 21:04:13 +00:00
|
|
|
b->keys[b->num_keys++] = key;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* 'Exported' functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
oxr_find_profile_for_device(struct oxr_logger *log,
|
|
|
|
struct oxr_instance *inst,
|
|
|
|
struct xrt_device *xdev,
|
|
|
|
struct oxr_interaction_profile **out_p)
|
|
|
|
{
|
|
|
|
if (xdev == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
enum xrt_device_name name = xdev->name;
|
|
|
|
|
|
|
|
//! @todo A lot more clever selecting the profile here.
|
|
|
|
switch (name) {
|
|
|
|
case XRT_DEVICE_HYDRA:
|
|
|
|
// clang-format off
|
|
|
|
interaction_profile_find(log, inst, inst->path_cache.khr_simple_controller, out_p);
|
|
|
|
// clang-format on
|
|
|
|
return;
|
|
|
|
case XRT_DEVICE_PSMV:
|
|
|
|
// clang-format off
|
|
|
|
interaction_profile_find(log, inst, inst->path_cache.khr_simple_controller, out_p);
|
2019-09-03 14:25:44 +00:00
|
|
|
interaction_profile_find(log, inst, inst->path_cache.mnd_ball_on_stick_controller, out_p);
|
2019-09-02 21:04:13 +00:00
|
|
|
// clang-format on
|
|
|
|
return;
|
2020-03-05 11:47:39 +00:00
|
|
|
case XRT_DEVICE_DAYDREAM:
|
|
|
|
interaction_profile_find(
|
|
|
|
log, inst, inst->path_cache.khr_simple_controller, out_p);
|
|
|
|
return;
|
2020-04-14 21:07:49 +00:00
|
|
|
case XRT_DEVICE_INDEX_CONTROLLER:
|
|
|
|
// clang-format off
|
|
|
|
interaction_profile_find(log, inst, inst->path_cache.khr_simple_controller, out_p);
|
|
|
|
interaction_profile_find(log, inst, inst->path_cache.valve_index_controller, out_p);
|
|
|
|
// clang-format on
|
|
|
|
return;
|
|
|
|
case XRT_DEVICE_VIVE_WAND:
|
|
|
|
// clang-format off
|
|
|
|
interaction_profile_find(log, inst, inst->path_cache.khr_simple_controller, out_p);
|
|
|
|
interaction_profile_find(log, inst, inst->path_cache.htc_vive_controller, out_p);
|
|
|
|
// clang-format on
|
|
|
|
return;
|
2019-09-02 21:04:13 +00:00
|
|
|
default: return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
oxr_binding_find_bindings_from_key(struct oxr_logger *log,
|
|
|
|
struct oxr_interaction_profile *p,
|
|
|
|
uint32_t key,
|
|
|
|
struct oxr_binding *bindings[32],
|
|
|
|
size_t *num_bindings)
|
|
|
|
{
|
|
|
|
if (p == NULL) {
|
|
|
|
*num_bindings = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! @todo This function should be a two call function, or handle more
|
|
|
|
//! then 32 bindings.
|
|
|
|
size_t num = 0;
|
|
|
|
|
|
|
|
for (size_t y = 0; y < p->num_bindings; y++) {
|
|
|
|
struct oxr_binding *b = &p->bindings[y];
|
|
|
|
|
|
|
|
for (size_t z = 0; z < b->num_keys; z++) {
|
|
|
|
if (b->keys[z] == key) {
|
|
|
|
bindings[num++] = b;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (num >= 32) {
|
|
|
|
*num_bindings = num;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*num_bindings = num;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
oxr_binding_destroy_all(struct oxr_logger *log, struct oxr_instance *inst)
|
|
|
|
{
|
|
|
|
for (size_t x = 0; x < inst->num_profiles; x++) {
|
|
|
|
struct oxr_interaction_profile *p = inst->profiles[x];
|
|
|
|
|
|
|
|
for (size_t y = 0; y < p->num_bindings; y++) {
|
|
|
|
struct oxr_binding *b = &p->bindings[y];
|
2019-11-13 20:53:16 +00:00
|
|
|
|
|
|
|
reset_binding_keys(b);
|
2019-09-02 21:04:13 +00:00
|
|
|
free(b->paths);
|
|
|
|
free(b->inputs);
|
|
|
|
free(b->outputs);
|
|
|
|
|
|
|
|
b->paths = NULL;
|
|
|
|
b->inputs = NULL;
|
|
|
|
b->outputs = NULL;
|
|
|
|
b->num_paths = 0;
|
|
|
|
b->num_inputs = 0;
|
|
|
|
b->num_outputs = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
free(p->bindings);
|
|
|
|
p->bindings = NULL;
|
|
|
|
p->num_bindings = 0;
|
2019-11-13 20:53:16 +00:00
|
|
|
|
|
|
|
free(p);
|
2019-09-02 21:04:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
free(inst->profiles);
|
|
|
|
inst->profiles = NULL;
|
|
|
|
inst->num_profiles = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Client functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_action_suggest_interaction_profile_bindings(
|
|
|
|
struct oxr_logger *log,
|
|
|
|
struct oxr_instance *inst,
|
|
|
|
const XrInteractionProfileSuggestedBinding *suggestedBindings)
|
|
|
|
{
|
|
|
|
struct oxr_interaction_profile *p = NULL;
|
|
|
|
XrPath path = suggestedBindings->interactionProfile;
|
|
|
|
const char *str;
|
|
|
|
size_t length;
|
|
|
|
|
|
|
|
// Check if this profile is valid.
|
|
|
|
if (!is_valid_interaction_profile(inst, path)) {
|
|
|
|
oxr_path_get_string(log, inst, path, &str, &length);
|
|
|
|
|
|
|
|
return oxr_error(log, XR_ERROR_PATH_UNSUPPORTED,
|
|
|
|
"(suggestedBindings->interactionProfile) "
|
|
|
|
"non-supported profile '%s'",
|
|
|
|
str);
|
|
|
|
}
|
|
|
|
|
|
|
|
interaction_profile_find_or_create(log, inst, path, &p);
|
|
|
|
|
|
|
|
// Valid path, but not used.
|
|
|
|
//! @todo Still needs to validate the paths.
|
|
|
|
if (p == NULL) {
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct oxr_binding *bindings = p->bindings;
|
|
|
|
size_t num_bindings = p->num_bindings;
|
|
|
|
|
|
|
|
//! @todo Validate keys **FIRST** then reset.
|
|
|
|
reset_all_keys(bindings, num_bindings);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < suggestedBindings->countSuggestedBindings; i++) {
|
|
|
|
const XrActionSuggestedBinding *s =
|
|
|
|
&suggestedBindings->suggestedBindings[i];
|
2020-04-28 22:51:39 +00:00
|
|
|
struct oxr_action *act =
|
|
|
|
XRT_CAST_OXR_HANDLE_TO_PTR(struct oxr_action *, s->action);
|
2019-09-02 21:04:13 +00:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
oxr_path_get_string(log, inst, s->binding, &str, &length);
|
|
|
|
fprintf(stderr, "\t\t%s %i -> %s\n", act->name, act->key, str);
|
|
|
|
#endif
|
|
|
|
|
2020-05-04 15:22:15 +00:00
|
|
|
add_key_to_matching_bindings(bindings, num_bindings, s->binding,
|
|
|
|
act->key);
|
2019-09-02 21:04:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_action_get_current_interaction_profile(
|
|
|
|
struct oxr_logger *log,
|
|
|
|
struct oxr_session *sess,
|
|
|
|
XrPath topLevelUserPath,
|
|
|
|
XrInteractionProfileState *interactionProfile)
|
|
|
|
{
|
|
|
|
struct oxr_instance *inst = sess->sys->inst;
|
|
|
|
|
|
|
|
if (!sess->actionsAttached) {
|
|
|
|
return oxr_error(log, XR_ERROR_ACTIONSET_NOT_ATTACHED,
|
|
|
|
" xrAttachSessionActionSets has not been "
|
|
|
|
"called on this session.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (topLevelUserPath == inst->path_cache.head) {
|
2019-12-02 23:05:50 +00:00
|
|
|
interactionProfile->interactionProfile = sess->head;
|
2019-09-02 21:04:13 +00:00
|
|
|
} else if (topLevelUserPath == inst->path_cache.left) {
|
2019-12-02 23:05:50 +00:00
|
|
|
interactionProfile->interactionProfile = sess->left;
|
2019-09-02 21:04:13 +00:00
|
|
|
} else if (topLevelUserPath == inst->path_cache.right) {
|
2019-12-02 23:05:50 +00:00
|
|
|
interactionProfile->interactionProfile = sess->right;
|
2019-09-02 21:04:13 +00:00
|
|
|
} else if (topLevelUserPath == inst->path_cache.gamepad) {
|
2019-12-02 23:05:50 +00:00
|
|
|
interactionProfile->interactionProfile = sess->gamepad;
|
2019-09-02 21:04:13 +00:00
|
|
|
} else {
|
|
|
|
return oxr_error(log, XR_ERROR_HANDLE_INVALID,
|
|
|
|
" not implemented");
|
|
|
|
}
|
2019-12-02 23:05:50 +00:00
|
|
|
return XR_SUCCESS;
|
2019-09-02 21:04:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_action_get_input_source_localized_name(
|
|
|
|
struct oxr_logger *log,
|
|
|
|
struct oxr_session *sess,
|
|
|
|
const XrInputSourceLocalizedNameGetInfo *getInfo,
|
|
|
|
uint32_t bufferCapacityInput,
|
|
|
|
uint32_t *bufferCountOutput,
|
|
|
|
char *buffer)
|
|
|
|
{
|
|
|
|
//! @todo Implement
|
|
|
|
return oxr_error(log, XR_ERROR_HANDLE_INVALID, " not implemented");
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_action_enumerate_bound_sources(struct oxr_logger *log,
|
|
|
|
struct oxr_session *sess,
|
|
|
|
uint64_t key,
|
|
|
|
uint32_t sourceCapacityInput,
|
|
|
|
uint32_t *sourceCountOutput,
|
|
|
|
XrPath *sources)
|
|
|
|
{
|
|
|
|
//! @todo Implement
|
|
|
|
return oxr_error(log, XR_ERROR_HANDLE_INVALID, " not implemented");
|
|
|
|
}
|