mirror of
https://gitlab.freedesktop.org/monado/monado.git
synced 2025-01-01 04:36:07 +00:00
d/wmr: Optionally average IMU samples for 3DoF tracker
Similar to how this was being done for SLAM. It's significantly less jittery and still has a good response. Smarter filters might benefit from raw measurements so the functionality can be enabled again with a checkbox.
This commit is contained in:
parent
10e39d79b8
commit
ed7c148a00
|
@ -397,7 +397,7 @@ dai_ts_to_monado_ts(dai::Timestamp &in)
|
|||
.count();
|
||||
}
|
||||
|
||||
// Look at wmr_source_push_imu_packet - that's where these averaging shenanigans come from ;)
|
||||
// Look at the WMR driver - that's where these averaging shenanigans come from ;)
|
||||
static void
|
||||
depthai_do_one_imu_frame(struct depthai_fs *depthai)
|
||||
{
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
// Copyright 2018, Philipp Zabel.
|
||||
// Copyright 2020-2021, N Madsen.
|
||||
// Copyright 2020-2021, Collabora, Ltd.
|
||||
// Copyright 2020-2022, Collabora, Ltd.
|
||||
// SPDX-License-Identifier: BSL-1.0
|
||||
/*!
|
||||
* @file
|
||||
|
@ -8,6 +8,7 @@
|
|||
* @author Philipp Zabel <philipp.zabel@gmail.com>
|
||||
* @author nima01 <nima_zero_one@protonmail.com>
|
||||
* @author Jakob Bornecrantz <jakob@collabora.com>
|
||||
* @author Mateo de Mayo <mateo.demayo@collabora.com>
|
||||
* @ingroup drv_wmr
|
||||
*/
|
||||
|
||||
|
@ -61,7 +62,6 @@
|
|||
#define CAMERA_FREQUENCY 30 //!< Observed value (OV7251)
|
||||
#define IMU_FREQUENCY 1000 //!< Observed value (ICM20602)
|
||||
#define IMU_SAMPLES_PER_PACKET 4 //!< There are 4 samples for each USB IMU packet
|
||||
#define SLAM_IMU_FREQUENCY IMU_FREQUENCY / IMU_SAMPLES_PER_PACKET
|
||||
|
||||
//! Specifies whether the user wants to use a SLAM tracker.
|
||||
DEBUG_GET_ONCE_BOOL_OPTION(wmr_slam, "WMR_SLAM", true)
|
||||
|
@ -304,7 +304,7 @@ hololens_handle_debug(struct wmr_hmd *wh, const unsigned char *buffer, int size)
|
|||
}
|
||||
|
||||
static void
|
||||
hololens_handle_sensors(struct wmr_hmd *wh, const unsigned char *buffer, int size)
|
||||
hololens_handle_sensors_avg(struct wmr_hmd *wh, const unsigned char *buffer, int size)
|
||||
{
|
||||
DRV_TRACE_MARKER();
|
||||
|
||||
|
@ -313,12 +313,58 @@ hololens_handle_sensors(struct wmr_hmd *wh, const unsigned char *buffer, int siz
|
|||
|
||||
hololens_sensors_decode_packet(wh, &wh->packet, buffer, size);
|
||||
|
||||
struct xrt_vec3 raw_gyro[4];
|
||||
struct xrt_vec3 raw_accel[4];
|
||||
struct xrt_vec3 calib_gyro[4];
|
||||
struct xrt_vec3 calib_accel[4];
|
||||
// Use a single averaged sample from all the samples in the packet
|
||||
struct xrt_vec3 avg_raw_accel = XRT_VEC3_ZERO;
|
||||
struct xrt_vec3 avg_raw_gyro = XRT_VEC3_ZERO;
|
||||
for (int i = 0; i < IMU_SAMPLES_PER_PACKET; i++) {
|
||||
struct xrt_vec3 a = XRT_VEC3_ZERO;
|
||||
struct xrt_vec3 g = XRT_VEC3_ZERO;
|
||||
vec3_from_hololens_accel(wh->packet.accel, i, &a);
|
||||
vec3_from_hololens_gyro(wh->packet.gyro, i, &g);
|
||||
math_vec3_accum(&a, &avg_raw_accel);
|
||||
math_vec3_accum(&g, &avg_raw_gyro);
|
||||
}
|
||||
math_vec3_scalar_mul(1.0f / IMU_SAMPLES_PER_PACKET, &avg_raw_accel);
|
||||
math_vec3_scalar_mul(1.0f / IMU_SAMPLES_PER_PACKET, &avg_raw_gyro);
|
||||
|
||||
for (int i = 0; i < 4; i++) {
|
||||
// Calibrate averaged sample
|
||||
struct xrt_vec3 avg_calib_accel = XRT_VEC3_ZERO;
|
||||
struct xrt_vec3 avg_calib_gyro = XRT_VEC3_ZERO;
|
||||
math_matrix_3x3_transform_vec3(&wh->config.sensors.accel.mix_matrix, &avg_raw_accel, &avg_calib_accel);
|
||||
math_matrix_3x3_transform_vec3(&wh->config.sensors.gyro.mix_matrix, &avg_raw_gyro, &avg_calib_gyro);
|
||||
math_vec3_accum(&wh->config.sensors.accel.bias_offsets, &avg_calib_accel);
|
||||
math_vec3_accum(&wh->config.sensors.gyro.bias_offsets, &avg_calib_gyro);
|
||||
math_quat_rotate_vec3(&wh->P_oxr_acc.orientation, &avg_calib_accel, &avg_calib_accel);
|
||||
math_quat_rotate_vec3(&wh->P_oxr_gyr.orientation, &avg_calib_gyro, &avg_calib_gyro);
|
||||
|
||||
// Fusion tracking
|
||||
os_mutex_lock(&wh->fusion.mutex);
|
||||
timepoint_ns t = wh->packet.gyro_timestamp[IMU_SAMPLES_PER_PACKET - 1] * WMR_MS_HOLOLENS_NS_PER_TICK;
|
||||
m_imu_3dof_update(&wh->fusion.i3dof, t, &avg_calib_accel, &avg_calib_gyro);
|
||||
wh->fusion.last_imu_timestamp_ns = now_ns;
|
||||
wh->fusion.last_angular_velocity = avg_calib_gyro;
|
||||
os_mutex_unlock(&wh->fusion.mutex);
|
||||
|
||||
// SLAM tracking
|
||||
wmr_source_push_imu_packet(wh->tracking.source, t, avg_raw_accel, avg_raw_gyro);
|
||||
}
|
||||
|
||||
static void
|
||||
hololens_handle_sensors_all(struct wmr_hmd *wh, const unsigned char *buffer, int size)
|
||||
{
|
||||
DRV_TRACE_MARKER();
|
||||
|
||||
// Get the timing as close to reading the packet as possible.
|
||||
uint64_t now_ns = os_monotonic_get_ns();
|
||||
|
||||
hololens_sensors_decode_packet(wh, &wh->packet, buffer, size);
|
||||
|
||||
struct xrt_vec3 raw_gyro[IMU_SAMPLES_PER_PACKET];
|
||||
struct xrt_vec3 raw_accel[IMU_SAMPLES_PER_PACKET];
|
||||
struct xrt_vec3 calib_gyro[IMU_SAMPLES_PER_PACKET];
|
||||
struct xrt_vec3 calib_accel[IMU_SAMPLES_PER_PACKET];
|
||||
|
||||
for (int i = 0; i < IMU_SAMPLES_PER_PACKET; i++) {
|
||||
struct xrt_vec3 *rg = &raw_gyro[i];
|
||||
struct xrt_vec3 *cg = &calib_gyro[i];
|
||||
vec3_from_hololens_gyro(wh->packet.gyro, i, rg);
|
||||
|
@ -336,7 +382,7 @@ hololens_handle_sensors(struct wmr_hmd *wh, const unsigned char *buffer, int siz
|
|||
|
||||
// Fusion tracking
|
||||
os_mutex_lock(&wh->fusion.mutex);
|
||||
for (int i = 0; i < 4; i++) {
|
||||
for (int i = 0; i < IMU_SAMPLES_PER_PACKET; i++) {
|
||||
m_imu_3dof_update( //
|
||||
&wh->fusion.i3dof, //
|
||||
wh->packet.gyro_timestamp[i] * WMR_MS_HOLOLENS_NS_PER_TICK, //
|
||||
|
@ -348,10 +394,22 @@ hololens_handle_sensors(struct wmr_hmd *wh, const unsigned char *buffer, int siz
|
|||
os_mutex_unlock(&wh->fusion.mutex);
|
||||
|
||||
// SLAM tracking
|
||||
//! @todo For now we are using raw_samples here because the centerline fix
|
||||
//! in the calibrated samples breaks SLAM systems. Handling of coordinate
|
||||
//! systems needs to be revisited.
|
||||
wmr_source_push_imu_packet(wh->tracking.source, wh->packet.gyro_timestamp, raw_accel, raw_gyro);
|
||||
for (int i = 0; i < IMU_SAMPLES_PER_PACKET; i++) {
|
||||
timepoint_ns t = wh->packet.gyro_timestamp[i] * WMR_MS_HOLOLENS_NS_PER_TICK;
|
||||
wmr_source_push_imu_packet(wh->tracking.source, t, raw_accel[i], raw_gyro[i]);
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
hololens_handle_sensors(struct wmr_hmd *wh, const unsigned char *buffer, int size)
|
||||
{
|
||||
if (wh->average_imus) {
|
||||
// Less overhead and jitter.
|
||||
hololens_handle_sensors_avg(wh, buffer, size);
|
||||
} else {
|
||||
// More sophisticated fusion algorithms might work better with raw data.
|
||||
hololens_handle_sensors_all(wh, buffer, size);
|
||||
}
|
||||
}
|
||||
|
||||
static bool
|
||||
|
@ -1338,8 +1396,11 @@ wmr_hmd_create_extra_calib(struct wmr_hmd *wh)
|
|||
math_matrix_4x4_isometry_from_pose(&P_imu_ht0, &T_imu_ht0);
|
||||
math_matrix_4x4_isometry_from_pose(&P_imu_ht1, &T_imu_ht1);
|
||||
|
||||
//! @note This might change during runtime but the calibration data will be already submitted
|
||||
double imu_frequency = wh->average_imus ? IMU_FREQUENCY / IMU_SAMPLES_PER_PACKET : IMU_FREQUENCY;
|
||||
|
||||
struct t_slam_calib_extras calib = {
|
||||
.imu_frequency = SLAM_IMU_FREQUENCY,
|
||||
.imu_frequency = imu_frequency,
|
||||
.cams =
|
||||
{
|
||||
{
|
||||
|
@ -1461,6 +1522,7 @@ wmr_hmd_setup_ui(struct wmr_hmd *wh)
|
|||
}
|
||||
u_var_add_pose(wh, &wh->pose, "Tracked Pose");
|
||||
u_var_add_pose(wh, &wh->offset, "Pose Offset");
|
||||
u_var_add_bool(wh, &wh->average_imus, "Average IMU samples");
|
||||
|
||||
u_var_add_gui_header(wh, NULL, "3DoF Tracking");
|
||||
m_imu_3dof_add_vars(&wh->fusion.i3dof, wh, "");
|
||||
|
@ -1732,8 +1794,9 @@ wmr_hmd_create(enum wmr_headset_type hmd_type,
|
|||
return;
|
||||
}
|
||||
|
||||
wh->pose = (struct xrt_pose){{0, 0, 0, 1}, {0, 0, 0}};
|
||||
wh->offset = (struct xrt_pose){{0, 0, 0, 1}, {0, 0, 0}};
|
||||
wh->pose = (struct xrt_pose)XRT_POSE_IDENTITY;
|
||||
wh->offset = (struct xrt_pose)XRT_POSE_IDENTITY;
|
||||
wh->average_imus = true;
|
||||
|
||||
/* Now that we have the config loaded, iterate the map of known headsets and see if we have
|
||||
* an entry for this specific headset (otherwise the generic entry will be used)
|
||||
|
|
|
@ -173,6 +173,9 @@ struct wmr_hmd
|
|||
//! Additional offset to apply to `pose`
|
||||
struct xrt_pose offset;
|
||||
|
||||
//! Average 4 IMU samples before sending them to the trackers
|
||||
bool average_imus;
|
||||
|
||||
struct
|
||||
{
|
||||
struct u_var_button hmd_screen_enable_btn;
|
||||
|
|
|
@ -74,7 +74,6 @@ struct wmr_source
|
|||
|
||||
bool is_running; //!< Whether the device is streaming
|
||||
bool first_imu_received; //!< Don't send frames until first IMU sample
|
||||
bool average_imus; //!< Average 4 IMU samples before sending them to the sinks
|
||||
time_duration_ns hw2mono; //!< Estimated offset from IMU to monotonic clock
|
||||
time_duration_ns cam_hw2mono; //!< Cache for hw2mono used in last left frame
|
||||
};
|
||||
|
@ -308,7 +307,6 @@ wmr_source_create(struct xrt_frame_context *xfctx, struct xrt_prober_device *dev
|
|||
|
||||
struct wmr_source *ws = U_TYPED_CALLOC(struct wmr_source);
|
||||
ws->log_level = debug_get_log_option_wmr_log();
|
||||
ws->average_imus = true;
|
||||
|
||||
// Setup xrt_fs
|
||||
struct xrt_fs *xfs = &ws->xfs;
|
||||
|
@ -341,7 +339,6 @@ wmr_source_create(struct xrt_frame_context *xfctx, struct xrt_prober_device *dev
|
|||
m_ff_vec3_f32_alloc(&ws->accel_ff, 1000);
|
||||
u_var_add_root(ws, WMR_SOURCE_STR, false);
|
||||
u_var_add_log_level(ws, &ws->log_level, "Log Level");
|
||||
u_var_add_bool(ws, &ws->average_imus, "Average IMU samples");
|
||||
u_var_add_ro_ff_vec3_f32(ws, ws->gyro_ff, "Gyroscope");
|
||||
u_var_add_ro_ff_vec3_f32(ws, ws->accel_ff, "Accelerometer");
|
||||
u_var_add_sink_debug(ws, &ws->ui_left_sink, "Left Camera");
|
||||
|
@ -359,37 +356,12 @@ wmr_source_create(struct xrt_frame_context *xfctx, struct xrt_prober_device *dev
|
|||
}
|
||||
|
||||
void
|
||||
wmr_source_push_imu_packet(struct xrt_fs *xfs,
|
||||
const uint64_t ts[4],
|
||||
struct xrt_vec3 accels[4],
|
||||
struct xrt_vec3 gyros[4])
|
||||
wmr_source_push_imu_packet(struct xrt_fs *xfs, timepoint_ns t, struct xrt_vec3 accel, struct xrt_vec3 gyro)
|
||||
{
|
||||
DRV_TRACE_MARKER();
|
||||
|
||||
struct wmr_source *ws = wmr_source_from_xfs(xfs);
|
||||
|
||||
if (ws->average_imus) {
|
||||
struct xrt_vec3 a = {0, 0, 0};
|
||||
struct xrt_vec3 g = {0, 0, 0};
|
||||
for (int i = 0; i < 4; i++) {
|
||||
math_vec3_accum(&accels[i], &a);
|
||||
math_vec3_accum(&gyros[i], &g);
|
||||
}
|
||||
math_vec3_scalar_mul(1.0f / 4, &a);
|
||||
math_vec3_scalar_mul(1.0f / 4, &g);
|
||||
|
||||
timepoint_ns t = (ts[3] + ts[0]) / 2 * WMR_MS_HOLOLENS_NS_PER_TICK;
|
||||
struct xrt_vec3_f64 accel = {a.x, a.y, a.z};
|
||||
struct xrt_vec3_f64 gyro = {g.x, g.y, g.z};
|
||||
struct xrt_imu_sample sample = {.timestamp_ns = t, .accel_m_s2 = accel, .gyro_rad_secs = gyro};
|
||||
xrt_sink_push_imu(&ws->imu_sink, &sample);
|
||||
} else {
|
||||
for (int i = 0; i < 4; i++) {
|
||||
timepoint_ns t = ts[i] * WMR_MS_HOLOLENS_NS_PER_TICK;
|
||||
struct xrt_vec3_f64 accel = {accels[i].x, accels[i].y, accels[i].z};
|
||||
struct xrt_vec3_f64 gyro = {gyros[i].x, gyros[i].y, gyros[i].z};
|
||||
struct xrt_imu_sample sample = {.timestamp_ns = t, .accel_m_s2 = accel, .gyro_rad_secs = gyro};
|
||||
struct xrt_vec3_f64 accel_f64 = {accel.x, accel.y, accel.z};
|
||||
struct xrt_vec3_f64 gyro_f64 = {gyro.x, gyro.y, gyro.z};
|
||||
struct xrt_imu_sample sample = {.timestamp_ns = t, .accel_m_s2 = accel_f64, .gyro_rad_secs = gyro_f64};
|
||||
xrt_sink_push_imu(&ws->imu_sink, &sample);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -12,6 +12,7 @@
|
|||
#include "wmr_config.h"
|
||||
#include "xrt/xrt_frameserver.h"
|
||||
#include "xrt/xrt_prober.h"
|
||||
#include "xrt/xrt_tracking.h"
|
||||
|
||||
/*!
|
||||
* WMR video/IMU data sources
|
||||
|
@ -31,11 +32,10 @@ wmr_source_create(struct xrt_frame_context *xfctx, struct xrt_prober_device *dev
|
|||
|
||||
//! @todo IMU data should be generated from within the data source, but right
|
||||
//! now we need this function because it is being generated from wmr_hmd
|
||||
//! @todo Should this method receive raw or calibrated samples? Currently
|
||||
//! receiving raw because Basalt can calibrate them, but other systems can't.
|
||||
void
|
||||
wmr_source_push_imu_packet(struct xrt_fs *xfs,
|
||||
const uint64_t ts[4],
|
||||
struct xrt_vec3 accels[4],
|
||||
struct xrt_vec3 gyros[4]);
|
||||
wmr_source_push_imu_packet(struct xrt_fs *xfs, timepoint_ns t, struct xrt_vec3 accel, struct xrt_vec3 gyro);
|
||||
|
||||
/*!
|
||||
* @}
|
||||
|
|
Loading…
Reference in a new issue