2020-01-13 23:30:27 +00:00
|
|
|
// Copyright 2019-2020, Collabora, Ltd.
|
2019-09-25 00:49:21 +00:00
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief OpenCV calibration helpers.
|
|
|
|
* @author Pete Black <pblack@collabora.com>
|
2019-11-16 23:17:11 +00:00
|
|
|
* @author Jakob Bornecrantz <jakob@collabora.com>
|
2020-01-13 23:30:27 +00:00
|
|
|
* @author Ryan Pavlik <ryan.pavlik@collabora.com>
|
2019-10-12 11:24:19 +00:00
|
|
|
* @ingroup aux_tracking
|
2019-09-25 00:49:21 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2019-11-16 23:17:11 +00:00
|
|
|
#ifndef __cplusplus
|
|
|
|
#error "This header is C++-only."
|
|
|
|
#endif
|
|
|
|
|
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>
|
|
|
|
|
|
|
|
|
2019-11-16 23:17:11 +00:00
|
|
|
/*!
|
2020-01-13 23:30:27 +00:00
|
|
|
* Save raw calibration data to file, hack until prober has storage for such
|
|
|
|
* things.
|
2019-11-16 23:17:11 +00:00
|
|
|
*/
|
|
|
|
extern "C" bool
|
2020-01-13 23:30:27 +00:00
|
|
|
t_file_save_raw_data_hack(struct t_stereo_camera_calibration *data);
|
2019-11-16 23:17:11 +00:00
|
|
|
|
|
|
|
/*!
|
2020-01-13 23:30:27 +00:00
|
|
|
* @brief Essential calibration data wrapped for C++.
|
|
|
|
*
|
|
|
|
* Just like the cv::Mat that it holds, this object does not own all the memory
|
|
|
|
* it points to!
|
2019-11-16 23:17:11 +00:00
|
|
|
*/
|
2020-01-13 23:30:27 +00:00
|
|
|
struct CameraCalibrationWrapper
|
2019-11-16 23:17:11 +00:00
|
|
|
{
|
2020-01-18 00:14:06 +00:00
|
|
|
t_camera_calibration &base;
|
2020-01-13 23:30:27 +00:00
|
|
|
xrt_size &image_size_pixels;
|
2020-01-15 16:20:18 +00:00
|
|
|
cv::Mat_<double> intrinsics_mat;
|
|
|
|
cv::Mat_<double> distortion_mat;
|
|
|
|
cv::Mat_<double> distortion_fisheye_mat;
|
2020-01-13 23:30:27 +00:00
|
|
|
bool &use_fisheye;
|
|
|
|
|
|
|
|
CameraCalibrationWrapper(t_camera_calibration &calib)
|
2020-01-18 00:14:06 +00:00
|
|
|
: base(calib), image_size_pixels(calib.image_size_pixels),
|
2020-01-15 16:20:18 +00:00
|
|
|
intrinsics_mat(3, 3, &calib.intrinsics[0][0]),
|
|
|
|
distortion_mat(XRT_DISTORTION_MAX_DIM, 1, &calib.distortion[0]),
|
|
|
|
distortion_fisheye_mat(4, 1, &calib.distortion_fisheye[0]),
|
2020-01-13 23:30:27 +00:00
|
|
|
use_fisheye(calib.use_fisheye)
|
|
|
|
{
|
|
|
|
assert(isDataStorageValid());
|
|
|
|
}
|
2019-11-16 23:17:11 +00:00
|
|
|
|
2020-01-13 23:30:27 +00:00
|
|
|
//! Try to verify nothing was reallocated.
|
|
|
|
bool
|
|
|
|
isDataStorageValid() const noexcept
|
|
|
|
{
|
|
|
|
return intrinsics_mat.size() == cv::Size(3, 3) &&
|
2020-01-18 00:14:06 +00:00
|
|
|
(double *)intrinsics_mat.data ==
|
|
|
|
&(base.intrinsics[0][0]) &&
|
|
|
|
|
2020-01-13 23:30:27 +00:00
|
|
|
distortion_mat.size() ==
|
|
|
|
cv::Size(1, XRT_DISTORTION_MAX_DIM) &&
|
2020-01-18 00:14:06 +00:00
|
|
|
(double *)distortion_mat.data == &(base.distortion[0]) &&
|
|
|
|
|
|
|
|
distortion_fisheye_mat.size() == cv::Size(1, 4) &&
|
|
|
|
(double *)distortion_fisheye_mat.data ==
|
|
|
|
&(base.distortion_fisheye[0]);
|
2020-01-13 23:30:27 +00:00
|
|
|
}
|
|
|
|
};
|
2019-11-16 23:27:32 +00:00
|
|
|
|
|
|
|
|
2020-01-13 23:30:27 +00:00
|
|
|
/*!
|
|
|
|
* @brief Essential stereo calibration data wrapped for C++.
|
|
|
|
*
|
|
|
|
* Just like the cv::Mat that it holds, this object does not own (all) the
|
|
|
|
* memory it points to!
|
|
|
|
*/
|
|
|
|
struct StereoCameraCalibrationWrapper
|
|
|
|
{
|
2020-01-18 00:14:06 +00:00
|
|
|
t_stereo_camera_calibration &base;
|
2020-01-16 14:57:36 +00:00
|
|
|
CameraCalibrationWrapper view[2];
|
2020-01-15 16:20:18 +00:00
|
|
|
cv::Mat_<double> camera_translation_mat;
|
|
|
|
cv::Mat_<double> camera_rotation_mat;
|
|
|
|
cv::Mat_<double> camera_essential_mat;
|
|
|
|
cv::Mat_<double> camera_fundamental_mat;
|
2020-01-13 23:30:27 +00:00
|
|
|
|
|
|
|
StereoCameraCalibrationWrapper(t_stereo_camera_calibration &stereo)
|
2020-01-18 00:14:06 +00:00
|
|
|
: base(stereo), view{CameraCalibrationWrapper{stereo.view[0]},
|
|
|
|
CameraCalibrationWrapper{stereo.view[1]}},
|
2020-01-15 16:20:18 +00:00
|
|
|
camera_translation_mat(3, 1, &stereo.camera_translation[0]),
|
|
|
|
camera_rotation_mat(3, 3, &stereo.camera_rotation[0][0]),
|
|
|
|
camera_essential_mat(3, 3, &stereo.camera_essential[0][0]),
|
|
|
|
camera_fundamental_mat(3, 3, &stereo.camera_fundamental[0][0])
|
2019-11-16 23:27:32 +00:00
|
|
|
{
|
2020-01-13 23:30:27 +00:00
|
|
|
assert(isDataStorageValid());
|
2019-11-16 23:27:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2020-01-13 23:30:27 +00:00
|
|
|
isDataStorageValid() const noexcept
|
2019-11-16 23:27:32 +00:00
|
|
|
{
|
2020-01-13 23:30:27 +00:00
|
|
|
return camera_translation_mat.size() == cv::Size(1, 3) &&
|
2020-01-18 00:14:06 +00:00
|
|
|
(double *)camera_translation_mat.data ==
|
|
|
|
&base.camera_translation[0] &&
|
|
|
|
|
2020-01-13 23:30:27 +00:00
|
|
|
camera_rotation_mat.size() == cv::Size(3, 3) &&
|
2020-01-18 00:14:06 +00:00
|
|
|
(double *)camera_rotation_mat.data ==
|
|
|
|
&base.camera_rotation[0][0] &&
|
2020-01-13 23:30:27 +00:00
|
|
|
|
|
|
|
camera_essential_mat.size() == cv::Size(3, 3) &&
|
2020-01-18 00:14:06 +00:00
|
|
|
(double *)camera_essential_mat.data ==
|
|
|
|
&base.camera_essential[0][0] &&
|
|
|
|
|
2020-01-13 23:30:27 +00:00
|
|
|
camera_fundamental_mat.size() == cv::Size(3, 3) &&
|
2020-01-18 00:14:06 +00:00
|
|
|
(double *)camera_fundamental_mat.data ==
|
|
|
|
&base.camera_fundamental[0][0] &&
|
|
|
|
|
2020-01-16 14:57:36 +00:00
|
|
|
view[0].isDataStorageValid() &&
|
|
|
|
view[1].isDataStorageValid();
|
2019-11-16 23:27:32 +00:00
|
|
|
}
|
2019-11-16 23:17:11 +00:00
|
|
|
};
|
2019-09-25 00:49:21 +00:00
|
|
|
|
2020-01-13 23:30:27 +00:00
|
|
|
|
2019-09-29 14:30:11 +00:00
|
|
|
/*!
|
2020-01-13 23:30:27 +00:00
|
|
|
* @brief An x,y pair of matrices for the remap() function.
|
|
|
|
*
|
|
|
|
* @see calibration_get_undistort_map
|
2019-09-29 14:30:11 +00:00
|
|
|
*/
|
2020-01-13 23:30:27 +00:00
|
|
|
struct RemapPair
|
2019-09-25 00:49:21 +00:00
|
|
|
{
|
2020-01-13 23:30:27 +00:00
|
|
|
cv::Mat remap_x;
|
|
|
|
cv::Mat remap_y;
|
|
|
|
};
|
2019-10-12 20:43:26 +00:00
|
|
|
|
2020-01-13 23:30:27 +00:00
|
|
|
/*!
|
|
|
|
* @brief Prepare undistortion/normalization remap structures for a rectilinear
|
|
|
|
* or fisheye image.
|
|
|
|
*
|
|
|
|
* @param calib A single camera calibration structure.
|
|
|
|
* @param rectify_transform_optional A rectification transform to apply, if
|
|
|
|
* desired.
|
|
|
|
* @param new_camera_matrix_optional Unlike OpenCV, the default/empty matrix
|
|
|
|
* here uses the input camera matrix as your output camera matrix.
|
|
|
|
*/
|
|
|
|
RemapPair
|
|
|
|
calibration_get_undistort_map(
|
|
|
|
t_camera_calibration &calib,
|
|
|
|
cv::InputArray rectify_transform_optional = cv::noArray(),
|
|
|
|
cv::Mat new_camera_matrix_optional = cv::Mat());
|
2019-10-12 20:43:26 +00:00
|
|
|
|
2020-01-16 14:57:36 +00:00
|
|
|
/*!
|
|
|
|
* @brief Rectification, rotation, projection data for a single view in a stereo
|
|
|
|
* pair.
|
|
|
|
*
|
|
|
|
* @see StereoRectificationMaps
|
|
|
|
*/
|
|
|
|
struct ViewRectification
|
|
|
|
{
|
|
|
|
RemapPair rectify;
|
|
|
|
cv::Mat rotation_mat = {};
|
|
|
|
cv::Mat projection_mat = {};
|
|
|
|
};
|
|
|
|
|
2020-01-13 23:30:27 +00:00
|
|
|
/*!
|
|
|
|
* @brief Rectification maps as well as transforms for a stereo camera.
|
|
|
|
*
|
|
|
|
* Computed in the constructor from saved calibration data.
|
|
|
|
*/
|
|
|
|
struct StereoRectificationMaps
|
|
|
|
{
|
2020-01-16 14:57:36 +00:00
|
|
|
ViewRectification view[2];
|
2019-10-12 20:43:26 +00:00
|
|
|
|
2020-01-13 23:30:27 +00:00
|
|
|
//! Disparity and position to camera world coordinates.
|
|
|
|
cv::Mat disparity_to_depth_mat = {};
|
2019-10-12 20:43:26 +00:00
|
|
|
|
2020-01-16 14:57:36 +00:00
|
|
|
/*!
|
|
|
|
* @brief Constructor - produces rectification data for a stereo camera
|
|
|
|
* based on calibration data.
|
|
|
|
*/
|
2020-01-13 23:30:27 +00:00
|
|
|
StereoRectificationMaps(t_stereo_camera_calibration &data);
|
2019-09-25 00:49:21 +00:00
|
|
|
};
|