头文件
#include "openMVG/graph/graph.hpp"
#include "openMVG/graph/graph_stats.hpp"
#include "openMVG/matching/indMatch.hpp"
#include "openMVG/matching/indMatch_utils.hpp"
#include "openMVG/matching/pairwiseAdjacencyDisplay.hpp"
#include "openMVG/matching_image_collection/Cascade_Hashing_Matcher_Regions.hpp"
#include "openMVG/matching_image_collection/Matcher_Regions.hpp"
#include "openMVG/matching_image_collection/Pair_Builder.hpp"
#include "openMVG/sfm/pipelines/sfm_features_provider.hpp"
#include "openMVG/sfm/pipelines/sfm_preemptive_regions_provider.hpp"
#include "openMVG/sfm/pipelines/sfm_regions_provider.hpp"
#include "openMVG/sfm/pipelines/sfm_regions_provider_cache.hpp"
#include "openMVG/sfm/sfm_data.hpp"
#include "openMVG/sfm/sfm_data_io.hpp"
#include "openMVG/stl/stl.hpp"
#include "openMVG/system/timer.hpp"
graph/graph.hpp
- 描述:这个头文件包含图(graph)的基本数据结构,如节点(node)和边(edge)的定义,以及图的构建、遍历等基本操作。
- 用途:在视觉重建、特征匹配等任务中,图结构经常用于表示图像之间的关系或特征之间的连接。
matching/indMatch.hpp
- 描述:这个头文件包含与特征匹配索引(index match)相关的数据结构或算法。
- 用途:在特征匹配过程中,通常需要将特征描述子与其索引(即它们在图像中的位置)一起处理。这个头文件可能定义了如何处理这些索引匹配。
matching/pairwiseAdjacencyDisplay.hpp
- 描述:这个头文件包含用于显示成对图像之间特征匹配结果的工具或函数。
- 用途:调试或可视化特征匹配结果
Cascade_Hashing_Matcher_Regions.hpp
- 描述:这个头文件包含一个使用级联哈希技术的特征匹配器,专门用于处理图像区域(如关键点周围的区域)。
- 用途:级联哈希技术通常用于加速大规模图像集合中的特征匹配。
Matcher_Regions.hpp
- 描述:定义了一个通用的图像区域匹配器。
- 用途:用于在图像集合中查找相似或匹配的区域。
Pair_Builder.hpp
- 描述:包含一个用于构建图像对的工具或类。
- 用途:在SfM中通常需要选择哪些图像对进行特征匹配。
sfm_features_provider.hpp
- 描述:定义了一个用于提供图像特征的接口或类。
- 用途:在SfM流程中,需要从图像中提取特征,这个头文件定义了如何提供这些特征。
sfm_preemptive_regions_provider.hpp
- 描述:这个头文件定义了一个预处理的图像区域提供器。
- 用途:在SfM的某些阶段,需要预先处理或过滤图像区域。
sfm_regions_provider_cache.hpp
- 描述:这个头文件定义了一个带缓存的图像区域提供器。
- 用途:在处理大型图像集合时,缓存已提取的特征或区域可以显著提高性能。
接下来进入到主函数
int main( int argc, char** argv )
{
CmdLine cmd;
std::string sSfM_Data_Filename;
std::string sOutputMatchesFilename = "";
float fDistRatio = 0.8f;
std::string sPredefinedPairList = "";
std::string sNearestMatchingMethod = "AUTO";
bool bForce = false;
unsigned int ui_max_cache_size = 0;
// Pre-emptive matching parameters
unsigned int ui_preemptive_feature_count = 200;
double preemptive_matching_percentage_threshold = 0.08;
//必须参数
cmd.add( make_option( 'i', sSfM_Data_Filename, "input_file" ) );
cmd.add( make_option( 'o', sOutputMatchesFilename, "output_file" ) );
cmd.add( make_option( 'p', sPredefinedPairList, "pair_list" ) );
// 可选参数
cmd.add( make_option( 'r', fDistRatio, "ratio" ) );
cmd.add( make_option( 'n', sNearestMatchingMethod, "nearest_matching_method" ) );
cmd.add( make_option( 'f', bForce, "force" ) );
cmd.add( make_option( 'c', ui_max_cache_size, "cache_size" ) );
// 预匹配
cmd.add( make_option( 'P', ui_preemptive_feature_count, "preemptive_feature_count") );
try
{
if ( argc == 1 )
throw std::string( "Invalid command line parameter." );
cmd.process( argc, argv );
}
catch ( const std::string& s )
{
OPENMVG_LOG_INFO
<< "Usage: " << argv[ 0 ] << '\n'
<< "[-i|--input_file] A SfM_Data file\n"
<< "[-o|--output_file] Output file where computed matches are stored\n"
<< "[-p|--pair_list] Pairs list file\n"
<< "\n[Optional]\n"
<< "[-f|--force] Force to recompute data]\n"
<< "[-r|--ratio] Distance ratio to discard non meaningful matches\n"
<< " 0.8: (default).\n"
<< "[-n|--nearest_matching_method]\n"
<< " AUTO: auto choice from regions type,\n"
<< " For Scalar based regions descriptor:\n"
<< " BRUTEFORCEL2: L2 BruteForce matching,\n"
<< " HNSWL2: L2 Approximate Matching with Hierarchical Navigable Small World graphs,\n"
<< " HNSWL1: L1 Approximate Matching with Hierarchical Navigable Small World graphs\n"
<< " tailored for quantized and histogram based descriptors (e.g uint8 RootSIFT)\n"
<< " ANNL2: L2 Approximate Nearest Neighbor matching,\n"
<< " CASCADEHASHINGL2: L2 Cascade Hashing matching.\n"
<< " FASTCASCADEHASHINGL2: (default)\n"
<< " L2 Cascade Hashing with precomputed hashed regions\n"
<< " (faster than CASCADEHASHINGL2 but use more memory).\n"
<< " For Binary based descriptor:\n"
<< " BRUTEFORCEHAMMING: BruteForce Hamming matching,\n"
<< " HNSWHAMMING: Hamming Approximate Matching with Hierarchical Navigable Small World graphs\n"
<< "[-c|--cache_size]\n"
<< " Use a regions cache (only cache_size regions will be stored in memory)\n"
<< " If not used, all regions will be load in memory."
<< "\n[Pre-emptive matching:]\n"
<< "[-P|--preemptive_feature_count] <NUMBER> Number of feature used for pre-emptive matching";
OPENMVG_LOG_INFO << s;
return EXIT_FAILURE;
}
OPENMVG_LOG_INFO << " You called : "
<< "\n"
<< argv[ 0 ] << "\n"
<< "--input_file " << sSfM_Data_Filename << "\n"
<< "--output_file " << sOutputMatchesFilename << "\n"
<< "--pair_list " << sPredefinedPairList << "\n"
<< "Optional parameters:"
<< "\n"
<< "--force " << bForce << "\n"
<< "--ratio " << fDistRatio << "\n"
<< "--nearest_matching_method " << sNearestMatchingMethod << "\n"
<< "--cache_size " << ((ui_max_cache_size == 0) ? "unlimited" : std::to_string(ui_max_cache_size)) << "\n"
<< "--preemptive_feature_used/count " << cmd.used('P') << " / " << ui_preemptive_feature_count;
if (cmd.used('P'))
{
OPENMVG_LOG_INFO << "--preemptive_feature_count " << ui_preemptive_feature_count;
}
if ( sOutputMatchesFilename.empty() )
{
OPENMVG_LOG_ERROR << "No output file set.";
return EXIT_FAILURE;
}
// -----------------------------
// 加载SFM视图和内参矩阵
// 计算描述符匹配
// 导出统计数据
// -----------------------------
//---------------------------------------
// 加载SFM视图
//---------------------------------------
SfM_Data sfm_data;
if (!Load(sfm_data, sSfM_Data_Filename, ESfM_Data(VIEWS|INTRINSICS))) {
OPENMVG_LOG_ERROR << "The input SfM_Data file \""<< sSfM_Data_Filename << "\" cannot be read.";
return EXIT_FAILURE;
}
const std::string sMatchesDirectory = stlplus::folder_part( sOutputMatchesFilename );
//---------------------------------------
// 加载SfM场景区域
//---------------------------------------
// 初始化图像描述文件中的regions_type(用于图像区域提取)
// 匹配的描述子初始化
using namespace openMVG::features;
const std::string sImage_describer = stlplus::create_filespec(sMatchesDirectory, "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<Regions_Provider> regions_provider;
if (ui_max_cache_size == 0)
{
// 将所有区域加载到内存中并存储在那里(默认)
regions_provider = std::make_shared<Regions_Provider>();
}
else
{
regions_provider = std::make_shared<Regions_Provider_Cache>(ui_max_cache_size);
}
// 该策略通过提前预测或推断哪些数据或区域将被需要,从而只加载那些真正必要的部分,而不是一次性加载所有数据或区域。
// 预匹配的特征点数量大于零,并且使用预匹配
if (ui_preemptive_feature_count > 0 && cmd.used('P'))
{
regions_provider = std::make_shared<Preemptive_Regions_Provider>(ui_preemptive_feature_count);
}
// Show the progress on the command line:
system::LoggerProgress progress;
if (!regions_provider->load(sfm_data, sMatchesDirectory, regions_type, &progress)) {
OPENMVG_LOG_ERROR << "Cannot load view regions from: " << sMatchesDirectory << ".";
return EXIT_FAILURE;
}
PairWiseMatches map_PutativeMatches;
// 这段代码的目的是从sfm_data中提取所有视图的文件路径和图像大小,并将这些信息分别存储在vec_fileNames和vec_imagesSize两个容器中。
std::vector<std::string> vec_fileNames;
std::vector<std::pair<size_t, size_t>> vec_imagesSize;
{
// 预先分配足够的内存来存储指定数量的元素
vec_fileNames.reserve(sfm_data.GetViews().size());
vec_imagesSize.reserve(sfm_data.GetViews().size());
for (const auto view_it : sfm_data.GetViews())
{
// 提取视图信息
const View * v = view_it.second.get();
// 构建文件路径
vec_fileNames.emplace_back(stlplus::create_filespec(sfm_data.s_root_path,
v->s_Img_path));
// 存储图像大小
vec_imagesSize.emplace_back(v->ui_width, v->ui_height);
}
}
OPENMVG_LOG_INFO << " - PUTATIVE MATCHES - ";
// 如果图像对存在则加载它们
if ( !bForce && ( stlplus::file_exists( sOutputMatchesFilename ) ) )
{
if ( !( Load( map_PutativeMatches, sOutputMatchesFilename ) ) )
{
OPENMVG_LOG_ERROR << "Cannot load input matches file";
return EXIT_FAILURE;
}
OPENMVG_LOG_INFO
<< "\t PREVIOUS RESULTS LOADED;"
<< " #pair: " << map_PutativeMatches.size();
}
else //不存在则匹配图像对
{
// 根据匹配请求的方法分配正确的匹配器
std::unique_ptr<Matcher> collectionMatcher;
if ( sNearestMatchingMethod == "AUTO" )
{
if ( regions_type->IsScalar() )
{
OPENMVG_LOG_INFO << "Using FAST_CASCADE_HASHING_L2 matcher";
collectionMatcher.reset(new Cascade_Hashing_Matcher_Regions(fDistRatio));
}
else
if (regions_type->IsBinary())
{
OPENMVG_LOG_INFO << "Using HNSWHAMMING matcher";
collectionMatcher.reset(new Matcher_Regions(fDistRatio, HNSW_HAMMING));
}
}
else
if (sNearestMatchingMethod == "BRUTEFORCEL2")
{
OPENMVG_LOG_INFO << "Using BRUTE_FORCE_L2 matcher";
collectionMatcher.reset(new Matcher_Regions(fDistRatio, BRUTE_FORCE_L2));
}
else
if (sNearestMatchingMethod == "BRUTEFORCEHAMMING")
{
OPENMVG_LOG_INFO << "Using BRUTE_FORCE_HAMMING matcher";
collectionMatcher.reset(new Matcher_Regions(fDistRatio, BRUTE_FORCE_HAMMING));
}
else
if (sNearestMatchingMethod == "HNSWL2")
{
OPENMVG_LOG_INFO << "Using HNSWL2 matcher";
collectionMatcher.reset(new Matcher_Regions(fDistRatio, HNSW_L2));
}
if (sNearestMatchingMethod == "HNSWL1")
{
OPENMVG_LOG_INFO << "Using HNSWL1 matcher";
collectionMatcher.reset(new Matcher_Regions(fDistRatio, HNSW_L1));
}
else
if (sNearestMatchingMethod == "HNSWHAMMING")
{
OPENMVG_LOG_INFO << "Using HNSWHAMMING matcher";
collectionMatcher.reset(new Matcher_Regions(fDistRatio, HNSW_HAMMING));
}
else
if (sNearestMatchingMethod == "ANNL2")
{
OPENMVG_LOG_INFO << "Using ANN_L2 matcher";
collectionMatcher.reset(new Matcher_Regions(fDistRatio, ANN_L2));
}
else
if (sNearestMatchingMethod == "CASCADEHASHINGL2")
{
OPENMVG_LOG_INFO << "Using CASCADE_HASHING_L2 matcher";
collectionMatcher.reset(new Matcher_Regions(fDistRatio, CASCADE_HASHING_L2));
}
else
if (sNearestMatchingMethod == "FASTCASCADEHASHINGL2")
{
OPENMVG_LOG_INFO << "Using FAST_CASCADE_HASHING_L2 matcher";
collectionMatcher.reset(new Cascade_Hashing_Matcher_Regions(fDistRatio));
}
if (!collectionMatcher)
{
OPENMVG_LOG_ERROR << "Invalid Nearest Neighbor method: " << sNearestMatchingMethod;
return EXIT_FAILURE;
}
// 匹配操作
system::Timer timer;
{
// From matching mode compute the pair list that have to be matched:
Pair_Set pairs;
if ( sPredefinedPairList.empty() )
{
OPENMVG_LOG_INFO << "No input pair file set. Use exhaustive match by default.";
const size_t NImage = sfm_data.GetViews().size();
pairs = exhaustivePairs( NImage );
}
else
if ( !loadPairs( sfm_data.GetViews().size(), sPredefinedPairList, pairs ) )
{
OPENMVG_LOG_ERROR << "Failed to load pairs from file: \"" << sPredefinedPairList << "\"";
return EXIT_FAILURE;
}
OPENMVG_LOG_INFO << "Running matching on #pairs: " << pairs.size();
// 匹配操作!!!
collectionMatcher->Match( regions_provider, pairs, map_PutativeMatches, &progress );
if (cmd.used('P')) // Preemptive filter
{
// Keep putative matches only if there is more than X matches
PairWiseMatches map_filtered_matches;
for (const auto & pairwisematches_it : map_PutativeMatches)
{
// 计算匹配的特征数量
const size_t putative_match_count = pairwisematches_it.second.size();
// 根据百分比阈值和特征数量计算一个阈值
const int match_count_threshold =
preemptive_matching_percentage_threshold * ui_preemptive_feature_count;
// TODO: Add an option to keeping X Best pairs
// 如果匹配的特征数量大于或等于这个阈值,则保留该对图像点
if (putative_match_count >= match_count_threshold) {
// the pair will be kept
map_filtered_matches.insert(pairwisematches_it);
}
}
map_PutativeMatches.clear();
std::swap(map_filtered_matches, map_PutativeMatches);
}
//---------------------------------------
//-- Export putative matches & pairs
//---------------------------------------
if ( !Save( map_PutativeMatches, std::string( sOutputMatchesFilename ) ) )
{
OPENMVG_LOG_ERROR
<< "Cannot save computed matches in: "
<< sOutputMatchesFilename;
return EXIT_FAILURE;
}
// Save pairs
const std::string sOutputPairFilename =
stlplus::create_filespec( sMatchesDirectory, "preemptive_pairs", "txt" );
if (!savePairs(
sOutputPairFilename,
getPairs(map_PutativeMatches)))
{
OPENMVG_LOG_ERROR
<< "Cannot save computed matches pairs in: "
<< sOutputPairFilename;
return EXIT_FAILURE;
}
}
OPENMVG_LOG_INFO << "Task (Regions Matching) done in (s): " << timer.elapsed();
}
OPENMVG_LOG_INFO << "#Putative pairs: " << map_PutativeMatches.size();
// -- export Putative View Graph statistics
graph::getGraphStatistics(sfm_data.GetViews().size(), getPairs(map_PutativeMatches));
//-- export putative matches Adjacency matrix
PairWiseMatchingToAdjacencyMatrixSVG( vec_fileNames.size(),
map_PutativeMatches,
stlplus::create_filespec( sMatchesDirectory, "PutativeAdjacencyMatrix", "svg" ) );
//-- export view pair graph once putative graph matches has been computed
{
std::set<IndexT> set_ViewIds;
std::transform( sfm_data.GetViews().begin(), sfm_data.GetViews().end(), std::inserter( set_ViewIds, set_ViewIds.begin() ), stl::RetrieveKey() );
graph::indexedGraph putativeGraph( set_ViewIds, getPairs( map_PutativeMatches ) );
graph::exportToGraphvizData(
stlplus::create_filespec( sMatchesDirectory, "putative_matches" ),
putativeGraph );
}
return EXIT_SUCCESS;
}
匹配算法如下:
void Cascade_Hashing_Matcher_Regions::Match
(
const std::shared_ptr<sfm::Regions_Provider> & regions_provider,
const Pair_Set & pairs,
PairWiseMatchesContainer & map_PutativeMatches, // the pairwise photometric corresponding points
system::ProgressInterface * my_progress_bar
)const
{
#ifdef OPENMVG_USE_OPENMP
OPENMVG_LOG_INFO << "Using the OPENMP thread interface";
#endif
if (!regions_provider)
return;
if (regions_provider->IsBinary())
return;
if (regions_provider->Type_id() == typeid(unsigned char).name())
{
impl::Match<unsigned char>(
*regions_provider.get(),
pairs,
f_dist_ratio_,
map_PutativeMatches,
my_progress_bar);
}
else
if (regions_provider->Type_id() == typeid(float).name())
{
impl::Match<float>(
*regions_provider.get(),
pairs,
f_dist_ratio_,
map_PutativeMatches,
my_progress_bar);
}
else
{
OPENMVG_LOG_ERROR << "Matcher not implemented for this region type: " << regions_provider->Type_id();
}
}
其中调用impl::Match函数进行匹配:
namespace impl
{
template <typename ScalarT>
void Match
(
const sfm::Regions_Provider & regions_provider,
const Pair_Set & pairs,
float fDistRatio,
PairWiseMatchesContainer & map_PutativeMatches, // the pairwise photometric corresponding points
system::ProgressInterface * my_progress_bar
)
{
if (!my_progress_bar)
my_progress_bar = &system::ProgressInterface::dummy();
my_progress_bar->Restart(pairs.size(), "- Matching -");
// Collect used view indexes
std::set<IndexT> used_index;
// Sort pairs according the first index to minimize later memory swapping
using Map_vectorT = std::map<IndexT, std::vector<IndexT>>;
Map_vectorT map_Pairs;
for (const auto & pair_idx : pairs)
{
map_Pairs[pair_idx.first].push_back(pair_idx.second);
used_index.insert(pair_idx.first);
used_index.insert(pair_idx.second);
}
using BaseMat = Eigen::Matrix<ScalarT, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor>;
// Init the cascade hasher
CascadeHasher cascade_hasher;
if (!used_index.empty())
{
const IndexT I = *used_index.begin();
const std::shared_ptr<features::Regions> regionsI = regions_provider.get(I);
const size_t dimension = regionsI->DescriptorLength();
cascade_hasher.Init(dimension);
}
std::map<IndexT, HashedDescriptions> hashed_base_;
// Compute the zero mean descriptor that will be used for hashing (one for all the image regions)
Eigen::VectorXf zero_mean_descriptor;
{
Eigen::MatrixXf matForZeroMean;
for (int i =0; i < used_index.size(); ++i)
{
std::set<IndexT>::const_iterator iter = used_index.begin();
std::advance(iter, i);
const IndexT I = *iter;
const std::shared_ptr<features::Regions> regionsI = regions_provider.get(I);
const ScalarT * tabI =
reinterpret_cast<const ScalarT*>(regionsI->DescriptorRawData());
const size_t dimension = regionsI->DescriptorLength();
if (i==0)
{
matForZeroMean.resize(used_index.size(), dimension);
matForZeroMean.fill(0.0f);
}
if (regionsI->RegionCount() > 0)
{
Eigen::Map<BaseMat> mat_I( (ScalarT*)tabI, regionsI->RegionCount(), dimension);
matForZeroMean.row(i) = CascadeHasher::GetZeroMeanDescriptor(mat_I);
}
}
zero_mean_descriptor = CascadeHasher::GetZeroMeanDescriptor(matForZeroMean);
}
// Index the input regions
#ifdef OPENMVG_USE_OPENMP
#pragma omp parallel for schedule(dynamic)
#endif
for (int i =0; i < used_index.size(); ++i)
{
std::set<IndexT>::const_iterator iter = used_index.begin();
std::advance(iter, i);
const IndexT I = *iter;
const std::shared_ptr<features::Regions> regionsI = regions_provider.get(I);
const ScalarT * tabI =
reinterpret_cast<const ScalarT*>(regionsI->DescriptorRawData());
const size_t dimension = regionsI->DescriptorLength();
Eigen::Map<BaseMat> mat_I( (ScalarT*)tabI, regionsI->RegionCount(), dimension);
#ifdef OPENMVG_USE_OPENMP
#pragma omp critical
#endif
{
hashed_base_[I] =
std::move(cascade_hasher.CreateHashedDescriptions(mat_I, zero_mean_descriptor));
}
}
// Perform matching between all the pairs
for (const auto & pair_it : map_Pairs)
{
if (my_progress_bar->hasBeenCanceled())
break;
const IndexT I = pair_it.first;
const std::vector<IndexT> & indexToCompare = pair_it.second;
const std::shared_ptr<features::Regions> regionsI = regions_provider.get(I);
if (regionsI->RegionCount() == 0)
{
(*my_progress_bar) += indexToCompare.size();
continue;
}
const std::vector<features::PointFeature> pointFeaturesI = regionsI->GetRegionsPositions();
const ScalarT * tabI =
reinterpret_cast<const ScalarT*>(regionsI->DescriptorRawData());
const size_t dimension = regionsI->DescriptorLength();
Eigen::Map<BaseMat> mat_I( (ScalarT*)tabI, regionsI->RegionCount(), dimension);
#ifdef OPENMVG_USE_OPENMP
#pragma omp parallel for schedule(dynamic)
#endif
for (int j = 0; j < (int)indexToCompare.size(); ++j)
{
if (my_progress_bar->hasBeenCanceled())
continue;
const size_t J = indexToCompare[j];
const std::shared_ptr<features::Regions> regionsJ = regions_provider.get(J);
if (regionsI->Type_id() != regionsJ->Type_id())
{
++(*my_progress_bar);
continue;
}
// Matrix representation of the query input data;
const ScalarT * tabJ = reinterpret_cast<const ScalarT*>(regionsJ->DescriptorRawData());
Eigen::Map<BaseMat> mat_J( (ScalarT*)tabJ, regionsJ->RegionCount(), dimension);
IndMatches pvec_indices;
using ResultType = typename Accumulator<ScalarT>::Type;
std::vector<ResultType> pvec_distances;
pvec_distances.reserve(regionsJ->RegionCount() * 2);
pvec_indices.reserve(regionsJ->RegionCount() * 2);
// Match the query descriptors to the database
cascade_hasher.Match_HashedDescriptions<BaseMat, ResultType>(
hashed_base_[J], mat_J,
hashed_base_[I], mat_I,
&pvec_indices, &pvec_distances);
std::vector<int> vec_nn_ratio_idx;
// Filter the matches using a distance ratio test:
// The probability that a match is correct is determined by taking
// the ratio of distance from the closest neighbor to the distance
// of the second closest.
matching::NNdistanceRatio(
pvec_distances.begin(), // distance start
pvec_distances.end(), // distance end
2, // Number of neighbor in iterator sequence (minimum required 2)
vec_nn_ratio_idx, // output (indices that respect the distance Ratio)
Square(fDistRatio));
matching::IndMatches vec_putative_matches;
vec_putative_matches.reserve(vec_nn_ratio_idx.size());
for (size_t k=0; k < vec_nn_ratio_idx.size(); ++k)
{
const size_t index = vec_nn_ratio_idx[k];
vec_putative_matches.emplace_back(pvec_indices[index*2].j_, pvec_indices[index*2].i_);
}
// Remove duplicates
matching::IndMatch::getDeduplicated(vec_putative_matches);
// Remove matches that have the same (X,Y) coordinates
const std::vector<features::PointFeature> pointFeaturesJ = regionsJ->GetRegionsPositions();
matching::IndMatchDecorator<float> matchDeduplicator(vec_putative_matches,
pointFeaturesI, pointFeaturesJ);
matchDeduplicator.getDeduplicated(vec_putative_matches);
#ifdef OPENMVG_USE_OPENMP
#pragma omp critical
#endif
{
if (!vec_putative_matches.empty())
{
map_PutativeMatches.insert(
{
{I,J},
std::move(vec_putative_matches)
});
}
}
++(*my_progress_bar);
}
}
}
} // namespace impl