2019-10-24 15:12:07 +00:00
|
|
|
// Copyright 2019, Collabora, Ltd.
|
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief PS Move tracker code that is expensive to compile.
|
|
|
|
*
|
|
|
|
* Typically built as a part of t_kalman.cpp to reduce incremental build times.
|
|
|
|
*
|
|
|
|
* @author Ryan Pavlik <ryan.pavlik@collabora.com>
|
|
|
|
* @author Pete Black <pblack@collabora.com>
|
|
|
|
* @author Jakob Bornecrantz <jakob@collabora.com>
|
|
|
|
* @ingroup aux_tracking
|
|
|
|
*/
|
|
|
|
|
2019-11-21 13:00:52 +00:00
|
|
|
#include "tracking/t_fusion.hpp"
|
|
|
|
#include "tracking/t_imu_fusion.hpp"
|
|
|
|
#include "tracking/t_tracker_psmv_fusion.hpp"
|
2019-10-24 15:12:07 +00:00
|
|
|
|
|
|
|
#include "math/m_api.h"
|
2019-11-21 13:15:38 +00:00
|
|
|
#include "math/m_eigen_interop.hpp"
|
2019-10-24 15:12:07 +00:00
|
|
|
|
|
|
|
#include "util/u_misc.h"
|
|
|
|
|
|
|
|
#include "flexkalman/AbsoluteOrientationMeasurement.h"
|
|
|
|
#include "flexkalman/FlexibleKalmanFilter.h"
|
|
|
|
#include "flexkalman/FlexibleUnscentedCorrect.h"
|
|
|
|
#include "flexkalman/PoseSeparatelyDampedConstantVelocity.h"
|
|
|
|
#include "flexkalman/PoseState.h"
|
|
|
|
|
|
|
|
|
2021-04-30 21:49:20 +00:00
|
|
|
namespace xrt::auxiliary::tracking {
|
2019-10-24 15:12:07 +00:00
|
|
|
|
2021-04-30 22:23:32 +00:00
|
|
|
using namespace xrt::auxiliary::math;
|
|
|
|
|
2021-04-30 21:49:20 +00:00
|
|
|
//! Anonymous namespace to hide implementation names
|
2019-10-24 15:12:07 +00:00
|
|
|
namespace {
|
2021-04-30 21:49:20 +00:00
|
|
|
|
|
|
|
using State = flexkalman::pose_externalized_rotation::State;
|
|
|
|
using ProcessModel = flexkalman::PoseSeparatelyDampedConstantVelocityProcessModel<State>;
|
|
|
|
|
|
|
|
struct TrackingInfo
|
|
|
|
{
|
|
|
|
bool valid{false};
|
|
|
|
bool tracked{false};
|
|
|
|
};
|
2019-10-24 15:12:07 +00:00
|
|
|
class PSMVFusion : public PSMVFusionInterface
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
EIGEN_MAKE_ALIGNED_OPERATOR_NEW
|
|
|
|
|
|
|
|
void
|
|
|
|
clear_position_tracked_flag() override;
|
|
|
|
|
|
|
|
void
|
2019-11-11 21:09:04 +00:00
|
|
|
process_imu_data(timepoint_ns timestamp_ns,
|
2019-10-24 15:12:07 +00:00
|
|
|
const struct xrt_tracking_sample *sample,
|
2021-01-14 14:13:48 +00:00
|
|
|
const struct xrt_vec3 *orientation_variance_optional) override;
|
2019-10-24 15:12:07 +00:00
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
process_3d_vision_data(timepoint_ns timestamp_ns,
|
|
|
|
const struct xrt_vec3 *position,
|
|
|
|
const struct xrt_vec3 *variance_optional,
|
|
|
|
const struct xrt_vec3 *lever_arm_optional,
|
|
|
|
float residual_limit) override;
|
2019-10-24 15:12:07 +00:00
|
|
|
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
get_prediction(timepoint_ns when_ns, struct xrt_space_relation *out_relation) override;
|
2019-10-24 15:12:07 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
void
|
|
|
|
reset_filter();
|
|
|
|
void
|
|
|
|
reset_filter_and_imu();
|
|
|
|
|
|
|
|
State filter_state;
|
|
|
|
ProcessModel process_model;
|
|
|
|
|
2021-04-30 21:49:20 +00:00
|
|
|
xrt::auxiliary::tracking::SimpleIMUFusion imu;
|
2019-10-24 15:12:07 +00:00
|
|
|
|
2019-11-11 21:09:04 +00:00
|
|
|
timepoint_ns filter_time_ns{0};
|
2019-10-24 15:12:07 +00:00
|
|
|
bool tracked{false};
|
|
|
|
TrackingInfo orientation_state;
|
|
|
|
TrackingInfo position_state;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
PSMVFusion::clear_position_tracked_flag()
|
|
|
|
{
|
|
|
|
position_state.tracked = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
PSMVFusion::reset_filter()
|
|
|
|
{
|
|
|
|
filter_state = State{};
|
|
|
|
tracked = false;
|
|
|
|
position_state = TrackingInfo{};
|
|
|
|
}
|
|
|
|
void
|
|
|
|
PSMVFusion::reset_filter_and_imu()
|
|
|
|
{
|
|
|
|
reset_filter();
|
|
|
|
orientation_state = TrackingInfo{};
|
|
|
|
imu = SimpleIMUFusion{};
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
PSMVFusion::process_imu_data(timepoint_ns timestamp_ns,
|
|
|
|
const struct xrt_tracking_sample *sample,
|
|
|
|
const struct xrt_vec3 *orientation_variance_optional)
|
2019-10-24 15:12:07 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
Eigen::Vector3d variance = Eigen::Vector3d::Constant(0.01);
|
|
|
|
if (orientation_variance_optional) {
|
2021-01-14 14:13:48 +00:00
|
|
|
variance = map_vec3(*orientation_variance_optional).cast<double>();
|
2019-10-24 15:12:07 +00:00
|
|
|
}
|
2021-01-14 14:13:48 +00:00
|
|
|
imu.handleAccel(map_vec3(sample->accel_m_s2).cast<double>(), timestamp_ns);
|
|
|
|
imu.handleGyro(map_vec3(sample->gyro_rad_secs).cast<double>(), timestamp_ns);
|
2019-10-24 15:12:07 +00:00
|
|
|
imu.postCorrect();
|
|
|
|
|
|
|
|
//! @todo use better measurements instead of the above "simple
|
|
|
|
//! fusion"
|
2019-11-11 21:09:04 +00:00
|
|
|
if (filter_time_ns != 0 && filter_time_ns != timestamp_ns) {
|
|
|
|
float dt = time_ns_to_s(timestamp_ns - filter_time_ns);
|
|
|
|
assert(dt > 0);
|
|
|
|
flexkalman::predict(filter_state, process_model, dt);
|
|
|
|
}
|
|
|
|
filter_time_ns = timestamp_ns;
|
2019-10-24 15:12:07 +00:00
|
|
|
auto meas = flexkalman::AbsoluteOrientationMeasurement{
|
|
|
|
// Must rotate by 180 to align
|
2021-01-14 14:13:48 +00:00
|
|
|
Eigen::Quaterniond(Eigen::AngleAxisd(EIGEN_PI, Eigen::Vector3d::UnitY())) * imu.getQuat(),
|
2019-10-24 15:12:07 +00:00
|
|
|
variance};
|
|
|
|
if (flexkalman::correctUnscented(filter_state, meas)) {
|
|
|
|
orientation_state.tracked = true;
|
|
|
|
orientation_state.valid = true;
|
|
|
|
} else {
|
2020-12-18 12:08:29 +00:00
|
|
|
U_LOG_E(
|
|
|
|
"Got non-finite something when filtering IMU - "
|
|
|
|
"resetting filter and IMU fusion!");
|
2019-10-24 15:12:07 +00:00
|
|
|
reset_filter_and_imu();
|
|
|
|
}
|
2019-11-11 23:39:33 +00:00
|
|
|
// 7200 deg/sec
|
2020-04-27 20:18:31 +00:00
|
|
|
constexpr double max_rad_per_sec = 20 * double(EIGEN_PI) * 2;
|
2021-01-14 14:13:48 +00:00
|
|
|
if (filter_state.angularVelocity().squaredNorm() > max_rad_per_sec * max_rad_per_sec) {
|
2020-12-18 12:08:29 +00:00
|
|
|
U_LOG_E(
|
|
|
|
"Got excessive angular velocity when filtering "
|
|
|
|
"IMU - resetting filter and IMU fusion!");
|
2019-11-11 23:39:33 +00:00
|
|
|
reset_filter_and_imu();
|
|
|
|
}
|
2019-10-24 15:12:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
PSMVFusion::process_3d_vision_data(timepoint_ns timestamp_ns,
|
|
|
|
const struct xrt_vec3 *position,
|
|
|
|
const struct xrt_vec3 *variance_optional,
|
|
|
|
const struct xrt_vec3 *lever_arm_optional,
|
|
|
|
float residual_limit)
|
2019-10-24 15:12:07 +00:00
|
|
|
{
|
|
|
|
Eigen::Vector3f pos = map_vec3(*position);
|
|
|
|
Eigen::Vector3d variance{1.e-4, 1.e-4, 4.e-4};
|
|
|
|
if (variance_optional) {
|
|
|
|
variance = map_vec3(*variance_optional).cast<double>();
|
|
|
|
}
|
|
|
|
Eigen::Vector3d lever_arm{0, 0.09, 0};
|
|
|
|
if (lever_arm_optional) {
|
2021-01-14 14:13:48 +00:00
|
|
|
lever_arm = map_vec3(*lever_arm_optional).cast<double>();
|
2019-10-24 15:12:07 +00:00
|
|
|
}
|
2021-04-30 21:49:20 +00:00
|
|
|
auto measurement = xrt::auxiliary::tracking::AbsolutePositionLeverArmMeasurement{pos.cast<double>(),
|
|
|
|
lever_arm, variance};
|
2019-10-24 15:12:07 +00:00
|
|
|
double resid = measurement.getResidual(filter_state).norm();
|
|
|
|
|
|
|
|
if (resid > residual_limit) {
|
|
|
|
// Residual arbitrarily "too large"
|
2020-12-18 12:08:29 +00:00
|
|
|
U_LOG_W(
|
|
|
|
"measurement residual is %f, resetting "
|
|
|
|
"filter state",
|
2019-10-24 15:12:07 +00:00
|
|
|
resid);
|
|
|
|
reset_filter();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (flexkalman::correctUnscented(filter_state, measurement)) {
|
|
|
|
tracked = true;
|
|
|
|
position_state.valid = true;
|
|
|
|
position_state.tracked = true;
|
|
|
|
} else {
|
2020-12-18 12:08:29 +00:00
|
|
|
U_LOG_W(
|
|
|
|
"Got non-finite something when filtering "
|
|
|
|
"tracker - resetting filter!");
|
2019-10-24 15:12:07 +00:00
|
|
|
reset_filter();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2021-01-14 14:13:48 +00:00
|
|
|
PSMVFusion::get_prediction(timepoint_ns when_ns, struct xrt_space_relation *out_relation)
|
2019-10-24 15:12:07 +00:00
|
|
|
{
|
|
|
|
if (out_relation == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Clear to sane values
|
|
|
|
U_ZERO(out_relation);
|
|
|
|
out_relation->pose.orientation.w = 1;
|
2019-11-11 21:09:04 +00:00
|
|
|
if (!tracked || filter_time_ns == 0) {
|
2019-10-24 15:12:07 +00:00
|
|
|
return;
|
|
|
|
}
|
2019-11-11 21:09:04 +00:00
|
|
|
float dt = time_ns_to_s(when_ns - filter_time_ns);
|
2021-01-14 14:13:48 +00:00
|
|
|
auto predicted_state = flexkalman::getPrediction(filter_state, process_model, dt);
|
2019-10-24 15:12:07 +00:00
|
|
|
|
2021-01-14 14:13:48 +00:00
|
|
|
map_vec3(out_relation->pose.position) = predicted_state.position().cast<float>();
|
|
|
|
map_quat(out_relation->pose.orientation) = predicted_state.getQuaternion().cast<float>();
|
|
|
|
map_vec3(out_relation->linear_velocity) = predicted_state.velocity().cast<float>();
|
|
|
|
map_vec3(out_relation->angular_velocity) = predicted_state.angularVelocity().cast<float>();
|
2019-10-24 15:12:07 +00:00
|
|
|
|
|
|
|
uint64_t flags = 0;
|
|
|
|
if (position_state.valid) {
|
|
|
|
flags |= XRT_SPACE_RELATION_POSITION_VALID_BIT;
|
|
|
|
flags |= XRT_SPACE_RELATION_LINEAR_VELOCITY_VALID_BIT;
|
|
|
|
if (position_state.tracked) {
|
2021-01-14 14:13:48 +00:00
|
|
|
flags |= XRT_SPACE_RELATION_POSITION_TRACKED_BIT;
|
2019-10-24 15:12:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (orientation_state.valid) {
|
|
|
|
flags |= XRT_SPACE_RELATION_ORIENTATION_VALID_BIT;
|
|
|
|
flags |= XRT_SPACE_RELATION_ANGULAR_VELOCITY_VALID_BIT;
|
|
|
|
if (orientation_state.tracked) {
|
2021-01-14 14:13:48 +00:00
|
|
|
flags |= XRT_SPACE_RELATION_ORIENTATION_TRACKED_BIT;
|
2019-10-24 15:12:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
out_relation->relation_flags = (xrt_space_relation_flags)flags;
|
|
|
|
}
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
|
|
std::unique_ptr<PSMVFusionInterface>
|
|
|
|
PSMVFusionInterface::create()
|
|
|
|
{
|
|
|
|
auto ret = std::make_unique<PSMVFusion>();
|
|
|
|
return ret;
|
|
|
|
}
|
2021-04-30 21:49:20 +00:00
|
|
|
} // namespace xrt::auxiliary::tracking
|