2019-05-07 12:47:18 +00:00
|
|
|
// Copyright 2018-2019, Collabora, Ltd.
|
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief Holds input related functions.
|
|
|
|
* @author Jakob Bornecrantz <jakob@collabora.com>
|
|
|
|
* @ingroup oxr_main
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
#include "util/u_debug.h"
|
|
|
|
#include "util/u_time.h"
|
|
|
|
#include "util/u_misc.h"
|
|
|
|
|
|
|
|
#include "xrt/xrt_compiler.h"
|
|
|
|
|
|
|
|
#include "oxr_objects.h"
|
|
|
|
#include "oxr_logger.h"
|
|
|
|
#include "oxr_handle.h"
|
|
|
|
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Pre declare functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void
|
|
|
|
oxr_session_get_source_set(struct oxr_session* sess,
|
|
|
|
XrActionSet actionSet,
|
|
|
|
struct oxr_source_set** src_set,
|
|
|
|
struct oxr_action_set** act_set);
|
|
|
|
|
|
|
|
static void
|
|
|
|
oxr_session_get_source(struct oxr_session* sess,
|
|
|
|
uint32_t act_key,
|
|
|
|
struct oxr_source** out_src);
|
|
|
|
|
|
|
|
static void
|
|
|
|
oxr_source_cache_update(struct oxr_logger* log,
|
|
|
|
struct oxr_session* sess,
|
|
|
|
struct oxr_action* act,
|
|
|
|
struct oxr_source_cache* cache,
|
|
|
|
int64_t time,
|
|
|
|
bool select);
|
|
|
|
|
|
|
|
static void
|
|
|
|
oxr_source_update(struct oxr_logger* log,
|
|
|
|
struct oxr_session* sess,
|
|
|
|
struct oxr_action* act,
|
|
|
|
int64_t time,
|
|
|
|
struct oxr_sub_paths sub_paths);
|
|
|
|
|
|
|
|
static void
|
|
|
|
oxr_source_bind_inputs(struct oxr_logger* log,
|
2019-09-02 21:04:13 +00:00
|
|
|
struct oxr_sink_logger* slog,
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_session* sess,
|
|
|
|
struct oxr_action* act,
|
|
|
|
struct oxr_source_cache* cache,
|
2019-09-02 21:04:13 +00:00
|
|
|
struct oxr_interaction_profile* profile,
|
|
|
|
enum oxr_sub_action_path sub_path);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
static XrResult
|
|
|
|
oxr_source_destroy_cb(struct oxr_logger* log, struct oxr_handle_base* hb);
|
|
|
|
|
|
|
|
static XrResult
|
|
|
|
oxr_source_create(struct oxr_logger* log,
|
2019-09-02 21:04:13 +00:00
|
|
|
struct oxr_source_set* src_set,
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_action* act,
|
2019-09-02 21:04:13 +00:00
|
|
|
struct oxr_interaction_profile* head,
|
|
|
|
struct oxr_interaction_profile* left,
|
|
|
|
struct oxr_interaction_profile* right,
|
|
|
|
struct oxr_interaction_profile* gamepad);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Action set functions
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static XrResult
|
|
|
|
oxr_action_set_destroy_cb(struct oxr_logger* log, struct oxr_handle_base* hb)
|
|
|
|
{
|
2019-05-07 12:47:18 +00:00
|
|
|
//! @todo Move to oxr_objects.h
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_action_set* act_set = (struct oxr_action_set*)hb;
|
|
|
|
|
|
|
|
free(act_set);
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_action_set_create(struct oxr_logger* log,
|
2019-07-13 16:17:57 +00:00
|
|
|
struct oxr_instance* inst,
|
2019-05-07 12:47:18 +00:00
|
|
|
const XrActionSetCreateInfo* createInfo,
|
|
|
|
struct oxr_action_set** out_act_set)
|
|
|
|
{
|
2019-05-07 12:47:18 +00:00
|
|
|
// Mod music for all!
|
2019-09-02 19:40:06 +00:00
|
|
|
static uint32_t key_gen = 1;
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
//! @todo Implement more fully.
|
|
|
|
struct oxr_action_set* act_set = NULL;
|
|
|
|
OXR_ALLOCATE_HANDLE_OR_RETURN(log, act_set, OXR_XR_DEBUG_ACTIONSET,
|
2019-07-13 16:17:57 +00:00
|
|
|
oxr_action_set_destroy_cb, &inst->handle);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
act_set->key = key_gen++;
|
2019-09-02 21:04:13 +00:00
|
|
|
|
2019-07-13 16:17:57 +00:00
|
|
|
act_set->inst = inst;
|
2019-09-02 21:04:13 +00:00
|
|
|
strncpy(act_set->name, createInfo->actionSetName,
|
|
|
|
sizeof(act_set->name));
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
*out_act_set = act_set;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Action functions
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static XrResult
|
|
|
|
oxr_action_destroy_cb(struct oxr_logger* log, struct oxr_handle_base* hb)
|
|
|
|
{
|
2019-05-07 12:47:18 +00:00
|
|
|
//! @todo Move to oxr_objects.h
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_action* act = (struct oxr_action*)hb;
|
|
|
|
|
|
|
|
free(act);
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_action_create(struct oxr_logger* log,
|
|
|
|
struct oxr_action_set* act_set,
|
|
|
|
const XrActionCreateInfo* createInfo,
|
|
|
|
struct oxr_action** out_act)
|
|
|
|
{
|
2019-07-13 16:17:57 +00:00
|
|
|
struct oxr_instance* inst = act_set->inst;
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_sub_paths sub_paths = {0};
|
|
|
|
|
|
|
|
// Mod music for all!
|
2019-09-02 19:40:06 +00:00
|
|
|
static uint32_t key_gen = 1;
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
oxr_classify_sub_action_paths(log, inst,
|
|
|
|
createInfo->countSubactionPaths,
|
|
|
|
createInfo->subactionPaths, &sub_paths);
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_action* act = NULL;
|
|
|
|
OXR_ALLOCATE_HANDLE_OR_RETURN(log, act, OXR_XR_DEBUG_ACTION,
|
|
|
|
oxr_action_destroy_cb, &act_set->handle);
|
2019-05-07 12:47:18 +00:00
|
|
|
act->key = key_gen++;
|
2019-05-07 12:47:18 +00:00
|
|
|
act->act_set = act_set;
|
2019-05-07 12:47:18 +00:00
|
|
|
act->sub_paths = sub_paths;
|
|
|
|
act->action_type = createInfo->actionType;
|
|
|
|
|
|
|
|
strncpy(act->name, createInfo->actionName, sizeof(act->name));
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
*out_act = act;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* "Exproted" helper functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
oxr_classify_sub_action_paths(struct oxr_logger* log,
|
|
|
|
struct oxr_instance* inst,
|
|
|
|
uint32_t num_subaction_paths,
|
|
|
|
const XrPath* subaction_paths,
|
|
|
|
struct oxr_sub_paths* sub_paths)
|
|
|
|
{
|
|
|
|
const char* str = NULL;
|
|
|
|
size_t length = 0;
|
|
|
|
|
|
|
|
// Reset the sub_paths completely.
|
2019-06-18 20:32:08 +00:00
|
|
|
U_ZERO(sub_paths);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
if (num_subaction_paths == 0) {
|
|
|
|
sub_paths->any = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < num_subaction_paths; i++) {
|
|
|
|
XrPath path = subaction_paths[i];
|
|
|
|
|
|
|
|
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 {
|
|
|
|
oxr_path_get_string(log, inst, path, &str, &length);
|
|
|
|
|
|
|
|
oxr_warn(log, " unrecognized sub action path '%s'",
|
|
|
|
str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_source_get_pose_input(struct oxr_logger* log,
|
|
|
|
struct oxr_session* sess,
|
|
|
|
uint32_t act_key,
|
|
|
|
const struct oxr_sub_paths* sub_paths,
|
|
|
|
struct oxr_source_input** out_input)
|
|
|
|
{
|
2019-07-24 23:52:07 +00:00
|
|
|
struct oxr_source* src = NULL;
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
oxr_session_get_source(sess, act_key, &src);
|
|
|
|
|
|
|
|
if (src == NULL) {
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Priority of inputs.
|
|
|
|
if (src->head.current.active && (sub_paths->head || sub_paths->any)) {
|
|
|
|
*out_input = src->head.inputs;
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
if (src->left.current.active && (sub_paths->left || sub_paths->any)) {
|
|
|
|
*out_input = src->left.inputs;
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
if (src->right.current.active && (sub_paths->right || sub_paths->any)) {
|
|
|
|
*out_input = src->right.inputs;
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
if (src->gamepad.current.active &&
|
|
|
|
(sub_paths->gamepad || sub_paths->any)) {
|
|
|
|
*out_input = src->gamepad.inputs;
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
if (src->user.current.active && (sub_paths->user || sub_paths->any)) {
|
|
|
|
*out_input = src->user.inputs;
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
2019-09-02 21:04:13 +00:00
|
|
|
* Not so hack functions.
|
2019-05-07 12:47:18 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
static bool
|
|
|
|
do_inputs(struct oxr_binding* bind,
|
|
|
|
struct xrt_device* xdev,
|
|
|
|
struct oxr_source_input inputs[16],
|
|
|
|
uint32_t* num_inputs)
|
2019-05-07 12:47:18 +00:00
|
|
|
{
|
|
|
|
struct xrt_input* input = NULL;
|
2019-09-02 21:04:13 +00:00
|
|
|
bool found = false;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < bind->num_inputs; i++) {
|
|
|
|
if (oxr_xdev_find_input(xdev, bind->inputs[i], &input)) {
|
|
|
|
uint32_t index = (*num_inputs)++;
|
|
|
|
inputs[index].input = input;
|
|
|
|
inputs[index].xdev = xdev;
|
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
do_outputs(struct oxr_binding* bind,
|
|
|
|
struct xrt_device* xdev,
|
|
|
|
struct oxr_source_output outputs[16],
|
|
|
|
uint32_t* num_outputs)
|
|
|
|
{
|
2019-05-07 12:47:18 +00:00
|
|
|
struct xrt_output* output = NULL;
|
2019-09-02 21:04:13 +00:00
|
|
|
bool found = false;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < bind->num_outputs; i++) {
|
|
|
|
if (oxr_xdev_find_output(xdev, bind->outputs[i], &output)) {
|
|
|
|
uint32_t index = (*num_outputs)++;
|
|
|
|
outputs[index].name = output->name;
|
|
|
|
outputs[index].xdev = xdev;
|
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
do_io_bindings(struct oxr_binding* b,
|
|
|
|
struct oxr_action* act,
|
|
|
|
struct xrt_device* xdev,
|
|
|
|
struct oxr_source_input inputs[16],
|
|
|
|
uint32_t* num_inputs,
|
|
|
|
struct oxr_source_output outputs[16],
|
|
|
|
uint32_t* num_outputs)
|
|
|
|
{
|
|
|
|
bool found = false;
|
|
|
|
|
|
|
|
if (act->action_type == XR_ACTION_TYPE_VIBRATION_OUTPUT) {
|
|
|
|
found |= do_outputs(b, xdev, outputs, num_outputs);
|
|
|
|
} else {
|
|
|
|
found |= do_inputs(b, xdev, inputs, num_inputs);
|
|
|
|
}
|
|
|
|
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
get_binding(struct oxr_logger* log,
|
|
|
|
struct oxr_sink_logger* slog,
|
|
|
|
struct oxr_session* sess,
|
|
|
|
struct oxr_action* act,
|
|
|
|
struct oxr_interaction_profile* profile,
|
|
|
|
enum oxr_sub_action_path sub_path,
|
|
|
|
struct oxr_source_input inputs[16],
|
|
|
|
uint32_t* num_inputs,
|
|
|
|
struct oxr_source_output outputs[16],
|
|
|
|
uint32_t* num_outputs)
|
|
|
|
{
|
2019-05-07 12:47:18 +00:00
|
|
|
struct xrt_device* xdev = NULL;
|
2019-09-02 21:04:13 +00:00
|
|
|
struct oxr_binding* bindings[32];
|
|
|
|
const char* profile_str;
|
|
|
|
const char* user_path_str;
|
|
|
|
size_t length;
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
//! @todo This probably falls on its head if the application doesn't use
|
|
|
|
//! sub action paths.
|
2019-05-07 12:47:18 +00:00
|
|
|
switch (sub_path) {
|
2019-09-02 21:04:13 +00:00
|
|
|
case OXR_SUB_ACTION_PATH_USER:
|
|
|
|
user_path_str = "/user";
|
|
|
|
xdev = NULL;
|
|
|
|
break;
|
|
|
|
case OXR_SUB_ACTION_PATH_HEAD:
|
|
|
|
user_path_str = "/user/head";
|
|
|
|
xdev = sess->sys->head;
|
|
|
|
break;
|
|
|
|
case OXR_SUB_ACTION_PATH_LEFT:
|
|
|
|
user_path_str = "/user/hand/left";
|
|
|
|
xdev = sess->sys->left;
|
|
|
|
break;
|
|
|
|
case OXR_SUB_ACTION_PATH_RIGHT:
|
|
|
|
user_path_str = "/user/hand/right";
|
|
|
|
xdev = sess->sys->right;
|
|
|
|
break;
|
|
|
|
case OXR_SUB_ACTION_PATH_GAMEPAD:
|
|
|
|
user_path_str = "/user/hand/gamepad";
|
|
|
|
xdev = NULL;
|
|
|
|
break;
|
2019-05-07 12:47:18 +00:00
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
oxr_slog(slog, "\tFor: %s\n", user_path_str);
|
|
|
|
|
|
|
|
if (xdev == NULL) {
|
|
|
|
oxr_slog(slog, "\t\tNo xdev!\n");
|
|
|
|
return;
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
if (profile == NULL) {
|
|
|
|
oxr_slog(slog, "\t\tNo profile!\n");
|
|
|
|
return;
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
oxr_path_get_string(log, sess->sys->inst, profile->path, &profile_str,
|
|
|
|
&length);
|
|
|
|
|
|
|
|
oxr_slog(slog, "\t\tProfile: %s\n", profile_str);
|
|
|
|
|
|
|
|
size_t num = 0;
|
|
|
|
oxr_binding_find_bindings_from_key(log, profile, act->key, bindings,
|
|
|
|
&num);
|
|
|
|
if (num == 0) {
|
|
|
|
oxr_slog(slog, "\t\tNo bindings\n");
|
|
|
|
return;
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
2019-08-16 15:52:15 +00:00
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
for (size_t i = 0; i < num; i++) {
|
|
|
|
const char* str = NULL;
|
|
|
|
struct oxr_binding* b = bindings[i];
|
|
|
|
|
|
|
|
// Just pick the first path.
|
|
|
|
oxr_path_get_string(log, sess->sys->inst, b->paths[0], &str,
|
|
|
|
&length);
|
|
|
|
oxr_slog(slog, "\t\t\tBinding: %s\n", str);
|
|
|
|
|
|
|
|
if (b->sub_path != sub_path) {
|
|
|
|
oxr_slog(slog, "\t\t\t\tRejected! (SUB PATH)\n");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool found = do_io_bindings(b, act, xdev, inputs, num_inputs,
|
|
|
|
outputs, num_outputs);
|
|
|
|
|
|
|
|
if (found) {
|
|
|
|
oxr_slog(slog, "\t\t\t\tBound!\n");
|
|
|
|
} else {
|
|
|
|
oxr_slog(slog, "\t\t\t\tRejected! (NO XDEV MAPPING)\n");
|
|
|
|
}
|
|
|
|
}
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Source set functions
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static XrResult
|
|
|
|
oxr_source_set_destroy_cb(struct oxr_logger* log, struct oxr_handle_base* hb)
|
|
|
|
{
|
|
|
|
//! @todo Move to oxr_objects.h
|
|
|
|
struct oxr_source_set* src_set = (struct oxr_source_set*)hb;
|
|
|
|
|
|
|
|
free(src_set);
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static XrResult
|
|
|
|
oxr_source_set_create(struct oxr_logger* log,
|
2019-07-13 16:17:57 +00:00
|
|
|
struct oxr_session* sess,
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_action_set* act_set,
|
|
|
|
struct oxr_source_set** out_src_set)
|
|
|
|
{
|
|
|
|
struct oxr_source_set* src_set = NULL;
|
|
|
|
OXR_ALLOCATE_HANDLE_OR_RETURN(log, src_set, OXR_XR_DEBUG_SOURCESET,
|
|
|
|
oxr_source_set_destroy_cb, &sess->handle);
|
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
src_set->sess = sess;
|
2019-05-07 12:47:18 +00:00
|
|
|
u_hashmap_int_insert(sess->act_sets, act_set->key, src_set);
|
|
|
|
|
|
|
|
*out_src_set = src_set;
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Source functions
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static XrResult
|
|
|
|
oxr_source_destroy_cb(struct oxr_logger* log, struct oxr_handle_base* hb)
|
|
|
|
{
|
|
|
|
//! @todo Move to oxr_objects.h
|
|
|
|
struct oxr_source* src = (struct oxr_source*)hb;
|
|
|
|
|
|
|
|
free(src->user.inputs);
|
|
|
|
free(src->user.outputs);
|
|
|
|
free(src->head.inputs);
|
|
|
|
free(src->head.outputs);
|
|
|
|
free(src->left.inputs);
|
|
|
|
free(src->left.outputs);
|
|
|
|
free(src->right.inputs);
|
|
|
|
free(src->right.outputs);
|
|
|
|
free(src->gamepad.inputs);
|
|
|
|
free(src->gamepad.outputs);
|
|
|
|
free(src);
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static XrResult
|
|
|
|
oxr_source_create(struct oxr_logger* log,
|
2019-09-02 21:04:13 +00:00
|
|
|
struct oxr_source_set* src_set,
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_action* act,
|
2019-09-02 21:04:13 +00:00
|
|
|
struct oxr_interaction_profile* head,
|
|
|
|
struct oxr_interaction_profile* left,
|
|
|
|
struct oxr_interaction_profile* right,
|
|
|
|
struct oxr_interaction_profile* gamepad)
|
2019-05-07 12:47:18 +00:00
|
|
|
{
|
2019-09-02 21:04:13 +00:00
|
|
|
struct oxr_session* sess = src_set->sess;
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_source* src = NULL;
|
2019-09-02 21:04:13 +00:00
|
|
|
struct oxr_sink_logger slog = {0};
|
2019-05-07 12:47:18 +00:00
|
|
|
OXR_ALLOCATE_HANDLE_OR_RETURN(log, src, OXR_XR_DEBUG_SOURCE,
|
2019-09-02 21:04:13 +00:00
|
|
|
oxr_source_destroy_cb, &src_set->handle);
|
|
|
|
|
|
|
|
u_hashmap_int_insert(src_set->sess->sources, act->key, src);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
// Start logging into a single buffer.
|
|
|
|
oxr_slog(&slog, ": Binding %s/%s\n", act->act_set->name, act->name);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
if (act->sub_paths.user || act->sub_paths.any) {
|
2019-09-02 21:04:13 +00:00
|
|
|
#if 0
|
|
|
|
oxr_source_bind_inputs(log, slog, sess, act, &src->user, user,
|
2019-05-07 12:47:18 +00:00
|
|
|
OXR_SUB_ACTION_PATH_USER);
|
2019-09-02 21:04:13 +00:00
|
|
|
#endif
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (act->sub_paths.head || act->sub_paths.any) {
|
2019-09-02 21:04:13 +00:00
|
|
|
oxr_source_bind_inputs(log, &slog, sess, act, &src->head, head,
|
2019-05-07 12:47:18 +00:00
|
|
|
OXR_SUB_ACTION_PATH_HEAD);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (act->sub_paths.left || act->sub_paths.any) {
|
2019-09-02 21:04:13 +00:00
|
|
|
oxr_source_bind_inputs(log, &slog, sess, act, &src->left, left,
|
2019-05-07 12:47:18 +00:00
|
|
|
OXR_SUB_ACTION_PATH_LEFT);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (act->sub_paths.right || act->sub_paths.any) {
|
2019-09-02 21:04:13 +00:00
|
|
|
oxr_source_bind_inputs(log, &slog, sess, act, &src->right,
|
|
|
|
right, OXR_SUB_ACTION_PATH_RIGHT);
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (act->sub_paths.gamepad || act->sub_paths.any) {
|
2019-09-02 21:04:13 +00:00
|
|
|
oxr_source_bind_inputs(log, &slog, sess, act, &src->gamepad,
|
|
|
|
gamepad, OXR_SUB_ACTION_PATH_GAMEPAD);
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
oxr_slog(&slog, "\tDone");
|
|
|
|
|
|
|
|
// Also frees all data.
|
|
|
|
oxr_log_slog(log, &slog);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
oxr_source_cache_stop_output(struct oxr_logger* log,
|
|
|
|
struct oxr_session* sess,
|
|
|
|
struct oxr_source_cache* cache)
|
|
|
|
{
|
|
|
|
// Set this as stopped.
|
|
|
|
cache->stop_output_time = 0;
|
|
|
|
|
|
|
|
union xrt_output_value value = {0};
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < cache->num_outputs; i++) {
|
|
|
|
struct oxr_source_output* output = &cache->outputs[i];
|
|
|
|
struct xrt_device* xdev = output->xdev;
|
|
|
|
|
|
|
|
xdev->set_output(xdev, output->name,
|
|
|
|
sess->sys->inst->timekeeping, &value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
oxr_source_cache_update(struct oxr_logger* log,
|
|
|
|
struct oxr_session* sess,
|
|
|
|
struct oxr_action* act,
|
|
|
|
struct oxr_source_cache* cache,
|
|
|
|
int64_t time,
|
|
|
|
bool selected)
|
|
|
|
{
|
|
|
|
struct oxr_source_state last = cache->current;
|
|
|
|
|
|
|
|
if (!selected) {
|
|
|
|
if (cache->stop_output_time > 0) {
|
|
|
|
oxr_source_cache_stop_output(log, sess, cache);
|
|
|
|
}
|
2019-06-18 19:03:42 +00:00
|
|
|
U_ZERO(&cache->current);
|
2019-05-07 12:47:18 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cache->num_outputs > 0 && cache->stop_output_time < time) {
|
|
|
|
oxr_source_cache_stop_output(log, sess, cache);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cache->num_inputs > 0) {
|
|
|
|
//! @todo Deal with other types and combine sources.
|
|
|
|
int64_t timestamp = cache->inputs[0].input->timestamp;
|
|
|
|
float value = cache->inputs[0].input->value.vec1.x;
|
|
|
|
|
|
|
|
bool changed = value != last.vec1.x;
|
|
|
|
cache->current.vec1.x = value;
|
|
|
|
|
|
|
|
if (last.active && changed) {
|
|
|
|
cache->current.timestamp = timestamp;
|
|
|
|
cache->current.changed = true;
|
|
|
|
} else if (last.active) {
|
|
|
|
cache->current.timestamp = last.timestamp;
|
|
|
|
cache->current.changed = false;
|
|
|
|
} else {
|
|
|
|
cache->current.timestamp = timestamp;
|
|
|
|
cache->current.changed = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
oxr_source_update(struct oxr_logger* log,
|
|
|
|
struct oxr_session* sess,
|
|
|
|
struct oxr_action* act,
|
|
|
|
int64_t time,
|
|
|
|
struct oxr_sub_paths sub_paths)
|
|
|
|
{
|
|
|
|
struct oxr_source* src = NULL;
|
|
|
|
|
|
|
|
oxr_session_get_source(sess, act->key, &src);
|
2019-09-02 21:04:13 +00:00
|
|
|
// This really shouldn't be happening.
|
2019-05-07 12:47:18 +00:00
|
|
|
if (src == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool select_head = sub_paths.head || sub_paths.any;
|
|
|
|
bool select_left = sub_paths.left || sub_paths.any;
|
|
|
|
bool select_right = sub_paths.right || sub_paths.any;
|
|
|
|
bool select_gamepad = sub_paths.gamepad || sub_paths.any;
|
|
|
|
|
|
|
|
oxr_source_cache_update(log, sess, act, &src->head, time, select_head);
|
|
|
|
oxr_source_cache_update(log, sess, act, &src->left, time, select_left);
|
|
|
|
oxr_source_cache_update(log, sess, act, &src->right, time,
|
|
|
|
select_right);
|
|
|
|
oxr_source_cache_update(log, sess, act, &src->gamepad, time,
|
|
|
|
select_gamepad);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
oxr_source_bind_inputs(struct oxr_logger* log,
|
2019-09-02 21:04:13 +00:00
|
|
|
struct oxr_sink_logger* slog,
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_session* sess,
|
|
|
|
struct oxr_action* act,
|
|
|
|
struct oxr_source_cache* cache,
|
2019-09-02 21:04:13 +00:00
|
|
|
struct oxr_interaction_profile* profile,
|
|
|
|
enum oxr_sub_action_path sub_path)
|
2019-05-07 12:47:18 +00:00
|
|
|
{
|
|
|
|
struct oxr_source_input inputs[16] = {0};
|
|
|
|
uint32_t num_inputs = 0;
|
|
|
|
struct oxr_source_output outputs[16] = {0};
|
|
|
|
uint32_t num_outputs = 0;
|
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
get_binding(log, slog, sess, act, profile, sub_path, inputs,
|
|
|
|
&num_inputs, outputs, &num_outputs);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
cache->current.active = false;
|
|
|
|
|
|
|
|
if (num_inputs > 0) {
|
|
|
|
cache->current.active = true;
|
|
|
|
cache->inputs =
|
|
|
|
U_TYPED_ARRAY_CALLOC(struct oxr_source_input, num_inputs);
|
|
|
|
for (uint32_t i = 0; i < num_inputs; i++) {
|
|
|
|
cache->inputs[i] = inputs[i];
|
|
|
|
}
|
|
|
|
cache->num_inputs = num_inputs;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (num_outputs > 0) {
|
|
|
|
cache->current.active = true;
|
|
|
|
cache->outputs =
|
|
|
|
U_TYPED_ARRAY_CALLOC(struct oxr_source_output, num_outputs);
|
|
|
|
for (uint32_t i = 0; i < num_outputs; i++) {
|
|
|
|
cache->outputs[i] = outputs[i];
|
|
|
|
}
|
|
|
|
cache->num_outputs = num_outputs;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Session functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
static void
|
|
|
|
oxr_session_get_source_set(struct oxr_session* sess,
|
|
|
|
XrActionSet actionSet,
|
|
|
|
struct oxr_source_set** src_set,
|
|
|
|
struct oxr_action_set** act_set)
|
|
|
|
{
|
|
|
|
void* ptr = NULL;
|
|
|
|
*act_set = (struct oxr_action_set*)actionSet;
|
|
|
|
|
|
|
|
int ret = u_hashmap_int_find(sess->act_sets, (*act_set)->key, &ptr);
|
|
|
|
if (ret == 0) {
|
|
|
|
*src_set = (struct oxr_source_set*)ptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
oxr_session_get_source(struct oxr_session* sess,
|
|
|
|
uint32_t act_key,
|
|
|
|
struct oxr_source** out_src)
|
|
|
|
{
|
|
|
|
void* ptr = NULL;
|
|
|
|
|
|
|
|
int ret = u_hashmap_int_find(sess->sources, act_key, &ptr);
|
|
|
|
if (ret == 0) {
|
|
|
|
*out_src = (struct oxr_source*)ptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
XrResult
|
|
|
|
oxr_session_attach_action_sets(struct oxr_logger* log,
|
|
|
|
struct oxr_session* sess,
|
|
|
|
const XrSessionActionSetsAttachInfo* bindInfo)
|
2019-05-07 12:47:18 +00:00
|
|
|
{
|
2019-09-02 21:04:13 +00:00
|
|
|
struct oxr_instance* inst = sess->sys->inst;
|
|
|
|
struct oxr_interaction_profile* head = NULL;
|
|
|
|
struct oxr_interaction_profile* left = NULL;
|
|
|
|
struct oxr_interaction_profile* right = NULL;
|
|
|
|
struct oxr_action_set* act_set = NULL;
|
|
|
|
struct oxr_source_set* src_set = NULL;
|
|
|
|
struct oxr_action* act = NULL;
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
oxr_find_profile_for_device(log, inst, sess->sys->head, &head);
|
|
|
|
oxr_find_profile_for_device(log, inst, sess->sys->left, &left);
|
|
|
|
oxr_find_profile_for_device(log, inst, sess->sys->right, &right);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
// Has any of the bound action sets been updated.
|
|
|
|
for (uint32_t i = 0; i < bindInfo->countActionSets; i++) {
|
|
|
|
act_set = (struct oxr_action_set*)bindInfo->actionSets[i];
|
|
|
|
act_set->attached = true;
|
|
|
|
|
|
|
|
oxr_source_set_create(log, sess, act_set, &src_set);
|
|
|
|
|
|
|
|
for (uint32_t k = 0; k < XRT_MAX_HANDLE_CHILDREN; k++) {
|
|
|
|
act = (struct oxr_action*)act_set->handle.children[k];
|
|
|
|
if (act == NULL) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
oxr_source_create(log, src_set, act, head, left, right,
|
|
|
|
NULL);
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
2019-09-02 21:04:13 +00:00
|
|
|
}
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
if (head != NULL) {
|
|
|
|
sess->head = head->path;
|
|
|
|
}
|
|
|
|
if (left != NULL) {
|
|
|
|
sess->left = left->path;
|
|
|
|
}
|
|
|
|
if (right != NULL) {
|
|
|
|
sess->right = right->path;
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
sess->actionsAttached = true;
|
|
|
|
|
2019-07-13 16:17:57 +00:00
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
XrResult
|
|
|
|
oxr_action_sync_data(struct oxr_logger* log,
|
|
|
|
struct oxr_session* sess,
|
|
|
|
uint32_t countActionSets,
|
|
|
|
const XrActiveActionSet* actionSets)
|
|
|
|
{
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_action_set* act_set = NULL;
|
|
|
|
struct oxr_source_set* src_set = NULL;
|
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
// Check that all action sets has been attached.
|
2019-05-07 12:47:18 +00:00
|
|
|
for (uint32_t i = 0; i < countActionSets; i++) {
|
|
|
|
oxr_session_get_source_set(sess, actionSets[i].actionSet,
|
|
|
|
&src_set, &act_set);
|
|
|
|
if (src_set == NULL) {
|
2019-09-02 21:04:13 +00:00
|
|
|
return oxr_error(
|
|
|
|
log, XR_ERROR_ACTIONSET_NOT_ATTACHED,
|
|
|
|
"(actionSets[%i].actionSet) has not been attached",
|
|
|
|
i);
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
2019-09-02 21:04:13 +00:00
|
|
|
}
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
// Synchronize outputs to this time.
|
|
|
|
int64_t now = time_state_get_now(sess->sys->inst->timekeeping);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-09-02 21:04:13 +00:00
|
|
|
// Loop over all xdev devices.
|
|
|
|
for (size_t i = 0; i < sess->sys->num_xdevs; i++) {
|
|
|
|
oxr_xdev_update(sess->sys->xdevs[i],
|
|
|
|
sess->sys->inst->timekeeping);
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Go over all action sets and update them.
|
|
|
|
for (uint32_t i = 0; i < countActionSets; i++) {
|
|
|
|
struct oxr_sub_paths sub_paths;
|
|
|
|
struct oxr_action_set* act_set =
|
|
|
|
(struct oxr_action_set*)actionSets[i].actionSet;
|
|
|
|
|
|
|
|
oxr_classify_sub_action_paths(log, sess->sys->inst, 1,
|
|
|
|
&actionSets[i].subactionPath,
|
|
|
|
&sub_paths);
|
|
|
|
|
2019-08-21 19:28:07 +00:00
|
|
|
for (uint32_t k = 0; k < XRT_MAX_HANDLE_CHILDREN; k++) {
|
2019-05-07 12:47:18 +00:00
|
|
|
// This assumes that all children of a
|
|
|
|
// action set are actions.
|
|
|
|
struct oxr_action* act =
|
|
|
|
(struct oxr_action*)act_set->handle.children[k];
|
|
|
|
|
|
|
|
if (act == NULL) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
oxr_source_update(log, sess, act, now, sub_paths);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
//! @todo Implement
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Action get functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
static void
|
|
|
|
get_state_from_state_bool(struct oxr_source_state* state,
|
|
|
|
XrActionStateBoolean* data)
|
|
|
|
{
|
|
|
|
data->currentState = state->boolean;
|
|
|
|
data->lastChangeTime = state->timestamp;
|
|
|
|
data->isActive = XR_TRUE;
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
bool value = state->boolean;
|
|
|
|
|
|
|
|
if (data->isActive) {
|
|
|
|
data->currentState |= value;
|
|
|
|
data->lastChangeTime = state->timestamp;
|
|
|
|
data->isActive = XR_TRUE;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
get_state_from_state_vec1(struct oxr_source_state* state,
|
2019-07-13 16:17:57 +00:00
|
|
|
XrActionStateFloat* data)
|
2019-05-07 12:47:18 +00:00
|
|
|
{
|
|
|
|
data->currentState = state->vec1.x;
|
|
|
|
data->lastChangeTime = state->timestamp;
|
|
|
|
data->isActive = XR_TRUE;
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
float value = state->vec1.x;
|
|
|
|
|
|
|
|
if (!data->isActive || (data->isActive && value > data->currentState)) {
|
|
|
|
data->currentState = value;
|
|
|
|
data->lastChangeTime = state->timestamp;
|
|
|
|
data->isActive = XR_TRUE;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
get_state_from_state_vec2(struct oxr_source_state* state,
|
|
|
|
XrActionStateVector2f* data)
|
|
|
|
{
|
|
|
|
data->currentState.x = state->vec2.x;
|
|
|
|
data->currentState.y = state->vec2.y;
|
|
|
|
data->lastChangeTime = state->timestamp;
|
|
|
|
data->isActive = XR_TRUE;
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
float value_x = state->vec2.x;
|
|
|
|
float value_y = state->vec2.y;
|
|
|
|
float distance = value_x * value_x + value_y * value_y;
|
|
|
|
float old_distance =
|
|
|
|
data->isActive ? data->currentState.x * data->currentState.x +
|
|
|
|
data->currentState.y * data->currentState.y
|
|
|
|
: 0.f;
|
|
|
|
|
|
|
|
if (!data->isActive || (data->isActive && distance > old_distance)) {
|
|
|
|
data->currentState.x = value_x;
|
|
|
|
data->currentState.y = value_y;
|
|
|
|
data->lastChangeTime = state->timestamp;
|
|
|
|
data->isActive = XR_TRUE;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
#define OXR_ACTION_GET_FILLER(TYPE) \
|
|
|
|
if (sub_paths.user && src->user.current.active) { \
|
|
|
|
get_state_from_state_##TYPE(&src->user.current, data); \
|
|
|
|
} \
|
|
|
|
if (sub_paths.head && src->head.current.active) { \
|
|
|
|
get_state_from_state_##TYPE(&src->head.current, data); \
|
|
|
|
} \
|
|
|
|
if (sub_paths.left && src->left.current.active) { \
|
|
|
|
get_state_from_state_##TYPE(&src->left.current, data); \
|
|
|
|
} \
|
|
|
|
if (sub_paths.right && src->right.current.active) { \
|
|
|
|
get_state_from_state_##TYPE(&src->right.current, data); \
|
|
|
|
} \
|
|
|
|
if (sub_paths.gamepad && src->gamepad.current.active) { \
|
|
|
|
get_state_from_state_##TYPE(&src->gamepad.current, data); \
|
|
|
|
}
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
XrResult
|
|
|
|
oxr_action_get_boolean(struct oxr_logger* log,
|
2019-07-13 16:17:57 +00:00
|
|
|
struct oxr_session* sess,
|
|
|
|
uint64_t key,
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_sub_paths sub_paths,
|
2019-05-07 12:47:18 +00:00
|
|
|
XrActionStateBoolean* data)
|
|
|
|
{
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_source* src = NULL;
|
|
|
|
|
2019-07-13 16:17:57 +00:00
|
|
|
oxr_session_get_source(sess, key, &src);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
data->isActive = XR_FALSE;
|
2019-06-18 19:03:42 +00:00
|
|
|
U_ZERO(&data->currentState);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
if (src == NULL) {
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! @todo support any subpath.
|
|
|
|
if (sub_paths.any) {
|
|
|
|
return oxr_error(log, XR_ERROR_HANDLE_INVALID,
|
|
|
|
"any path not implemented!");
|
|
|
|
}
|
|
|
|
|
|
|
|
OXR_ACTION_GET_FILLER(bool);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_action_get_vector1f(struct oxr_logger* log,
|
2019-07-13 16:17:57 +00:00
|
|
|
struct oxr_session* sess,
|
|
|
|
uint64_t key,
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_sub_paths sub_paths,
|
2019-07-13 16:17:57 +00:00
|
|
|
XrActionStateFloat* data)
|
2019-05-07 12:47:18 +00:00
|
|
|
{
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_source* src = NULL;
|
|
|
|
|
2019-07-13 16:17:57 +00:00
|
|
|
oxr_session_get_source(sess, key, &src);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
data->isActive = XR_FALSE;
|
2019-06-18 19:03:42 +00:00
|
|
|
U_ZERO(&data->currentState);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
if (src == NULL) {
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! @todo support any subpath.
|
|
|
|
if (sub_paths.any) {
|
|
|
|
return oxr_error(log, XR_ERROR_HANDLE_INVALID,
|
|
|
|
"any path not implemented!");
|
|
|
|
}
|
|
|
|
|
|
|
|
OXR_ACTION_GET_FILLER(vec1);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_action_get_vector2f(struct oxr_logger* log,
|
2019-07-13 16:17:57 +00:00
|
|
|
struct oxr_session* sess,
|
|
|
|
uint64_t key,
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_sub_paths sub_paths,
|
2019-05-07 12:47:18 +00:00
|
|
|
XrActionStateVector2f* data)
|
|
|
|
{
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_source* src = NULL;
|
|
|
|
|
2019-07-13 16:17:57 +00:00
|
|
|
oxr_session_get_source(sess, key, &src);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
data->isActive = XR_FALSE;
|
2019-06-18 19:03:42 +00:00
|
|
|
U_ZERO(&data->currentState);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
if (src == NULL) {
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! @todo support any subpath.
|
|
|
|
if (sub_paths.any) {
|
|
|
|
return oxr_error(log, XR_ERROR_HANDLE_INVALID,
|
|
|
|
"any path not implemented!");
|
|
|
|
}
|
|
|
|
|
|
|
|
OXR_ACTION_GET_FILLER(vec2);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_action_get_pose(struct oxr_logger* log,
|
2019-07-13 16:17:57 +00:00
|
|
|
struct oxr_session* sess,
|
|
|
|
uint64_t key,
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_sub_paths sub_paths,
|
2019-05-07 12:47:18 +00:00
|
|
|
XrActionStatePose* data)
|
|
|
|
{
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_source* src = NULL;
|
|
|
|
|
2019-07-13 16:17:57 +00:00
|
|
|
oxr_session_get_source(sess, key, &src);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
data->isActive = XR_FALSE;
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
if (src == NULL) {
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sub_paths.user || sub_paths.any) {
|
|
|
|
data->isActive |= src->user.current.active;
|
|
|
|
}
|
|
|
|
if (sub_paths.head || sub_paths.any) {
|
|
|
|
data->isActive |= src->head.current.active;
|
|
|
|
}
|
|
|
|
if (sub_paths.left || sub_paths.any) {
|
|
|
|
data->isActive |= src->left.current.active;
|
|
|
|
}
|
|
|
|
if (sub_paths.right || sub_paths.any) {
|
|
|
|
data->isActive |= src->right.current.active;
|
|
|
|
}
|
|
|
|
if (sub_paths.gamepad || sub_paths.any) {
|
|
|
|
data->isActive |= src->gamepad.current.active;
|
|
|
|
}
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Haptic feedback functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
static void
|
|
|
|
set_source_output_vibration(struct oxr_session* sess,
|
|
|
|
struct oxr_source_cache* cache,
|
|
|
|
int64_t stop,
|
|
|
|
const XrHapticVibration* data)
|
|
|
|
{
|
|
|
|
cache->stop_output_time = stop;
|
|
|
|
|
|
|
|
union xrt_output_value value = {0};
|
|
|
|
value.vibration.frequency = data->frequency;
|
|
|
|
value.vibration.amplitude = data->amplitude;
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < cache->num_outputs; i++) {
|
|
|
|
struct oxr_source_output* output = &cache->outputs[i];
|
|
|
|
struct xrt_device* xdev = output->xdev;
|
|
|
|
|
|
|
|
xdev->set_output(xdev, output->name,
|
|
|
|
sess->sys->inst->timekeeping, &value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-13 16:17:57 +00:00
|
|
|
|
|
|
|
|
2019-05-07 12:47:18 +00:00
|
|
|
XrResult
|
|
|
|
oxr_action_apply_haptic_feedback(struct oxr_logger* log,
|
2019-07-13 16:17:57 +00:00
|
|
|
struct oxr_session* sess,
|
|
|
|
uint64_t key,
|
|
|
|
struct oxr_sub_paths sub_paths,
|
2019-05-07 12:47:18 +00:00
|
|
|
const XrHapticBaseHeader* hapticEvent)
|
|
|
|
{
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_source* src = NULL;
|
|
|
|
|
2019-07-13 16:17:57 +00:00
|
|
|
oxr_session_get_source(sess, key, &src);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
if (src == NULL) {
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
const XrHapticVibration* data = (const XrHapticVibration*)hapticEvent;
|
|
|
|
|
|
|
|
int64_t now = time_state_get_now(sess->sys->inst->timekeeping);
|
|
|
|
int64_t stop = data->duration <= 0 ? now : now + data->duration;
|
|
|
|
|
|
|
|
// clang-format off
|
|
|
|
if (src->user.current.active && (sub_paths.user || sub_paths.any)) {
|
|
|
|
set_source_output_vibration(sess, &src->user, stop, data);
|
|
|
|
}
|
|
|
|
if (src->head.current.active && (sub_paths.head || sub_paths.any)) {
|
|
|
|
set_source_output_vibration(sess, &src->head, stop, data);
|
|
|
|
}
|
|
|
|
if (src->left.current.active && (sub_paths.left || sub_paths.any)) {
|
|
|
|
set_source_output_vibration(sess, &src->left, stop, data);
|
|
|
|
}
|
|
|
|
if (src->right.current.active && (sub_paths.right || sub_paths.any)) {
|
|
|
|
set_source_output_vibration(sess, &src->right, stop, data);
|
|
|
|
}
|
|
|
|
if (src->gamepad.current.active && (sub_paths.gamepad || sub_paths.any)) {
|
|
|
|
set_source_output_vibration(sess, &src->gamepad, stop, data);
|
|
|
|
}
|
|
|
|
// clang-format on
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
XrResult
|
|
|
|
oxr_action_stop_haptic_feedback(struct oxr_logger* log,
|
2019-07-13 16:17:57 +00:00
|
|
|
struct oxr_session* sess,
|
|
|
|
uint64_t key,
|
|
|
|
struct oxr_sub_paths sub_paths)
|
2019-05-07 12:47:18 +00:00
|
|
|
{
|
2019-05-07 12:47:18 +00:00
|
|
|
struct oxr_source* src = NULL;
|
|
|
|
|
2019-07-13 16:17:57 +00:00
|
|
|
oxr_session_get_source(sess, key, &src);
|
2019-05-07 12:47:18 +00:00
|
|
|
|
|
|
|
if (src == NULL) {
|
|
|
|
return XR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// clang-format off
|
|
|
|
if (src->user.current.active && (sub_paths.user || sub_paths.any)) {
|
|
|
|
oxr_source_cache_stop_output(log, sess, &src->user);
|
|
|
|
}
|
|
|
|
if (src->head.current.active && (sub_paths.head || sub_paths.any)) {
|
|
|
|
oxr_source_cache_stop_output(log, sess, &src->head);
|
|
|
|
}
|
|
|
|
if (src->left.current.active && (sub_paths.left || sub_paths.any)) {
|
|
|
|
oxr_source_cache_stop_output(log, sess, &src->left);
|
|
|
|
}
|
|
|
|
if (src->right.current.active && (sub_paths.right || sub_paths.any)) {
|
|
|
|
oxr_source_cache_stop_output(log, sess, &src->right);
|
|
|
|
}
|
|
|
|
if (src->gamepad.current.active && (sub_paths.gamepad || sub_paths.any)) {
|
|
|
|
oxr_source_cache_stop_output(log, sess, &src->gamepad);
|
|
|
|
}
|
|
|
|
// clang-format on
|
|
|
|
|
|
|
|
return XR_SUCCESS;
|
2019-05-07 12:47:18 +00:00
|
|
|
}
|