2019-09-25 00:49:21 +00:00
|
|
|
// Copyright 2019, Collabora, Ltd.
|
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief OpenCV calibration helpers.
|
|
|
|
* @author Pete Black <pblack@collabora.com>
|
2019-10-12 11:24:19 +00:00
|
|
|
* @ingroup aux_tracking
|
2019-09-25 00:49:21 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2019-09-29 14:30:11 +00:00
|
|
|
#include "tracking/t_tracking.h"
|
|
|
|
|
2019-09-25 00:49:21 +00:00
|
|
|
#include <opencv2/opencv.hpp>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2019-09-29 14:30:11 +00:00
|
|
|
/*!
|
|
|
|
* What calibration data that is saved down to file.
|
|
|
|
*/
|
|
|
|
struct CalibrationRawData : t_calibration_raw_data
|
2019-09-25 00:49:21 +00:00
|
|
|
{
|
2019-10-12 20:43:26 +00:00
|
|
|
public:
|
|
|
|
cv::Mat camera_rotation_mat = {};
|
|
|
|
cv::Mat camera_translation_mat = {};
|
|
|
|
cv::Mat camera_essential_mat = {};
|
|
|
|
cv::Mat camera_fundamental_mat = {};
|
|
|
|
|
|
|
|
cv::Mat disparity_to_depth_mat = {};
|
|
|
|
|
|
|
|
cv::Mat l_intrinsics_mat = {};
|
|
|
|
cv::Mat l_distortion_mat = {};
|
|
|
|
cv::Mat l_distortion_fisheye_mat = {};
|
|
|
|
cv::Mat l_rotation_mat = {};
|
|
|
|
cv::Mat l_projection_mat = {};
|
|
|
|
cv::Mat r_intrinsics_mat = {};
|
|
|
|
cv::Mat r_distortion_mat = {};
|
|
|
|
cv::Mat r_distortion_fisheye_mat = {};
|
|
|
|
cv::Mat r_rotation_mat = {};
|
|
|
|
cv::Mat r_projection_mat = {};
|
|
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
CalibrationRawData()
|
|
|
|
{
|
|
|
|
// clang-format off
|
|
|
|
camera_translation_mat = cv::Mat(3, 1, CV_64F, &camera_translation[0]);
|
2019-10-13 09:48:07 +00:00
|
|
|
camera_rotation_mat = cv::Mat(3, 3, CV_64F, &camera_rotation[0][0]);
|
|
|
|
camera_essential_mat = cv::Mat(3, 3, CV_64F, &camera_essential[0][0]);
|
|
|
|
camera_fundamental_mat = cv::Mat(3, 3, CV_64F, &camera_fundamental[0][0]);
|
|
|
|
|
|
|
|
l_intrinsics_mat = cv::Mat(3, 3, CV_64F, &l_intrinsics[0][0]);
|
|
|
|
l_distortion_mat = cv::Mat(1, 5, CV_64F, &l_distortion[0]);
|
2019-11-14 17:05:16 +00:00
|
|
|
l_distortion_fisheye_mat = cv::Mat(1, 4, CV_64F, &l_distortion_fisheye[0]);
|
2019-10-13 09:48:07 +00:00
|
|
|
l_rotation_mat = cv::Mat(3, 3, CV_64F, &l_rotation[0][0]);
|
|
|
|
l_projection_mat = cv::Mat(3, 4, CV_64F, &l_projection[0][0]);
|
|
|
|
|
|
|
|
r_intrinsics_mat = cv::Mat(3, 3, CV_64F, &r_intrinsics[0][0]);
|
|
|
|
r_distortion_mat = cv::Mat(1, 5, CV_64F, &r_distortion[0]);
|
2019-11-14 17:05:16 +00:00
|
|
|
r_distortion_fisheye_mat = cv::Mat(1, 4, CV_64F, &r_distortion_fisheye[0]);
|
2019-10-13 09:48:07 +00:00
|
|
|
r_rotation_mat = cv::Mat(3, 3, CV_64F, &r_rotation[0][0]);
|
|
|
|
r_projection_mat = cv::Mat(3, 4, CV_64F, &r_projection[0][0]);
|
2019-10-12 20:43:26 +00:00
|
|
|
// clang-format on
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
isDataStorageValid()
|
|
|
|
{
|
|
|
|
return camera_rotation_mat.size() == cv::Size(3, 3) &&
|
|
|
|
camera_translation_mat.size() == cv::Size(1, 3) &&
|
|
|
|
camera_essential_mat.size() == cv::Size(3, 3) &&
|
2019-10-13 09:48:07 +00:00
|
|
|
camera_fundamental_mat.size() == cv::Size(3, 3) &&
|
|
|
|
l_intrinsics_mat.size() == cv::Size(3, 3) &&
|
|
|
|
l_distortion_mat.size() == cv::Size(5, 1) &&
|
2019-11-14 17:05:16 +00:00
|
|
|
l_distortion_fisheye_mat.size() == cv::Size(4, 1) &&
|
2019-10-13 09:48:07 +00:00
|
|
|
l_rotation_mat.size() == cv::Size(3, 3) &&
|
|
|
|
l_projection_mat.size() == cv::Size(4, 3) &&
|
|
|
|
r_intrinsics_mat.size() == cv::Size(3, 3) &&
|
|
|
|
r_distortion_mat.size() == cv::Size(5, 1) &&
|
2019-11-14 17:05:16 +00:00
|
|
|
r_distortion_fisheye_mat.size() == cv::Size(4, 1) &&
|
2019-10-13 09:48:07 +00:00
|
|
|
r_rotation_mat.size() == cv::Size(3, 3) &&
|
|
|
|
r_projection_mat.size() == cv::Size(4, 3);
|
2019-10-12 20:43:26 +00:00
|
|
|
}
|
2019-09-25 00:49:21 +00:00
|
|
|
};
|
|
|
|
|
2019-09-29 14:30:11 +00:00
|
|
|
struct CalibrationData : t_calibration_data
|
2019-09-25 00:49:21 +00:00
|
|
|
{
|
2019-09-29 14:30:11 +00:00
|
|
|
cv::Mat l_undistort_map_x = {};
|
|
|
|
cv::Mat l_undistort_map_y = {};
|
|
|
|
cv::Mat l_rectify_map_x = {};
|
|
|
|
cv::Mat l_rectify_map_y = {};
|
|
|
|
cv::Mat r_undistort_map_x = {};
|
|
|
|
cv::Mat r_undistort_map_y = {};
|
|
|
|
cv::Mat r_rectify_map_x = {};
|
|
|
|
cv::Mat r_rectify_map_y = {};
|
|
|
|
cv::Mat disparity_to_depth = {};
|
|
|
|
};
|
2019-09-27 11:28:37 +00:00
|
|
|
|
2019-09-29 14:30:11 +00:00
|
|
|
extern "C" bool
|
|
|
|
t_file_load_stereo_calibration_v1_hack(struct t_calibration_data **out_data);
|
2019-09-27 11:28:37 +00:00
|
|
|
|
2019-09-29 14:30:11 +00:00
|
|
|
extern "C" bool
|
|
|
|
t_file_save_raw_data_hack(struct t_calibration_raw_data *raw_data);
|
2019-09-25 00:49:21 +00:00
|
|
|
|
|
|
|
XRT_MAYBE_UNUSED static bool
|
2019-09-29 10:43:08 +00:00
|
|
|
calibration_get_stereo(const char *configuration_filename,
|
2019-09-25 00:49:21 +00:00
|
|
|
uint32_t frame_w,
|
|
|
|
uint32_t frame_h,
|
|
|
|
bool use_fisheye,
|
2019-09-29 10:43:08 +00:00
|
|
|
cv::Mat *l_undistort_map_x,
|
|
|
|
cv::Mat *l_undistort_map_y,
|
|
|
|
cv::Mat *l_rectify_map_x,
|
|
|
|
cv::Mat *l_rectify_map_y,
|
|
|
|
cv::Mat *r_undistort_map_x,
|
|
|
|
cv::Mat *r_undistort_map_y,
|
|
|
|
cv::Mat *r_rectify_map_x,
|
|
|
|
cv::Mat *r_rectify_map_y,
|
|
|
|
cv::Mat *disparity_to_depth)
|
2019-09-25 00:49:21 +00:00
|
|
|
{
|
2019-09-29 14:30:11 +00:00
|
|
|
t_calibration_data *data_c;
|
|
|
|
bool ok = t_file_load_stereo_calibration_v1_hack(&data_c);
|
2019-09-25 00:49:21 +00:00
|
|
|
|
2019-09-29 14:30:11 +00:00
|
|
|
if (!ok) {
|
2019-09-25 00:49:21 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-09-29 14:30:11 +00:00
|
|
|
CalibrationData *data = (CalibrationData *)data_c;
|
2019-09-25 00:49:21 +00:00
|
|
|
|
2019-09-29 14:30:11 +00:00
|
|
|
*l_undistort_map_x = data->l_undistort_map_x;
|
|
|
|
*l_undistort_map_y = data->l_undistort_map_y;
|
|
|
|
*l_rectify_map_x = data->l_rectify_map_x;
|
|
|
|
*l_rectify_map_y = data->l_rectify_map_y;
|
|
|
|
*r_undistort_map_x = data->r_undistort_map_x;
|
|
|
|
*r_undistort_map_y = data->r_undistort_map_y;
|
|
|
|
*r_rectify_map_x = data->r_rectify_map_x;
|
|
|
|
*r_rectify_map_y = data->r_rectify_map_y;
|
|
|
|
*disparity_to_depth = data->disparity_to_depth;
|
2019-09-25 00:49:21 +00:00
|
|
|
|
2019-09-29 14:30:11 +00:00
|
|
|
t_calibration_data_free(data_c);
|
2019-09-25 00:49:21 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! @todo Move this as it is a generic helper
|
2019-09-27 11:28:37 +00:00
|
|
|
XRT_MAYBE_UNUSED static int
|
2019-09-29 10:43:08 +00:00
|
|
|
mkpath(char *path)
|
2019-09-25 00:49:21 +00:00
|
|
|
{
|
|
|
|
char tmp[PATH_MAX]; //!< @todo PATH_MAX probably not strictly correct
|
2019-09-29 10:43:08 +00:00
|
|
|
char *p = nullptr;
|
2019-09-25 00:49:21 +00:00
|
|
|
size_t len;
|
|
|
|
|
|
|
|
snprintf(tmp, sizeof(tmp), "%s", path);
|
|
|
|
len = strlen(tmp) - 1;
|
|
|
|
if (tmp[len] == '/') {
|
|
|
|
tmp[len] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (p = tmp + 1; *p; p++) {
|
|
|
|
if (*p == '/') {
|
|
|
|
*p = 0;
|
|
|
|
if (mkdir(tmp, S_IRWXU) < 0 && errno != EEXIST)
|
|
|
|
return -1;
|
|
|
|
*p = '/';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mkdir(tmp, S_IRWXU) < 0 && errno != EEXIST) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|