mirror of
https://gitlab.freedesktop.org/monado/monado.git
synced 2025-01-19 13:18:32 +00:00
t/euroc: Add euroc dataset recorder
This commit is contained in:
parent
e245f4e5a9
commit
91ecbdb7fb
|
@ -118,6 +118,8 @@ if(XRT_HAVE_OPENCV)
|
|||
tracking/t_debug_hsv_filter.cpp
|
||||
tracking/t_debug_hsv_picker.cpp
|
||||
tracking/t_debug_hsv_viewer.cpp
|
||||
tracking/t_euroc_recorder.cpp
|
||||
tracking/t_euroc_recorder.h
|
||||
tracking/t_file.cpp
|
||||
tracking/t_frame_cv_mat_wrapper.cpp
|
||||
tracking/t_frame_cv_mat_wrapper.hpp
|
||||
|
|
|
@ -215,6 +215,8 @@ if build_tracking
|
|||
'tracking/t_debug_hsv_filter.cpp',
|
||||
'tracking/t_debug_hsv_picker.cpp',
|
||||
'tracking/t_debug_hsv_viewer.cpp',
|
||||
'tracking/t_euroc_recorder.cpp',
|
||||
'tracking/t_euroc_recorder.h',
|
||||
'tracking/t_file.cpp',
|
||||
'tracking/t_frame_cv_mat_wrapper.cpp',
|
||||
'tracking/t_frame_cv_mat_wrapper.hpp',
|
||||
|
|
257
src/xrt/auxiliary/tracking/t_euroc_recorder.cpp
Normal file
257
src/xrt/auxiliary/tracking/t_euroc_recorder.cpp
Normal file
|
@ -0,0 +1,257 @@
|
|||
// Copyright 2021, Collabora, Ltd.
|
||||
// SPDX-License-Identifier: BSL-1.0
|
||||
/*!
|
||||
* @file
|
||||
* @brief EuRoC dataset recorder utility.
|
||||
* @author Mateo de Mayo <mateo.demayo@collabora.com>
|
||||
* @ingroup aux_tracking
|
||||
*/
|
||||
|
||||
#include "t_euroc_recorder.h"
|
||||
|
||||
#include "os/os_time.h"
|
||||
#include "util/u_frame.h"
|
||||
#include "util/u_sink.h"
|
||||
|
||||
#include <cassert>
|
||||
#include <ctime>
|
||||
#include <filesystem>
|
||||
#include <fstream>
|
||||
#include <string>
|
||||
#include <queue>
|
||||
#include <iomanip>
|
||||
|
||||
#include <opencv2/imgcodecs.hpp>
|
||||
|
||||
using std::ofstream;
|
||||
using std::queue;
|
||||
using std::string;
|
||||
using std::filesystem::create_directories;
|
||||
|
||||
struct euroc_recorder
|
||||
{
|
||||
struct xrt_frame_node node;
|
||||
string path; //!< Destination path for the dataset
|
||||
bool first_received; //!< Whether we have received the first sample
|
||||
|
||||
// Cloner sinks: copy frame to heap for quick release of the original
|
||||
struct xrt_slam_sinks cloner_queues; //!< Queue sinks that write into cloner sinks
|
||||
struct xrt_imu_sink cloner_imu_sink;
|
||||
struct xrt_frame_sink cloner_left_sink;
|
||||
struct xrt_frame_sink cloner_right_sink;
|
||||
|
||||
// Writer sinks: write copied frame to disk
|
||||
struct xrt_slam_sinks writer_queues; //!< Queue sinks that write into writer sinks
|
||||
struct xrt_imu_sink writer_imu_sink;
|
||||
struct xrt_frame_sink writer_left_sink;
|
||||
struct xrt_frame_sink writer_right_sink;
|
||||
|
||||
queue<xrt_imu_sample> imu_queue{}; //!< IMU pushes get saved here and are delayed until left_frame pushes
|
||||
};
|
||||
|
||||
|
||||
/*
|
||||
*
|
||||
* Writer sinks functionality
|
||||
*
|
||||
*/
|
||||
|
||||
static void
|
||||
euroc_recorder_try_mkfiles(struct euroc_recorder *er)
|
||||
{
|
||||
// Create directory structure and files only on first received frame
|
||||
if (er->first_received) {
|
||||
return;
|
||||
}
|
||||
er->first_received = true;
|
||||
|
||||
string path = er->path;
|
||||
|
||||
create_directories(path + "/mav0/imu0");
|
||||
ofstream imu_csv{path + "/mav0/imu0/data.csv"};
|
||||
imu_csv << "#timestamp [ns],w_RS_S_x [rad s^-1],w_RS_S_y [rad s^-1],w_RS_S_z [rad s^-1],"
|
||||
"a_RS_S_x [m s^-2],a_RS_S_y [m s^-2],a_RS_S_z [m s^-2]\r\n";
|
||||
|
||||
create_directories(path + "/mav0/cam0/data");
|
||||
ofstream left_cam_csv{path + "/mav0/cam0/data.csv"};
|
||||
left_cam_csv << "#timestamp [ns],filename\r\n";
|
||||
|
||||
create_directories(path + "/mav0/cam1/data");
|
||||
ofstream right_cam_csv{path + "/mav0/cam1/data.csv"};
|
||||
right_cam_csv << "#timestamp [ns],filename\r\n";
|
||||
}
|
||||
|
||||
extern "C" void
|
||||
euroc_recorder_save_imu(xrt_imu_sink *sink, struct xrt_imu_sample *sample)
|
||||
{
|
||||
euroc_recorder *er = container_of(sink, euroc_recorder, writer_imu_sink);
|
||||
euroc_recorder_try_mkfiles(er);
|
||||
|
||||
timepoint_ns ts = sample->timestamp_ns;
|
||||
xrt_vec3_f64 a = sample->accel_m_s2;
|
||||
xrt_vec3_f64 w = sample->gyro_rad_secs;
|
||||
|
||||
std::ofstream imu_csv{string(er->path) + "/mav0/imu0/data.csv", std::ios::app};
|
||||
imu_csv.setf(std::ios::fixed);
|
||||
imu_csv << std::setprecision(20);
|
||||
imu_csv << ts << ",";
|
||||
imu_csv << w.x << "," << w.y << "," << w.z << ",";
|
||||
imu_csv << a.x << "," << a.y << "," << a.z << "\r\n";
|
||||
}
|
||||
|
||||
static void
|
||||
euroc_recorder_save_frame(euroc_recorder *er, struct xrt_frame *frame, bool is_left)
|
||||
{
|
||||
euroc_recorder_try_mkfiles(er);
|
||||
|
||||
string path = string(er->path);
|
||||
string cam_name = is_left ? "cam0" : "cam1";
|
||||
uint64_t ts = frame->timestamp;
|
||||
|
||||
ofstream cam_csv{path + "/mav0/" + cam_name + "/data.csv", std::ios::app};
|
||||
cam_csv << ts << "," << ts << ".png\r\n";
|
||||
|
||||
assert(frame->format == XRT_FORMAT_L8 || frame->format == XRT_FORMAT_R8G8B8); // Only formats supported
|
||||
auto img_type = frame->format == XRT_FORMAT_L8 ? CV_8UC1 : CV_8UC3;
|
||||
string img_path = path + "/mav0/" + cam_name + "/data/" + std::to_string(ts) + ".png";
|
||||
cv::Mat img{(int)frame->height, (int)frame->width, img_type, frame->data};
|
||||
cv::imwrite(img_path, img);
|
||||
}
|
||||
|
||||
extern "C" void
|
||||
euroc_recorder_save_left(struct xrt_frame_sink *sink, struct xrt_frame *frame)
|
||||
{
|
||||
euroc_recorder *er = container_of(sink, euroc_recorder, writer_left_sink);
|
||||
euroc_recorder_save_frame(er, frame, true);
|
||||
|
||||
// Also, write queued IMU samples to disk now.
|
||||
while (!er->imu_queue.empty()) {
|
||||
xrt_imu_sample imu = er->imu_queue.front();
|
||||
xrt_sink_push_imu(&er->writer_imu_sink, &imu);
|
||||
er->imu_queue.pop();
|
||||
}
|
||||
}
|
||||
|
||||
extern "C" void
|
||||
euroc_recorder_save_right(struct xrt_frame_sink *sink, struct xrt_frame *frame)
|
||||
{
|
||||
euroc_recorder *er = container_of(sink, euroc_recorder, writer_right_sink);
|
||||
euroc_recorder_save_frame(er, frame, false);
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
*
|
||||
* Cloner sinks functionality
|
||||
*
|
||||
*/
|
||||
|
||||
extern "C" void
|
||||
euroc_recorder_receive_imu(xrt_imu_sink *sink, struct xrt_imu_sample *sample)
|
||||
{
|
||||
// Contrary to frame sinks, we don't have separately threaded queues for IMU
|
||||
// sinks so we use an std::queue to temporarily store IMU samples, later we
|
||||
// write them to disk when writing left frames.
|
||||
euroc_recorder *er = container_of(sink, euroc_recorder, cloner_imu_sink);
|
||||
er->imu_queue.push(*sample);
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
euroc_recorder_receive_frame(euroc_recorder *er, struct xrt_frame *src_frame, bool is_left)
|
||||
{
|
||||
// Let's clone the frame so that we can release the src_frame quickly
|
||||
xrt_frame *copy = nullptr;
|
||||
u_frame_clone(src_frame, ©);
|
||||
|
||||
xrt_sink_push_frame(is_left ? er->writer_queues.left : er->writer_queues.right, copy);
|
||||
}
|
||||
|
||||
extern "C" void
|
||||
euroc_recorder_receive_left(struct xrt_frame_sink *sink, struct xrt_frame *frame)
|
||||
{
|
||||
euroc_recorder *er = container_of(sink, euroc_recorder, cloner_left_sink);
|
||||
euroc_recorder_receive_frame(er, frame, true);
|
||||
}
|
||||
|
||||
extern "C" void
|
||||
euroc_recorder_receive_right(struct xrt_frame_sink *sink, struct xrt_frame *frame)
|
||||
{
|
||||
euroc_recorder *er = container_of(sink, euroc_recorder, cloner_right_sink);
|
||||
euroc_recorder_receive_frame(er, frame, false);
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
*
|
||||
* Frame node functionality
|
||||
*
|
||||
*/
|
||||
|
||||
extern "C" void
|
||||
euroc_recorder_node_break_apart(struct xrt_frame_node *node)
|
||||
{}
|
||||
|
||||
extern "C" void
|
||||
euroc_recorder_node_destroy(struct xrt_frame_node *node)
|
||||
{
|
||||
struct euroc_recorder *er = container_of(node, struct euroc_recorder, node);
|
||||
delete er;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
*
|
||||
* Exported functions
|
||||
*
|
||||
*/
|
||||
|
||||
extern "C" xrt_slam_sinks *
|
||||
euroc_recorder_create(struct xrt_frame_context *xfctx, const char *record_path)
|
||||
{
|
||||
struct euroc_recorder *er = new euroc_recorder{};
|
||||
|
||||
struct xrt_frame_node *xfn = &er->node;
|
||||
xfn->break_apart = euroc_recorder_node_break_apart;
|
||||
xfn->destroy = euroc_recorder_node_destroy;
|
||||
xrt_frame_context_add(xfctx, xfn);
|
||||
|
||||
// Determine dataset path
|
||||
if (record_path != nullptr) {
|
||||
er->path = record_path;
|
||||
} else {
|
||||
time_t seconds = os_realtime_get_ns() / U_1_000_000_000;
|
||||
constexpr size_t size = sizeof("YYYYMMDDHHmmss");
|
||||
char datetime[size] = {0};
|
||||
strftime(datetime, size, "%Y%m%d%H%M%S", localtime(&seconds));
|
||||
string default_path = string{"euroc_recording_"} + datetime;
|
||||
er->path = default_path;
|
||||
}
|
||||
|
||||
// Setup sink pipeline
|
||||
|
||||
// First, make the public queues that will clone frames in memory so that
|
||||
// original frames can be released as soon as possible. Not doing this could
|
||||
// result in frame queues from the user being filled up.
|
||||
u_sink_queue_create(xfctx, 0, &er->cloner_left_sink, &er->cloner_queues.left);
|
||||
u_sink_queue_create(xfctx, 0, &er->cloner_right_sink, &er->cloner_queues.right);
|
||||
er->cloner_queues.imu = &er->cloner_imu_sink;
|
||||
|
||||
// Clone samples into heap and release original samples right after
|
||||
er->cloner_imu_sink.push_imu = euroc_recorder_receive_imu;
|
||||
er->cloner_left_sink.push_frame = euroc_recorder_receive_left;
|
||||
er->cloner_right_sink.push_frame = euroc_recorder_receive_right;
|
||||
|
||||
// Then, make a queue to save frame sinks to disk in a separate thread
|
||||
u_sink_queue_create(xfctx, 0, &er->writer_left_sink, &er->writer_queues.left);
|
||||
u_sink_queue_create(xfctx, 0, &er->writer_right_sink, &er->writer_queues.right);
|
||||
er->writer_queues.imu = nullptr;
|
||||
|
||||
// Write cloned samples to disk with these
|
||||
er->writer_imu_sink.push_imu = euroc_recorder_save_imu;
|
||||
er->writer_left_sink.push_frame = euroc_recorder_save_left;
|
||||
er->writer_right_sink.push_frame = euroc_recorder_save_right;
|
||||
|
||||
xrt_slam_sinks *public_sinks = &er->cloner_queues;
|
||||
return public_sinks;
|
||||
}
|
33
src/xrt/auxiliary/tracking/t_euroc_recorder.h
Normal file
33
src/xrt/auxiliary/tracking/t_euroc_recorder.h
Normal file
|
@ -0,0 +1,33 @@
|
|||
// Copyright 2021, Collabora, Ltd.
|
||||
// SPDX-License-Identifier: BSL-1.0
|
||||
/*!
|
||||
* @file
|
||||
* @brief EuRoC dataset recorder utility.
|
||||
* @author Mateo de Mayo <mateo.demayo@collabora.com>
|
||||
* @ingroup aux_tracking
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include "xrt/xrt_tracking.h"
|
||||
#include "xrt/xrt_frame.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
/*!
|
||||
* @brief Create SLAM sinks to record samples in EuRoC format.
|
||||
*
|
||||
* @param xfctx Frame context for the sinks.
|
||||
* @param record_path Directory name to save the dataset or NULL for a default based on the current datetime.
|
||||
* @return struct xrt_slam_sinks* Sinks to push samples to for recording.
|
||||
*
|
||||
* @ingroup aux_tracking
|
||||
*/
|
||||
struct xrt_slam_sinks *
|
||||
euroc_recorder_create(struct xrt_frame_context *xfctx, const char *record_path);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
Loading…
Reference in a new issue