#include "openMVG/cameras/Camera_Common.hpp"
#include "openMVG/cameras/Cameras_Common_command_line_helper.hpp"
#include "openMVG/sfm/pipelines/sfm_features_provider.hpp"
#include "openMVG/sfm/pipelines/sfm_matches_provider.hpp"
#include "openMVG/sfm/sfm_data.hpp"
#include "openMVG/sfm/sfm_data_io.hpp"
#include "openMVG/sfm/sfm_report.hpp"
#include "openMVG/sfm/sfm_view.hpp"
#include "openMVG/system/timer.hpp"
#include "openMVG/types.hpp"
// SfM Engines
#include "openMVG/sfm/pipelines/sequential/sequential_SfM.hpp"
#include "openMVG/sfm/pipelines/sequential/sequential_SfM2.hpp"
#include "openMVG/sfm/pipelines/sequential/SfmSceneInitializerMaxPair.hpp"
#include "openMVG/sfm/pipelines/sequential/SfmSceneInitializerStellar.hpp"
#include "openMVG/sfm/pipelines/global/GlobalSfM_rotation_averaging.hpp"
#include "openMVG/sfm/pipelines/global/GlobalSfM_translation_averaging.hpp"
#include "openMVG/sfm/pipelines/global/sfm_global_engine_relative_motions.hpp"
#include "third_party/cmdLine/cmdLine.h"
#include "third_party/stlplus3/filesystemSimplified/file_system.hpp"
#include <cstdlib>
#include <memory>
#include <string>
#include <utility>
using namespace openMVG;
using namespace openMVG::cameras;
using namespace openMVG::sfm;
//SFM初始化方式,INCREMENTALV2版本才能用到
enum class ESfMSceneInitializer
{
INITIALIZE_EXISTING_POSES,
INITIALIZE_MAX_PAIR,
INITIALIZE_AUTO_PAIR,
INITIALIZE_STELLAR
};
//SFM Engine选用
enum class ESfMEngine
{
INCREMENTAL,
INCREMENTALV2,
GLOBAL
};
bool StringToEnum
(
const std::string & str,
ESfMEngine & sfm_engine
)
{
const std::map<std::string, ESfMEngine> string_to_enum_mapping =
{
{"INCREMENTAL", ESfMEngine::INCREMENTAL},
{"INCREMENTALV2", ESfMEngine::INCREMENTALV2},
{"GLOBAL", ESfMEngine::GLOBAL},
};
const auto it = string_to_enum_mapping.find(str);
if (it == string_to_enum_mapping.end())
return false;
sfm_engine = it->second;
return true;
}
bool StringToEnum
(
const std::string & str,
ESfMSceneInitializer & scene_initializer
)
{
const std::map<std::string, ESfMSceneInitializer> string_to_enum_mapping =
{
{"EXISTING_POSE", ESfMSceneInitializer::INITIALIZE_EXISTING_POSES},
{"MAX_PAIR", ESfMSceneInitializer::INITIALIZE_MAX_PAIR},
{"AUTO_PAIR", ESfMSceneInitializer::INITIALIZE_AUTO_PAIR},
{"STELLAR", ESfMSceneInitializer::INITIALIZE_STELLAR},
};
const auto it = string_to_enum_mapping.find(str);
if (it == string_to_enum_mapping.end())
return false;
scene_initializer = it->second;
return true;
}
//给两个文件名字,找到两个文件对应的View的索引
bool computeIndexFromImageNames(
const SfM_Data & sfm_data,
const std::pair<std::string,std::string>& initialPairName,
Pair& initialPairIndex)
{
if (initialPairName.first == initialPairName.second)
{
OPENMVG_LOG_ERROR << "Invalid image names. You cannot use the same image to initialize a pair.";
return false;
}
initialPairIndex = {UndefinedIndexT, UndefinedIndexT};
/// List views filenames and find the one that correspond to the user ones:
for (Views::const_iterator it = sfm_data.GetViews().begin();
it != sfm_data.GetViews().end(); ++it)
{
const View * v = it->second.get();
const std::string filename = stlplus::filename_part(v->s_Img_path);
if (filename == initialPairName.first)
{
initialPairIndex.first = v->id_view;
}
else{
if (filename == initialPairName.second)
{
initialPairIndex.second = v->id_view;
}
}
}
return (initialPairIndex.first != UndefinedIndexT &&
initialPairIndex.second != UndefinedIndexT);
}
int main(int argc, char **argv)
{
using namespace std;
OPENMVG_LOG_INFO
<< "\n-----------------------------------------------------------"
<< "\n Structure from Motion:"
<< "\n-----------------------------------------------------------";
CmdLine cmd;
// Common options:
std::string
filename_sfm_data,
directory_match,
filename_match,
directory_output,
engine_name = "INCREMENTAL";
// Bundle adjustment options:
std::string sIntrinsic_refinement_options = "ADJUST_ALL";
std::string sExtrinsic_refinement_options = "ADJUST_ALL";
bool b_use_motion_priors = false;
// Incremental SfM options
int triangulation_method = static_cast<int>(ETriangulationMethod::DEFAULT);
int resection_method = static_cast<int>(resection::SolverType::DEFAULT);
int user_camera_model = PINHOLE_CAMERA_RADIAL3;
// SfM v1
std::pair<std::string,std::string> initial_pair_string("","");
// SfM v2
std::string sfm_initializer_method = "STELLAR";
// Global SfM options
int rotation_averaging_method = int (ROTATION_AVERAGING_L2);
int translation_averaging_method = int (TRANSLATION_AVERAGING_SOFTL1);
// Common options
cmd.add( make_option('i', filename_sfm_data, "input_file") );
cmd.add( make_option('m', directory_match, "match_dir") );
cmd.add( make_option('M', filename_match, "match_file") );
cmd.add( make_option('o', directory_output, "output_dir") );
cmd.add( make_option('s', engine_name, "sfm_engine") );
// Bundle adjustment options
cmd.add( make_option('f', sIntrinsic_refinement_options, "refine_intrinsic_config") );
cmd.add( make_option('e', sExtrinsic_refinement_options, "refine_extrinsic_config") );
cmd.add( make_switch('P', "prior_usage") );
// Incremental SfM pipeline options
cmd.add( make_option('t', triangulation_method, "triangulation_method"));
cmd.add( make_option('r', resection_method, "resection_method"));
cmd.add( make_option('c', user_camera_model, "camera_model") );
// Incremental SfM2
cmd.add( make_option('S', sfm_initializer_method, "sfm_initializer") );
// Incremental SfM1
cmd.add( make_option('a', initial_pair_string.first, "initial_pair_a") );
cmd.add( make_option('b', initial_pair_string.second, "initial_pair_b") );
// Global SfM
cmd.add( make_option('r', rotation_averaging_method, "rotationAveraging") );
cmd.add( make_option('t', translation_averaging_method, "translationAveraging") );
try {
if (argc == 1) throw std::string("Invalid parameter.");
cmd.process(argc, argv);
} catch (const std::string& s) {
OPENMVG_LOG_INFO << "Usage: " << argv[0] << '\n'
<< "[Required]\n"
<< "[-i|--input_file] path to a SfM_Data scene\n"
<< "[-m|--match_dir] path to the matches that corresponds to the provided SfM_Data scene\n"
<< "[-o|--output_dir] path where the output data will be stored\n"
<< "[-s|--sfm_engine] Type of SfM Engine to use for the reconstruction\n"
<< "\t INCREMENTAL : add image sequentially to a 2 view seed\n"
<< "\t INCREMENTALV2 : add image sequentially to a 2 or N view seed (experimental)\n"
<< "\t GLOBAL : initialize globally rotation and translations\n"
<< "\n\n"
<< "[Optional parameters]\n"
<< "\n\n"
<< "[Common]\n"
<< "[-M|--match_file] path to the match file to use (i.e matches.f.txt or matches.f.bin)\n"
<< "[-f|--refine_extrinsic_config] Intrinsic parameters refinement option\n"
<< "\t ADJUST_ALL -> refine all existing parameters (default) \n"
<< "\t NONE -> intrinsic parameters are held as constant\n"
<< "\t ADJUST_FOCAL_LENGTH -> refine only the focal length\n"
<< "\t ADJUST_PRINCIPAL_POINT -> refine only the principal point position\n"
<< "\t ADJUST_DISTORTION -> refine only the distortion coefficient(s) (if any)\n"
<< "\t -> NOTE: options can be combined thanks to '|'\n"
<< "\t ADJUST_FOCAL_LENGTH|ADJUST_PRINCIPAL_POINT\n"
<< "\t\t-> refine the focal length & the principal point position\n"
<< "\t ADJUST_FOCAL_LENGTH|ADJUST_DISTORTION\n"
<< "\t\t-> refine the focal length & the distortion coefficient(s) (if any)\n"
<< "\t ADJUST_PRINCIPAL_POINT|ADJUST_DISTORTION\n"
<< "\t\t-> refine the principal point position & the distortion coefficient(s) (if any)\n"
<< "[-e|--refine_extrinsic_config] Extrinsic parameters refinement option\n"
<< "\t ADJUST_ALL -> refine all existing parameters (default) \n"
<< "\t NONE -> extrinsic parameters are held as constant\n"
<< "[-P|--prior_usage] Enable usage of motion priors (i.e GPS positions) (default: false)\n"
<< "\n\n"
<< "[Engine specifics]\n"
<< "\n\n"
<< "[INCREMENTAL]\n"
<< "\t[-a|--initial_pair_a] filename of the first image (without path)\n"
<< "\t[-b|--initial_pair_b] filename of the second image (without path)\n"
<< "\t[-c|--camera_model] Camera model type for view with unknown intrinsic:\n"
<< "\t\t 1: Pinhole \n"
<< "\t\t 2: Pinhole radial 1\n"
<< "\t\t 3: Pinhole radial 3 (default)\n"
<< "\t\t 4: Pinhole radial 3 + tangential 2\n"
<< "\t\t 5: Pinhole fisheye\n"
<< "\t[--triangulation_method] triangulation method (default=" << triangulation_method << "):\n"
<< "\t\t" << static_cast<int>(ETriangulationMethod::DIRECT_LINEAR_TRANSFORM) << ": DIRECT_LINEAR_TRANSFORM\n"
<< "\t\t" << static_cast<int>(ETriangulationMethod::L1_ANGULAR) << ": L1_ANGULAR\n"
<< "\t\t" << static_cast<int>(ETriangulationMethod::LINFINITY_ANGULAR) << ": LINFINITY_ANGULAR\n"
<< "\t\t" << static_cast<int>(ETriangulationMethod::INVERSE_DEPTH_WEIGHTED_MIDPOINT) << ": INVERSE_DEPTH_WEIGHTED_MIDPOINT\n"
<< "\t[--resection_method] resection/pose estimation method (default=" << resection_method << "):\n"
<< "\t\t" << static_cast<int>(resection::SolverType::DLT_6POINTS) << ": DIRECT_LINEAR_TRANSFORM 6Points | does not use intrinsic data\n"
<< "\t\t" << static_cast<int>(resection::SolverType::P3P_KE_CVPR17) << ": P3P_KE_CVPR17\n"
<< "\t\t" << static_cast<int>(resection::SolverType::P3P_KNEIP_CVPR11) << ": P3P_KNEIP_CVPR11\n"
<< "\t\t" << static_cast<int>(resection::SolverType::P3P_NORDBERG_ECCV18) << ": P3P_NORDBERG_ECCV18\n"
<< "\t\t" << static_cast<int>(resection::SolverType::UP2P_KUKELOVA_ACCV10) << ": UP2P_KUKELOVA_ACCV10 | 2Points | upright camera\n"
<< "\n\n"
<< "[INCREMENTALV2]\n"
<< "\t[-S|--sfm_initializer] Choose the SfM initializer method:\n"
<< "\t\t 'EXISTING_POSE'-> Initialize the reconstruction from the existing sfm_data camera poses\n"
<< "\t\t 'MAX_PAIR'-> Initialize the reconstruction from the pair that has the most of matches\n"
<< "\t\t 'AUTO_PAIR'-> Initialize the reconstruction with a pair selected automatically\n"
<< "\t\t 'STELLAR'-> Initialize the reconstruction with a 'stellar' reconstruction\n"
<< "\t[-c|--camera_model] Camera model type for view with unknown intrinsic:\n"
<< "\t\t 1: Pinhole \n"
<< "\t\t 2: Pinhole radial 1\n"
<< "\t\t 3: Pinhole radial 3 (default)\n"
<< "\t\t 4: Pinhole radial 3 + tangential 2\n"
<< "\t\t 5: Pinhole fisheye\n"
<< "\t[--triangulation_method] triangulation method (default=" << triangulation_method << "):\n"
<< "\t\t" << static_cast<int>(ETriangulationMethod::DIRECT_LINEAR_TRANSFORM) << ": DIRECT_LINEAR_TRANSFORM\n"
<< "\t\t" << static_cast<int>(ETriangulationMethod::L1_ANGULAR) << ": L1_ANGULAR\n"
<< "\t\t" << static_cast<int>(ETriangulationMethod::LINFINITY_ANGULAR) << ": LINFINITY_ANGULAR\n"
<< "\t\t" << static_cast<int>(ETriangulationMethod::INVERSE_DEPTH_WEIGHTED_MIDPOINT) << ": INVERSE_DEPTH_WEIGHTED_MIDPOINT\n"
<< "\t[--resection_method] resection/pose estimation method (default=" << resection_method << "):\n"
<< "\t\t" << static_cast<int>(resection::SolverType::DLT_6POINTS) << ": DIRECT_LINEAR_TRANSFORM 6Points | does not use intrinsic data\n"
<< "\t\t" << static_cast<int>(resection::SolverType::P3P_KE_CVPR17) << ": P3P_KE_CVPR17\n"
<< "\t\t" << static_cast<int>(resection::SolverType::P3P_KNEIP_CVPR11) << ": P3P_KNEIP_CVPR11\n"
<< "\t\t" << static_cast<int>(resection::SolverType::P3P_NORDBERG_ECCV18) << ": P3P_NORDBERG_ECCV18\n"
<< "\t\t" << static_cast<int>(resection::SolverType::UP2P_KUKELOVA_ACCV10) << ": UP2P_KUKELOVA_ACCV10 | 2Points | upright camera\n"
<< "\n\n"
<< "[GLOBAL]\n"
<< "\t[-r|--rotationAveraging]\n"
<< "\t\t 1 -> L1 minimization\n"
<< "\t\t 2 -> L2 minimization (default)\n"
<< "\t[-t|--translationAveraging]:\n"
<< "\t\t 1 -> L1 minimization\n"
<< "\t\t 2 -> L2 minimization of sum of squared Chordal distances\n"
<< "\t\t 3 -> SoftL1 minimization (default)\n";
OPENMVG_LOG_ERROR << s;
return EXIT_FAILURE;
}
b_use_motion_priors = cmd.used('P');
// Check validity of command line parameters:
if ( !isValid(static_cast<ETriangulationMethod>(triangulation_method))) {
OPENMVG_LOG_ERROR << "Invalid triangulation method";
return EXIT_FAILURE;
}
if ( !isValid(openMVG::cameras::EINTRINSIC(user_camera_model)) ) {
OPENMVG_LOG_ERROR << "Invalid camera type";
return EXIT_FAILURE;
}
const cameras::Intrinsic_Parameter_Type intrinsic_refinement_options =
cameras::StringTo_Intrinsic_Parameter_Type(sIntrinsic_refinement_options);
if (intrinsic_refinement_options == static_cast<cameras::Intrinsic_Parameter_Type>(0) )
{
OPENMVG_LOG_ERROR << "Invalid input for Bundle Adjustment Intrinsic parameter refinement option";
return EXIT_FAILURE;
}
const sfm::Extrinsic_Parameter_Type extrinsic_refinement_options =
sfm::StringTo_Extrinsic_Parameter_Type(sExtrinsic_refinement_options);
if (extrinsic_refinement_options == static_cast<sfm::Extrinsic_Parameter_Type>(0) )
{
OPENMVG_LOG_ERROR << "Invalid input for the Bundle Adjustment Extrinsic parameter refinement option";
return EXIT_FAILURE;
}
ESfMSceneInitializer scene_initializer_enum;
if (!StringToEnum(sfm_initializer_method, scene_initializer_enum))
{
OPENMVG_LOG_ERROR << "Invalid input for the SfM initializer option";
return EXIT_FAILURE;
}
ESfMEngine sfm_engine_type;
if (!StringToEnum(engine_name, sfm_engine_type))
{
OPENMVG_LOG_ERROR << "Invalid input for the SfM Engine type";
return EXIT_FAILURE;
}
if (rotation_averaging_method < ROTATION_AVERAGING_L1 ||
rotation_averaging_method > ROTATION_AVERAGING_L2 ) {
OPENMVG_LOG_ERROR << "Rotation averaging method is invalid";
return EXIT_FAILURE;
}
if (translation_averaging_method < TRANSLATION_AVERAGING_L1 ||
translation_averaging_method > TRANSLATION_AVERAGING_SOFTL1 ) {
OPENMVG_LOG_ERROR << "Translation averaging method is invalid";
return EXIT_FAILURE;
}
if (directory_output.empty()) {
OPENMVG_LOG_ERROR << "It is an invalid output directory";
return EXIT_FAILURE;
}
//跟之前一样,读取SFM场景数据
SfM_Data sfm_data;
const ESfM_Data sfm_data_loading_etypes =
scene_initializer_enum == ESfMSceneInitializer::INITIALIZE_EXISTING_POSES ?
ESfM_Data(VIEWS|INTRINSICS|EXTRINSICS) : ESfM_Data(VIEWS|INTRINSICS);
if (!Load(sfm_data, filename_sfm_data, sfm_data_loading_etypes)) {
OPENMVG_LOG_ERROR << "The input SfM_Data file \""<< filename_sfm_data << "\" cannot be read.";
return EXIT_FAILURE;
}
if (!stlplus::folder_exists(directory_output))
{
if (!stlplus::folder_create(directory_output))
{
OPENMVG_LOG_ERROR << "Cannot create the output directory";
return EXIT_FAILURE;
}
}
//读取特征
if (directory_match.empty() && !filename_match.empty() && stlplus::file_exists(filename_match))
{
directory_match = stlplus::folder_part(filename_match);
filename_match = stlplus::filename_part(filename_match);
}
using namespace openMVG::features;
const std::string sImage_describer = stlplus::create_filespec(directory_match, "image_describer", "json");
std::unique_ptr<Regions> regions_type = Init_region_type_from_file(sImage_describer);
if (!regions_type)
{
OPENMVG_LOG_ERROR << "Invalid: " << sImage_describer << " regions type file.";
return EXIT_FAILURE;
}
std::shared_ptr<Features_Provider> feats_provider = std::make_shared<Features_Provider>();
if (!feats_provider->load(sfm_data, directory_match, regions_type)) {
OPENMVG_LOG_ERROR << "Cannot load view corresponding features in directory: " << directory_match << ".";
return EXIT_FAILURE;
}
// 读取匹配
std::shared_ptr<Matches_Provider> matches_provider = std::make_shared<Matches_Provider>();
if // Try to read the provided match filename or the default one (matches.f.txt/bin)
(
!(matches_provider->load(sfm_data, stlplus::create_filespec(directory_match, filename_match)) ||
matches_provider->load(sfm_data, stlplus::create_filespec(directory_match, "matches.f.txt")) ||
matches_provider->load(sfm_data, stlplus::create_filespec(directory_match, "matches.f.bin")) ||
matches_provider->load(sfm_data, stlplus::create_filespec(directory_match, "matches.e.txt")) ||
matches_provider->load(sfm_data, stlplus::create_filespec(directory_match, "matches.e.bin")))
)
{
OPENMVG_LOG_ERROR << "Cannot load the match file.";
return EXIT_FAILURE;
}
//场景初始化器
std::unique_ptr<SfMSceneInitializer> scene_initializer;
switch(scene_initializer_enum)
{
case ESfMSceneInitializer::INITIALIZE_AUTO_PAIR:
OPENMVG_LOG_ERROR << "Not yet implemented.";
return EXIT_FAILURE;
break;
case ESfMSceneInitializer::INITIALIZE_MAX_PAIR:
scene_initializer.reset(new SfMSceneInitializerMaxPair(sfm_data,
feats_provider.get(),
matches_provider.get()));
break;
case ESfMSceneInitializer::INITIALIZE_EXISTING_POSES:
scene_initializer.reset(new SfMSceneInitializer(sfm_data,
feats_provider.get(),
matches_provider.get()));
break;
case ESfMSceneInitializer::INITIALIZE_STELLAR:
scene_initializer.reset(new SfMSceneInitializerStellar(sfm_data,
feats_provider.get(),
matches_provider.get()));
break;
default:
OPENMVG_LOG_ERROR << "Unknown SFM Scene initializer method";
return EXIT_FAILURE;
}
if (!scene_initializer)
{
OPENMVG_LOG_ERROR << "Invalid scene initializer.";
return EXIT_FAILURE;
}
//SFM引擎,前面都是数据准备部分,INCREMENTAL,INCREMENTALV2,GLOBAL版本
std::unique_ptr<ReconstructionEngine> sfm_engine;
switch (sfm_engine_type)
{
case ESfMEngine::INCREMENTAL:
{
SequentialSfMReconstructionEngine * engine =
new SequentialSfMReconstructionEngine(
sfm_data,
directory_output,
stlplus::create_filespec(directory_output, "Reconstruction_Report.html"));
// Configuration:
engine->SetFeaturesProvider(feats_provider.get());
engine->SetMatchesProvider(matches_provider.get());
// Configure reconstruction parameters
engine->SetUnknownCameraType(EINTRINSIC(user_camera_model));
engine->Set_Use_Motion_Prior(b_use_motion_priors);
engine->SetTriangulationMethod(static_cast<ETriangulationMethod>(triangulation_method));
engine->SetResectionMethod(static_cast<resection::SolverType>(resection_method));
// Handle Initial pair parameter
if (!initial_pair_string.first.empty() && !initial_pair_string.second.empty())
{
Pair initial_pair_index;
if (!computeIndexFromImageNames(sfm_data, initial_pair_string, initial_pair_index))
{
OPENMVG_LOG_ERROR << "Could not find the initial pairs <" << initial_pair_string.first
<< ", " << initial_pair_string.second << ">!";
return EXIT_FAILURE;
}
engine->setInitialPair(initial_pair_index);
}
sfm_engine.reset(engine);
}
break;
case ESfMEngine::INCREMENTALV2:
{
SequentialSfMReconstructionEngine2 * engine =
new SequentialSfMReconstructionEngine2(
scene_initializer.get(),
sfm_data,
directory_output,
stlplus::create_filespec(directory_output, "Reconstruction_Report.html"));
// Configuration:
engine->SetFeaturesProvider(feats_provider.get());
engine->SetMatchesProvider(matches_provider.get());
// Configure reconstruction parameters
engine->Set_Intrinsics_Refinement_Type(intrinsic_refinement_options);
engine->SetUnknownCameraType(EINTRINSIC(user_camera_model));
engine->Set_Use_Motion_Prior(b_use_motion_priors);
engine->SetTriangulationMethod(static_cast<ETriangulationMethod>(triangulation_method));
engine->SetResectionMethod(static_cast<resection::SolverType>(resection_method));
sfm_engine.reset(engine);
}
break;
case ESfMEngine::GLOBAL:
{
GlobalSfMReconstructionEngine_RelativeMotions * engine =
new GlobalSfMReconstructionEngine_RelativeMotions(
sfm_data,
directory_output,
stlplus::create_filespec(directory_output, "Reconstruction_Report.html"));
// Configuration:
engine->SetFeaturesProvider(feats_provider.get());
engine->SetMatchesProvider(matches_provider.get());
// Configure reconstruction parameters
engine->Set_Intrinsics_Refinement_Type(intrinsic_refinement_options);
engine->Set_Use_Motion_Prior(b_use_motion_priors);
// Configure motion averaging method
engine->SetRotationAveragingMethod(ERotationAveragingMethod(rotation_averaging_method));
engine->SetTranslationAveragingMethod(ETranslationAveragingMethod(translation_averaging_method));
sfm_engine.reset(engine);
}
break;
default:
break;
}
if (!sfm_engine)
{
OPENMVG_LOG_ERROR << "Cannot create the requested SfM Engine.";
return EXIT_FAILURE;
}
sfm_engine->Set_Intrinsics_Refinement_Type(intrinsic_refinement_options);
sfm_engine->Set_Extrinsics_Refinement_Type(extrinsic_refinement_options);
//---------------------------------------
// Sequential reconstruction process
//---------------------------------------
openMVG::system::Timer timer;
if (sfm_engine->Process())
{
OPENMVG_LOG_INFO << " Total Sfm took (s): " << timer.elapsed();
OPENMVG_LOG_INFO << "...Generating SfM_Report.html";
Generate_SfM_Report(sfm_engine->Get_SfM_Data(),
stlplus::create_filespec(directory_output, "SfMReconstruction_Report.html"));
//-- Export to disk computed scene (data & viewable results)
OPENMVG_LOG_INFO << "...Export SfM_Data to disk.";
Save(sfm_engine->Get_SfM_Data(),
stlplus::create_filespec(directory_output, "sfm_data", ".bin"),
ESfM_Data(ALL));
Save(sfm_engine->Get_SfM_Data(),
stlplus::create_filespec(directory_output, "cloud_and_poses", ".ply"),
ESfM_Data(ALL));
return EXIT_SUCCESS;
}
return EXIT_FAILURE;
}
软件步骤6:OpenMVG实现增量式和全局式SFM
最新推荐文章于 2024-05-14 11:28:23 发布