mirror of
https://gitlab.freedesktop.org/monado/monado.git
synced 2025-01-17 04:15:44 +00:00
m/space: Space graph functions
This commit is contained in:
parent
ca91695722
commit
fe94dec5ea
|
@ -23,6 +23,8 @@ set(MATH_SOURCE_FILES
|
|||
math/m_permutation.c
|
||||
math/m_permutation.h
|
||||
math/m_quatexpmap.cpp
|
||||
math/m_space.cpp
|
||||
math/m_space.h
|
||||
math/m_vec2.h
|
||||
math/m_vec3.h
|
||||
)
|
||||
|
|
296
src/xrt/auxiliary/math/m_space.cpp
Normal file
296
src/xrt/auxiliary/math/m_space.cpp
Normal file
|
@ -0,0 +1,296 @@
|
|||
// Copyright 2020, Collabora, Ltd.
|
||||
// SPDX-License-Identifier: BSL-1.0
|
||||
/*!
|
||||
* @file
|
||||
* @brief Functions for manipulating a @ref xrt_space_graph struct.
|
||||
* @author Jakob Bornecrantz <jakob@collabora.com>
|
||||
* @ingroup aux_math
|
||||
*/
|
||||
|
||||
#include "util/u_misc.h"
|
||||
|
||||
#include "math/m_api.h"
|
||||
#include "math/m_vec2.h"
|
||||
#include "math/m_vec3.h"
|
||||
#include "math/m_space.h"
|
||||
|
||||
#include <stdio.h>
|
||||
|
||||
|
||||
/*
|
||||
*
|
||||
* Dump functions.
|
||||
*
|
||||
*/
|
||||
|
||||
static void
|
||||
dump_relation(const struct xrt_space_relation *r)
|
||||
{
|
||||
fprintf(stderr, "%04x", r->relation_flags);
|
||||
|
||||
if (r->relation_flags & XRT_SPACE_RELATION_POSITION_VALID_BIT) {
|
||||
fprintf(stderr, " P{%f %f %f}", r->pose.position.x,
|
||||
r->pose.position.y, r->pose.position.z);
|
||||
}
|
||||
|
||||
if (r->relation_flags & XRT_SPACE_RELATION_ORIENTATION_VALID_BIT) {
|
||||
fprintf(stderr, " O{%f %f %f %f}", r->pose.orientation.x,
|
||||
r->pose.orientation.y, r->pose.orientation.z,
|
||||
r->pose.orientation.w);
|
||||
}
|
||||
|
||||
if (r->relation_flags & XRT_SPACE_RELATION_LINEAR_VELOCITY_VALID_BIT) {
|
||||
fprintf(stderr, " LV{%f %f %f}", r->linear_velocity.x,
|
||||
r->linear_velocity.y, r->linear_velocity.z);
|
||||
}
|
||||
|
||||
if (r->relation_flags &
|
||||
XRT_SPACE_RELATION_LINEAR_ACCELERATION_VALID_BIT) {
|
||||
fprintf(stderr, " LA{%f %f %f}", r->linear_acceleration.x,
|
||||
r->linear_acceleration.y, r->linear_acceleration.z);
|
||||
}
|
||||
|
||||
if (r->relation_flags & XRT_SPACE_RELATION_ANGULAR_VELOCITY_VALID_BIT) {
|
||||
fprintf(stderr, " AV{%f %f %f}", r->angular_velocity.x,
|
||||
r->angular_velocity.y, r->angular_velocity.z);
|
||||
}
|
||||
|
||||
if (r->relation_flags &
|
||||
XRT_SPACE_RELATION_ANGULAR_ACCELERATION_VALID_BIT) {
|
||||
fprintf(stderr, " AA{%f %f %f}", r->angular_acceleration.x,
|
||||
r->angular_acceleration.y, r->angular_acceleration.z);
|
||||
}
|
||||
|
||||
fprintf(stderr, "\n");
|
||||
}
|
||||
|
||||
static void
|
||||
dump_graph(const struct xrt_space_graph *xsg)
|
||||
{
|
||||
fprintf(stderr, "%s %u\n", __func__, xsg->num_steps);
|
||||
for (uint32_t i = 0; i < xsg->num_steps; i++) {
|
||||
const struct xrt_space_relation *r = &xsg->steps[i];
|
||||
fprintf(stderr, "\t%2u: ", i);
|
||||
dump_relation(r);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
*
|
||||
* Helper functions.
|
||||
*
|
||||
*/
|
||||
|
||||
static bool
|
||||
has_step_with_no_pose(const struct xrt_space_graph *xsg)
|
||||
{
|
||||
const enum xrt_space_relation_flags pose_flags =
|
||||
(enum xrt_space_relation_flags)(
|
||||
XRT_SPACE_RELATION_POSITION_VALID_BIT |
|
||||
XRT_SPACE_RELATION_ORIENTATION_VALID_BIT);
|
||||
|
||||
for (uint32_t i = 0; i < xsg->num_steps; i++) {
|
||||
const struct xrt_space_relation *r = &xsg->steps[i];
|
||||
if ((r->relation_flags & pose_flags) == 0) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
struct flags
|
||||
{
|
||||
unsigned int has_orientation : 1;
|
||||
unsigned int has_position : 1;
|
||||
unsigned int has_linear_velocity : 1;
|
||||
unsigned int has_angular_velocity : 1;
|
||||
unsigned int has_tracked_orientation : 1;
|
||||
unsigned int has_tracked_position : 1;
|
||||
};
|
||||
|
||||
flags
|
||||
get_flags(const struct xrt_space_relation *r)
|
||||
{
|
||||
// clang-format off
|
||||
flags flags = {};
|
||||
flags.has_orientation = (r->relation_flags & XRT_SPACE_RELATION_ORIENTATION_VALID_BIT) != 0;
|
||||
flags.has_position = (r->relation_flags & XRT_SPACE_RELATION_POSITION_VALID_BIT) != 0;
|
||||
flags.has_linear_velocity = (r->relation_flags & XRT_SPACE_RELATION_LINEAR_VELOCITY_VALID_BIT) != 0;
|
||||
flags.has_angular_velocity = (r->relation_flags & XRT_SPACE_RELATION_ANGULAR_VELOCITY_VALID_BIT) != 0;
|
||||
flags.has_tracked_orientation = (r->relation_flags & XRT_SPACE_RELATION_ORIENTATION_TRACKED_BIT) != 0;
|
||||
flags.has_tracked_position = (r->relation_flags & XRT_SPACE_RELATION_POSITION_TRACKED_BIT) != 0;
|
||||
// clang-format on
|
||||
|
||||
return flags;
|
||||
}
|
||||
|
||||
static void
|
||||
make_valid_pose(flags flags,
|
||||
const struct xrt_pose *in_pose,
|
||||
struct xrt_pose *out_pose)
|
||||
{
|
||||
if (flags.has_orientation) {
|
||||
out_pose->orientation = in_pose->orientation;
|
||||
} else {
|
||||
out_pose->orientation.x = 0.0f;
|
||||
out_pose->orientation.y = 0.0f;
|
||||
out_pose->orientation.z = 0.0f;
|
||||
out_pose->orientation.w = 1.0f;
|
||||
}
|
||||
|
||||
if (flags.has_position) {
|
||||
out_pose->position = in_pose->position;
|
||||
} else {
|
||||
out_pose->position.x = 0.0f;
|
||||
out_pose->position.y = 0.0f;
|
||||
out_pose->position.z = 0.0f;
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
apply_relation(const struct xrt_space_relation *a,
|
||||
const struct xrt_space_relation *b,
|
||||
struct xrt_space_relation *out_relation)
|
||||
{
|
||||
flags af = get_flags(a);
|
||||
flags bf = get_flags(b);
|
||||
|
||||
flags nf = {};
|
||||
struct xrt_pose pose = {};
|
||||
struct xrt_vec3 linear_velocity = {};
|
||||
struct xrt_vec3 angular_velocity = {};
|
||||
|
||||
|
||||
/*
|
||||
* Linear velocity.
|
||||
*/
|
||||
|
||||
if (af.has_linear_velocity) {
|
||||
nf.has_linear_velocity = true;
|
||||
struct xrt_vec3 tmp = {};
|
||||
math_quat_rotate_vec3(&b->pose.orientation, // Base rotation
|
||||
&a->linear_velocity, // In base space
|
||||
&tmp); // Output
|
||||
linear_velocity += tmp;
|
||||
}
|
||||
|
||||
if (bf.has_linear_velocity) {
|
||||
nf.has_linear_velocity = true;
|
||||
linear_velocity += b->linear_velocity;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* Angular velocity.
|
||||
*/
|
||||
|
||||
if (af.has_angular_velocity) {
|
||||
nf.has_angular_velocity = true;
|
||||
struct xrt_vec3 tmp = {};
|
||||
|
||||
math_quat_rotate_derivative(
|
||||
&b->pose.orientation, // Base rotation
|
||||
&a->angular_velocity, // In base space
|
||||
&tmp); // Output
|
||||
|
||||
angular_velocity += tmp;
|
||||
}
|
||||
|
||||
if (bf.has_angular_velocity) {
|
||||
nf.has_angular_velocity = true;
|
||||
nf.has_linear_velocity = true;
|
||||
angular_velocity += b->angular_velocity;
|
||||
|
||||
struct xrt_vec3 rotated_position = {};
|
||||
struct xrt_vec3 position = {};
|
||||
struct xrt_quat orientation = {};
|
||||
struct xrt_vec3 tangental_velocity = {};
|
||||
|
||||
position = a->pose.position; // In the base space
|
||||
orientation = b->pose.orientation; // Base space
|
||||
|
||||
math_quat_rotate_vec3(&orientation, // Rotation
|
||||
&position, // Vector
|
||||
&rotated_position); // Result
|
||||
|
||||
math_vec3_cross(&b->angular_velocity, // A
|
||||
&rotated_position, // B
|
||||
&tangental_velocity); // Result
|
||||
|
||||
linear_velocity += tangental_velocity;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* Apply the pose part.
|
||||
*/
|
||||
|
||||
struct xrt_pose body_pose = {};
|
||||
struct xrt_pose base_pose = {};
|
||||
make_valid_pose(af, &a->pose, &body_pose);
|
||||
make_valid_pose(bf, &b->pose, &base_pose);
|
||||
math_pose_transform(&base_pose, &body_pose, &pose);
|
||||
|
||||
|
||||
/*
|
||||
* Write everything out.
|
||||
*/
|
||||
|
||||
int new_flags = 0;
|
||||
new_flags |= XRT_SPACE_RELATION_POSITION_VALID_BIT;
|
||||
new_flags |= XRT_SPACE_RELATION_ORIENTATION_VALID_BIT;
|
||||
if (af.has_tracked_position || bf.has_tracked_position) {
|
||||
new_flags |= XRT_SPACE_RELATION_POSITION_TRACKED_BIT;
|
||||
}
|
||||
if (af.has_tracked_position || bf.has_tracked_position) {
|
||||
new_flags |= XRT_SPACE_RELATION_ORIENTATION_TRACKED_BIT;
|
||||
}
|
||||
if (nf.has_linear_velocity) {
|
||||
new_flags |= XRT_SPACE_RELATION_LINEAR_VELOCITY_VALID_BIT;
|
||||
}
|
||||
if (nf.has_angular_velocity) {
|
||||
new_flags |= XRT_SPACE_RELATION_ANGULAR_VELOCITY_VALID_BIT;
|
||||
}
|
||||
|
||||
struct xrt_space_relation tmp = {};
|
||||
tmp.relation_flags = (enum xrt_space_relation_flags)new_flags;
|
||||
tmp.pose = pose;
|
||||
tmp.linear_velocity = linear_velocity;
|
||||
tmp.angular_velocity = angular_velocity;
|
||||
|
||||
*out_relation = tmp;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
*
|
||||
* Exported functions.
|
||||
*
|
||||
*/
|
||||
|
||||
void
|
||||
m_space_graph_resolve(const struct xrt_space_graph *xsg,
|
||||
struct xrt_space_relation *out_relation)
|
||||
{
|
||||
if (xsg->num_steps == 0 || has_step_with_no_pose(xsg)) {
|
||||
U_ZERO(out_relation);
|
||||
return;
|
||||
}
|
||||
|
||||
struct xrt_space_relation r = xsg->steps[0];
|
||||
for (uint32_t i = 1; i < xsg->num_steps; i++) {
|
||||
apply_relation(&r, &xsg->steps[i], &r);
|
||||
}
|
||||
|
||||
#if 0
|
||||
dump_graph(xsg);
|
||||
fprintf(stderr, "\tRR: ");
|
||||
dump_relation(&r);
|
||||
#else
|
||||
(void)dump_graph;
|
||||
#endif
|
||||
|
||||
*out_relation = r;
|
||||
}
|
173
src/xrt/auxiliary/math/m_space.h
Normal file
173
src/xrt/auxiliary/math/m_space.h
Normal file
|
@ -0,0 +1,173 @@
|
|||
// Copyright 2020, Collabora, Ltd.
|
||||
// SPDX-License-Identifier: BSL-1.0
|
||||
/*!
|
||||
* @file
|
||||
* @brief Functions for manipulating @ref xrt_pose, @ref xrt_space_relation and
|
||||
* @ref xrt_space_graph structs.
|
||||
* @author Jakob Bornecrantz <jakob@collabora.com>
|
||||
* @ingroup aux_math
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include "xrt/xrt_defines.h"
|
||||
|
||||
#include "math/m_api.h"
|
||||
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
|
||||
/*!
|
||||
* @ingroup aux_math
|
||||
* @{
|
||||
*/
|
||||
|
||||
/*
|
||||
*
|
||||
* Pose functions.
|
||||
*
|
||||
*/
|
||||
|
||||
static inline bool
|
||||
m_pose_is_identity(struct xrt_pose *pose)
|
||||
{
|
||||
struct xrt_pose p = *pose;
|
||||
|
||||
if ((p.position.x == 0.0f || p.position.x == -0.0f) && // x
|
||||
(p.position.y == 0.0f || p.position.y == -0.0f) && // y
|
||||
(p.position.z == 0.0f || p.position.z == -0.0f) && // z
|
||||
(p.orientation.x == 0.0f || p.orientation.x == -0.0f) && // x
|
||||
(p.orientation.y == 0.0f || p.orientation.y == -0.0f) && // y
|
||||
(p.orientation.z == 0.0f || p.orientation.z == -0.0f) && // z
|
||||
(p.orientation.w == 1.0f || p.orientation.w == -1.0f) // w
|
||||
) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
*
|
||||
* Space relation functions.
|
||||
*
|
||||
*/
|
||||
|
||||
static inline void
|
||||
m_space_relation_from_pose(const struct xrt_pose *pose,
|
||||
struct xrt_space_relation *out_relation)
|
||||
{
|
||||
enum xrt_space_relation_flags flags = (enum xrt_space_relation_flags)(
|
||||
XRT_SPACE_RELATION_ORIENTATION_VALID_BIT |
|
||||
XRT_SPACE_RELATION_POSITION_VALID_BIT);
|
||||
struct xrt_space_relation relation = {
|
||||
flags, *pose, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0},
|
||||
};
|
||||
|
||||
*out_relation = relation;
|
||||
}
|
||||
|
||||
static inline void
|
||||
m_space_relation_ident(struct xrt_space_relation *out_relation)
|
||||
{
|
||||
struct xrt_pose identity = {
|
||||
{0, 0, 0, 1},
|
||||
{0, 0, 0},
|
||||
};
|
||||
|
||||
m_space_relation_from_pose(&identity, out_relation);
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
*
|
||||
* Space graph functions.
|
||||
*
|
||||
*/
|
||||
|
||||
/*!
|
||||
* Reserve a step in the graph and return a pointer to the relation.
|
||||
*/
|
||||
static inline struct xrt_space_relation *
|
||||
m_space_graph_reserve(struct xrt_space_graph *xsg)
|
||||
{
|
||||
if (xsg->num_steps < XRT_SPACE_GRAPHS_MAX) {
|
||||
return &xsg->steps[xsg->num_steps++];
|
||||
} else {
|
||||
return NULL;
|
||||
}
|
||||
}
|
||||
|
||||
/*!
|
||||
* Flattens a space graph into a single relation.
|
||||
*/
|
||||
static inline void
|
||||
m_space_graph_add_relation(struct xrt_space_graph *xsg,
|
||||
const struct xrt_space_relation *relation)
|
||||
{
|
||||
if (xsg->num_steps >= XRT_SPACE_GRAPHS_MAX) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (relation->relation_flags == 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
xsg->steps[xsg->num_steps++] = *relation;
|
||||
}
|
||||
|
||||
static inline void
|
||||
m_space_graph_add_pose(struct xrt_space_graph *xsg, const struct xrt_pose *pose)
|
||||
{
|
||||
struct xrt_space_relation relation;
|
||||
m_space_relation_from_pose(pose, &relation);
|
||||
m_space_graph_add_relation(xsg, &relation);
|
||||
}
|
||||
|
||||
static inline void
|
||||
m_space_graph_add_pose_if_not_identity(struct xrt_space_graph *xsg,
|
||||
const struct xrt_pose *pose)
|
||||
{
|
||||
struct xrt_pose p = *pose;
|
||||
|
||||
if (m_pose_is_identity(&p)) {
|
||||
return;
|
||||
}
|
||||
|
||||
m_space_graph_add_pose(xsg, &p);
|
||||
}
|
||||
|
||||
static inline void
|
||||
m_space_graph_add_inverted_pose_if_not_identity(struct xrt_space_graph *xsg,
|
||||
const struct xrt_pose *pose)
|
||||
{
|
||||
struct xrt_pose p = *pose;
|
||||
|
||||
if (m_pose_is_identity(&p)) {
|
||||
return;
|
||||
}
|
||||
|
||||
struct xrt_pose invert;
|
||||
math_pose_invert(&p, &invert);
|
||||
m_space_graph_add_pose(xsg, &invert);
|
||||
}
|
||||
|
||||
/*!
|
||||
* Flattens a space graph into a single relation.
|
||||
*/
|
||||
void
|
||||
m_space_graph_resolve(const struct xrt_space_graph *xsg,
|
||||
struct xrt_space_relation *out_relation);
|
||||
|
||||
/*!
|
||||
* @}
|
||||
*/
|
||||
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
|
@ -125,6 +125,8 @@ lib_aux_math = static_library(
|
|||
'math/m_permutation.c',
|
||||
'math/m_permutation.h',
|
||||
'math/m_quatexpmap.cpp',
|
||||
'math/m_space.cpp',
|
||||
'math/m_space.h',
|
||||
'math/m_vec2.h',
|
||||
'math/m_vec3.h',
|
||||
),
|
||||
|
|
Loading…
Reference in a new issue