You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1493 lines
53 KiB
1493 lines
53 KiB
/* |
|
* InterfaceCOLMAP.cpp |
|
* |
|
* Copyright (c) 2014-2018 SEACAVE |
|
* |
|
* Author(s): |
|
* |
|
* cDc <cdc.seacave@gmail.com> |
|
* |
|
* |
|
* This program is free software: you can redistribute it and/or modify |
|
* it under the terms of the GNU Affero General Public License as published by |
|
* the Free Software Foundation, either version 3 of the License, or |
|
* (at your option) any later version. |
|
* |
|
* This program is distributed in the hope that it will be useful, |
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
* GNU Affero General Public License for more details. |
|
* |
|
* You should have received a copy of the GNU Affero General Public License |
|
* along with this program. If not, see <http://www.gnu.org/licenses/>. |
|
* |
|
* |
|
* Additional Terms: |
|
* |
|
* You are required to preserve legal notices and author attributions in |
|
* that material or in the Appropriate Legal Notices displayed by works |
|
* containing it. |
|
*/ |
|
|
|
#include "../../libs/MVS/Common.h" |
|
#include "../../libs/MVS/Scene.h" |
|
#include <boost/program_options.hpp> |
|
#include "endian.h" |
|
|
|
using namespace MVS; |
|
|
|
|
|
// D E F I N E S /////////////////////////////////////////////////// |
|
|
|
#define APPNAME _T("InterfaceCOLMAP") |
|
#define MVS_FILE_EXTENSION _T(".mvs") |
|
#define COLMAP_IMAGES_FOLDER _T("images/") |
|
#define COLMAP_SPARSE_FOLDER _T("sparse/") |
|
#define COLMAP_CAMERAS_TXT COLMAP_SPARSE_FOLDER _T("cameras.txt") |
|
#define COLMAP_IMAGES_TXT COLMAP_SPARSE_FOLDER _T("images.txt") |
|
#define COLMAP_POINTS_TXT COLMAP_SPARSE_FOLDER _T("points3D.txt") |
|
#define COLMAP_CAMERAS_BIN COLMAP_SPARSE_FOLDER _T("cameras.bin") |
|
#define COLMAP_IMAGES_BIN COLMAP_SPARSE_FOLDER _T("images.bin") |
|
#define COLMAP_POINTS_BIN COLMAP_SPARSE_FOLDER _T("points3D.bin") |
|
#define COLMAP_DENSE_POINTS _T("fused.ply") |
|
#define COLMAP_DENSE_POINTS_VISIBILITY _T("fused.ply.vis") |
|
#define COLMAP_STEREO_FOLDER _T("stereo/") |
|
#define COLMAP_FUSION COLMAP_STEREO_FOLDER _T("fusion.cfg") |
|
#define COLMAP_PATCHMATCH COLMAP_STEREO_FOLDER _T("patch-match.cfg") |
|
#define COLMAP_STEREO_CONSISTENCYGRAPHS_FOLDER COLMAP_STEREO_FOLDER _T("consistency_graphs/") |
|
#define COLMAP_STEREO_DEPTHMAPS_FOLDER COLMAP_STEREO_FOLDER _T("depth_maps/") |
|
#define COLMAP_STEREO_NORMALMAPS_FOLDER COLMAP_STEREO_FOLDER _T("normal_maps/") |
|
|
|
|
|
// S T R U C T S /////////////////////////////////////////////////// |
|
|
|
namespace { |
|
|
|
namespace OPT { |
|
bool IsFromOpenMVS; // conversion direction |
|
bool bNormalizeIntrinsics; |
|
bool bForceSparsePointCloud; |
|
String strInputFileName; |
|
String strPointCloudFileName; |
|
String strOutputFileName; |
|
String strImageFolder; |
|
unsigned nArchiveType; |
|
int nProcessPriority; |
|
unsigned nMaxThreads; |
|
String strConfigFileName; |
|
boost::program_options::variables_map vm; |
|
} // namespace OPT |
|
|
|
class Application { |
|
public: |
|
Application() {} |
|
~Application() { CleanUp(); } |
|
|
|
bool Initialize(size_t argc, LPCTSTR* argv); |
|
void CleanUp(); |
|
}; // Application |
|
|
|
// Initializes the application and parses command-line parameters |
|
bool Application::Initialize(size_t argc, LPCTSTR* argv) |
|
{ |
|
// Open the log file and console for output |
|
OPEN_LOG(); |
|
OPEN_LOGCONSOLE(); |
|
|
|
// group of options allowed only on command line |
|
boost::program_options::options_description generic("Generic options"); |
|
generic.add_options() |
|
("help,h", "imports SfM or MVS scene stored in COLMAP undistoreted format OR exports MVS scene to COLMAP format") |
|
("working-folder,w", boost::program_options::value<std::string>(&WORKING_FOLDER), "working directory (default current directory)") |
|
("config-file,c", boost::program_options::value<std::string>(&OPT::strConfigFileName)->default_value(APPNAME _T(".cfg")), "file name containing program options") |
|
("archive-type", boost::program_options::value(&OPT::nArchiveType)->default_value(ARCHIVE_MVS), "project archive type: -1-interface, 0-text, 1-binary, 2-compressed binary") |
|
("process-priority", boost::program_options::value(&OPT::nProcessPriority)->default_value(-1), "process priority (below normal by default)") |
|
("max-threads", boost::program_options::value(&OPT::nMaxThreads)->default_value(0), "maximum number of threads (0 for using all available cores)") |
|
#if TD_VERBOSE != TD_VERBOSE_OFF |
|
("verbosity,v", boost::program_options::value(&g_nVerbosityLevel)->default_value( |
|
#if TD_VERBOSE == TD_VERBOSE_DEBUG |
|
3 |
|
#else |
|
2 |
|
#endif |
|
), "verbosity level") |
|
#endif |
|
; |
|
|
|
// group of options allowed both on command line and in config file |
|
boost::program_options::options_description configOptions("Main options"); |
|
configOptions.add_options() |
|
("input-file,i", boost::program_options::value<std::string>(&OPT::strInputFileName), "input COLMAP folder containing cameras, images and points files OR input MVS project file") |
|
("pointcloud-file,p", boost::program_options::value<std::string>(&OPT::strPointCloudFileName), "point-cloud with views file name (overwrite existing point-cloud)") |
|
("output-file,o", boost::program_options::value<std::string>(&OPT::strOutputFileName), "output filename for storing the MVS project") |
|
("image-folder", boost::program_options::value<std::string>(&OPT::strImageFolder)->default_value(COLMAP_IMAGES_FOLDER), "folder to the undistorted images") |
|
("normalize,f", boost::program_options::value(&OPT::bNormalizeIntrinsics)->default_value(false), "normalize intrinsics while exporting to MVS format") |
|
("force-points,e", boost::program_options::value(&OPT::bForceSparsePointCloud)->default_value(false), "force exporting point-cloud as sparse points also even if dense point-cloud detected") |
|
; |
|
|
|
// Combine all command-line options |
|
boost::program_options::options_description cmdline_options; |
|
cmdline_options.add(generic).add(configOptions); |
|
|
|
// Only config file options |
|
boost::program_options::options_description config_file_options; |
|
config_file_options.add(configOptions); |
|
|
|
boost::program_options::positional_options_description positionalOptions; |
|
positionalOptions.add("input-file", -1); |
|
|
|
try { |
|
// parse command line options |
|
boost::program_options::store(boost::program_options::command_line_parser((int)argc, argv).options(cmdline_options).positional(positionalOptions).run(), OPT::vm); |
|
boost::program_options::notify(OPT::vm); |
|
|
|
// Initialize working folder |
|
INIT_WORKING_FOLDER; |
|
|
|
// Parse configuration file if exists |
|
std::ifstream configFile(MAKE_PATH_SAFE(OPT::strConfigFileName)); |
|
if (configFile) { |
|
boost::program_options::store(parse_config_file(configFile, config_file_options), OPT::vm); |
|
boost::program_options::notify(OPT::vm); |
|
} |
|
} |
|
catch (const std::exception& e) { |
|
LOG(e.what()); |
|
return false; |
|
} |
|
|
|
// Open and configure the log file |
|
OPEN_LOGFILE(MAKE_PATH(APPNAME _T("-")+Util::getUniqueName(0)+_T(".log"))); |
|
|
|
// print application details: version and command line |
|
Util::LogBuild(); |
|
LOG(_T("Command line: ") APPNAME _T("%s"), Util::CommandLineToString(argc, argv).c_str()); |
|
|
|
// Check and validate paths |
|
Util::ensureValidPath(OPT::strInputFileName); |
|
Util::ensureValidPath(OPT::strPointCloudFileName); |
|
|
|
// Handle help option or invalid command |
|
const bool bInvalidCommand(OPT::strInputFileName.empty()); |
|
if (OPT::vm.count("help") || bInvalidCommand) { |
|
boost::program_options::options_description visible("Available options"); |
|
visible.add(generic).add(configOptions); |
|
GET_LOG() << _T("\n" |
|
"Import/export 3D reconstruction from COLMAP (TXT/BIN format) and to COLMAP (TXT format). \n" |
|
"In order to import a scene, run COLMAP SfM and next undistort the images (only PINHOLE\n" |
|
"camera model supported for the moment)." |
|
"\n") |
|
<< visible; |
|
} |
|
|
|
if (bInvalidCommand) |
|
return false; |
|
|
|
// initialize optional options |
|
Util::ensureValidFolderPath(OPT::strImageFolder); |
|
Util::ensureValidPath(OPT::strOutputFileName); |
|
OPT::strImageFolder = MAKE_PATH_FULL(WORKING_FOLDER_FULL, OPT::strImageFolder); |
|
|
|
// Determine if the source file is an OpenMVS project by checking the file extension |
|
const String inputFileExtension(Util::getFileExt(OPT::strInputFileName).ToLower()); |
|
OPT::IsFromOpenMVS = (inputFileExtension == MVS_FILE_EXTENSION); |
|
|
|
// Set default output filename based on input filename if not explicitly specified |
|
if (OPT::IsFromOpenMVS) { |
|
if (OPT::strOutputFileName.empty()) |
|
OPT::strOutputFileName = Util::getFilePath(OPT::strInputFileName); |
|
} else { |
|
// Ensure the input filename ends with a directory slash if needed |
|
Util::ensureFolderSlash(OPT::strInputFileName); |
|
if (OPT::strOutputFileName.empty()) |
|
OPT::strOutputFileName = OPT::strInputFileName + _T("scene") MVS_FILE_EXTENSION; |
|
} |
|
|
|
MVS::Initialize(APPNAME, OPT::nMaxThreads, OPT::nProcessPriority); |
|
return true; |
|
} |
|
|
|
// finalize application instance |
|
void Application::CleanUp() |
|
{ |
|
MVS::Finalize(); |
|
|
|
CLOSE_LOGFILE(); |
|
CLOSE_LOGCONSOLE(); |
|
CLOSE_LOG(); |
|
} |
|
|
|
namespace COLMAP { |
|
|
|
using namespace colmap; |
|
|
|
// See colmap/src/util/types.h |
|
typedef uint32_t camera_t; |
|
typedef uint32_t image_t; |
|
typedef uint64_t image_pair_t; |
|
typedef uint32_t point2D_t; |
|
typedef uint64_t point3D_t; |
|
|
|
const std::vector<String> mapCameraModel = { |
|
"SIMPLE_PINHOLE", |
|
"PINHOLE", |
|
"SIMPLE_RADIAL", |
|
"RADIAL", |
|
"OPENCV", |
|
"OPENCV_FISHEYE", |
|
"FULL_OPENCV", |
|
"FOV", |
|
"SIMPLE_RADIAL_FISHEYE", |
|
"RADIAL_FISHEYE", |
|
"THIN_PRISM_FISHEYE" |
|
}; |
|
|
|
// tools |
|
bool NextLine(std::istream& stream, std::istringstream& in, bool bIgnoreEmpty=true) { |
|
String line; |
|
do { |
|
std::getline(stream, line); |
|
Util::strTrim(line, _T(" ")); |
|
if (stream.fail()) |
|
return false; |
|
} while (((bIgnoreEmpty && line.empty()) || line[0u] == '#') && stream.good()); |
|
in.clear(); |
|
in.str(line); |
|
return true; |
|
} |
|
// structure describing a camera |
|
struct Camera { |
|
uint32_t ID; // ID of the camera |
|
String model; // camera model name |
|
uint32_t width, height; // camera resolution |
|
std::vector<REAL> params; // camera parameters |
|
uint64_t numCameras{0}; // only for binary format |
|
|
|
Camera() {} |
|
Camera(uint32_t _ID) : ID(_ID) {} |
|
bool operator < (const Camera& rhs) const { return ID < rhs.ID; } |
|
|
|
struct CameraHash { |
|
size_t operator()(const Camera& camera) const { |
|
const size_t h1(std::hash<String>()(camera.model)); |
|
const size_t h2(std::hash<uint32_t>()(camera.width)); |
|
const size_t h3(std::hash<uint32_t>()(camera.height)); |
|
size_t h(h1 ^ ((h2 ^ (h3 << 1)) << 1)); |
|
for (REAL p: camera.params) |
|
h = std::hash<REAL>()(p) ^ (h << 1); |
|
return h; |
|
} |
|
}; |
|
struct CameraEqualTo { |
|
bool operator()(const Camera& _Left, const Camera& _Right) const { |
|
return _Left.model == _Right.model && |
|
_Left.width == _Right.width && _Left.height == _Right.height && |
|
_Left.params == _Right.params; |
|
} |
|
}; |
|
|
|
bool Read(std::istream& stream, bool binary) { |
|
if (binary) |
|
return ReadBIN(stream); |
|
return ReadTXT(stream); |
|
} |
|
|
|
bool Write(std::ostream& stream, bool binary) { |
|
if (binary) |
|
return WriteBIN(stream); |
|
return WriteTXT(stream); |
|
} |
|
|
|
// Camera list with one line of data per camera: |
|
// CAMERA_ID, MODEL, WIDTH, HEIGHT, PARAMS[] |
|
bool ReadTXT(std::istream& stream) { |
|
std::istringstream in; |
|
if (!NextLine(stream, in)) |
|
return false; |
|
in >> ID >> model >> width >> height; |
|
if (in.fail()) |
|
return false; |
|
--ID; |
|
if (model != _T("PINHOLE")) |
|
return false; |
|
params.resize(4); |
|
in >> params[0] >> params[1] >> params[2] >> params[3]; |
|
return !in.fail(); |
|
} |
|
|
|
// See: colmap/src/base/reconstruction.cc |
|
// void Reconstruction::ReadCamerasBinary(const std::string& path) |
|
bool ReadBIN(std::istream& stream) { |
|
if (stream.peek() == EOF) |
|
return false; |
|
|
|
if (numCameras == 0) { |
|
// Read the first entry in the binary file |
|
numCameras = ReadBinaryLittleEndian<uint64_t>(&stream); |
|
} |
|
|
|
ID = ReadBinaryLittleEndian<camera_t>(&stream)-1; |
|
model = mapCameraModel[ReadBinaryLittleEndian<int>(&stream)]; |
|
width = (uint32_t)ReadBinaryLittleEndian<uint64_t>(&stream); |
|
height = (uint32_t)ReadBinaryLittleEndian<uint64_t>(&stream); |
|
if (model != _T("PINHOLE")) |
|
return false; |
|
params.resize(4); |
|
ReadBinaryLittleEndian<double>(&stream, ¶ms); |
|
return true; |
|
} |
|
|
|
bool WriteTXT(std::ostream& out) const { |
|
out << ID+1 << _T(" ") << model << _T(" ") << width << _T(" ") << height; |
|
if (out.fail()) |
|
return false; |
|
for (REAL param: params) { |
|
out << _T(" ") << param; |
|
if (out.fail()) |
|
return false; |
|
} |
|
out << std::endl; |
|
return true; |
|
} |
|
|
|
bool WriteBIN(std::ostream& stream) { |
|
if (numCameras != 0) { |
|
// Write the first entry in the binary file |
|
WriteBinaryLittleEndian<uint64_t>(&stream, numCameras); |
|
numCameras = 0; |
|
} |
|
|
|
WriteBinaryLittleEndian<camera_t>(&stream, ID+1); |
|
const int64 modelId(std::distance(mapCameraModel.begin(), std::find(mapCameraModel.begin(), mapCameraModel.end(), model))); |
|
WriteBinaryLittleEndian<int>(&stream, (int)modelId); |
|
WriteBinaryLittleEndian<uint64_t>(&stream, width); |
|
WriteBinaryLittleEndian<uint64_t>(&stream, height); |
|
for (REAL param: params) |
|
WriteBinaryLittleEndian<double>(&stream, param); |
|
return !stream.fail(); |
|
} |
|
}; |
|
typedef std::vector<Camera> Cameras; |
|
// structure describing an image |
|
struct Image { |
|
struct Proj { |
|
Eigen::Vector2f p; |
|
uint32_t idPoint; |
|
}; |
|
uint32_t ID; // ID of the image |
|
Eigen::Quaterniond q; // rotation |
|
Eigen::Vector3d t; // translation |
|
uint32_t idCamera; // ID of the associated camera |
|
String name; // image file name |
|
std::vector<Proj> projs; // known image projections |
|
uint64_t numRegImages{0}; // only for binary format |
|
|
|
Image() {} |
|
Image(uint32_t _ID) : ID(_ID) {} |
|
bool operator < (const Image& rhs) const { return ID < rhs.ID; } |
|
|
|
bool Read(std::istream& stream, bool binary) { |
|
if (binary) |
|
return ReadBIN(stream); |
|
return ReadTXT(stream); |
|
} |
|
|
|
bool Write(std::ostream& stream, bool binary) { |
|
if (binary) |
|
return WriteBIN(stream); |
|
return WriteTXT(stream); |
|
} |
|
|
|
// Image list with two lines of data per image: |
|
// IMAGE_ID, QW, QX, QY, QZ, TX, TY, TZ, CAMERA_ID, NAME |
|
// POINTS2D[] as (X, Y, POINT3D_ID) |
|
bool ReadTXT(std::istream& stream) { |
|
std::istringstream in; |
|
if (!NextLine(stream, in)) |
|
return false; |
|
in >> ID |
|
>> q.w() >> q.x() >> q.y() >> q.z() |
|
>> t(0) >> t(1) >> t(2) |
|
>> idCamera >> name; |
|
if (in.fail()) |
|
return false; |
|
--ID; --idCamera; |
|
Util::ensureValidPath(name); |
|
if (!NextLine(stream, in, false)) |
|
return false; |
|
projs.clear(); |
|
while (true) { |
|
Proj proj; |
|
in >> proj.p(0) >> proj.p(1) >> (int&)proj.idPoint; |
|
if (in.fail()) |
|
break; |
|
--proj.idPoint; |
|
projs.emplace_back(proj); |
|
} |
|
return true; |
|
} |
|
|
|
// See: colmap/src/base/reconstruction.cc |
|
// void Reconstruction::ReadImagesBinary(const std::string& path) |
|
bool ReadBIN(std::istream& stream) { |
|
if (stream.peek() == EOF) |
|
return false; |
|
|
|
if (!numRegImages) { |
|
// Read the first entry in the binary file |
|
numRegImages = ReadBinaryLittleEndian<uint64_t>(&stream); |
|
} |
|
|
|
ID = ReadBinaryLittleEndian<image_t>(&stream)-1; |
|
q.w() = ReadBinaryLittleEndian<double>(&stream); |
|
q.x() = ReadBinaryLittleEndian<double>(&stream); |
|
q.y() = ReadBinaryLittleEndian<double>(&stream); |
|
q.z() = ReadBinaryLittleEndian<double>(&stream); |
|
t(0) = ReadBinaryLittleEndian<double>(&stream); |
|
t(1) = ReadBinaryLittleEndian<double>(&stream); |
|
t(2) = ReadBinaryLittleEndian<double>(&stream); |
|
idCamera = ReadBinaryLittleEndian<camera_t>(&stream)-1; |
|
|
|
name = ""; |
|
while (true) { |
|
char nameChar; |
|
stream.read(&nameChar, 1); |
|
if (nameChar == '\0') |
|
break; |
|
name += nameChar; |
|
} |
|
Util::ensureValidPath(name); |
|
|
|
const size_t numPoints2D = ReadBinaryLittleEndian<uint64_t>(&stream); |
|
projs.clear(); |
|
for (size_t j = 0; j < numPoints2D; ++j) { |
|
Proj proj; |
|
proj.p(0) = (float)ReadBinaryLittleEndian<double>(&stream); |
|
proj.p(1) = (float)ReadBinaryLittleEndian<double>(&stream); |
|
proj.idPoint = (uint32_t)ReadBinaryLittleEndian<point3D_t>(&stream)-1; |
|
projs.emplace_back(proj); |
|
} |
|
return true; |
|
} |
|
|
|
bool WriteTXT(std::ostream& out) const { |
|
out << ID+1 << _T(" ") |
|
<< q.w() << _T(" ") << q.x() << _T(" ") << q.y() << _T(" ") << q.z() << _T(" ") |
|
<< t(0) << _T(" ") << t(1) << _T(" ") << t(2) << _T(" ") |
|
<< idCamera+1 << _T(" ") << name |
|
<< std::endl; |
|
for (const Proj& proj: projs) { |
|
out << proj.p(0) << _T(" ") << proj.p(1) << _T(" ") << (int)proj.idPoint+1 << _T(" "); |
|
if (out.fail()) |
|
return false; |
|
} |
|
out << std::endl; |
|
return !out.fail(); |
|
} |
|
|
|
bool WriteBIN(std::ostream& stream) { |
|
if (numRegImages != 0) { |
|
// Write the first entry in the binary file |
|
WriteBinaryLittleEndian<uint64_t>(&stream, numRegImages); |
|
numRegImages = 0; |
|
} |
|
|
|
WriteBinaryLittleEndian<image_t>(&stream, ID+1); |
|
|
|
WriteBinaryLittleEndian<double>(&stream, q.w()); |
|
WriteBinaryLittleEndian<double>(&stream, q.x()); |
|
WriteBinaryLittleEndian<double>(&stream, q.y()); |
|
WriteBinaryLittleEndian<double>(&stream, q.z()); |
|
|
|
WriteBinaryLittleEndian<double>(&stream, t(0)); |
|
WriteBinaryLittleEndian<double>(&stream, t(1)); |
|
WriteBinaryLittleEndian<double>(&stream, t(2)); |
|
|
|
WriteBinaryLittleEndian<camera_t>(&stream, idCamera+1); |
|
|
|
stream.write(name.c_str(), name.size()+1); |
|
|
|
WriteBinaryLittleEndian<uint64_t>(&stream, projs.size()); |
|
for (const Proj& proj: projs) { |
|
WriteBinaryLittleEndian<double>(&stream, proj.p(0)); |
|
WriteBinaryLittleEndian<double>(&stream, proj.p(1)); |
|
WriteBinaryLittleEndian<point3D_t>(&stream, proj.idPoint+1); |
|
} |
|
return !stream.fail(); |
|
} |
|
}; |
|
typedef std::vector<Image> Images; |
|
// structure describing a 3D point |
|
struct Point { |
|
struct Track { |
|
uint32_t idImage; |
|
uint32_t idProj; |
|
}; |
|
uint32_t ID; // ID of the point |
|
Interface::Pos3f p; // position |
|
Interface::Col3 c; // BGR color |
|
float e; // error |
|
std::vector<Track> tracks; // point track |
|
uint64_t numPoints3D{0}; // only for binary format |
|
|
|
Point() {} |
|
Point(uint32_t _ID) : ID(_ID) {} |
|
bool operator < (const Image& rhs) const { return ID < rhs.ID; } |
|
|
|
bool Read(std::istream& stream, bool binary) { |
|
if (binary) |
|
return ReadBIN(stream); |
|
return ReadTXT(stream); |
|
} |
|
|
|
bool Write(std::ostream& stream, bool binary) { |
|
if (binary) |
|
return WriteBIN(stream); |
|
return WriteTXT(stream); |
|
} |
|
|
|
// 3D point list with one line of data per point: |
|
// POINT3D_ID, X, Y, Z, R, G, B, ERROR, TRACK[] as (IMAGE_ID, POINT2D_IDX) |
|
bool ReadTXT(std::istream& stream) { |
|
std::istringstream in; |
|
if (!NextLine(stream, in)) |
|
return false; |
|
int r,g,b; |
|
in >> ID |
|
>> p.x >> p.y >> p.z |
|
>> r >> g >> b |
|
>> e; |
|
c.x = CLAMP(b,0,255); |
|
c.y = CLAMP(g,0,255); |
|
c.z = CLAMP(r,0,255); |
|
if (in.fail()) |
|
return false; |
|
--ID; |
|
tracks.clear(); |
|
while (true) { |
|
Track track; |
|
in >> track.idImage >> track.idProj; |
|
if (in.fail()) |
|
break; |
|
--track.idImage; --track.idProj; |
|
tracks.emplace_back(track); |
|
} |
|
return !tracks.empty(); |
|
} |
|
|
|
// See: colmap/src/base/reconstruction.cc |
|
// void Reconstruction::ReadPoints3DBinary(const std::string& path) |
|
bool ReadBIN(std::istream& stream) { |
|
if (stream.peek() == EOF) |
|
return false; |
|
|
|
if (!numPoints3D) { |
|
// Read the first entry in the binary file |
|
numPoints3D = ReadBinaryLittleEndian<uint64_t>(&stream); |
|
} |
|
|
|
int r,g,b; |
|
ID = (uint32_t)ReadBinaryLittleEndian<point3D_t>(&stream)-1; |
|
p.x = (float)ReadBinaryLittleEndian<double>(&stream); |
|
p.y = (float)ReadBinaryLittleEndian<double>(&stream); |
|
p.z = (float)ReadBinaryLittleEndian<double>(&stream); |
|
r = ReadBinaryLittleEndian<uint8_t>(&stream); |
|
g = ReadBinaryLittleEndian<uint8_t>(&stream); |
|
b = ReadBinaryLittleEndian<uint8_t>(&stream); |
|
e = (float)ReadBinaryLittleEndian<double>(&stream); |
|
c.x = CLAMP(b,0,255); |
|
c.y = CLAMP(g,0,255); |
|
c.z = CLAMP(r,0,255); |
|
|
|
const size_t trackLength = ReadBinaryLittleEndian<uint64_t>(&stream); |
|
tracks.clear(); |
|
for (size_t j = 0; j < trackLength; ++j) { |
|
Track track; |
|
track.idImage = ReadBinaryLittleEndian<image_t>(&stream)-1; |
|
track.idProj = ReadBinaryLittleEndian<point2D_t>(&stream)-1; |
|
tracks.emplace_back(track); |
|
} |
|
return !tracks.empty(); |
|
} |
|
|
|
bool WriteTXT(std::ostream& out) const { |
|
ASSERT(!tracks.empty()); |
|
const int r(c.z),g(c.y),b(c.x); |
|
out << ID+1 << _T(" ") |
|
<< p.x << _T(" ") << p.y << _T(" ") << p.z << _T(" ") |
|
<< r << _T(" ") << g << _T(" ") << b << _T(" ") |
|
<< e << _T(" "); |
|
for (const Track& track: tracks) { |
|
out << track.idImage+1 << _T(" ") << track.idProj+1 << _T(" "); |
|
if (out.fail()) |
|
return false; |
|
} |
|
out << std::endl; |
|
return !out.fail(); |
|
} |
|
|
|
bool WriteBIN(std::ostream& stream) { |
|
ASSERT(!tracks.empty()); |
|
if (numPoints3D != 0) { |
|
// Write the first entry in the binary file |
|
WriteBinaryLittleEndian<uint64_t>(&stream, numPoints3D); |
|
numPoints3D = 0; |
|
} |
|
|
|
WriteBinaryLittleEndian<point3D_t>(&stream, ID+1); |
|
WriteBinaryLittleEndian<double>(&stream, p.x); |
|
WriteBinaryLittleEndian<double>(&stream, p.y); |
|
WriteBinaryLittleEndian<double>(&stream, p.z); |
|
WriteBinaryLittleEndian<uint8_t>(&stream, c.z); |
|
WriteBinaryLittleEndian<uint8_t>(&stream, c.y); |
|
WriteBinaryLittleEndian<uint8_t>(&stream, c.x); |
|
WriteBinaryLittleEndian<double>(&stream, e); |
|
|
|
WriteBinaryLittleEndian<uint64_t>(&stream, tracks.size()); |
|
for (const Track& track: tracks) { |
|
WriteBinaryLittleEndian<image_t>(&stream, track.idImage+1); |
|
WriteBinaryLittleEndian<point2D_t>(&stream, track.idProj+1); |
|
} |
|
return !stream.fail(); |
|
} |
|
}; |
|
typedef std::vector<Point> Points; |
|
// structure describing an 2D dynamic matrix |
|
template <typename T> |
|
struct Mat { |
|
size_t width_ = 0; |
|
size_t height_ = 0; |
|
size_t depth_ = 0; |
|
std::vector<T> data_; |
|
|
|
size_t GetNumBytes() const { |
|
return data_.size() * sizeof(T); |
|
} |
|
const T* GetChannelPtr(size_t c) const { |
|
return data_.data()+width_*height_*c; |
|
} |
|
|
|
// See: colmap/src/mvs/mat.h |
|
void Read(const std::string& path) { |
|
std::streampos pos; { |
|
std::fstream text_file(path, std::ios::in | std::ios::binary); |
|
char unused_char; |
|
text_file >> width_ >> unused_char >> height_ >> unused_char >> depth_ >> |
|
unused_char; |
|
pos = text_file.tellg(); |
|
} |
|
data_.resize(width_ * height_ * depth_); |
|
std::fstream binary_file(path, std::ios::in | std::ios::binary); |
|
binary_file.seekg(pos); |
|
ReadBinaryLittleEndian<T>(&binary_file, &data_); |
|
} |
|
void Write(const std::string& path) const { |
|
{ |
|
std::fstream text_file(path, std::ios::out); |
|
text_file << width_ << "&" << height_ << "&" << depth_ << "&"; |
|
} |
|
std::fstream binary_file(path, std::ios::out | std::ios::binary | std::ios::app); |
|
WriteBinaryLittleEndian<T>(&binary_file, data_); |
|
} |
|
}; |
|
} // namespace COLMAP |
|
|
|
typedef Eigen::Matrix<double,3,3,Eigen::RowMajor> EMat33d; |
|
typedef Eigen::Matrix<double,3,1> EVec3d; |
|
|
|
/** |
|
* Determines whether to use the TXT or BIN version of a file based on availability. |
|
* It tries to open the TXT file first; if unsuccessful, it attempts to open the BIN file. |
|
* |
|
* @param filenameTXT The path to the TXT file. |
|
* @param filenameBIN The path to the BIN file. |
|
* @param fileStream Reference to the ifstream object that will be used for file operations. |
|
* @param chosenFilename Reference to a string that will store the path of the chosen file. |
|
* @param isBinaryFormat Reference to a boolean that indicates whether the chosen file is binary. |
|
* @return true if either file was successfully opened, false otherwise. |
|
*/ |
|
bool DetermineInputSource(const String& filenameTXT, const String& filenameBIN, std::ifstream& fileStream, String& filenameCamera, bool& isBinaryFormat) |
|
{ |
|
// Try to open the TXT file first. |
|
fileStream.open(filenameTXT); |
|
if (fileStream.good()) { |
|
filenameCamera = filenameTXT; |
|
isBinaryFormat = false; |
|
return true; |
|
} |
|
fileStream.open(filenameBIN, std::ios::binary); |
|
if (fileStream.good()) { |
|
filenameCamera = filenameBIN; |
|
isBinaryFormat = true; |
|
return true; |
|
} |
|
VERBOSE("error: unable to open file '%s'", filenameTXT.c_str()); |
|
VERBOSE("error: unable to open file '%s'", filenameBIN.c_str()); |
|
return false; |
|
} |
|
|
|
|
|
bool ImportScene(const String& inputFolder, const String& outputFolder, Interface& scene, PointCloud& pointcloud) |
|
{ |
|
// Define a map to store camera IDs and associated platform index. |
|
typedef std::unordered_map<uint32_t,uint32_t> CamerasMap; |
|
CamerasMap mapCameras; |
|
{ |
|
// Determine the camera file source between TXT and BIN formats. |
|
const String filenameCamerasTXT(inputFolder+COLMAP_CAMERAS_TXT); |
|
const String filenameCamerasBIN(inputFolder+COLMAP_CAMERAS_BIN); |
|
std::ifstream cameraFile; |
|
bool isBinaryFormat; |
|
String filenameCamera; |
|
if (!DetermineInputSource(filenameCamerasTXT, filenameCamerasBIN, cameraFile, filenameCamera, isBinaryFormat)) { |
|
return false; // Exit if file source determination fails. |
|
} |
|
LOG_OUT() << "Reading cameras: " << filenameCamera << std::endl; |
|
|
|
typedef std::unordered_map<COLMAP::Camera,uint32_t,COLMAP::Camera::CameraHash,COLMAP::Camera::CameraEqualTo> CamerasSet; |
|
CamerasSet setCameras; |
|
COLMAP::Camera colmapCamera; |
|
while (cameraFile.good() && colmapCamera.Read(cameraFile, isBinaryFormat)) { |
|
const auto setIt(setCameras.emplace(colmapCamera, (uint32_t)scene.platforms.size())); |
|
mapCameras.emplace(colmapCamera.ID, setIt.first->second); |
|
if (!setIt.second) { |
|
// reuse existing platform |
|
continue; |
|
} |
|
// create new platform |
|
Interface::Platform platform; |
|
platform.name = String::FormatString(_T("platform%03u"), colmapCamera.ID); // only one camera per platform supported |
|
Interface::Platform::Camera camera; |
|
camera.name = colmapCamera.model; |
|
camera.K = Interface::Mat33d::eye(); |
|
// account for different pixel center conventions as COLMAP uses pixel center at (0.5,0.5) |
|
camera.K(0,0) = colmapCamera.params[0]; |
|
camera.K(1,1) = colmapCamera.params[1]; |
|
camera.K(0,2) = colmapCamera.params[2]-REAL(0.5); |
|
camera.K(1,2) = colmapCamera.params[3]-REAL(0.5); |
|
camera.R = Interface::Mat33d::eye(); |
|
camera.C = Interface::Pos3d(0,0,0); |
|
if (OPT::bNormalizeIntrinsics) { |
|
// normalize camera intrinsics |
|
camera.K = Camera::ScaleK<double>(camera.K, 1.0/Camera::GetNormalizationScale(colmapCamera.width, colmapCamera.height)); |
|
} else { |
|
camera.width = colmapCamera.width; |
|
camera.height = colmapCamera.height; |
|
} |
|
platform.cameras.emplace_back(camera); |
|
scene.platforms.emplace_back(platform); |
|
} |
|
} |
|
if (mapCameras.empty()) { |
|
VERBOSE("error: no valid cameras (make sure they are in PINHOLE model)"); |
|
return false; |
|
} |
|
|
|
// Function to read and process image data from input files |
|
typedef std::map<COLMAP::Image, uint32_t> ImagesMap; |
|
ImagesMap mapImages; |
|
{ |
|
// Define a map to associate COLMAP image structures with scene image indices |
|
const String filenameImagesTXT(inputFolder+COLMAP_IMAGES_TXT); |
|
const String filenameImagesBIN(inputFolder+COLMAP_IMAGES_BIN); |
|
std::ifstream file; |
|
bool binary; |
|
String filenameImages; |
|
if (!DetermineInputSource(filenameImagesTXT, filenameImagesBIN, file, filenameImages, binary)) { |
|
return false; |
|
} |
|
LOG_OUT() << "Reading images: " << filenameImages << std::endl; |
|
|
|
// Read image data from the file and store in the scene structure |
|
COLMAP::Image imageColmap; |
|
while (file.good() && imageColmap.Read(file, binary)) { |
|
mapImages.emplace(imageColmap, (uint32_t)scene.images.size()); |
|
Interface::Platform::Pose pose; |
|
Eigen::Map<EMat33d>(pose.R.val) = imageColmap.q.toRotationMatrix(); |
|
EnsureRotationMatrix((Matrix3x3d&)pose.R); |
|
Eigen::Map<EVec3d>(&pose.C.x) = -(imageColmap.q.inverse() * imageColmap.t); |
|
|
|
Interface::Image image; |
|
image.name = MAKE_PATH_REL(outputFolder,OPT::strImageFolder+imageColmap.name); |
|
image.platformID = mapCameras.at(imageColmap.idCamera); |
|
image.cameraID = 0; |
|
image.ID = imageColmap.ID; |
|
|
|
Interface::Platform& platform = scene.platforms[image.platformID]; |
|
image.poseID = (uint32_t)platform.poses.size(); |
|
platform.poses.emplace_back(pose); |
|
scene.images.emplace_back(image); |
|
} |
|
} |
|
|
|
// read points list |
|
const String filenameDensePoints(inputFolder+COLMAP_DENSE_POINTS); |
|
const String filenameDenseVisPoints(inputFolder+COLMAP_DENSE_POINTS_VISIBILITY); |
|
{ |
|
// parse sparse point-cloud |
|
const String filenamePointsTXT(inputFolder+COLMAP_POINTS_TXT); |
|
const String filenamePointsBIN(inputFolder+COLMAP_POINTS_BIN); |
|
std::ifstream file; |
|
bool binary; |
|
String filenamePoints; |
|
if (!DetermineInputSource(filenamePointsTXT, filenamePointsBIN, file, filenamePoints, binary)) { |
|
return false; |
|
} |
|
LOG_OUT() << "Reading points: " << filenamePoints << std::endl; |
|
COLMAP::Point point; |
|
while (file.good() && point.Read(file, binary)) { |
|
Interface::Vertex vertex; |
|
vertex.X = point.p; |
|
for (const COLMAP::Point::Track& track: point.tracks) { |
|
Interface::Vertex::View view; |
|
view.imageID = mapImages.at(COLMAP::Image(track.idImage)); |
|
view.confidence = 0; |
|
vertex.views.emplace_back(view); |
|
} |
|
std::sort(vertex.views.begin(), vertex.views.end(), |
|
[](const Interface::Vertex::View& view0, const Interface::Vertex::View& view1) { return view0.imageID < view1.imageID; }); |
|
scene.vertices.emplace_back(std::move(vertex)); |
|
scene.verticesColor.emplace_back(Interface::Color{point.c}); |
|
} |
|
} |
|
pointcloud.Release(); |
|
if (File::access(filenameDensePoints) && File::access(filenameDenseVisPoints)) { |
|
// parse dense point-cloud |
|
LOG_OUT() << "Reading points: " << filenameDensePoints << " and " << filenameDenseVisPoints << std::endl; |
|
if (!pointcloud.Load(filenameDensePoints)) { |
|
VERBOSE("error: unable to open file '%s'", filenameDensePoints.c_str()); |
|
return false; |
|
} |
|
File file(filenameDenseVisPoints, File::READ, File::OPEN); |
|
if (!file.isOpen()) { |
|
VERBOSE("error: unable to open file '%s'", filenameDenseVisPoints.c_str()); |
|
return false; |
|
} |
|
uint64_t numPoints(0); |
|
file.read(&numPoints, sizeof(uint64_t)); |
|
if (pointcloud.GetSize() != numPoints) { |
|
VERBOSE("error: point-cloud and visibility have different size"); |
|
return false; |
|
} |
|
pointcloud.pointViews.resize(numPoints); |
|
for (size_t i=0; i<numPoints; ++i) { |
|
PointCloud::ViewArr& views = pointcloud.pointViews[i]; |
|
uint32_t numViews(0); |
|
file.read(&numViews, sizeof(uint32_t)); |
|
for (uint32_t v=0; v<numViews; ++v) { |
|
uint32_t imageID; |
|
file.read(&imageID, sizeof(uint32_t)); |
|
views.emplace_back(imageID); |
|
} |
|
views.Sort(); |
|
} |
|
} |
|
|
|
// read depth-maps |
|
const String pathDepthMaps(inputFolder+COLMAP_STEREO_DEPTHMAPS_FOLDER); |
|
const String pathNormalMaps(inputFolder+COLMAP_STEREO_NORMALMAPS_FOLDER); |
|
if (File::isFolder(pathDepthMaps) && File::isFolder(pathNormalMaps)) { |
|
// read patch-match list |
|
CLISTDEF2IDX(IIndexArr,IIndex) imagesNeighbors((IIndex)scene.images.size()); |
|
{ |
|
const String filenameFusion(inputFolder+COLMAP_PATCHMATCH); |
|
LOG_OUT() << "Reading patch-match configuration: " << filenameFusion << std::endl; |
|
std::ifstream file(filenameFusion); |
|
if (!file.good()) { |
|
VERBOSE("error: unable to open file '%s'", filenameFusion.c_str()); |
|
return false; |
|
} |
|
while (true) { |
|
String imageName, neighbors; |
|
std::getline(file, imageName); |
|
std::getline(file, neighbors); |
|
if (file.fail() || imageName.empty() || neighbors.empty()) |
|
break; |
|
const ImagesMap::const_iterator it_image = std::find_if(mapImages.begin(), mapImages.end(), |
|
[&imageName](const ImagesMap::value_type& image) { |
|
return image.first.name == imageName; |
|
}); |
|
if (it_image == mapImages.end()) |
|
continue; |
|
IIndexArr& imageNeighbors = imagesNeighbors[it_image->second]; |
|
CLISTDEF2(String) neighborNames; |
|
Util::strSplit(neighbors, _T(','), neighborNames); |
|
FOREACH(i, neighborNames) { |
|
String& neighborName = neighborNames[i]; |
|
Util::strTrim(neighborName, _T(" ")); |
|
const ImagesMap::const_iterator it_neighbor = std::find_if(mapImages.begin(), mapImages.end(), |
|
[&neighborName](const ImagesMap::value_type& image) { |
|
return image.first.name == neighborName; |
|
}); |
|
if (it_neighbor == mapImages.end()) { |
|
if (i == 0) |
|
break; |
|
continue; |
|
} |
|
imageNeighbors.emplace_back(scene.images[it_neighbor->second].ID); |
|
} |
|
} |
|
} |
|
LOG_OUT() << "Reading depth-maps/normal-maps: " << pathDepthMaps << " and " << pathNormalMaps << std::endl; |
|
Util::ensureFolder(outputFolder); |
|
const String strType[] = {".geometric.bin", ".photometric.bin"}; |
|
FOREACH(idx, scene.images) { |
|
const Interface::Image& image = scene.images[idx]; |
|
COLMAP::Mat<float> colDepthMap, colNormalMap; |
|
const String filenameImage(Util::getFileNameExt(image.name)); |
|
for (int i=0; i<2; ++i) { |
|
const String filenameDepthMaps(pathDepthMaps+filenameImage+strType[i]); |
|
if (File::isFile(filenameDepthMaps)) { |
|
colDepthMap.Read(filenameDepthMaps); |
|
const String filenameNormalMaps(pathNormalMaps+filenameImage+strType[i]); |
|
if (File::isFile(filenameNormalMaps)) { |
|
colNormalMap.Read(filenameNormalMaps); |
|
} |
|
break; |
|
} |
|
} |
|
if (!colDepthMap.data_.empty()) { |
|
IIndexArr IDs = {image.ID}; |
|
IDs.Join(imagesNeighbors[(IIndex)idx]); |
|
const Interface::Platform& platform = scene.platforms[image.platformID]; |
|
const Interface::Platform::Pose pose(platform.GetPose(image.cameraID, image.poseID)); |
|
const Interface::Mat33d K(platform.GetFullK(image.cameraID, (uint32_t)colDepthMap.width_, (uint32_t)colDepthMap.height_)); |
|
MVS::DepthMap depthMap((int)colDepthMap.height_, (int)colDepthMap.width_); |
|
memcpy(depthMap.getData(), colDepthMap.data_.data(), colDepthMap.GetNumBytes()); |
|
MVS::NormalMap normalMap; |
|
if (!colNormalMap.data_.empty()) { |
|
normalMap.create((int)colNormalMap.height_, (int)colNormalMap.width_); |
|
cv::merge(std::vector<cv::Mat>{ |
|
cv::Mat((int)colNormalMap.height_, (int)colNormalMap.width_, CV_32F, (void*)colNormalMap.GetChannelPtr(0)), |
|
cv::Mat((int)colNormalMap.height_, (int)colNormalMap.width_, CV_32F, (void*)colNormalMap.GetChannelPtr(1)), |
|
cv::Mat((int)colNormalMap.height_, (int)colNormalMap.width_, CV_32F, (void*)colNormalMap.GetChannelPtr(2)) |
|
}, normalMap); |
|
} |
|
MVS::ConfidenceMap confMap; |
|
MVS::ViewsMap viewsMap; |
|
const auto depthMM(std::minmax_element(colDepthMap.data_.cbegin(), colDepthMap.data_.cend())); |
|
const MVS::Depth dMin(*depthMM.first), dMax(*depthMM.second); |
|
if (!ExportDepthDataRaw(outputFolder+String::FormatString("depth%04u.dmap", image.ID), MAKE_PATH_FULL(outputFolder, image.name), IDs, depthMap.size(), K, pose.R, pose.C, dMin, dMax, depthMap, normalMap, confMap, viewsMap)) |
|
return false; |
|
} |
|
} |
|
} |
|
return true; |
|
} |
|
|
|
|
|
bool ImportPointCloud(const String& strPointCloudFileName, Interface& scene) |
|
{ |
|
PointCloud pointcloud; |
|
if (!pointcloud.Load(strPointCloudFileName)) { |
|
VERBOSE("error: cannot load point-cloud file"); |
|
return false; |
|
} |
|
if (!pointcloud.IsValid()) { |
|
VERBOSE("error: loaded point-cloud does not have visibility information"); |
|
return false; |
|
} |
|
// replace scene point-cloud with the loaded one |
|
scene.vertices.clear(); |
|
scene.verticesColor.clear(); |
|
scene.verticesNormal.clear(); |
|
scene.vertices.reserve(pointcloud.points.size()); |
|
if (!pointcloud.colors.empty()) |
|
scene.verticesColor.reserve(pointcloud.points.size()); |
|
if (!pointcloud.normals.empty()) |
|
scene.verticesNormal.reserve(pointcloud.points.size()); |
|
FOREACH(i, pointcloud.points) { |
|
Interface::Vertex vertex; |
|
vertex.X = pointcloud.points[i]; |
|
vertex.views.reserve(pointcloud.pointViews[i].size()); |
|
FOREACH(j, pointcloud.pointViews[i]) { |
|
Interface::Vertex::View& view = vertex.views.emplace_back(); |
|
view.imageID = pointcloud.pointViews[i][j]; |
|
view.confidence = (pointcloud.pointWeights.empty() ? 0.f : pointcloud.pointWeights[i][j]); |
|
} |
|
scene.vertices.emplace_back(std::move(vertex)); |
|
if (!pointcloud.colors.empty()) { |
|
const Pixel8U& c = pointcloud.colors[i]; |
|
scene.verticesColor.emplace_back(Interface::Color{Interface::Col3{c.b, c.g, c.r}}); |
|
} |
|
if (!pointcloud.normals.empty()) |
|
scene.verticesNormal.emplace_back(Interface::Normal{pointcloud.normals[i]}); |
|
} |
|
return true; |
|
} |
|
|
|
bool ExportScene(const String& strFolder, const Interface& scene, bool bForceSparsePointCloud = false, bool binary = true) |
|
{ |
|
Util::ensureFolder(strFolder+COLMAP_SPARSE_FOLDER); |
|
|
|
// write camera list |
|
CLISTDEF0IDX(KMatrix,uint32_t) Ks; |
|
CLISTDEF0IDX(COLMAP::Camera,uint32_t) cams; |
|
{ |
|
// Construct the filename for camera data based on the format preference |
|
const String filenameCameras(strFolder+(binary?COLMAP_CAMERAS_BIN:COLMAP_CAMERAS_TXT)); |
|
LOG_OUT() << "Writing cameras: " << filenameCameras << std::endl; |
|
std::ofstream file(filenameCameras, binary ? std::ios::trunc|std::ios::binary : std::ios::trunc); |
|
|
|
if (!file.good()) { |
|
VERBOSE("error: unable to open file '%s'", filenameCameras.c_str()); |
|
return false; |
|
} |
|
|
|
COLMAP::Camera cam; |
|
if (binary) { |
|
cam.numCameras = 0; |
|
for (const Interface::Platform& platform: scene.platforms) |
|
cam.numCameras += (uint32_t)platform.cameras.size(); |
|
} else { |
|
file << _T("# Camera list with one line of data per camera:") << std::endl; |
|
file << _T("# CAMERA_ID, MODEL, WIDTH, HEIGHT, PARAMS[]") << std::endl; |
|
} |
|
cam.model = _T("PINHOLE"); |
|
cam.params.resize(4); |
|
for (uint32_t ID=0; ID<(uint32_t)scene.platforms.size(); ++ID) { |
|
const Interface::Platform& platform = scene.platforms[ID]; |
|
ASSERT(platform.cameras.size() == 1); // only one camera per platform supported |
|
const Interface::Platform::Camera& camera = platform.cameras[0]; |
|
cam.ID = ID; |
|
KMatrix K; |
|
if (camera.width == 0 || camera.height == 0) { |
|
// find one image using this camera |
|
const Interface::Image* pImage(NULL); |
|
for (uint32_t i=0; i<(uint32_t)scene.images.size(); ++i) { |
|
const Interface::Image& image = scene.images[i]; |
|
if (image.platformID == ID && image.cameraID == 0 && image.poseID != NO_ID) { |
|
pImage = ℑ |
|
break; |
|
} |
|
} |
|
if (pImage == NULL) { |
|
LOG("error: no image using camera %u of platform %u", 0, ID); |
|
continue; |
|
} |
|
IMAGEPTR ptrImage(Image::ReadImageHeader(MAKE_PATH_SAFE(pImage->name))); |
|
if (ptrImage == NULL) |
|
return false; |
|
cam.width = ptrImage->GetWidth(); |
|
cam.height = ptrImage->GetHeight(); |
|
// unnormalize camera intrinsics |
|
K = platform.GetFullK(0, cam.width, cam.height); |
|
} else { |
|
cam.width = camera.width; |
|
cam.height = camera.height; |
|
K = camera.K; |
|
} |
|
// account for different pixel center conventions as COLMAP uses pixel center at (0.5,0.5) |
|
cam.params[0] = K(0,0); |
|
cam.params[1] = K(1,1); |
|
cam.params[2] = K(0,2)+REAL(0.5); |
|
cam.params[3] = K(1,2)+REAL(0.5); |
|
if (!cam.Write(file, binary)) |
|
return false; |
|
Ks.emplace_back(K); |
|
cams.emplace_back(cam); |
|
} |
|
} |
|
|
|
// create images list |
|
COLMAP::Images images; |
|
CameraArr cameras; |
|
float maxNumPointsSparse(0); |
|
const float avgViewsPerPoint(3.f); |
|
const uint32_t avgResolutionSmallView(640*480), avgResolutionLargeView(6000*4000); |
|
const uint32_t avgPointsPerSmallView(3000), avgPointsPerLargeView(12000); |
|
{ |
|
images.resize(scene.images.size()); |
|
cameras.resize((unsigned)scene.images.size()); |
|
for (uint32_t ID=0; ID<(uint32_t)scene.images.size(); ++ID) { |
|
const Interface::Image& image = scene.images[ID]; |
|
if (image.poseID == NO_ID) |
|
continue; |
|
const Interface::Platform& platform = scene.platforms[image.platformID]; |
|
const Interface::Platform::Pose& pose = platform.poses[image.poseID]; |
|
ASSERT(image.cameraID == 0); |
|
COLMAP::Image& img = images[ID]; |
|
img.ID = ID; |
|
img.q = Eigen::Quaterniond(Eigen::Map<const EMat33d>(pose.R.val)); |
|
img.t = -(img.q * Eigen::Map<const EVec3d>(&pose.C.x)); |
|
img.idCamera = image.platformID; |
|
img.name = MAKE_PATH_REL(OPT::strImageFolder, MAKE_PATH_FULL(WORKING_FOLDER_FULL, image.name)); |
|
Camera& camera = cameras[ID]; |
|
camera.K = Ks[image.platformID]; |
|
camera.R = pose.R; |
|
camera.C = pose.C; |
|
camera.ComposeP(); |
|
const COLMAP::Camera& cam = cams[image.platformID]; |
|
const uint32_t resolutionView(cam.width*cam.height); |
|
const float linearFactor(float(avgResolutionLargeView-resolutionView)/(avgResolutionLargeView-avgResolutionSmallView)); |
|
maxNumPointsSparse += (avgPointsPerSmallView+(avgPointsPerLargeView-avgPointsPerSmallView)*linearFactor)/avgViewsPerPoint; |
|
} |
|
} |
|
|
|
// auto-select dense or sparse mode based on number of points |
|
const bool bSparsePointCloud(scene.vertices.size() < (size_t)maxNumPointsSparse); |
|
if (bSparsePointCloud || bForceSparsePointCloud) { |
|
// write points list |
|
{ |
|
const String filenamePoints(strFolder+(binary?COLMAP_POINTS_BIN:COLMAP_POINTS_TXT)); |
|
LOG_OUT() << "Writing points: " << filenamePoints << std::endl; |
|
std::ofstream file(filenamePoints, binary ? std::ios::trunc|std::ios::binary : std::ios::trunc); |
|
if (!file.good()) { |
|
VERBOSE("error: unable to open file '%s'", filenamePoints.c_str()); |
|
return false; |
|
} |
|
uint64_t numPoints3D = 0; |
|
if (binary) { |
|
numPoints3D = scene.vertices.size(); |
|
} else { |
|
file << _T("# 3D point list with one line of data per point:") << std::endl; |
|
file << _T("# POINT3D_ID, X, Y, Z, R, G, B, ERROR, TRACK[] as (IMAGE_ID, POINT2D_IDX)") << std::endl; |
|
} |
|
for (uint32_t ID=0; ID<(uint32_t)scene.vertices.size(); ++ID) { |
|
const Interface::Vertex& vertex = scene.vertices[ID]; |
|
COLMAP::Point point; |
|
point.ID = ID; |
|
point.p = vertex.X; |
|
for (const Interface::Vertex::View& view: vertex.views) { |
|
COLMAP::Image& img = images[view.imageID]; |
|
point.tracks.emplace_back(COLMAP::Point::Track{view.imageID, (uint32_t)img.projs.size()}); |
|
COLMAP::Image::Proj proj; |
|
proj.idPoint = ID; |
|
const Point3 X(vertex.X); |
|
ProjectVertex_3x4_3_2(cameras[view.imageID].P.val, X.ptr(), proj.p.data()); |
|
// account for different pixel center conventions as COLMAP uses pixel center at (0.5,0.5) |
|
proj.p[0] += REAL(0.5); |
|
proj.p[1] += REAL(0.5); |
|
img.projs.emplace_back(proj); |
|
} |
|
point.c = scene.verticesColor.empty() ? Interface::Col3(255,255,255) : scene.verticesColor[ID].c; |
|
point.e = 0; |
|
if (numPoints3D != 0) { |
|
point.numPoints3D = numPoints3D; |
|
numPoints3D = 0; |
|
} |
|
if (!point.Write(file, binary)) |
|
return false; |
|
} |
|
} |
|
|
|
Util::ensureFolder(strFolder+COLMAP_STEREO_FOLDER); |
|
|
|
// write fusion list |
|
{ |
|
const String filenameFusion(strFolder+COLMAP_FUSION); |
|
LOG_OUT() << "Writing fusion configuration: " << filenameFusion << std::endl; |
|
std::ofstream file(filenameFusion); |
|
if (!file.good()) { |
|
VERBOSE("error: unable to open file '%s'", filenameFusion.c_str()); |
|
return false; |
|
} |
|
for (const COLMAP::Image& img: images) { |
|
if (img.projs.empty()) |
|
continue; |
|
file << img.name << std::endl; |
|
if (file.fail()) |
|
return false; |
|
} |
|
} |
|
|
|
// write patch-match list |
|
{ |
|
const String filenameFusion(strFolder+COLMAP_PATCHMATCH); |
|
LOG_OUT() << "Writing patch-match configuration: " << filenameFusion << std::endl; |
|
std::ofstream file(filenameFusion); |
|
if (!file.good()) { |
|
VERBOSE("error: unable to open file '%s'", filenameFusion.c_str()); |
|
return false; |
|
} |
|
for (const COLMAP::Image& img: images) { |
|
if (img.projs.empty()) |
|
continue; |
|
file << img.name << std::endl; |
|
if (file.fail()) |
|
return false; |
|
file << _T("__auto__, 20") << std::endl; |
|
if (file.fail()) |
|
return false; |
|
} |
|
} |
|
|
|
Util::ensureFolder(strFolder+COLMAP_STEREO_CONSISTENCYGRAPHS_FOLDER); |
|
Util::ensureFolder(strFolder+COLMAP_STEREO_DEPTHMAPS_FOLDER); |
|
Util::ensureFolder(strFolder+COLMAP_STEREO_NORMALMAPS_FOLDER); |
|
} |
|
if (!bSparsePointCloud) { |
|
// export dense point-cloud |
|
const String filenameDensePoints(strFolder+COLMAP_DENSE_POINTS); |
|
const String filenameDenseVisPoints(strFolder+COLMAP_DENSE_POINTS_VISIBILITY); |
|
LOG_OUT() << "Writing points: " << filenameDensePoints << " and " << filenameDenseVisPoints << std::endl; |
|
File file(filenameDenseVisPoints, File::WRITE, File::CREATE | File::TRUNCATE); |
|
if (!file.isOpen()) { |
|
VERBOSE("error: unable to write file '%s'", filenameDenseVisPoints.c_str()); |
|
return false; |
|
} |
|
const uint64_t numPoints(scene.vertices.size()); |
|
file.write(&numPoints, sizeof(uint64_t)); |
|
PointCloud pointcloud; |
|
for (size_t i=0; i<numPoints; ++i) { |
|
const Interface::Vertex& vertex = scene.vertices[i]; |
|
pointcloud.points.emplace_back(vertex.X); |
|
if (!scene.verticesNormal.empty()) |
|
pointcloud.normals.emplace_back(scene.verticesNormal[i].n); |
|
if (!scene.verticesColor.empty()) |
|
pointcloud.colors.emplace_back(scene.verticesColor[i].c); |
|
const uint32_t numViews((uint32_t)vertex.views.size()); |
|
file.write(&numViews, sizeof(uint32_t)); |
|
for (uint32_t v=0; v<numViews; ++v) { |
|
const Interface::Vertex::View& view = vertex.views[v]; |
|
file.write(&view.imageID, sizeof(uint32_t)); |
|
} |
|
} |
|
if (!pointcloud.Save(filenameDensePoints, false, true)) { |
|
VERBOSE("error: unable to write file '%s'", filenameDensePoints.c_str()); |
|
return false; |
|
} |
|
} |
|
|
|
// write images list |
|
{ |
|
const String filenameImages(strFolder+(binary?COLMAP_IMAGES_BIN:COLMAP_IMAGES_TXT)); |
|
LOG_OUT() << "Writing images: " << filenameImages << std::endl; |
|
std::ofstream file(filenameImages, binary ? std::ios::trunc|std::ios::binary : std::ios::trunc); |
|
if (!file.good()) { |
|
VERBOSE("error: unable to open file '%s'", filenameImages.c_str()); |
|
return false; |
|
} |
|
uint64_t numRegImages = 0; |
|
if (binary) { |
|
for (const COLMAP::Image& img: images) { |
|
if (bSparsePointCloud && img.projs.empty()) |
|
continue; |
|
++numRegImages; |
|
} |
|
} else { |
|
file << _T("# Image list with two lines of data per image:") << std::endl; |
|
file << _T("# IMAGE_ID, QW, QX, QY, QZ, TX, TY, TZ, CAMERA_ID, NAME") << std::endl; |
|
file << _T("# POINTS2D[] as (X, Y, POINT3D_ID)") << std::endl; |
|
} |
|
for (COLMAP::Image& img: images) { |
|
if (bSparsePointCloud && img.projs.empty()) |
|
continue; |
|
if (numRegImages != 0) { |
|
img.numRegImages = numRegImages; |
|
numRegImages = 0; |
|
} |
|
if (!img.Write(file, binary)) |
|
return false; |
|
} |
|
} |
|
return true; |
|
} |
|
|
|
|
|
// export camera intrinsics in txt format |
|
bool ExportIntrinsicsTxt(const String& fileName, const Interface& scene) |
|
{ |
|
LOG_OUT() << "Writing intrinsics: " << fileName << std::endl; |
|
uint32_t idxValidK(NO_ID); |
|
for (uint32_t ID=0; ID<(uint32_t)scene.images.size(); ++ID) { |
|
const Interface::Image& image = scene.images[ID]; |
|
if (!image.IsValid()) |
|
continue; |
|
if (idxValidK == NO_ID) { |
|
idxValidK = ID; |
|
continue; |
|
} |
|
if (scene.GetK(idxValidK) != scene.GetK(ID)) { |
|
VERBOSE("error: multiple camera models"); |
|
return false; |
|
} |
|
} |
|
if (idxValidK == NO_ID) |
|
return false; |
|
const Interface::Image& image = scene.images[idxValidK]; |
|
String imagefileName(image.name); |
|
Util::ensureValidPath(imagefileName); |
|
imagefileName = MAKE_PATH_FULL(WORKING_FOLDER_FULL, imagefileName); |
|
IMAGEPTR pImage = Image::ReadImageHeader(imagefileName); |
|
if (!pImage) { |
|
VERBOSE("error: unable to open image file '%s'", imagefileName.c_str()); |
|
return false; |
|
} |
|
const Interface::Mat33d& K = scene.platforms[image.platformID].GetFullK(image.cameraID, pImage->GetWidth(), pImage->GetHeight()); |
|
Eigen::Matrix4d K4(Eigen::Matrix4d::Identity()); |
|
K4.topLeftCorner<3,3>() = Eigen::Map<const EMat33d>(K.val); |
|
Util::ensureFolder(fileName); |
|
std::ofstream out(fileName); |
|
if (!out.good()) { |
|
VERBOSE("error: unable to open file '%s'", fileName.c_str()); |
|
return false; |
|
} |
|
out << std::setprecision(12); |
|
out << K4(0,0) << _T(" ") << K4(0,1) << _T(" ") << K4(0,2) << _T(" ") << K4(0,3) << _T("\n"); |
|
out << K4(1,0) << _T(" ") << K4(1,1) << _T(" ") << K4(1,2) << _T(" ") << K4(1,3) << _T("\n"); |
|
out << K4(2,0) << _T(" ") << K4(2,1) << _T(" ") << K4(2,2) << _T(" ") << K4(2,3) << _T("\n"); |
|
out << K4(3,0) << _T(" ") << K4(3,1) << _T(" ") << K4(3,2) << _T(" ") << K4(3,3) << _T("\n"); |
|
return !out.fail(); |
|
} |
|
|
|
|
|
// export image poses in log format |
|
// see: http://redwood-data.org/indoor/fileformat.html |
|
// to support: https://www.tanksandtemples.org/tutorial |
|
bool ExportImagesLog(const String& fileName, const Interface& scene) |
|
{ |
|
LOG_OUT() << "Writing poses: " << fileName << std::endl; |
|
Util::ensureFolder(fileName); |
|
std::ofstream out(fileName); |
|
if (!out.good()) { |
|
VERBOSE("error: unable to open file '%s'", fileName.c_str()); |
|
return false; |
|
} |
|
out << std::setprecision(12); |
|
IIndexArr orderedImages((uint32_t)scene.images.size()); |
|
std::iota(orderedImages.begin(), orderedImages.end(), 0u); |
|
orderedImages.Sort([&scene](IIndex i, IIndex j) { |
|
return scene.images[i].ID < scene.images[j].ID; |
|
}); |
|
for (IIndex ID: orderedImages) { |
|
const Interface::Image& image = scene.images[ID]; |
|
Eigen::Matrix3d R(Eigen::Matrix3d::Identity()); |
|
Eigen::Vector3d t(Eigen::Vector3d::Zero()); |
|
if (image.poseID != NO_ID) { |
|
const Interface::Platform& platform = scene.platforms[image.platformID]; |
|
const Interface::Platform::Pose& pose = platform.poses[image.poseID]; |
|
R = Eigen::Map<const EMat33d>(pose.R.val).transpose(); |
|
t = Eigen::Map<const EVec3d>(&pose.C.x); |
|
} |
|
Eigen::Matrix4d T(Eigen::Matrix4d::Identity()); |
|
T.topLeftCorner<3,3>() = R; |
|
T.topRightCorner<3,1>() = t; |
|
out << ID << _T(" ") << ID << _T(" ") << 0 << _T("\n"); |
|
out << T(0,0) << _T(" ") << T(0,1) << _T(" ") << T(0,2) << _T(" ") << T(0,3) << _T("\n"); |
|
out << T(1,0) << _T(" ") << T(1,1) << _T(" ") << T(1,2) << _T(" ") << T(1,3) << _T("\n"); |
|
out << T(2,0) << _T(" ") << T(2,1) << _T(" ") << T(2,2) << _T(" ") << T(2,3) << _T("\n"); |
|
out << T(3,0) << _T(" ") << T(3,1) << _T(" ") << T(3,2) << _T(" ") << T(3,3) << _T("\n"); |
|
} |
|
return !out.fail(); |
|
} |
|
|
|
|
|
// export poses in Strecha camera format: |
|
// Strecha model is P = K[R^T|-R^T t] |
|
// our model is P = K[R|t], t = -RC |
|
bool ExportImagesCamera(const String& pathName, const Interface& scene) |
|
{ |
|
LOG_OUT() << "Writing poses: " << pathName << std::endl; |
|
Util::ensureFolder(pathName); |
|
for (uint32_t ID=0; ID<(uint32_t)scene.images.size(); ++ID) { |
|
const Interface::Image& image = scene.images[ID]; |
|
String imageFileName(image.name); |
|
Util::ensureValidPath(imageFileName); |
|
const String fileName(pathName+Util::getFileNameExt(imageFileName)+".camera"); |
|
std::ofstream out(fileName); |
|
if (!out.good()) { |
|
VERBOSE("error: unable to open file '%s'", fileName.c_str()); |
|
return false; |
|
} |
|
out << std::setprecision(12); |
|
KMatrix K(KMatrix::IDENTITY); |
|
RMatrix R(RMatrix::IDENTITY); |
|
CMatrix t(CMatrix::ZERO); |
|
unsigned width(0), height(0); |
|
if (image.platformID != NO_ID && image.cameraID != NO_ID) { |
|
const Interface::Platform& platform = scene.platforms[image.platformID]; |
|
const Interface::Platform::Camera& camera = platform.cameras[image.cameraID]; |
|
if (camera.HasResolution()) { |
|
width = camera.width; |
|
height = camera.height; |
|
K = camera.K; |
|
} else { |
|
IMAGEPTR pImage = Image::ReadImageHeader(image.name); |
|
width = pImage->GetWidth(); |
|
height = pImage->GetHeight(); |
|
K = platform.GetFullK(image.cameraID, width, height); |
|
} |
|
if (image.poseID != NO_ID) { |
|
const Interface::Platform::Pose& pose = platform.poses[image.poseID]; |
|
R = pose.R.t(); |
|
t = pose.C; |
|
} |
|
} |
|
out << K(0,0) << _T(" ") << K(0,1) << _T(" ") << K(0,2) << _T("\n"); |
|
out << K(1,0) << _T(" ") << K(1,1) << _T(" ") << K(1,2) << _T("\n"); |
|
out << K(2,0) << _T(" ") << K(2,1) << _T(" ") << K(2,2) << _T("\n"); |
|
out << _T("0 0 0") << _T("\n"); |
|
out << R(0,0) << _T(" ") << R(0,1) << _T(" ") << R(0,2) << _T("\n"); |
|
out << R(1,0) << _T(" ") << R(1,1) << _T(" ") << R(1,2) << _T("\n"); |
|
out << R(2,0) << _T(" ") << R(2,1) << _T(" ") << R(2,2) << _T("\n"); |
|
out << t.x << _T(" ") << t.y << _T(" ") << t.z << _T("\n"); |
|
out << width << _T(" ") << height << _T("\n"); |
|
if (out.fail()) { |
|
VERBOSE("error: unable to write file '%s'", fileName.c_str()); |
|
return false; |
|
} |
|
} |
|
return true; |
|
} |
|
|
|
} // unnamed namespace |
|
|
|
// Main function with command-line arguments |
|
int main(int argc, LPCTSTR* argv) |
|
{ |
|
#ifdef _DEBUGINFO |
|
// set _crtBreakAlloc index to stop in <dbgheap.c> at allocation |
|
_CrtSetDbgFlag(_CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF);// | _CRTDBG_CHECK_ALWAYS_DF); |
|
#endif |
|
|
|
Application app; // Instance of the application |
|
if (!app.Initialize(argc, argv)) // Initialize app with command line arguments |
|
return EXIT_FAILURE; // Exit if initialization fails |
|
|
|
TD_TIMER_START(); |
|
|
|
if (OPT::IsFromOpenMVS) { |
|
// read MVS input data |
|
Interface scene; // Scene data structure for MVS input |
|
if (!ARCHIVE::SerializeLoad(scene, MAKE_PATH_SAFE(OPT::strInputFileName))) |
|
return EXIT_FAILURE; // Exit if loading fails |
|
|
|
// Check output file type and export accordingly |
|
if (Util::getFileExt(OPT::strOutputFileName) == _T(".log")) { |
|
// write poses in log format |
|
ExportIntrinsicsTxt(MAKE_PATH_FULL(WORKING_FOLDER_FULL, String("intrinsics.txt")), scene); |
|
ExportImagesLog(MAKE_PATH_SAFE(OPT::strOutputFileName), scene); |
|
} else |
|
if (Util::getFileExt(OPT::strOutputFileName) == _T(".camera")) { |
|
// write poses in Strecha camera format |
|
ExportImagesCamera((OPT::strOutputFileName=Util::getFileFullName(MAKE_PATH_FULL(WORKING_FOLDER_FULL, OPT::strOutputFileName)))+PATH_SEPARATOR, scene); |
|
} else { |
|
// write COLMAP input data |
|
if (!OPT::strPointCloudFileName.empty() && !ImportPointCloud(MAKE_PATH_SAFE(OPT::strPointCloudFileName), scene)) |
|
return EXIT_FAILURE; |
|
Util::ensureFolderSlash(OPT::strOutputFileName); |
|
ExportScene(MAKE_PATH_SAFE(OPT::strOutputFileName), scene, OPT::bForceSparsePointCloud); |
|
} |
|
VERBOSE("Input data exported: %u images & %u vertices (%s)", scene.images.size(), scene.vertices.size(), TD_TIMER_GET_FMT().c_str()); |
|
} else { |
|
// read COLMAP input data |
|
Interface scene; |
|
const String strOutFolder(Util::getFilePath(MAKE_PATH_FULL(WORKING_FOLDER_FULL, OPT::strOutputFileName))); |
|
PointCloud pointcloud; |
|
if (!ImportScene(MAKE_PATH_SAFE(OPT::strInputFileName), strOutFolder, scene, pointcloud)) |
|
return EXIT_FAILURE; |
|
// write MVS input data |
|
Util::ensureFolder(strOutFolder); |
|
if (!ARCHIVE::SerializeSave(scene, MAKE_PATH_SAFE(OPT::strOutputFileName))) |
|
return EXIT_FAILURE; |
|
if (!pointcloud.IsEmpty() && !pointcloud.Save(MAKE_PATH_SAFE(Util::getFileFullName(OPT::strOutputFileName)) + _T(".ply"), true)) |
|
return EXIT_FAILURE; |
|
VERBOSE("Exported data: %u images, %u points%s (%s)", |
|
scene.images.size(), scene.vertices.size(), pointcloud.IsEmpty()?"":String::FormatString(", %d dense points", pointcloud.GetSize()).c_str(), |
|
TD_TIMER_GET_FMT().c_str()); |
|
} |
|
|
|
return EXIT_SUCCESS; |
|
} |
|
/*----------------------------------------------------------------*/
|
|
|