// Copyright 2018-2019, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief Holds input related functions. * @author Jakob Bornecrantz * @ingroup oxr_main */ #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" #include #include #include #include #include /* * * 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_source_cache *cache, int64_t time, bool select); static void oxr_source_update(struct oxr_logger *log, struct oxr_session *sess, struct oxr_source *src, int64_t time, struct oxr_sub_paths sub_paths); static void oxr_source_bind_inputs(struct oxr_logger *log, struct oxr_sink_logger *slog, struct oxr_session *sess, struct oxr_action *act, struct oxr_source_cache *cache, struct oxr_interaction_profile *profile, enum oxr_sub_action_path sub_path); static XrResult oxr_source_destroy_cb(struct oxr_logger *log, struct oxr_handle_base *hb); static XrResult oxr_source_create(struct oxr_logger *log, struct oxr_source_set *src_set, struct oxr_action *act, struct oxr_interaction_profile *head, struct oxr_interaction_profile *left, struct oxr_interaction_profile *right, struct oxr_interaction_profile *gamepad); /* * * Action set functions * */ static XrResult oxr_action_set_destroy_cb(struct oxr_logger *log, struct oxr_handle_base *hb) { //! @todo Move to oxr_objects.h 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, struct oxr_instance *inst, const XrActionSetCreateInfo *createInfo, struct oxr_action_set **out_act_set) { // Mod music for all! static uint32_t key_gen = 1; //! @todo Implement more fully. struct oxr_action_set *act_set = NULL; OXR_ALLOCATE_HANDLE_OR_RETURN(log, act_set, OXR_XR_DEBUG_ACTIONSET, oxr_action_set_destroy_cb, &inst->handle); act_set->key = key_gen++; act_set->inst = inst; strncpy(act_set->name, createInfo->actionSetName, sizeof(act_set->name)); *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) { //! @todo Move to oxr_objects.h 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) { struct oxr_instance *inst = act_set->inst; struct oxr_sub_paths sub_paths = {0}; // Mod music for all! static uint32_t key_gen = 1; oxr_classify_sub_action_paths(log, inst, createInfo->countSubactionPaths, createInfo->subactionPaths, &sub_paths); struct oxr_action *act = NULL; OXR_ALLOCATE_HANDLE_OR_RETURN(log, act, OXR_XR_DEBUG_ACTION, oxr_action_destroy_cb, &act_set->handle); act->key = key_gen++; act->act_set = act_set; act->sub_paths = sub_paths; act->action_type = createInfo->actionType; strncpy(act->name, createInfo->actionName, sizeof(act->name)); *out_act = act; return XR_SUCCESS; } /* * * "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. U_ZERO(sub_paths); 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) { struct oxr_source *src = NULL; 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; } /* * * Not so hack functions. * */ static bool do_inputs(struct oxr_binding *bind, struct xrt_device *xdev, struct oxr_source_input inputs[16], uint32_t *num_inputs) { struct xrt_input *input = NULL; 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) { struct xrt_output *output = NULL; 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) { struct xrt_device *xdev = NULL; struct oxr_binding *bindings[32]; const char *profile_str; const char *user_path_str; size_t length; //! @todo This probably falls on its head if the application doesn't use //! sub action paths. switch (sub_path) { 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; default: break; } oxr_slog(slog, "\tFor: %s\n", user_path_str); if (xdev == NULL) { oxr_slog(slog, "\t\tNo xdev!\n"); return; } if (profile == NULL) { oxr_slog(slog, "\t\tNo profile!\n"); return; } 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; } 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"); } } } /* * * 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, struct oxr_session *sess, 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); src_set->sess = sess; u_hashmap_int_insert(sess->act_sets, act_set->key, src_set); src_set->next = sess->src_set_list; sess->src_set_list = 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, struct oxr_source_set *src_set, struct oxr_action *act, struct oxr_interaction_profile *head, struct oxr_interaction_profile *left, struct oxr_interaction_profile *right, struct oxr_interaction_profile *gamepad) { struct oxr_session *sess = src_set->sess; struct oxr_source *src = NULL; struct oxr_sink_logger slog = {0}; OXR_ALLOCATE_HANDLE_OR_RETURN(log, src, OXR_XR_DEBUG_SOURCE, oxr_source_destroy_cb, &src_set->handle); u_hashmap_int_insert(src_set->sess->sources, act->key, src); // Need to copy this. src->action_type = act->action_type; // Start logging into a single buffer. oxr_slog(&slog, ": Binding %s/%s\n", act->act_set->name, act->name); if (act->sub_paths.user || act->sub_paths.any) { #if 0 oxr_source_bind_inputs(log, slog, sess, act, &src->user, user, OXR_SUB_ACTION_PATH_USER); #endif } if (act->sub_paths.head || act->sub_paths.any) { oxr_source_bind_inputs(log, &slog, sess, act, &src->head, head, OXR_SUB_ACTION_PATH_HEAD); } if (act->sub_paths.left || act->sub_paths.any) { oxr_source_bind_inputs(log, &slog, sess, act, &src->left, left, OXR_SUB_ACTION_PATH_LEFT); } if (act->sub_paths.right || act->sub_paths.any) { oxr_source_bind_inputs(log, &slog, sess, act, &src->right, right, OXR_SUB_ACTION_PATH_RIGHT); } if (act->sub_paths.gamepad || act->sub_paths.any) { oxr_source_bind_inputs(log, &slog, sess, act, &src->gamepad, gamepad, OXR_SUB_ACTION_PATH_GAMEPAD); } oxr_slog(&slog, "\tDone"); // Also frees all data. if (sess->sys->inst->debug_bindings) { oxr_log_slog(log, &slog); } else { oxr_slog_abort(&slog); } 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_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); } U_ZERO(&cache->current); return; } if (cache->num_outputs > 0) { cache->current.active = true; if (cache->stop_output_time < time) { oxr_source_cache_stop_output(log, sess, cache); } } if (cache->num_inputs > 0) { /*! * @todo This logic should be a lot more smarter. */ // If the input is not active signal that. if (!cache->inputs[0].input->active) { // Reset all state. U_ZERO(&cache->current); return; } // Signal that the input is active, always set just to be sure. cache->current.active = true; /*! * @todo Combine multiple sources for a single subaction path * and convert type as required. */ struct xrt_input *input = cache->inputs[0].input; int64_t timestamp = input->timestamp; bool changed = false; switch (XRT_GET_INPUT_TYPE(input->name)) { case XRT_INPUT_TYPE_VEC1_ZERO_TO_ONE: case XRT_INPUT_TYPE_VEC1_MINUS_ONE_TO_ONE: { changed = (input->value.vec1.x != last.vec1.x); cache->current.vec1.x = input->value.vec1.x; break; } case XRT_INPUT_TYPE_VEC2_MINUS_ONE_TO_ONE: { changed = (input->value.vec2.x != last.vec2.x) || (input->value.vec2.y != last.vec2.y); cache->current.vec2.x = input->value.vec2.x; cache->current.vec2.y = input->value.vec2.y; break; } #if 0 case XRT_INPUT_TYPE_VEC3_MINUS_ONE_TO_ONE: { changed = (input->value.vec3.x != last.vec3.x) || (input->value.vec3.y != last.vec3.y) || (input->value.vec3.z != last.vec3.z); cache->current.vec3.x = input->value.vec3.x; cache->current.vec3.y = input->value.vec3.y; cache->current.vec3.z = input->value.vec3.z; break; } #endif case XRT_INPUT_TYPE_BOOLEAN: { changed = (input->value.boolean != last.boolean); cache->current.boolean = input->value.boolean; break; } case XRT_INPUT_TYPE_POSE: return; default: // Should not end up here. assert(false); } 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; } } } #define BOOL_CHECK(NAME) \ if (src->NAME.current.active) { \ active |= true; \ value |= src->NAME.current.boolean; \ timestamp = src->NAME.current.timestamp; \ } #define VEC1_CHECK(NAME) \ if (src->NAME.current.active) { \ active |= true; \ if (value < src->NAME.current.vec1.x) { \ value = src->NAME.current.vec1.x; \ timestamp = src->NAME.current.timestamp; \ } \ } #define VEC2_CHECK(NAME) \ if (src->NAME.current.active) { \ active |= true; \ float curr_x = src->NAME.current.vec2.x; \ float curr_y = src->NAME.current.vec2.y; \ float curr_d = curr_x * curr_x + curr_y * curr_y; \ if (distance < curr_d) { \ x = curr_x; \ y = curr_y; \ distance = curr_d; \ timestamp = src->NAME.current.timestamp; \ } \ } static void oxr_source_update(struct oxr_logger *log, struct oxr_session *sess, struct oxr_source *src, int64_t time, struct oxr_sub_paths sub_paths) { // This really shouldn't be happening. if (src == NULL) { return; } //! @todo "/user" sub-action path. bool select_any = sub_paths.any; 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; // clang-format off oxr_source_cache_update(log, sess, &src->head, time, select_head); oxr_source_cache_update(log, sess, &src->left, time, select_left); oxr_source_cache_update(log, sess, &src->right, time, select_right); oxr_source_cache_update(log, sess, &src->gamepad, time, select_gamepad); // clang-format on if (!select_any) { U_ZERO(&src->any_state); return; } /* * Any state. */ struct oxr_source_state last = src->any_state; bool active = false; bool changed = false; XrTime timestamp = 0; switch (src->action_type) { case XR_ACTION_TYPE_BOOLEAN_INPUT: { bool value = false; BOOL_CHECK(user); BOOL_CHECK(head); BOOL_CHECK(left); BOOL_CHECK(right); BOOL_CHECK(gamepad); changed = last.boolean != value; src->any_state.boolean = value; break; } case XR_ACTION_TYPE_FLOAT_INPUT: { float value = -2.0; VEC1_CHECK(user); VEC1_CHECK(head); VEC1_CHECK(left); VEC1_CHECK(right); VEC1_CHECK(gamepad); changed = last.vec1.x != value; src->any_state.vec1.x = value; break; } case XR_ACTION_TYPE_VECTOR2F_INPUT: { float x = 0.0; float y = 0.0; float distance = -1.0; VEC2_CHECK(user); VEC2_CHECK(head); VEC2_CHECK(left); VEC2_CHECK(right); VEC2_CHECK(gamepad); changed = last.vec2.x != x || last.vec2.y != y; src->any_state.vec2.x = x; src->any_state.vec2.y = y; break; } default: case XR_ACTION_TYPE_POSE_INPUT: case XR_ACTION_TYPE_VIBRATION_OUTPUT: // Nothing to do //! @todo You sure? return; } if (!active) { U_ZERO(&src->any_state); } else if (last.active && changed) { src->any_state.timestamp = timestamp; src->any_state.changed = true; src->any_state.active = true; } else if (last.active) { src->any_state.timestamp = last.timestamp; src->any_state.changed = false; src->any_state.active = true; } else { src->any_state.timestamp = timestamp; src->any_state.changed = false; src->any_state.active = true; } } static void oxr_source_bind_inputs(struct oxr_logger *log, struct oxr_sink_logger *slog, struct oxr_session *sess, struct oxr_action *act, struct oxr_source_cache *cache, struct oxr_interaction_profile *profile, enum oxr_sub_action_path sub_path) { struct oxr_source_input inputs[16] = {0}; uint32_t num_inputs = 0; struct oxr_source_output outputs[16] = {0}; uint32_t num_outputs = 0; get_binding(log, slog, sess, act, profile, sub_path, inputs, &num_inputs, outputs, &num_outputs); 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; } } /* * * Session 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) { 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; } } XrResult oxr_session_attach_action_sets(struct oxr_logger *log, struct oxr_session *sess, const XrSessionActionSetsAttachInfo *bindInfo) { 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; 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); // 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); } } if (head != NULL) { sess->head = head->path; } if (left != NULL) { sess->left = left->path; } if (right != NULL) { sess->right = right->path; } sess->actionsAttached = true; return oxr_session_success_result(sess); } XrResult oxr_action_sync_data(struct oxr_logger *log, struct oxr_session *sess, uint32_t countActionSets, const XrActiveActionSet *actionSets) { struct oxr_action_set *act_set = NULL; struct oxr_source_set *src_set = NULL; // Check that all action sets has been attached. 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) { return oxr_error( log, XR_ERROR_ACTIONSET_NOT_ATTACHED, "(actionSets[%i].actionSet) has not been attached", i); } } // Synchronize outputs to this time. int64_t now = time_state_get_now(sess->sys->inst->timekeeping); // 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); } // Reset all requested source sets. src_set = sess->src_set_list; while (src_set != NULL) { U_ZERO(&src_set->requested_sub_paths); // Grab the next one. src_set = src_set->next; } // Go over all action sets and update them. for (uint32_t i = 0; i < countActionSets; i++) { struct oxr_sub_paths sub_paths; oxr_session_get_source_set(sess, actionSets[i].actionSet, &src_set, &act_set); assert(src_set != NULL); oxr_classify_sub_action_paths(log, sess->sys->inst, 1, &actionSets[i].subactionPath, &sub_paths); src_set->requested_sub_paths.any |= sub_paths.any; src_set->requested_sub_paths.user |= sub_paths.user; src_set->requested_sub_paths.head |= sub_paths.head; src_set->requested_sub_paths.left |= sub_paths.left; src_set->requested_sub_paths.right |= sub_paths.right; src_set->requested_sub_paths.gamepad |= sub_paths.gamepad; } // Reset all source sets. src_set = sess->src_set_list; while (src_set != NULL) { struct oxr_sub_paths sub_paths = src_set->requested_sub_paths; for (uint32_t k = 0; k < XRT_MAX_HANDLE_CHILDREN; k++) { // This assumes that all children of a // source set are actions. struct oxr_source *src = (struct oxr_source *)src_set->handle.children[k]; if (src == NULL) { continue; } oxr_source_update(log, sess, src, now, sub_paths); } // Grab the next one. src_set = src_set->next; } return oxr_session_success_focused_result(sess); } /* * * Action get functions. * */ static void get_state_from_state_bool(struct oxr_source_state *state, XrActionStateBoolean *data) { data->currentState = state->boolean; data->lastChangeTime = state->timestamp; data->changedSinceLastSync = state->changed; data->isActive = XR_TRUE; } static void get_state_from_state_vec1(struct oxr_source_state *state, XrActionStateFloat *data) { data->currentState = state->vec1.x; data->lastChangeTime = state->timestamp; data->changedSinceLastSync = state->changed; data->isActive = XR_TRUE; } 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->changedSinceLastSync = state->changed; data->isActive = XR_TRUE; } #define OXR_ACTION_GET_FILLER(TYPE) \ if (sub_paths.any && src->any_state.active) { \ get_state_from_state_##TYPE(&src->any_state, data); \ } \ 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); \ } XrResult oxr_action_get_boolean(struct oxr_logger *log, struct oxr_session *sess, uint64_t key, struct oxr_sub_paths sub_paths, XrActionStateBoolean *data) { struct oxr_source *src = NULL; oxr_session_get_source(sess, key, &src); data->isActive = XR_FALSE; U_ZERO(&data->currentState); if (src == NULL) { return oxr_session_success_result(sess); } OXR_ACTION_GET_FILLER(bool); return oxr_session_success_result(sess); } XrResult oxr_action_get_vector1f(struct oxr_logger *log, struct oxr_session *sess, uint64_t key, struct oxr_sub_paths sub_paths, XrActionStateFloat *data) { struct oxr_source *src = NULL; oxr_session_get_source(sess, key, &src); data->isActive = XR_FALSE; U_ZERO(&data->currentState); if (src == NULL) { return oxr_session_success_result(sess); } OXR_ACTION_GET_FILLER(vec1); return oxr_session_success_result(sess); } XrResult oxr_action_get_vector2f(struct oxr_logger *log, struct oxr_session *sess, uint64_t key, struct oxr_sub_paths sub_paths, XrActionStateVector2f *data) { struct oxr_source *src = NULL; oxr_session_get_source(sess, key, &src); data->isActive = XR_FALSE; U_ZERO(&data->currentState); if (src == NULL) { return oxr_session_success_result(sess); } OXR_ACTION_GET_FILLER(vec2); return oxr_session_success_result(sess); } XrResult oxr_action_get_pose(struct oxr_logger *log, struct oxr_session *sess, uint64_t key, struct oxr_sub_paths sub_paths, XrActionStatePose *data) { struct oxr_source *src = NULL; oxr_session_get_source(sess, key, &src); data->isActive = XR_FALSE; if (src == NULL) { return oxr_session_success_result(sess); } 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; } return oxr_session_success_result(sess); } /* * * Haptic feedback functions. * */ 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); } } XrResult oxr_action_apply_haptic_feedback(struct oxr_logger *log, struct oxr_session *sess, uint64_t key, struct oxr_sub_paths sub_paths, const XrHapticBaseHeader *hapticEvent) { struct oxr_source *src = NULL; oxr_session_get_source(sess, key, &src); if (src == NULL) { return oxr_session_success_result(sess); } 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 oxr_session_success_result(sess); } XrResult oxr_action_stop_haptic_feedback(struct oxr_logger *log, struct oxr_session *sess, uint64_t key, struct oxr_sub_paths sub_paths) { struct oxr_source *src = NULL; oxr_session_get_source(sess, key, &src); if (src == NULL) { return oxr_session_success_result(sess); } // 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 oxr_session_success_result(sess); }