t/euroc: Add euroc dataset recorder

This commit is contained in:
Mateo de Mayo 2021-11-30 09:57:21 -03:00 committed by Jakob Bornecrantz
parent e245f4e5a9
commit 91ecbdb7fb
4 changed files with 294 additions and 0 deletions

View file

@ -118,6 +118,8 @@ if(XRT_HAVE_OPENCV)
tracking/t_debug_hsv_filter.cpp tracking/t_debug_hsv_filter.cpp
tracking/t_debug_hsv_picker.cpp tracking/t_debug_hsv_picker.cpp
tracking/t_debug_hsv_viewer.cpp tracking/t_debug_hsv_viewer.cpp
tracking/t_euroc_recorder.cpp
tracking/t_euroc_recorder.h
tracking/t_file.cpp tracking/t_file.cpp
tracking/t_frame_cv_mat_wrapper.cpp tracking/t_frame_cv_mat_wrapper.cpp
tracking/t_frame_cv_mat_wrapper.hpp tracking/t_frame_cv_mat_wrapper.hpp

View file

@ -215,6 +215,8 @@ if build_tracking
'tracking/t_debug_hsv_filter.cpp', 'tracking/t_debug_hsv_filter.cpp',
'tracking/t_debug_hsv_picker.cpp', 'tracking/t_debug_hsv_picker.cpp',
'tracking/t_debug_hsv_viewer.cpp', 'tracking/t_debug_hsv_viewer.cpp',
'tracking/t_euroc_recorder.cpp',
'tracking/t_euroc_recorder.h',
'tracking/t_file.cpp', 'tracking/t_file.cpp',
'tracking/t_frame_cv_mat_wrapper.cpp', 'tracking/t_frame_cv_mat_wrapper.cpp',
'tracking/t_frame_cv_mat_wrapper.hpp', 'tracking/t_frame_cv_mat_wrapper.hpp',

View 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, &copy);
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;
}

View 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