/** BSD 3-Clause License This file is part of the Basalt project. https://gitlab.com/VladyslavUsenko/basalt.git Copyright (c) 2019, Vladyslav Usenko and Nikolaus Demmel. All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. * Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include namespace basalt { CamCalib::CamCalib(const std::string &dataset_path, const std::string &dataset_type, const std::string &aprilgrid_path, const std::string &cache_path, const std::string &cache_dataset_name, int skip_images, const std::vector &cam_types, bool show_gui) : dataset_path(dataset_path), dataset_type(dataset_type), april_grid(aprilgrid_path), cache_path(ensure_trailing_slash(cache_path)), cache_dataset_name(cache_dataset_name), skip_images(skip_images), cam_types(cam_types), show_gui(show_gui), show_frame("ui.show_frame", 0, 0, 1500), show_corners("ui.show_corners", true, false, true), show_corners_rejected("ui.show_corners_rejected", false, false, true), show_init_reproj("ui.show_init_reproj", false, false, true), show_opt("ui.show_opt", true, false, true), show_vign("ui.show_vign", false, false, true), show_ids("ui.show_ids", false, false, true), huber_thresh("ui.huber_thresh", 4.0, 0.1, 10.0), opt_intr("ui.opt_intr", true, false, true), opt_until_convg("ui.opt_until_converge", false, false, true), stop_thresh("ui.stop_thresh", 1e-8, 1e-10, 0.01, true) { if (show_gui) initGui(); if (!fs::exists(cache_path)) { fs::create_directory(cache_path); } pangolin::ColourWheel cw; for (int i = 0; i < 20; i++) { cam_colors.emplace_back(cw.GetUniqueColour()); } } CamCalib::~CamCalib() { if (processing_thread) { processing_thread->join(); } } void CamCalib::initGui() { pangolin::CreateWindowAndBind("Main", 1600, 1000); pangolin::CreatePanel("ui").SetBounds(0.0, 1.0, 0.0, pangolin::Attach::Pix(UI_WIDTH)); img_view_display = &pangolin::CreateDisplay() .SetBounds(0.5, 1.0, pangolin::Attach::Pix(UI_WIDTH), 1.0) .SetLayout(pangolin::LayoutEqual); pangolin::View &vign_plot_display = pangolin::CreateDisplay().SetBounds(0.0, 0.5, 0.72, 1.0); vign_plotter.reset(new pangolin::Plotter(&vign_data_log, 0.0, 1000.0, 0.0, 1.0, 0.01f, 0.01f)); vign_plot_display.AddDisplay(*vign_plotter); pangolin::View &polar_error_display = pangolin::CreateDisplay().SetBounds( 0.0, 0.5, pangolin::Attach::Pix(UI_WIDTH), 0.43); polar_plotter.reset( new pangolin::Plotter(nullptr, 0.0, 120.0, 0.0, 1.0, 0.01f, 0.01f)); polar_error_display.AddDisplay(*polar_plotter); pangolin::View &azimuthal_plot_display = pangolin::CreateDisplay().SetBounds(0.0, 0.5, 0.45, 0.7); azimuth_plotter.reset( new pangolin::Plotter(nullptr, -180.0, 180.0, 0.0, 1.0, 0.01f, 0.01f)); azimuthal_plot_display.AddDisplay(*azimuth_plotter); pangolin::Var> load_dataset( "ui.load_dataset", std::bind(&CamCalib::loadDataset, this)); pangolin::Var> detect_corners( "ui.detect_corners", std::bind(&CamCalib::detectCorners, this)); pangolin::Var> init_cam_intrinsics( "ui.init_cam_intr", std::bind(&CamCalib::initCamIntrinsics, this)); pangolin::Var> init_cam_poses( "ui.init_cam_poses", std::bind(&CamCalib::initCamPoses, this)); pangolin::Var> init_cam_extrinsics( "ui.init_cam_extr", std::bind(&CamCalib::initCamExtrinsics, this)); pangolin::Var> init_opt( "ui.init_opt", std::bind(&CamCalib::initOptimization, this)); pangolin::Var> optimize( "ui.optimize", std::bind(&CamCalib::optimize, this)); pangolin::Var> save_calib( "ui.save_calib", std::bind(&CamCalib::saveCalib, this)); pangolin::Var> compute_vign( "ui.compute_vign", std::bind(&CamCalib::computeVign, this)); setNumCameras(1); } void CamCalib::computeVign() { Eigen::aligned_vector optical_centers; for (size_t i = 0; i < calib_opt->calib->intrinsics.size(); i++) { optical_centers.emplace_back( calib_opt->calib->intrinsics[i].getParam().segment<2>(2)); } std::map> reprojected_vignette2; for (size_t i = 0; i < vio_dataset->get_image_timestamps().size(); i++) { int64_t timestamp_ns = vio_dataset->get_image_timestamps()[i]; const std::vector img_vec = vio_dataset->get_image_data(timestamp_ns); for (size_t j = 0; j < calib_opt->calib->intrinsics.size(); j++) { TimeCamId tcid(timestamp_ns, j); auto it = reprojected_vignette.find(tcid); if (it != reprojected_vignette.end() && img_vec[j].img.get()) { Eigen::aligned_vector rv; rv.resize(it->second.corners_proj.size()); for (size_t k = 0; k < it->second.corners_proj.size(); k++) { Eigen::Vector2d pos = it->second.corners_proj[k]; rv[k].head<2>() = pos; if (img_vec[j].img->InBounds(pos[0], pos[1], 1) && it->second.corners_proj_success[k]) { double val = img_vec[j].img->interp(pos); val /= std::numeric_limits::max(); if (img_vec[j].exposure > 0) { val *= 0.001 / img_vec[j].exposure; // bring to common exposure } rv[k][2] = val; } else { rv[k][2] = -1; } } reprojected_vignette2.emplace(tcid, rv); } } } VignetteEstimator ve(vio_dataset, optical_centers, calib_opt->calib->resolution, reprojected_vignette2, april_grid); ve.optimize(); ve.compute_error(&reprojected_vignette_error); std::vector> vign_data; ve.compute_data_log(vign_data); vign_data_log.Clear(); for (const auto &v : vign_data) vign_data_log.Log(v); { vign_plotter->ClearSeries(); vign_plotter->ClearMarkers(); for (size_t i = 0; i < calib_opt->calib->intrinsics.size(); i++) { vign_plotter->AddSeries("$i", "$" + std::to_string(2 * i), pangolin::DrawingModeLine, cam_colors[i], "vignette camera " + std::to_string(i)); } vign_plotter->ScaleViewSmooth(vign_data_log.Samples() / 1000.0f, 1.0f, 0.0f, 0.5f); } ve.save_vign_png(cache_path); calib_opt->setVignette(ve.get_vign_param()); std::cout << "Saved vignette png files to " << cache_path << std::endl; } void CamCalib::setNumCameras(size_t n) { while (img_view.size() < n && show_gui) { std::shared_ptr iv(new pangolin::ImageView); size_t idx = img_view.size(); img_view.push_back(iv); img_view_display->AddDisplay(*iv); iv->extern_draw_function = std::bind(&CamCalib::drawImageOverlay, this, std::placeholders::_1, idx); } } void CamCalib::renderingLoop() { while (!pangolin::ShouldQuit()) { glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); if (vio_dataset.get()) { if (show_frame.GuiChanged()) { size_t frame_id = static_cast(show_frame); int64_t timestamp = vio_dataset->get_image_timestamps()[frame_id]; const std::vector &img_vec = vio_dataset->get_image_data(timestamp); for (size_t cam_id = 0; cam_id < vio_dataset->get_num_cams(); cam_id++) if (img_vec[cam_id].img.get()) { pangolin::GlPixFormat fmt; fmt.glformat = GL_LUMINANCE; fmt.gltype = GL_UNSIGNED_SHORT; fmt.scalable_internal_format = GL_LUMINANCE16; img_view[cam_id]->SetImage( img_vec[cam_id].img->ptr, img_vec[cam_id].img->w, img_vec[cam_id].img->h, img_vec[cam_id].img->pitch, fmt); } else { img_view[cam_id]->Clear(); } } } if (opt_until_convg) { bool converged = optimizeWithParam(true); if (converged) opt_until_convg = false; } pangolin::FinishFrame(); } } void CamCalib::computeProjections() { reprojected_corners.clear(); reprojected_vignette.clear(); if (!calib_opt.get() || !vio_dataset.get()) return; constexpr int ANGLE_BIN_SIZE = 2; std::vector> polar_sum( calib_opt->calib->intrinsics.size()); std::vector> polar_num( calib_opt->calib->intrinsics.size()); std::vector> azimuth_sum( calib_opt->calib->intrinsics.size()); std::vector> azimuth_num( calib_opt->calib->intrinsics.size()); for (size_t i = 0; i < calib_opt->calib->intrinsics.size(); i++) { polar_sum[i].setZero(); polar_num[i].setZero(); azimuth_sum[i].setZero(); azimuth_num[i].setZero(); } for (size_t j = 0; j < vio_dataset->get_image_timestamps().size(); ++j) { int64_t timestamp_ns = vio_dataset->get_image_timestamps()[j]; for (size_t i = 0; i < calib_opt->calib->intrinsics.size(); i++) { TimeCamId tcid(timestamp_ns, i); ProjectedCornerData rc, rv; Eigen::aligned_vector polar_azimuthal_angle; Sophus::SE3d T_c_w_ = (calib_opt->getT_w_i(timestamp_ns) * calib_opt->calib->T_i_c[i]) .inverse(); Eigen::Matrix4d T_c_w = T_c_w_.matrix(); calib_opt->calib->intrinsics[i].project( april_grid.aprilgrid_corner_pos_3d, T_c_w, rc.corners_proj, rc.corners_proj_success, polar_azimuthal_angle); calib_opt->calib->intrinsics[i].project( april_grid.aprilgrid_vignette_pos_3d, T_c_w, rv.corners_proj, rv.corners_proj_success); reprojected_corners.emplace(tcid, rc); reprojected_vignette.emplace(tcid, rv); // Compute reprojection histogrames over polar and azimuth angle auto it = calib_corners.find(tcid); if (it != calib_corners.end()) { for (size_t k = 0; k < it->second.corners.size(); k++) { size_t id = it->second.corner_ids[k]; if (rc.corners_proj_success[id]) { double error = (it->second.corners[k] - rc.corners_proj[id]).norm(); size_t polar_bin = 180 * polar_azimuthal_angle[id][0] / (M_PI * ANGLE_BIN_SIZE); polar_sum[tcid.cam_id][polar_bin] += error; polar_num[tcid.cam_id][polar_bin] += 1; size_t azimuth_bin = 180 / ANGLE_BIN_SIZE + (180.0 * polar_azimuthal_angle[id][1]) / (M_PI * ANGLE_BIN_SIZE); azimuth_sum[tcid.cam_id][azimuth_bin] += error; azimuth_num[tcid.cam_id][azimuth_bin] += 1; } } } } } while (polar_data_log.size() < calib_opt->calib->intrinsics.size()) { polar_data_log.emplace_back(new pangolin::DataLog); } while (azimuth_data_log.size() < calib_opt->calib->intrinsics.size()) { azimuth_data_log.emplace_back(new pangolin::DataLog); } constexpr int MIN_POINTS_HIST = 3; polar_plotter->ClearSeries(); azimuth_plotter->ClearSeries(); for (size_t c = 0; c < calib_opt->calib->intrinsics.size(); c++) { polar_data_log[c]->Clear(); azimuth_data_log[c]->Clear(); for (int i = 0; i < polar_sum[c].rows(); i++) { if (polar_num[c][i] > MIN_POINTS_HIST) { double x_coord = ANGLE_BIN_SIZE * i + ANGLE_BIN_SIZE / 2.0; double mean_reproj = polar_sum[c][i] / polar_num[c][i]; polar_data_log[c]->Log(x_coord, mean_reproj); } } polar_plotter->AddSeries( "$0", "$1", pangolin::DrawingModeLine, cam_colors[c], "mean error(pix) vs polar angle(deg) for cam" + std::to_string(c), polar_data_log[c].get()); for (int i = 0; i < azimuth_sum[c].rows(); i++) { if (azimuth_num[c][i] > MIN_POINTS_HIST) { double x_coord = ANGLE_BIN_SIZE * i + ANGLE_BIN_SIZE / 2.0 - 180.0; double mean_reproj = azimuth_sum[c][i] / azimuth_num[c][i]; azimuth_data_log[c]->Log(x_coord, mean_reproj); } } azimuth_plotter->AddSeries( "$0", "$1", pangolin::DrawingModeLine, cam_colors[c], "mean error(pix) vs azimuth angle(deg) for cam" + std::to_string(c), azimuth_data_log[c].get()); } } void CamCalib::detectCorners() { if (processing_thread) { processing_thread->join(); processing_thread.reset(); } processing_thread.reset(new std::thread([this]() { std::cout << "Started detecting corners" << std::endl; CalibHelper::detectCorners(this->vio_dataset, this->calib_corners, this->calib_corners_rejected); std::string path = cache_path + cache_dataset_name + "_detected_corners.cereal"; std::ofstream os(path, std::ios::binary); cereal::BinaryOutputArchive archive(os); archive(this->calib_corners); archive(this->calib_corners_rejected); std::cout << "Done detecting corners. Saved them here: " << path << std::endl; })); if (!show_gui) { processing_thread->join(); processing_thread.reset(); } } void CamCalib::initCamIntrinsics() { if (calib_corners.empty()) { std::cerr << "No corners detected. Press detect_corners to start corner " "detection." << std::endl; return; } std::cout << "Started camera intrinsics initialization" << std::endl; if (!calib_opt) calib_opt.reset(new PosesOptimization); calib_opt->resetCalib(vio_dataset->get_num_cams(), cam_types); std::vector cam_initialized(vio_dataset->get_num_cams(), false); int inc = 1; if (vio_dataset->get_image_timestamps().size() > 100) inc = 3; for (size_t j = 0; j < vio_dataset->get_num_cams(); j++) { for (size_t i = 0; i < vio_dataset->get_image_timestamps().size(); i += inc) { const int64_t timestamp_ns = vio_dataset->get_image_timestamps()[i]; const std::vector &img_vec = vio_dataset->get_image_data(timestamp_ns); TimeCamId tcid(timestamp_ns, j); if (calib_corners.find(tcid) != calib_corners.end()) { CalibCornerData cid = calib_corners.at(tcid); Eigen::Vector4d init_intr; bool success = CalibHelper::initializeIntrinsics( cid.corners, cid.corner_ids, april_grid, img_vec[j].img->w, img_vec[j].img->h, init_intr); if (success) { cam_initialized[j] = true; calib_opt->calib->intrinsics[j].setFromInit(init_intr); break; } } } } // Try perfect pinhole initialization for cameras that are not initalized. for (size_t j = 0; j < vio_dataset->get_num_cams(); j++) { if (!cam_initialized[j]) { std::vector pinhole_corners; int w = 0; int h = 0; for (size_t i = 0; i < vio_dataset->get_image_timestamps().size(); i += inc) { const int64_t timestamp_ns = vio_dataset->get_image_timestamps()[i]; const std::vector &img_vec = vio_dataset->get_image_data(timestamp_ns); TimeCamId tcid(timestamp_ns, j); auto it = calib_corners.find(tcid); if (it != calib_corners.end()) { if (it->second.corners.size() > 8) { pinhole_corners.emplace_back(&it->second); } } w = img_vec[j].img->w; h = img_vec[j].img->h; } BASALT_ASSERT(w > 0 && h > 0); Eigen::Vector4d init_intr; bool success = CalibHelper::initializeIntrinsicsPinhole( pinhole_corners, april_grid, w, h, init_intr); if (success) { cam_initialized[j] = true; std::cout << "Initialized camera " << j << " with pinhole model. You should set pinhole model for " "this camera!" << std::endl; calib_opt->calib->intrinsics[j].setFromInit(init_intr); } } } std::cout << "Done camera intrinsics initialization:" << std::endl; for (size_t j = 0; j < vio_dataset->get_num_cams(); j++) { std::cout << "Cam " << j << ": " << calib_opt->calib->intrinsics[j].getParam().transpose() << std::endl; } // set resolution { size_t img_idx = 1; int64_t t_ns = vio_dataset->get_image_timestamps()[img_idx]; auto img_data = vio_dataset->get_image_data(t_ns); // Find the frame with all valid images while (img_idx < vio_dataset->get_image_timestamps().size()) { bool img_data_valid = true; for (size_t i = 0; i < vio_dataset->get_num_cams(); i++) { if (!img_data[i].img.get()) img_data_valid = false; } if (!img_data_valid) { img_idx++; int64_t t_ns_new = vio_dataset->get_image_timestamps()[img_idx]; img_data = vio_dataset->get_image_data(t_ns_new); } else { break; } } Eigen::aligned_vector res; for (size_t i = 0; i < vio_dataset->get_num_cams(); i++) { res.emplace_back(img_data[i].img->w, img_data[i].img->h); } calib_opt->setResolution(res); } } void CamCalib::initCamPoses() { if (calib_corners.empty()) { std::cerr << "No corners detected. Press detect_corners to start corner " "detection." << std::endl; return; } if (!calib_opt.get() || !calib_opt->calibInitialized()) { std::cerr << "No initial intrinsics. Press init_intrinsics initialize " "intrinsics" << std::endl; return; } if (processing_thread) { processing_thread->join(); processing_thread.reset(); } std::cout << "Started initial camera pose computation " << std::endl; CalibHelper::initCamPoses(calib_opt->calib, april_grid.aprilgrid_corner_pos_3d, this->calib_corners, this->calib_init_poses); std::string path = cache_path + cache_dataset_name + "_init_poses.cereal"; std::ofstream os(path, std::ios::binary); cereal::BinaryOutputArchive archive(os); archive(this->calib_init_poses); std::cout << "Done initial camera pose computation. Saved them here: " << path << std::endl; } void CamCalib::initCamExtrinsics() { if (calib_init_poses.empty()) { std::cerr << "No initial camera poses. Press init_cam_poses initialize " "camera poses " << std::endl; return; } if (!calib_opt.get() || !calib_opt->calibInitialized()) { std::cerr << "No initial intrinsics. Press init_intrinsics initialize " "intrinsics" << std::endl; return; } // Camera graph. Stores the edge from i to j with weight w and timestamp. i // and j should be sorted; std::map, std::pair> cam_graph; // Construct the graph. for (size_t i = 0; i < vio_dataset->get_image_timestamps().size(); i++) { int64_t timestamp_ns = vio_dataset->get_image_timestamps()[i]; for (size_t cam_i = 0; cam_i < vio_dataset->get_num_cams(); cam_i++) { TimeCamId tcid_i(timestamp_ns, cam_i); auto it = calib_init_poses.find(tcid_i); if (it == calib_init_poses.end() || it->second.num_inliers < MIN_CORNERS) continue; for (size_t cam_j = cam_i + 1; cam_j < vio_dataset->get_num_cams(); cam_j++) { TimeCamId tcid_j(timestamp_ns, cam_j); auto it2 = calib_init_poses.find(tcid_j); if (it2 == calib_init_poses.end() || it2->second.num_inliers < MIN_CORNERS) continue; std::pair edge_id(cam_i, cam_j); int curr_weight = cam_graph[edge_id].first; int new_weight = std::min(it->second.num_inliers, it2->second.num_inliers); if (curr_weight < new_weight) { cam_graph[edge_id] = std::make_pair(new_weight, timestamp_ns); } } } } std::vector cameras_initialized(vio_dataset->get_num_cams(), false); cameras_initialized[0] = true; size_t last_camera = 0; calib_opt->calib->T_i_c[0] = Sophus::SE3d(); // Identity auto next_max_weight_edge = [&](size_t cam_id) { int max_weight = -1; std::pair res(-1, -1); for (size_t i = 0; i < vio_dataset->get_num_cams(); i++) { if (cameras_initialized[i]) continue; std::pair edge_id; if (i < cam_id) { edge_id = std::make_pair(i, cam_id); } else if (i > cam_id) { edge_id = std::make_pair(cam_id, i); } auto it = cam_graph.find(edge_id); if (it != cam_graph.end() && max_weight < it->second.first) { max_weight = it->second.first; res.first = i; res.second = it->second.second; } } return res; }; for (size_t i = 0; i < vio_dataset->get_num_cams() - 1; i++) { std::pair res = next_max_weight_edge(last_camera); std::cout << "Initializing camera pair " << last_camera << " " << res.first << std::endl; if (res.first >= 0) { size_t new_camera = res.first; TimeCamId tcid_last(res.second, last_camera); TimeCamId tcid_new(res.second, new_camera); calib_opt->calib->T_i_c[new_camera] = calib_opt->calib->T_i_c[last_camera] * calib_init_poses.at(tcid_last).T_a_c.inverse() * calib_init_poses.at(tcid_new).T_a_c; last_camera = new_camera; cameras_initialized[last_camera] = true; } } std::cout << "Done camera extrinsics initialization:" << std::endl; for (size_t j = 0; j < vio_dataset->get_num_cams(); j++) { std::cout << "T_c0_c" << j << ":\n" << calib_opt->calib->T_i_c[j].matrix() << std::endl; } } // namespace basalt void CamCalib::initOptimization() { if (!calib_opt) { std::cerr << "Calibration is not initialized. Initialize calibration first!" << std::endl; return; } if (calib_init_poses.empty()) { std::cerr << "No initial camera poses. Press init_cam_poses initialize " "camera poses " << std::endl; return; } calib_opt->setAprilgridCorners3d(april_grid.aprilgrid_corner_pos_3d); std::unordered_set invalid_frames; for (const auto &kv : calib_corners) { if (kv.second.corner_ids.size() < MIN_CORNERS) invalid_frames.insert(kv.first); } for (size_t j = 0; j < vio_dataset->get_image_timestamps().size(); ++j) { int64_t timestamp_ns = vio_dataset->get_image_timestamps()[j]; int max_inliers = -1; int max_inliers_idx = -1; for (size_t cam_id = 0; cam_id < calib_opt->calib->T_i_c.size(); cam_id++) { TimeCamId tcid(timestamp_ns, cam_id); const auto cp_it = calib_init_poses.find(tcid); if (cp_it != calib_init_poses.end()) { if ((int)cp_it->second.num_inliers > max_inliers) { max_inliers = cp_it->second.num_inliers; max_inliers_idx = cam_id; } } } if (max_inliers >= (int)MIN_CORNERS) { TimeCamId tcid(timestamp_ns, max_inliers_idx); const auto cp_it = calib_init_poses.find(tcid); // Initial pose calib_opt->addPoseMeasurement( timestamp_ns, cp_it->second.T_a_c * calib_opt->calib->T_i_c[max_inliers_idx].inverse()); } else { // Set all frames invalid if we do not have initial pose for (size_t cam_id = 0; cam_id < calib_opt->calib->T_i_c.size(); cam_id++) { invalid_frames.emplace(timestamp_ns, cam_id); } } } for (const auto &kv : calib_corners) { if (invalid_frames.count(kv.first) == 0) calib_opt->addAprilgridMeasurement(kv.first.frame_id, kv.first.cam_id, kv.second.corners, kv.second.corner_ids); } calib_opt->init(); computeProjections(); std::cout << "Initialized optimization." << std::endl; } // namespace basalt void CamCalib::loadDataset() { basalt::DatasetIoInterfacePtr dataset_io = basalt::DatasetIoFactory::getDatasetIo(dataset_type); dataset_io->read(dataset_path); vio_dataset = dataset_io->get_data(); setNumCameras(vio_dataset->get_num_cams()); if (skip_images > 1) { std::vector new_image_timestamps; for (size_t i = 0; i < vio_dataset->get_image_timestamps().size(); i++) { if (i % skip_images == 0) new_image_timestamps.push_back(vio_dataset->get_image_timestamps()[i]); } vio_dataset->get_image_timestamps() = new_image_timestamps; } // load detected corners if they exist { std::string path = cache_path + cache_dataset_name + "_detected_corners.cereal"; std::ifstream is(path, std::ios::binary); if (is.good()) { cereal::BinaryInputArchive archive(is); calib_corners.clear(); calib_corners_rejected.clear(); archive(calib_corners); archive(calib_corners_rejected); std::cout << "Loaded detected corners from: " << path << std::endl; } else { std::cout << "No pre-processed detected corners found" << std::endl; } } // load initial poses if they exist { std::string path = cache_path + cache_dataset_name + "_init_poses.cereal"; std::ifstream is(path, std::ios::binary); if (is.good()) { cereal::BinaryInputArchive archive(is); calib_init_poses.clear(); archive(calib_init_poses); std::cout << "Loaded initial poses from: " << path << std::endl; } else { std::cout << "No pre-processed initial poses found" << std::endl; } } // load calibration if exist { if (!calib_opt) calib_opt.reset(new PosesOptimization); calib_opt->loadCalib(cache_path); } reprojected_corners.clear(); reprojected_vignette.clear(); if (show_gui) { show_frame = 0; show_frame.Meta().range[1] = vio_dataset->get_image_timestamps().size() - 1; show_frame.Meta().gui_changed = true; } } void CamCalib::optimize() { optimizeWithParam(true); } bool CamCalib::optimizeWithParam(bool print_info, std::map *stats) { if (calib_init_poses.empty()) { std::cerr << "No initial camera poses. Press init_cam_poses initialize " "camera poses " << std::endl; return true; } if (!calib_opt.get() || !calib_opt->calibInitialized()) { std::cerr << "No initial intrinsics. Press init_intrinsics initialize " "intrinsics" << std::endl; return true; } bool converged = true; if (calib_opt) { // calib_opt->compute_projections(); double error; double reprojection_error; int num_points; auto start = std::chrono::high_resolution_clock::now(); converged = calib_opt->optimize(opt_intr, huber_thresh, stop_thresh, error, num_points, reprojection_error); auto finish = std::chrono::high_resolution_clock::now(); if (stats) { stats->clear(); stats->emplace("energy_error", error); stats->emplace("num_points", num_points); stats->emplace("mean_energy_error", error / num_points); stats->emplace("reprojection_error", reprojection_error); stats->emplace("mean_reprojection_error", reprojection_error / num_points); } if (print_info) { std::cout << "==================================" << std::endl; for (size_t i = 0; i < vio_dataset->get_num_cams(); i++) { std::cout << "intrinsics " << i << ": " << calib_opt->calib->intrinsics[i].getParam().transpose() << std::endl; std::cout << "T_i_c" << i << ":\n" << calib_opt->calib->T_i_c[i].matrix() << std::endl; } std::cout << "Current error: " << error << " num_points " << num_points << " mean_error " << error / num_points << " reprojection_error " << reprojection_error << " mean reprojection " << reprojection_error / num_points << " opt_time " << std::chrono::duration_cast( finish - start) .count() << "ms." << std::endl; if (converged) std::cout << "Optimization Converged !!" << std::endl; std::cout << "==================================" << std::endl; } if (show_gui) { computeProjections(); } } return converged; } void CamCalib::saveCalib() { if (calib_opt) { calib_opt->saveCalib(cache_path); std::cout << "Saved calibration in " << cache_path << "calibration.json" << std::endl; } } void CamCalib::drawImageOverlay(pangolin::View &v, size_t cam_id) { UNUSED(v); size_t frame_id = show_frame; if (vio_dataset && frame_id < vio_dataset->get_image_timestamps().size()) { int64_t timestamp_ns = vio_dataset->get_image_timestamps()[frame_id]; TimeCamId tcid(timestamp_ns, cam_id); if (show_corners) { glLineWidth(1.0); glColor3f(1.0, 0.0, 0.0); glEnable(GL_BLEND); glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); if (calib_corners.find(tcid) != calib_corners.end()) { const CalibCornerData &cr = calib_corners.at(tcid); const CalibCornerData &cr_rej = calib_corners_rejected.at(tcid); for (size_t i = 0; i < cr.corners.size(); i++) { // the radius is the threshold used for maximum displacement. The // search region is slightly larger. const float radius = static_cast(cr.radii[i]); const Eigen::Vector2f c = cr.corners[i].cast(); pangolin::glDrawCirclePerimeter(c[0], c[1], radius); if (show_ids) pangolin::GlFont::I().Text("%d", cr.corner_ids[i]).Draw(c[0], c[1]); } pangolin::GlFont::I() .Text("Detected %d corners (%d rejected)", cr.corners.size(), cr_rej.corners.size()) .Draw(5, 50); if (show_corners_rejected) { glColor3f(1.0, 0.5, 0.0); for (size_t i = 0; i < cr_rej.corners.size(); i++) { // the radius is the threshold used for maximum displacement. The // search region is slightly larger. const float radius = static_cast(cr_rej.radii[i]); const Eigen::Vector2f c = cr_rej.corners[i].cast(); pangolin::glDrawCirclePerimeter(c[0], c[1], radius); if (show_ids) pangolin::GlFont::I() .Text("%d", cr_rej.corner_ids[i]) .Draw(c[0], c[1]); } } } else { glLineWidth(1.0); pangolin::GlFont::I().Text("Corners not processed").Draw(5, 50); } } if (show_init_reproj) { glLineWidth(1.0); glColor3f(1.0, 1.0, 0.0); glEnable(GL_BLEND); glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); if (calib_init_poses.find(tcid) != calib_init_poses.end()) { const CalibInitPoseData &cr = calib_init_poses.at(tcid); for (size_t i = 0; i < cr.reprojected_corners.size(); i++) { Eigen::Vector2d c = cr.reprojected_corners[i]; pangolin::glDrawCirclePerimeter(c[0], c[1], 3.0); if (show_ids) pangolin::GlFont::I().Text("%d", i).Draw(c[0], c[1]); } pangolin::GlFont::I() .Text("Initial pose with %d inliers", cr.num_inliers) .Draw(5, 100); } else { pangolin::GlFont::I().Text("Initial pose not processed").Draw(5, 100); } } if (show_opt) { glLineWidth(1.0); glColor3f(1.0, 0.0, 1.0); glEnable(GL_BLEND); glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); if (reprojected_corners.find(tcid) != reprojected_corners.end()) { if (calib_corners.count(tcid) > 0 && calib_corners.at(tcid).corner_ids.size() >= MIN_CORNERS) { const auto &rc = reprojected_corners.at(tcid); for (size_t i = 0; i < rc.corners_proj.size(); i++) { if (!rc.corners_proj_success[i]) continue; Eigen::Vector2d c = rc.corners_proj[i]; pangolin::glDrawCirclePerimeter(c[0], c[1], 3.0); if (show_ids) pangolin::GlFont::I().Text("%d", i).Draw(c[0], c[1]); } } else { pangolin::GlFont::I().Text("Too few corners detected.").Draw(5, 150); } } } if (show_vign) { glLineWidth(1.0); glColor3f(1.0, 1.0, 0.0); glEnable(GL_BLEND); glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); if (reprojected_vignette.find(tcid) != reprojected_vignette.end()) { if (calib_corners.count(tcid) > 0 && calib_corners.at(tcid).corner_ids.size() >= MIN_CORNERS) { const auto &rc = reprojected_vignette.at(tcid); bool has_errors = false; auto it = reprojected_vignette_error.find(tcid); if (it != reprojected_vignette_error.end()) has_errors = true; for (size_t i = 0; i < rc.corners_proj.size(); i++) { if (!rc.corners_proj_success[i]) continue; Eigen::Vector2d c = rc.corners_proj[i].head<2>(); pangolin::glDrawCirclePerimeter(c[0], c[1], 3.0); if (show_ids) { if (has_errors) { pangolin::GlFont::I() .Text("%d(%f)", i, it->second[i]) .Draw(c[0], c[1]); } else { pangolin::GlFont::I().Text("%d", i).Draw(c[0], c[1]); } } } } else { pangolin::GlFont::I().Text("Too few corners detected.").Draw(5, 200); } } } } } bool CamCalib::hasCorners() const { return !calib_corners.empty(); } } // namespace basalt