2019-03-18 05:52:32 +00:00
|
|
|
// Copyright 2019, Collabora, Ltd.
|
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief C interface to math library.
|
|
|
|
* @author Jakob Bornecrantz <jakob@collabora.com>
|
|
|
|
*
|
|
|
|
* @see xrt_vec3
|
|
|
|
* @see xrt_quat
|
|
|
|
* @see xrt_pose
|
|
|
|
* @see xrt_space_relation
|
2019-04-06 11:29:47 +00:00
|
|
|
* @ingroup aux_math
|
2019-03-18 05:52:32 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "xrt/xrt_defines.h"
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2019-04-06 11:29:47 +00:00
|
|
|
/*!
|
|
|
|
* @defgroup aux_math Math
|
|
|
|
* @ingroup aux
|
|
|
|
*
|
|
|
|
* @brief C interface to some transform-related math functions.
|
|
|
|
*/
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
/*!
|
|
|
|
* @dir auxiliary/math
|
2019-04-06 11:29:47 +00:00
|
|
|
* @ingroup aux
|
2019-03-18 05:52:32 +00:00
|
|
|
*
|
|
|
|
* @brief C interface to some transform-related math functions.
|
|
|
|
*/
|
|
|
|
|
2019-09-16 15:35:45 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Defines.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* Standard gravity acceleration constant.
|
|
|
|
*
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
#define MATH_GRAVITY_M_S2 (9.8066)
|
|
|
|
|
2019-04-06 11:29:47 +00:00
|
|
|
|
2019-04-01 16:10:41 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Hash functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* Generate a hash value from the given string, trailing zero not included.
|
|
|
|
*
|
2019-06-18 16:18:33 +00:00
|
|
|
* Hashing function used is not specified so no guarantee of staying the same
|
2019-04-01 16:10:41 +00:00
|
|
|
* between different versions of the software, or even when the same version
|
|
|
|
* is compiled on different platforms/libc++ as it might use std::hash.
|
|
|
|
*
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
size_t
|
|
|
|
math_hash_string(const char *str_c, size_t length);
|
|
|
|
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Vector functions
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-07-21 12:35:41 +00:00
|
|
|
/*!
|
|
|
|
* Check if this vec3 is valid for math operations.
|
|
|
|
*
|
|
|
|
* @relates xrt_vec3
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
bool
|
|
|
|
math_vec3_validate(const struct xrt_vec3 *vec3);
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
/*!
|
|
|
|
* Accumulate a vector by adding in-place.
|
|
|
|
*
|
|
|
|
* Logically, *inAndOut += *additional
|
|
|
|
* OK if the two arguments are the same addresses.
|
|
|
|
*
|
|
|
|
* @relates xrt_vec3
|
2019-04-06 11:29:47 +00:00
|
|
|
* @ingroup aux_math
|
2019-03-18 05:52:32 +00:00
|
|
|
*/
|
|
|
|
void
|
|
|
|
math_vec3_accum(const struct xrt_vec3 *additional, struct xrt_vec3 *inAndOut);
|
|
|
|
|
2020-10-12 00:09:19 +00:00
|
|
|
/*!
|
|
|
|
* Subtract from a vector in-place.
|
|
|
|
*
|
|
|
|
* Logically, *inAndOut -= *subtrahend
|
|
|
|
* OK if the two arguments are the same addresses.
|
|
|
|
*
|
|
|
|
* @relates xrt_vec3
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
math_vec3_subtract(const struct xrt_vec3 *subtrahend, struct xrt_vec3 *inAndOut);
|
2020-10-12 00:09:19 +00:00
|
|
|
|
|
|
|
/*!
|
|
|
|
* Multiply a vector in-place.
|
|
|
|
*
|
|
|
|
* Logically, *inAndOut *= scalar
|
|
|
|
*
|
|
|
|
* @relates xrt_vec3
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
math_vec3_scalar_mul(float scalar, struct xrt_vec3 *inAndOut);
|
|
|
|
|
2019-12-29 21:00:26 +00:00
|
|
|
/*!
|
|
|
|
* Cross product of a vector.
|
|
|
|
*
|
|
|
|
* @relates xrt_vec3
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
math_vec3_cross(const struct xrt_vec3 *l, const struct xrt_vec3 *r, struct xrt_vec3 *result);
|
2019-12-29 21:00:26 +00:00
|
|
|
|
2020-06-19 11:38:21 +00:00
|
|
|
|
|
|
|
/*!
|
|
|
|
* Normalize a vec3 in place.
|
|
|
|
*
|
|
|
|
* @relates xrt_vec3
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
math_vec3_normalize(struct xrt_vec3 *in);
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Quat functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2020-03-13 19:14:18 +00:00
|
|
|
/*!
|
|
|
|
* Create a rotation from a angle in radians and a vector.
|
|
|
|
*
|
|
|
|
* @relates xrt_quat
|
|
|
|
* @relates xrt_vec3
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
math_quat_from_angle_vector(float angle_rads, const struct xrt_vec3 *vector, struct xrt_quat *result);
|
2020-03-13 19:14:18 +00:00
|
|
|
|
2019-12-25 14:13:34 +00:00
|
|
|
/*!
|
|
|
|
* Create a rotation from a 3x3 rotation matrix.
|
|
|
|
*
|
|
|
|
* @relates xrt_quat
|
|
|
|
* @relates xrt_matrix_3x3
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
math_quat_from_matrix_3x3(const struct xrt_matrix_3x3 *mat, struct xrt_quat *result);
|
2019-12-25 14:13:34 +00:00
|
|
|
|
2019-12-29 21:00:26 +00:00
|
|
|
/*!
|
|
|
|
* Create a rotation from two vectors plus x and z, by creating a rotation
|
|
|
|
* matrix by crossing z and x to get the y axis.
|
|
|
|
*
|
|
|
|
* @relates xrt_quat
|
|
|
|
* @relates xrt_vec3
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
math_quat_from_plus_x_z(const struct xrt_vec3 *plus_x, const struct xrt_vec3 *plus_z, struct xrt_quat *result);
|
2019-12-29 21:00:26 +00:00
|
|
|
|
2019-07-21 12:35:41 +00:00
|
|
|
/*!
|
|
|
|
* Check if this quat can be used in transformation operations.
|
|
|
|
*
|
|
|
|
* @relates xrt_quat
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
bool
|
2019-08-16 21:54:58 +00:00
|
|
|
math_quat_validate(const struct xrt_quat *quat);
|
2019-07-21 12:35:41 +00:00
|
|
|
|
2020-08-30 14:25:46 +00:00
|
|
|
/*!
|
|
|
|
* Invert a quaternion.
|
|
|
|
*
|
|
|
|
* @relates xrt_quat
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
math_quat_invert(const struct xrt_quat *quat, struct xrt_quat *out_quat);
|
|
|
|
|
2019-06-21 20:43:45 +00:00
|
|
|
/*!
|
|
|
|
* Normalize a quaternion.
|
|
|
|
*
|
|
|
|
* @relates xrt_quat
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
math_quat_normalize(struct xrt_quat *inout);
|
|
|
|
|
2020-06-16 13:39:16 +00:00
|
|
|
/*!
|
|
|
|
* Normalizes a quaternion if it has accumulated float precision errors.
|
|
|
|
* Returns true if the quaternion was already normalized or was normalized after
|
|
|
|
* being found within a small float precision tolerance.
|
|
|
|
* Returns false if the quaternion was not at all normalized.
|
|
|
|
*
|
|
|
|
* @relates xrt_quat
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
bool
|
|
|
|
math_quat_ensure_normalized(struct xrt_quat *inout);
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
/*!
|
|
|
|
* Rotate a vector.
|
|
|
|
*
|
|
|
|
* @relates xrt_quat
|
|
|
|
* @relatesalso xrt_vec3
|
2019-04-06 11:29:47 +00:00
|
|
|
* @ingroup aux_math
|
2019-03-18 05:52:32 +00:00
|
|
|
*/
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
math_quat_rotate_vec3(const struct xrt_quat *left, const struct xrt_vec3 *right, struct xrt_vec3 *result);
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
/*!
|
|
|
|
* Rotate a quaternion (compose rotations).
|
|
|
|
*
|
|
|
|
* @relates xrt_quat
|
2019-04-06 11:29:47 +00:00
|
|
|
* @ingroup aux_math
|
2019-03-18 05:52:32 +00:00
|
|
|
*/
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
math_quat_rotate(const struct xrt_quat *left, const struct xrt_quat *right, struct xrt_quat *result);
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* Integrate an angular velocity vector (exponential map) and apply to a
|
|
|
|
* quaternion.
|
|
|
|
*
|
2019-03-22 15:32:57 +00:00
|
|
|
* ang_vel and dt should share the same units of time, and the ang_vel
|
2019-03-18 05:52:32 +00:00
|
|
|
* vector should be in radians per unit of time.
|
|
|
|
*
|
|
|
|
* @relates xrt_quat
|
|
|
|
* @relatesalso xrt_vec3
|
2019-04-06 11:29:47 +00:00
|
|
|
* @ingroup aux_math
|
2019-03-18 05:52:32 +00:00
|
|
|
*/
|
|
|
|
void
|
|
|
|
math_quat_integrate_velocity(const struct xrt_quat *quat,
|
|
|
|
const struct xrt_vec3 *ang_vel,
|
2019-11-12 17:39:15 +00:00
|
|
|
float dt,
|
2019-03-18 05:52:32 +00:00
|
|
|
struct xrt_quat *result);
|
2019-03-22 15:32:57 +00:00
|
|
|
|
|
|
|
/*!
|
|
|
|
* Compute an angular velocity vector (exponential map format) by taking the
|
|
|
|
* finite difference of two quaternions.
|
2019-03-27 13:34:58 +00:00
|
|
|
*
|
2019-03-22 15:32:57 +00:00
|
|
|
* quat1 is the orientation dt time after the orientation was quat0
|
|
|
|
*
|
|
|
|
* out_ang_vel and dt share the same units of time, and out_ang_vel is be in
|
|
|
|
* radians per unit of time.
|
|
|
|
*
|
|
|
|
* @relates xrt_quat
|
|
|
|
* @relatesalso xrt_vec3
|
2019-04-06 11:29:47 +00:00
|
|
|
* @ingroup aux_math
|
2019-03-22 15:32:57 +00:00
|
|
|
*/
|
|
|
|
void
|
|
|
|
math_quat_finite_difference(const struct xrt_quat *quat0,
|
|
|
|
const struct xrt_quat *quat1,
|
2019-11-12 17:39:15 +00:00
|
|
|
float dt,
|
2019-03-22 15:32:57 +00:00
|
|
|
struct xrt_vec3 *out_ang_vel);
|
|
|
|
|
2020-09-03 23:33:36 +00:00
|
|
|
/*!
|
|
|
|
* Used to rotate a derivative like a angular velocity.
|
|
|
|
*
|
|
|
|
* @relates xrt_quat
|
|
|
|
* @relatesalso xrt_vec3
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
math_quat_rotate_derivative(const struct xrt_quat *rot, const struct xrt_vec3 *deriv, struct xrt_vec3 *result);
|
2020-09-03 23:33:36 +00:00
|
|
|
|
2020-03-16 14:12:39 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
2020-05-05 12:55:03 +00:00
|
|
|
* Matrix functions
|
2020-03-16 14:12:39 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2020-11-27 15:45:51 +00:00
|
|
|
/*!
|
|
|
|
* Multiply Matrix2x2.
|
|
|
|
*
|
|
|
|
* @relates xrt_matrix_2x2
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
math_matrix_2x2_multiply(const struct xrt_matrix_2x2 *left,
|
|
|
|
const struct xrt_matrix_2x2 *right,
|
|
|
|
struct xrt_matrix_2x2 *result);
|
|
|
|
|
2020-03-16 14:12:39 +00:00
|
|
|
void
|
|
|
|
math_matrix_3x3_transform_vec3(const struct xrt_matrix_3x3 *left,
|
|
|
|
const struct xrt_vec3 *right,
|
|
|
|
struct xrt_vec3 *result);
|
|
|
|
|
2020-05-05 12:55:03 +00:00
|
|
|
/*!
|
|
|
|
* Initialize Matrix4x4 with identity.
|
|
|
|
*
|
|
|
|
* @relates xrt_matrix_4x4
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
math_matrix_4x4_identity(struct xrt_matrix_4x4 *result);
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* Multiply Matrix4x4.
|
|
|
|
*
|
|
|
|
* @relates xrt_matrix_4x4
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
math_matrix_4x4_multiply(const struct xrt_matrix_4x4 *left,
|
|
|
|
const struct xrt_matrix_4x4 *right,
|
|
|
|
struct xrt_matrix_4x4 *result);
|
2020-05-07 14:41:04 +00:00
|
|
|
|
|
|
|
/*!
|
|
|
|
* Compute view matrix from xrt_pose.
|
|
|
|
*
|
|
|
|
* @relates xrt_matrix_4x4
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
math_matrix_4x4_view_from_pose(const struct xrt_pose *pose, struct xrt_matrix_4x4 *result);
|
2020-05-15 15:51:00 +00:00
|
|
|
|
|
|
|
/*!
|
|
|
|
* Compute quad layer model matrix from xrt_pose and xrt_vec2 size.
|
|
|
|
*
|
|
|
|
* @relates xrt_matrix_4x4
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
math_matrix_4x4_model(const struct xrt_pose *pose, const struct xrt_vec3 *size, struct xrt_matrix_4x4 *result);
|
2020-03-16 14:12:39 +00:00
|
|
|
|
2020-09-08 17:41:24 +00:00
|
|
|
/*!
|
|
|
|
* Compute inverse view projection matrix,
|
|
|
|
* using only the starting 3x3 block of the view.
|
|
|
|
*
|
|
|
|
* @relates xrt_matrix_4x4
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
math_matrix_4x4_inverse_view_projection(const struct xrt_matrix_4x4 *view,
|
|
|
|
const struct xrt_matrix_4x4 *projection,
|
|
|
|
struct xrt_matrix_4x4 *result);
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Pose functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* Check if this pose can be used in transformation operations.
|
|
|
|
*
|
|
|
|
* @relates xrt_pose
|
2019-04-06 11:29:47 +00:00
|
|
|
* @ingroup aux_math
|
2019-03-18 05:52:32 +00:00
|
|
|
*/
|
|
|
|
bool
|
|
|
|
math_pose_validate(const struct xrt_pose *pose);
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* Invert pose.
|
|
|
|
*
|
|
|
|
* OK if input and output are the same addresses.
|
|
|
|
*
|
|
|
|
* @relates xrt_pose
|
2019-04-06 11:29:47 +00:00
|
|
|
* @ingroup aux_math
|
2019-03-18 05:52:32 +00:00
|
|
|
*/
|
|
|
|
void
|
|
|
|
math_pose_invert(const struct xrt_pose *pose, struct xrt_pose *outPose);
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* Apply a rigid-body transformation to a pose.
|
|
|
|
*
|
|
|
|
* OK if input and output are the same addresses.
|
|
|
|
*
|
|
|
|
* @relates xrt_pose
|
2019-04-06 11:29:47 +00:00
|
|
|
* @ingroup aux_math
|
2019-03-18 05:52:32 +00:00
|
|
|
*/
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
math_pose_transform(const struct xrt_pose *transform, const struct xrt_pose *pose, struct xrt_pose *outPose);
|
2019-03-18 05:52:32 +00:00
|
|
|
|
2019-12-29 21:00:26 +00:00
|
|
|
/*!
|
|
|
|
* Apply a rigid-body transformation to a point.
|
|
|
|
*
|
2020-01-30 14:55:21 +00:00
|
|
|
* The input point and output may be the same pointer.
|
|
|
|
*
|
2019-12-29 21:00:26 +00:00
|
|
|
* @relates xrt_pose
|
|
|
|
* @relates xrt_vec3
|
|
|
|
* @ingroup aux_math
|
|
|
|
*/
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
math_pose_transform_point(const struct xrt_pose *transform, const struct xrt_vec3 *point, struct xrt_vec3 *out_point);
|
2019-12-29 21:00:26 +00:00
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
2020-09-04 10:33:56 +00:00
|
|
|
* Optics functions.
|
2019-03-18 05:52:32 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*!
|
|
|
|
* Perform the computations from
|
|
|
|
* "Computing Half-Fields-Of-View from Simpler Display Models",
|
|
|
|
* to get half-FOVs from things we can retrieve from other APIs.
|
|
|
|
* The origin is in the lower-left corner of the display, so w_1 is the width to
|
|
|
|
* the left of CoP, and h_1 is the height below CoP.
|
|
|
|
*
|
|
|
|
* If vertfov_total is set to 0, it will be computed from h_total.
|
|
|
|
*
|
|
|
|
* Distances are in arbitrary but consistent units. Angles are in radians.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* In the diagram below, treating it like a FOV for horizontal,
|
|
|
|
* the top angle is horizfov_total, the length of the bottom
|
|
|
|
* is w_total, and the distance between the vertical line and the left corner is
|
|
|
|
* w_1. Vertical is similar - h_1 is above the center line.
|
|
|
|
* The triangle need not be symmetrical, despite how the diagram looks.
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* horizfov_total
|
|
|
|
* *
|
|
|
|
* angle_left (neg) -> / | \ <- angle_right
|
|
|
|
* / | \
|
|
|
|
* / | \
|
|
|
|
* / | \
|
|
|
|
* -------------
|
|
|
|
* [ w_1 ]
|
|
|
|
* [ --- w --- ]
|
|
|
|
*
|
|
|
|
* ------- --- |\
|
|
|
|
* | \
|
|
|
|
* h_1 | \ angle_up
|
|
|
|
* h_total ___ |-------* vertfov_total
|
|
|
|
* | / angle_down (neg)
|
|
|
|
* | /
|
|
|
|
* | /
|
|
|
|
* ------- |/
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* @return true if successful.
|
2019-04-06 11:29:47 +00:00
|
|
|
* @ingroup aux_math
|
2019-03-18 05:52:32 +00:00
|
|
|
*/
|
|
|
|
bool
|
|
|
|
math_compute_fovs(double w_total,
|
|
|
|
double w_1,
|
|
|
|
double horizfov_total,
|
|
|
|
double h_total,
|
|
|
|
double h_1,
|
|
|
|
double vertfov_total,
|
|
|
|
struct xrt_fov *fov);
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|