mirror of
https://gitlab.freedesktop.org/monado/monado.git
synced 2025-01-17 04:15:44 +00:00
d/vive: Add vive_source to convert v4l2 timestamps into monotonic clock
Use a sink in the middle of the stream to correct for v4l2 timestamps with hardware timestamps to monotonic clock. This sink, together with other utilities related to data streaming, lives in a new vive_source entity, with similar functionality to wmr_source or rs_source. The vive_source lifetime is managed by the builder xfctx, which prevents deallocation dependencies between vive_device and the v4l2_fs to cause segfaults.
This commit is contained in:
parent
13d90bff77
commit
6e16959098
|
@ -177,6 +177,8 @@ if(XRT_BUILD_DRIVER_VIVE)
|
||||||
vive/vive_controller.c
|
vive/vive_controller.c
|
||||||
vive/vive_lighthouse.h
|
vive/vive_lighthouse.h
|
||||||
vive/vive_lighthouse.c
|
vive/vive_lighthouse.c
|
||||||
|
vive/vive_source.h
|
||||||
|
vive/vive_source.c
|
||||||
)
|
)
|
||||||
target_link_libraries(
|
target_link_libraries(
|
||||||
drv_vive
|
drv_vive
|
||||||
|
|
|
@ -13,6 +13,7 @@
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <asm/byteorder.h>
|
#include <asm/byteorder.h>
|
||||||
|
|
||||||
|
#include "util/u_debug.h"
|
||||||
#include "util/u_time.h"
|
#include "util/u_time.h"
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -27,7 +28,11 @@
|
||||||
#define VIVE_WARN(d, ...) U_LOG_IFL_W(d->log_level, __VA_ARGS__)
|
#define VIVE_WARN(d, ...) U_LOG_IFL_W(d->log_level, __VA_ARGS__)
|
||||||
#define VIVE_ERROR(d, ...) U_LOG_IFL_E(d->log_level, __VA_ARGS__)
|
#define VIVE_ERROR(d, ...) U_LOG_IFL_E(d->log_level, __VA_ARGS__)
|
||||||
|
|
||||||
|
DEBUG_GET_ONCE_LOG_OPTION(vive_log, "VIVE_LOG", U_LOGGING_WARN)
|
||||||
|
|
||||||
#define VIVE_CLOCK_FREQ 48e6 // 48 MHz
|
#define VIVE_CLOCK_FREQ 48e6 // 48 MHz
|
||||||
|
#define CAMERA_FREQUENCY 54
|
||||||
|
#define IMU_FREQUENCY 1000
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Helper function to convert raw device ticks to nanosecond timestamps.
|
* Helper function to convert raw device ticks to nanosecond timestamps.
|
||||||
|
@ -45,12 +50,10 @@ ticks_to_ns(uint32_t sample_ticks_raw, uint32_t *inout_prev_ticks, timepoint_ns
|
||||||
uint32_t sample_ticks = __le32_to_cpu(sample_ticks_raw);
|
uint32_t sample_ticks = __le32_to_cpu(sample_ticks_raw);
|
||||||
uint32_t prev_ticks = *inout_prev_ticks;
|
uint32_t prev_ticks = *inout_prev_ticks;
|
||||||
|
|
||||||
uint32_t delta_ticks = 0;
|
// From the C standard https://www.open-std.org/jtc1/sc22/wg14/www/docs/n2912.pdf
|
||||||
if (prev_ticks < sample_ticks) {
|
// "A computation involving unsigned operands can never produce an overflow,
|
||||||
delta_ticks = sample_ticks - prev_ticks;
|
// because arithmetic for the unsigned type is performed modulo 2^N"
|
||||||
} else { // Handle overflow
|
uint32_t delta_ticks = sample_ticks - prev_ticks;
|
||||||
delta_ticks = (UINT32_MAX - prev_ticks) + sample_ticks;
|
|
||||||
}
|
|
||||||
|
|
||||||
const double one_tick_in_s = (1 / VIVE_CLOCK_FREQ);
|
const double one_tick_in_s = (1 / VIVE_CLOCK_FREQ);
|
||||||
const double one_tick_in_ns = one_tick_in_s * U_TIME_1S_IN_NS;
|
const double one_tick_in_ns = one_tick_in_s * U_TIME_1S_IN_NS;
|
||||||
|
|
|
@ -53,8 +53,6 @@
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
DEBUG_GET_ONCE_LOG_OPTION(vive_log, "VIVE_LOG", U_LOGGING_WARN)
|
|
||||||
|
|
||||||
enum vive_controller_input_index
|
enum vive_controller_input_index
|
||||||
{
|
{
|
||||||
// common inputs
|
// common inputs
|
||||||
|
|
|
@ -28,10 +28,10 @@
|
||||||
#include "vive.h"
|
#include "vive.h"
|
||||||
#include "vive_device.h"
|
#include "vive_device.h"
|
||||||
#include "vive_protocol.h"
|
#include "vive_protocol.h"
|
||||||
|
#include "vive_source.h"
|
||||||
|
#include "xrt/xrt_tracking.h"
|
||||||
|
|
||||||
|
|
||||||
DEBUG_GET_ONCE_LOG_OPTION(vive_log, "VIVE_LOG", U_LOGGING_WARN)
|
|
||||||
|
|
||||||
static bool
|
static bool
|
||||||
vive_mainboard_power_off(struct vive_device *d);
|
vive_mainboard_power_off(struct vive_device *d);
|
||||||
|
|
||||||
|
@ -416,6 +416,8 @@ update_imu(struct vive_device *d, const void *buffer)
|
||||||
rel.pose.orientation = d->fusion.i3dof.rot;
|
rel.pose.orientation = d->fusion.i3dof.rot;
|
||||||
m_relation_history_push(d->fusion.relation_hist, &rel, now_ns);
|
m_relation_history_push(d->fusion.relation_hist, &rel, now_ns);
|
||||||
os_mutex_unlock(&d->fusion.mutex);
|
os_mutex_unlock(&d->fusion.mutex);
|
||||||
|
|
||||||
|
vive_source_push_imu_packet(d->source, d->imu.last_sample_ts_ns, acceleration, angular_velocity);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -537,13 +539,14 @@ _decode_pulse_report(struct vive_device *d, const void *buffer)
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sensor_id == 0xfd) {
|
if (sensor_id == 0xfd) { // Camera frame timestamp
|
||||||
/* TODO: handle camera sync timestamp */
|
vive_source_push_frame_ticks(d->source, pulse->timestamp);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sensor_id == 0xfb) {
|
if (sensor_id == 0xfb) {
|
||||||
/* TODO: Only turns on when the camera is running but not every frame. */
|
/* TODO: Only turns on when the camera is running but not every frame. It
|
||||||
|
* seems to come with every 16h frame on an Index (~3.37hz) */
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -868,7 +871,7 @@ vive_setup_trackers(struct vive_device *d, struct vive_tracking_status status)
|
||||||
}
|
}
|
||||||
|
|
||||||
// SLAM tracker and hand tracker, if enabled, should've been initialized in
|
// SLAM tracker and hand tracker, if enabled, should've been initialized in
|
||||||
// the lighthouse builder.
|
// the lighthouse builder. The out_sinks fields will be set there as well.
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -878,7 +881,8 @@ vive_device_create(struct os_hid_device *mainboard_dev,
|
||||||
struct os_hid_device *sensors_dev,
|
struct os_hid_device *sensors_dev,
|
||||||
struct os_hid_device *watchman_dev,
|
struct os_hid_device *watchman_dev,
|
||||||
enum VIVE_VARIANT variant,
|
enum VIVE_VARIANT variant,
|
||||||
struct vive_tracking_status tstatus)
|
struct vive_tracking_status tstatus,
|
||||||
|
struct vive_source *vs)
|
||||||
{
|
{
|
||||||
XRT_TRACE_MARKER();
|
XRT_TRACE_MARKER();
|
||||||
|
|
||||||
|
@ -1000,6 +1004,7 @@ vive_device_create(struct os_hid_device *mainboard_dev,
|
||||||
os_thread_helper_init(&d->sensors_thread);
|
os_thread_helper_init(&d->sensors_thread);
|
||||||
os_thread_helper_init(&d->watchman_thread);
|
os_thread_helper_init(&d->watchman_thread);
|
||||||
|
|
||||||
|
d->source = vs;
|
||||||
d->pose = (struct xrt_pose)XRT_POSE_IDENTITY;
|
d->pose = (struct xrt_pose)XRT_POSE_IDENTITY;
|
||||||
d->offset = (struct xrt_pose)XRT_POSE_IDENTITY;
|
d->offset = (struct xrt_pose)XRT_POSE_IDENTITY;
|
||||||
|
|
||||||
|
@ -1024,8 +1029,6 @@ vive_device_create(struct os_hid_device *mainboard_dev,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
vive_setup_trackers(d, tstatus);
|
|
||||||
|
|
||||||
switch (d->config.variant) {
|
switch (d->config.variant) {
|
||||||
case VIVE_VARIANT_VIVE: snprintf(d->base.str, XRT_DEVICE_NAME_LEN, "HTC Vive (vive)"); break;
|
case VIVE_VARIANT_VIVE: snprintf(d->base.str, XRT_DEVICE_NAME_LEN, "HTC Vive (vive)"); break;
|
||||||
case VIVE_VARIANT_PRO: snprintf(d->base.str, XRT_DEVICE_NAME_LEN, "HTC Vive Pro (vive)"); break;
|
case VIVE_VARIANT_PRO: snprintf(d->base.str, XRT_DEVICE_NAME_LEN, "HTC Vive Pro (vive)"); break;
|
||||||
|
@ -1034,6 +1037,13 @@ vive_device_create(struct os_hid_device *mainboard_dev,
|
||||||
}
|
}
|
||||||
snprintf(d->base.serial, XRT_DEVICE_NAME_LEN, "%s", d->config.firmware.device_serial_number);
|
snprintf(d->base.serial, XRT_DEVICE_NAME_LEN, "%s", d->config.firmware.device_serial_number);
|
||||||
|
|
||||||
|
bool trackers_set = vive_setup_trackers(d, tstatus);
|
||||||
|
if (!trackers_set) {
|
||||||
|
VIVE_ERROR(d, "Failed to setup trackers");
|
||||||
|
vive_device_destroy((struct xrt_device *)d);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
ret = os_thread_helper_start(&d->sensors_thread, vive_sensors_run_thread, d);
|
ret = os_thread_helper_start(&d->sensors_thread, vive_sensors_run_thread, d);
|
||||||
if (ret != 0) {
|
if (ret != 0) {
|
||||||
VIVE_ERROR(d, "Failed to start sensors thread!");
|
VIVE_ERROR(d, "Failed to start sensors thread!");
|
||||||
|
|
|
@ -100,6 +100,9 @@ struct vive_device
|
||||||
//! Whether to track the HMD with 6dof SLAM or fallback to the 3dof tracker
|
//! Whether to track the HMD with 6dof SLAM or fallback to the 3dof tracker
|
||||||
bool slam_over_3dof;
|
bool slam_over_3dof;
|
||||||
|
|
||||||
|
//! In charge of managing raw samples, redirects them for tracking
|
||||||
|
struct vive_source *source;
|
||||||
|
|
||||||
//! Last tracked pose
|
//! Last tracked pose
|
||||||
struct xrt_pose pose;
|
struct xrt_pose pose;
|
||||||
|
|
||||||
|
@ -129,8 +132,8 @@ vive_device_create(struct os_hid_device *mainboard_dev,
|
||||||
struct os_hid_device *sensors_dev,
|
struct os_hid_device *sensors_dev,
|
||||||
struct os_hid_device *watchman_dev,
|
struct os_hid_device *watchman_dev,
|
||||||
enum VIVE_VARIANT variant,
|
enum VIVE_VARIANT variant,
|
||||||
struct vive_tracking_status tstatus);
|
struct vive_tracking_status tstatus,
|
||||||
|
struct vive_source *vs);
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
|
|
|
@ -66,6 +66,7 @@ init_vive1(struct xrt_prober *xp,
|
||||||
size_t device_count,
|
size_t device_count,
|
||||||
enum u_logging_level log_level,
|
enum u_logging_level log_level,
|
||||||
struct vive_tracking_status tstatus,
|
struct vive_tracking_status tstatus,
|
||||||
|
struct vive_source *vs,
|
||||||
struct xrt_device **out_xdev)
|
struct xrt_device **out_xdev)
|
||||||
{
|
{
|
||||||
log_vive_device(log_level, xp, dev);
|
log_vive_device(log_level, xp, dev);
|
||||||
|
@ -120,7 +121,7 @@ init_vive1(struct xrt_prober *xp,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
struct vive_device *d =
|
struct vive_device *d =
|
||||||
vive_device_create(mainboard_dev, sensors_dev, watchman_dev, VIVE_VARIANT_VIVE, tstatus);
|
vive_device_create(mainboard_dev, sensors_dev, watchman_dev, VIVE_VARIANT_VIVE, tstatus, vs);
|
||||||
if (d == NULL) {
|
if (d == NULL) {
|
||||||
free(sensors_dev);
|
free(sensors_dev);
|
||||||
free(mainboard_dev);
|
free(mainboard_dev);
|
||||||
|
@ -139,6 +140,7 @@ init_vive_pro(struct xrt_prober *xp,
|
||||||
size_t device_count,
|
size_t device_count,
|
||||||
enum u_logging_level log_level,
|
enum u_logging_level log_level,
|
||||||
struct vive_tracking_status tstatus,
|
struct vive_tracking_status tstatus,
|
||||||
|
struct vive_source *vs,
|
||||||
struct xrt_device **out_xdev)
|
struct xrt_device **out_xdev)
|
||||||
{
|
{
|
||||||
XRT_TRACE_MARKER();
|
XRT_TRACE_MARKER();
|
||||||
|
@ -195,7 +197,8 @@ init_vive_pro(struct xrt_prober *xp,
|
||||||
free(sensors_dev);
|
free(sensors_dev);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
struct vive_device *d = vive_device_create(mainboard_dev, sensors_dev, watchman_dev, VIVE_VARIANT_PRO, tstatus);
|
struct vive_device *d =
|
||||||
|
vive_device_create(mainboard_dev, sensors_dev, watchman_dev, VIVE_VARIANT_PRO, tstatus, vs);
|
||||||
if (d == NULL) {
|
if (d == NULL) {
|
||||||
free(sensors_dev);
|
free(sensors_dev);
|
||||||
free(mainboard_dev);
|
free(mainboard_dev);
|
||||||
|
@ -214,6 +217,7 @@ init_valve_index(struct xrt_prober *xp,
|
||||||
size_t device_count,
|
size_t device_count,
|
||||||
enum u_logging_level log_level,
|
enum u_logging_level log_level,
|
||||||
struct vive_tracking_status tstatus,
|
struct vive_tracking_status tstatus,
|
||||||
|
struct vive_source *vs,
|
||||||
struct vive_config **out_vive_config,
|
struct vive_config **out_vive_config,
|
||||||
struct xrt_device **out_xdevs)
|
struct xrt_device **out_xdevs)
|
||||||
{
|
{
|
||||||
|
@ -252,7 +256,7 @@ init_valve_index(struct xrt_prober *xp,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct vive_device *d = vive_device_create(NULL, sensors_dev, watchman_dev, VIVE_VARIANT_INDEX, tstatus);
|
struct vive_device *d = vive_device_create(NULL, sensors_dev, watchman_dev, VIVE_VARIANT_INDEX, tstatus, vs);
|
||||||
if (d == NULL) {
|
if (d == NULL) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -272,6 +276,7 @@ vive_found(struct xrt_prober *xp,
|
||||||
size_t index,
|
size_t index,
|
||||||
cJSON *attached_data,
|
cJSON *attached_data,
|
||||||
struct vive_tracking_status tstatus,
|
struct vive_tracking_status tstatus,
|
||||||
|
struct vive_source *vs,
|
||||||
struct vive_config **out_vive_config,
|
struct vive_config **out_vive_config,
|
||||||
struct xrt_device **out_xdev)
|
struct xrt_device **out_xdev)
|
||||||
{
|
{
|
||||||
|
@ -289,10 +294,12 @@ vive_found(struct xrt_prober *xp,
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (dev->product_id) {
|
switch (dev->product_id) {
|
||||||
case VIVE_PID: return init_vive1(xp, dev, devices, device_count, log_level, tstatus, out_xdev);
|
case VIVE_PID: return init_vive1(xp, dev, devices, device_count, log_level, tstatus, vs, out_xdev);
|
||||||
case VIVE_PRO_MAINBOARD_PID: return init_vive_pro(xp, dev, devices, device_count, log_level, tstatus, out_xdev);
|
case VIVE_PRO_MAINBOARD_PID:
|
||||||
|
return init_vive_pro(xp, dev, devices, device_count, log_level, tstatus, vs, out_xdev);
|
||||||
case VIVE_PRO_LHR_PID:
|
case VIVE_PRO_LHR_PID:
|
||||||
return init_valve_index(xp, dev, devices, device_count, log_level, tstatus, out_vive_config, out_xdev);
|
return init_valve_index(xp, dev, devices, device_count, log_level, tstatus, vs, out_vive_config,
|
||||||
|
out_xdev);
|
||||||
default: U_LOG_E("No product ids matched %.4x", dev->product_id); return 0;
|
default: U_LOG_E("No product ids matched %.4x", dev->product_id); return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -54,6 +54,7 @@ vive_found(struct xrt_prober *xp,
|
||||||
size_t index,
|
size_t index,
|
||||||
cJSON *attached_data,
|
cJSON *attached_data,
|
||||||
struct vive_tracking_status tstatus,
|
struct vive_tracking_status tstatus,
|
||||||
|
struct vive_source *vs,
|
||||||
struct vive_config **out_vive_config,
|
struct vive_config **out_vive_config,
|
||||||
struct xrt_device **out_xdev);
|
struct xrt_device **out_xdev);
|
||||||
|
|
||||||
|
|
240
src/xrt/drivers/vive/vive_source.c
Normal file
240
src/xrt/drivers/vive/vive_source.c
Normal file
|
@ -0,0 +1,240 @@
|
||||||
|
// Copyright 2022, Collabora, Ltd.
|
||||||
|
// SPDX-License-Identifier: BSL-1.0
|
||||||
|
/*!
|
||||||
|
* @file
|
||||||
|
* @brief Interface for vive data sources
|
||||||
|
* @author Mateo de Mayo <mateo.demayo@collabora.com>
|
||||||
|
* @ingroup drv_vive
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "os/os_threading.h"
|
||||||
|
#include "util/u_deque.h"
|
||||||
|
#include "util/u_logging.h"
|
||||||
|
#include "xrt/xrt_frame.h"
|
||||||
|
#include "xrt/xrt_tracking.h"
|
||||||
|
|
||||||
|
#include "vive.h"
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Manages the data streaming state related to a vive headset.
|
||||||
|
*
|
||||||
|
* @implements xrt_frame_node
|
||||||
|
*/
|
||||||
|
struct vive_source
|
||||||
|
{
|
||||||
|
struct xrt_frame_node node;
|
||||||
|
enum u_logging_level log_level;
|
||||||
|
|
||||||
|
// Sinks
|
||||||
|
struct xrt_frame_sink sbs_sink; //!< Intermediate sink for SBS frames
|
||||||
|
struct xrt_imu_sink imu_sink; //!< Intermediate sink for IMU samples
|
||||||
|
struct xrt_slam_sinks in_sinks; //!< Pointers to intermediate sinks
|
||||||
|
struct xrt_slam_sinks out_sinks; //!< Pointers to downstream sinks
|
||||||
|
|
||||||
|
// V4L2 frame streaming state
|
||||||
|
bool timestamps_have_been_zero_until_now; //!< First v4l2 frames are zeroed
|
||||||
|
bool waiting_for_first_nonempty_frame; //!< Whether the first good frame has been received
|
||||||
|
|
||||||
|
// Frame timestamps
|
||||||
|
struct u_deque_timepoint_ns frame_timestamps; //! Queue of yet unused frame hw timestamps
|
||||||
|
struct os_mutex frame_timestamps_lock; //! Lock for accessing frame_timestamps
|
||||||
|
uint32_t last_frame_ticks; //! Last frame timestamp in device ticks
|
||||||
|
timepoint_ns last_frame_ts_ns; //! Last frame timestamp in device nanoseconds
|
||||||
|
|
||||||
|
// Clock offsets
|
||||||
|
time_duration_ns hw2mono; //!< Estimated offset from IMU to monotonic clock
|
||||||
|
time_duration_ns hw2v4l2; //!< Estimated offset from IMU to V4L2 clock
|
||||||
|
};
|
||||||
|
|
||||||
|
//! Given a sample from two timestamp domains a and b that should have been
|
||||||
|
//! sampled as close as possible, together with an estimate of the offset
|
||||||
|
//! between a clock and b clock (or zero), it applies a smoothing average on the
|
||||||
|
//! estimated offset and returns a in b clock.
|
||||||
|
//! @todo Copy of clock_hw2mono in wmr_source.c, unify into a utility.
|
||||||
|
static inline timepoint_ns
|
||||||
|
clock_offset_a2b(double freq, timepoint_ns a, timepoint_ns b, time_duration_ns *inout_a2b)
|
||||||
|
{
|
||||||
|
// Totally arbitrary way of computing alpha, if you have a better one, replace it
|
||||||
|
const double alpha = 1.0 - 12.5 / freq; // Weight to put on accumulated a2b
|
||||||
|
time_duration_ns old_a2b = *inout_a2b;
|
||||||
|
time_duration_ns got_a2b = b - a;
|
||||||
|
time_duration_ns new_a2b = old_a2b * alpha + got_a2b * (1.0 - alpha);
|
||||||
|
if (old_a2b == 0) { // a2b has not been set yet
|
||||||
|
new_a2b = got_a2b;
|
||||||
|
}
|
||||||
|
*inout_a2b = new_a2b;
|
||||||
|
return a + new_a2b;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
*
|
||||||
|
* Vive source methods
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
//! Find the best corresponding hw timestamp from this v4l2 frame, return
|
||||||
|
//! whether it was found.
|
||||||
|
bool
|
||||||
|
vive_source_try_convert_v4l2_timestamp(struct vive_source *vs, struct xrt_frame *xf)
|
||||||
|
{
|
||||||
|
assert(xf->timestamp != 0 || vs->timestamps_have_been_zero_until_now);
|
||||||
|
if (xf->timestamp == 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
vs->timestamps_have_been_zero_until_now = false;
|
||||||
|
|
||||||
|
struct u_deque_timepoint_ns vive_timestamps = vs->frame_timestamps;
|
||||||
|
struct os_mutex *vive_timestamps_lock = &vs->frame_timestamps_lock;
|
||||||
|
|
||||||
|
timepoint_ns v4l2_ts = xf->timestamp;
|
||||||
|
|
||||||
|
size_t vive_ts_count = u_deque_timepoint_ns_size(vive_timestamps);
|
||||||
|
if (vive_ts_count == 0) { // This seems to happen in some runs
|
||||||
|
// This code assumes vive_timestamps will always be populated before v4l2
|
||||||
|
// receives a frame, thus if we reach this, this assumption has failed.
|
||||||
|
VIVE_ERROR(vs, "Received a v4l2 frame but thwere are no vive timestamps to use");
|
||||||
|
VIVE_ERROR(vs, "Will continue, but frame timestamps could be off by one");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
os_mutex_lock(vive_timestamps_lock);
|
||||||
|
|
||||||
|
// Find i in vive_timestamps that would be closer to xf->timestamp in v4l2 clock
|
||||||
|
int closer_i = -1;
|
||||||
|
timepoint_ns vive_ts = -1;
|
||||||
|
time_duration_ns min_distance = INT64_MAX;
|
||||||
|
for (size_t i = 0; i < vive_ts_count; i++) {
|
||||||
|
vive_ts = u_deque_timepoint_ns_at(vive_timestamps, i);
|
||||||
|
timepoint_ns v4l2_ts_est = vive_ts + vs->hw2v4l2;
|
||||||
|
time_duration_ns distance = llabs(v4l2_ts_est - v4l2_ts);
|
||||||
|
if (distance < min_distance) {
|
||||||
|
closer_i = i;
|
||||||
|
min_distance = distance;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Discard missed frames and set vive_timestamp to use in this frame
|
||||||
|
timepoint_ns vive_timestamp = 0;
|
||||||
|
for (; closer_i >= 0; closer_i--) {
|
||||||
|
u_deque_timepoint_ns_pop_front(vive_timestamps, &vive_timestamp);
|
||||||
|
}
|
||||||
|
|
||||||
|
os_mutex_unlock(vive_timestamps_lock);
|
||||||
|
|
||||||
|
// Our estimate is within a reasonable time distance
|
||||||
|
assert(min_distance < U_TIME_1S_IN_NS / CAMERA_FREQUENCY || vs->waiting_for_first_nonempty_frame);
|
||||||
|
vs->waiting_for_first_nonempty_frame = false;
|
||||||
|
|
||||||
|
// Update estimate of hw2v4l2 clock offset, only used for matching timestamps
|
||||||
|
clock_offset_a2b(CAMERA_FREQUENCY, vive_timestamp, xf->timestamp, &vs->hw2v4l2);
|
||||||
|
|
||||||
|
// Use vive_timestamp and put it in monotonic clock
|
||||||
|
xf->timestamp = vive_timestamp + vs->hw2mono; // Notice that we don't use hw2v4l2
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
vive_source_receive_sbs_frame(struct xrt_frame_sink *sink, struct xrt_frame *xf)
|
||||||
|
{
|
||||||
|
struct vive_source *vs = container_of(sink, struct vive_source, sbs_sink);
|
||||||
|
bool should_push = vive_source_try_convert_v4l2_timestamp(vs, xf);
|
||||||
|
|
||||||
|
if (!should_push) {
|
||||||
|
VIVE_TRACE(vs, "skipped sbs img t=%ld source_t=%ld", xf->timestamp, xf->source_timestamp);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
VIVE_TRACE(vs, "sbs img t=%ld source_t=%ld", xf->timestamp, xf->source_timestamp);
|
||||||
|
|
||||||
|
if (vs->out_sinks.left) { // The split into left right will happen downstream
|
||||||
|
xrt_sink_push_frame(vs->out_sinks.left, xf);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
vive_source_receive_imu_sample(struct xrt_imu_sink *sink, struct xrt_imu_sample *s)
|
||||||
|
{
|
||||||
|
struct vive_source *vs = container_of(sink, struct vive_source, imu_sink);
|
||||||
|
s->timestamp_ns = clock_offset_a2b(IMU_FREQUENCY, s->timestamp_ns, os_monotonic_get_ns(), &vs->hw2mono);
|
||||||
|
timepoint_ns ts = s->timestamp_ns;
|
||||||
|
struct xrt_vec3_f64 a = s->accel_m_s2;
|
||||||
|
struct xrt_vec3_f64 w = s->gyro_rad_secs;
|
||||||
|
VIVE_TRACE(vs, "imu t=%ld a=(%f %f %f) w=(%f %f %f)", ts, a.x, a.y, a.z, w.x, w.y, w.z);
|
||||||
|
|
||||||
|
if (vs->out_sinks.imu) {
|
||||||
|
xrt_sink_push_imu(vs->out_sinks.imu, s);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
vive_source_node_break_apart(struct xrt_frame_node *node)
|
||||||
|
{}
|
||||||
|
|
||||||
|
static void
|
||||||
|
vive_source_node_destroy(struct xrt_frame_node *node)
|
||||||
|
{
|
||||||
|
struct vive_source *vs = container_of(node, struct vive_source, node);
|
||||||
|
os_mutex_destroy(&vs->frame_timestamps_lock);
|
||||||
|
u_deque_timepoint_ns_destroy(&vs->frame_timestamps);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*!
|
||||||
|
*
|
||||||
|
* Exported functions
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
struct vive_source *
|
||||||
|
vive_source_create(struct xrt_frame_context *xfctx)
|
||||||
|
{
|
||||||
|
struct vive_source *vs = U_TYPED_CALLOC(struct vive_source);
|
||||||
|
vs->log_level = debug_get_log_option_vive_log();
|
||||||
|
|
||||||
|
// Setup sinks
|
||||||
|
vs->sbs_sink.push_frame = vive_source_receive_sbs_frame;
|
||||||
|
vs->imu_sink.push_imu = vive_source_receive_imu_sample;
|
||||||
|
vs->in_sinks.left = &vs->sbs_sink;
|
||||||
|
vs->in_sinks.right = NULL;
|
||||||
|
vs->in_sinks.imu = &vs->imu_sink;
|
||||||
|
|
||||||
|
vs->timestamps_have_been_zero_until_now = true;
|
||||||
|
vs->waiting_for_first_nonempty_frame = true;
|
||||||
|
|
||||||
|
vs->frame_timestamps = u_deque_timepoint_ns_create();
|
||||||
|
os_mutex_init(&vs->frame_timestamps_lock);
|
||||||
|
|
||||||
|
// Setup node
|
||||||
|
struct xrt_frame_node *xfn = &vs->node;
|
||||||
|
xfn->break_apart = vive_source_node_break_apart;
|
||||||
|
xfn->destroy = vive_source_node_destroy;
|
||||||
|
xrt_frame_context_add(xfctx, &vs->node);
|
||||||
|
|
||||||
|
VIVE_DEBUG(vs, "Vive source created");
|
||||||
|
|
||||||
|
return vs;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
vive_source_push_imu_packet(struct vive_source *vs, timepoint_ns t, struct xrt_vec3 a, struct xrt_vec3 g)
|
||||||
|
{
|
||||||
|
struct xrt_vec3_f64 a64 = {a.x, a.y, a.z};
|
||||||
|
struct xrt_vec3_f64 g64 = {g.x, g.y, g.z};
|
||||||
|
struct xrt_imu_sample sample = {.timestamp_ns = t, .accel_m_s2 = a64, .gyro_rad_secs = g64};
|
||||||
|
xrt_sink_push_imu(&vs->imu_sink, &sample);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
vive_source_push_frame_ticks(struct vive_source *vs, timepoint_ns ticks)
|
||||||
|
{
|
||||||
|
ticks_to_ns(ticks, &vs->last_frame_ticks, &vs->last_frame_ts_ns);
|
||||||
|
u_deque_timepoint_ns_push_back(vs->frame_timestamps, vs->last_frame_ts_ns);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
vive_source_hook_into_sinks(struct vive_source *vs, struct xrt_slam_sinks *sinks)
|
||||||
|
{
|
||||||
|
vs->out_sinks = *sinks;
|
||||||
|
sinks->left = vs->in_sinks.left;
|
||||||
|
}
|
45
src/xrt/drivers/vive/vive_source.h
Normal file
45
src/xrt/drivers/vive/vive_source.h
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
// Copyright 2022, Collabora, Ltd.
|
||||||
|
// SPDX-License-Identifier: BSL-1.0
|
||||||
|
/*!
|
||||||
|
* @file
|
||||||
|
* @brief Interface for vive data sources
|
||||||
|
* @author Mateo de Mayo <mateo.demayo@collabora.com>
|
||||||
|
* @ingroup drv_vive
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "xrt/xrt_frame.h"
|
||||||
|
#include "xrt/xrt_tracking.h"
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Vive data sources
|
||||||
|
*
|
||||||
|
* @addtogroup drv_vive
|
||||||
|
* @{
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
struct vive_source *
|
||||||
|
vive_source_create(struct xrt_frame_context *xfctx);
|
||||||
|
|
||||||
|
void
|
||||||
|
vive_source_push_imu_packet(struct vive_source *vs, timepoint_ns t, struct xrt_vec3 a, struct xrt_vec3 g);
|
||||||
|
|
||||||
|
void
|
||||||
|
vive_source_push_frame_ticks(struct vive_source *vs, timepoint_ns ticks);
|
||||||
|
|
||||||
|
void
|
||||||
|
vive_source_hook_into_sinks(struct vive_source *vs, struct xrt_slam_sinks *sinks);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* @}
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
|
@ -36,6 +36,7 @@
|
||||||
#ifdef XRT_BUILD_DRIVER_VIVE
|
#ifdef XRT_BUILD_DRIVER_VIVE
|
||||||
#include "vive/vive_prober.h"
|
#include "vive/vive_prober.h"
|
||||||
#include "vive/vive_device.h"
|
#include "vive/vive_device.h"
|
||||||
|
#include "vive/vive_source.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef XRT_BUILD_DRIVER_SURVIVE
|
#ifdef XRT_BUILD_DRIVER_SURVIVE
|
||||||
|
@ -94,7 +95,6 @@ struct index_camera_finder
|
||||||
{
|
{
|
||||||
struct xrt_fs *xfs;
|
struct xrt_fs *xfs;
|
||||||
struct xrt_frame_context *xfctx;
|
struct xrt_frame_context *xfctx;
|
||||||
bool found;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -124,19 +124,6 @@ get_selected_mode(struct xrt_fs *xfs)
|
||||||
return selected_mode;
|
return selected_mode;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool
|
|
||||||
visual_inertial_stream_start(struct xrt_fs *xfs, struct xrt_slam_sinks *sinks)
|
|
||||||
{
|
|
||||||
// Stream frames
|
|
||||||
struct xrt_frame_sink *sbs_sink = sinks->left;
|
|
||||||
bool success = xrt_fs_stream_start(xfs, sbs_sink, XRT_FS_CAPTURE_TYPE_TRACKING, get_selected_mode(xfs));
|
|
||||||
if (!success) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
on_video_device(struct xrt_prober *xp,
|
on_video_device(struct xrt_prober *xp,
|
||||||
struct xrt_prober_device *pdev,
|
struct xrt_prober_device *pdev,
|
||||||
|
@ -462,13 +449,18 @@ stream_data_sources(struct u_system_devices *usysd, struct xrt_prober *xp, struc
|
||||||
}
|
}
|
||||||
|
|
||||||
bool success = false;
|
bool success = false;
|
||||||
if (lhs.slam_enabled) {
|
|
||||||
LH_ASSERT(false, "Not implemented");
|
|
||||||
} else {
|
|
||||||
uint32_t mode = get_selected_mode(finder.xfs);
|
uint32_t mode = get_selected_mode(finder.xfs);
|
||||||
success = xrt_fs_stream_start(finder.xfs, sinks.left, XRT_FS_CAPTURE_TYPE_TRACKING, mode);
|
|
||||||
|
// If SLAM is enabled (only on vive driver) we intercept the data sink
|
||||||
|
if (lhs.slam_enabled) {
|
||||||
|
struct vive_device *d = (struct vive_device *)usysd->base.roles.head;
|
||||||
|
LH_ASSERT_(d != NULL && d->source != NULL);
|
||||||
|
struct vive_source *vs = d->source;
|
||||||
|
vive_source_hook_into_sinks(vs, &sinks);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
success = xrt_fs_stream_start(finder.xfs, sinks.left, XRT_FS_CAPTURE_TYPE_TRACKING, mode);
|
||||||
|
|
||||||
if (!success) {
|
if (!success) {
|
||||||
LH_ERROR("Unable to start data streaming");
|
LH_ERROR("Unable to start data streaming");
|
||||||
xrt_frame_context_destroy_nodes(&usysd->xfctx);
|
xrt_frame_context_destroy_nodes(&usysd->xfctx);
|
||||||
|
@ -551,7 +543,8 @@ lighthouse_open_system(struct xrt_builder *xb,
|
||||||
case VIVE_PID:
|
case VIVE_PID:
|
||||||
case VIVE_PRO_MAINBOARD_PID:
|
case VIVE_PRO_MAINBOARD_PID:
|
||||||
case VIVE_PRO_LHR_PID: {
|
case VIVE_PRO_LHR_PID: {
|
||||||
int num_devices = vive_found(xp, xpdevs, xpdev_count, i, NULL, tstatus, &hmd_config,
|
struct vive_source *vs = vive_source_create(&usysd->xfctx);
|
||||||
|
int num_devices = vive_found(xp, xpdevs, xpdev_count, i, NULL, tstatus, vs, &hmd_config,
|
||||||
&usysd->base.xdevs[usysd->base.xdev_count]);
|
&usysd->base.xdevs[usysd->base.xdev_count]);
|
||||||
usysd->base.xdev_count += num_devices;
|
usysd->base.xdev_count += num_devices;
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue