-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathvoxec_main.cpp
158 lines (131 loc) · 4.85 KB
/
voxec_main.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
#include "voxec.h"
#include "json_logger.h"
#include <boost/optional.hpp>
#include <boost/program_options.hpp>
#include <boost/algorithm/string/replace.hpp>
#include <fstream>
#include <iostream>
namespace po = boost::program_options;
int main(int argc, char** argv) {
std::unique_ptr<std::ofstream> log_file;
po::options_description opts("Command line options");
opts.add_options()
("quiet,q", "limit output on stdout to progress indication")
("log-file", po::value<std::string>(), "filename to log json message")
("threads,t", po::value<size_t>(), "number of parallel processing threads")
("size,d", po::value<double>(), "voxel size in meters")
("padding,p", po::value<size_t>(), "padding around geometry bounding box (default=32)")
("chunk,c", po::value<size_t>(), "chunk size in number of voxels")
("mmap,m", "use memory-mapped files instead of pure RAM")
("mesh", "emit obj mesh for the last instruction")
("with-pointcloud", "emit point cloud for every voxel grid variable")
("no-vox", "do not write binary dumps")
("help,h", "emit usage information and exit")
("input-file", po::value<std::string>(), "input IFC file")
("output-file", po::value<std::string>(), "output voxel file");
po::positional_options_description positional_options;
positional_options.add("input-file", 1);
positional_options.add("output-file", 1);
po::variables_map vmap;
try {
po::store(po::command_line_parser(argc, argv).
options(opts).positional(positional_options).run(), vmap);
} catch (const std::exception& e) {
json_logger::message(json_logger::LOG_FATAL, "invalid command line {error}", {
{"error", {{"message", std::string(e.what())}}}
});
return 1;
}
if (vmap.count("help")) {
std::cout << "voxec: Voxelization Toolkit Execution Runtime" << std::endl;
std::cout << opts << std::endl << std::endl;
std::cout << "Available commands:" << std::endl << std::endl;
auto& m = voxel_operation_map::map();
for (auto& p : m) {
voxel_operation* op = voxel_operation_map::create(p.first);
std::cout << "### " << p.first << "()" << std::endl << std::endl;
std::cout << "name|required|type" << std::endl;
std::cout << "---|---|---" << std::endl;
for (auto& spec : op->arg_names()) {
std::cout << spec.name << "|" << (spec.required ? "Y" : "n") << "|" << boost::replace_all_copy(spec.type, "|", ",") << std::endl;
}
std::cout << std::endl << std::endl;
}
return 0;
}
if (!vmap.count("input-file")) {
json_logger::message(json_logger::LOG_FATAL, "input file not specified");
return 1;
}
double d = 0.05;
if (!vmap.count("size")) {
json_logger::message(json_logger::LOG_NOTICE, "using default size 0.05m");
} else {
d = vmap["size"].as<double>();
}
if (vmap.count("padding")) {
set_padding(vmap["padding"].as<size_t>());
}
boost::optional<size_t> threads, chunk;
if (vmap.count("threads")) {
threads = vmap["threads"].as<size_t>();
}
if (vmap.count("chunk")) {
chunk = vmap["chunk"].as<size_t>();
}
if (vmap.count("log-file")) {
const std::string log_filename = vmap["log-file"].as<std::string>();
log_file = std::make_unique<std::ofstream>(log_filename.c_str());
json_logger::register_output(json_logger::FMT_JSON, &*log_file);
}
const std::string input_filename = vmap["input-file"].as<std::string>();
const bool with_mesh = vmap.count("mesh") != 0;
const bool quiet = vmap.count("quiet") != 0;
const bool no_vox = vmap.count("no-vox") != 0;
const bool with_pointcloud = vmap.count("with-pointcloud") != 0;
if (!quiet) {
json_logger::register_output(json_logger::FMT_TEXT, &std::cerr);
}
std::ifstream ifs(input_filename.c_str(), std::ios::binary);
if (!ifs.good()) {
json_logger::message(json_logger::LOG_FATAL, "unable to open file {file}", {
{"file", {{"text", input_filename}}}
});
return 1;
}
ifs.seekg(0, ifs.end);
size_t size = ifs.tellg();
ifs.seekg(0, ifs.beg);
ifs >> std::noskipws;
boost::spirit::istream_iterator first(ifs), last;
auto f = first;
voxelfile_parser<decltype(first)> parser;
std::vector<statement_or_function_def> tree;
phrase_parse(first, last, parser, blank, tree);
if (std::distance(f, first) != size) {
json_logger::message(json_logger::LOG_FATAL, "parse errors in voxelfile at {offset}", {
{"offset", {{"value", (long) std::distance(f, first)}}}
});
return 1;
}
boost::optional<std::string> error;
try {
run(tree, d, threads.get_value_or(1), chunk.get_value_or(128), with_mesh, quiet, no_vox, with_pointcloud);
} catch (const std::runtime_error& e) {
error = std::string(e.what());
} catch (const Standard_Failure& e) {
if (e.GetMessageString()) {
error = std::string(e.GetMessageString());
} else {
error.emplace();
}
} catch (...) {
error.emplace();
}
if (error) {
json_logger::message(json_logger::LOG_FATAL, "encountered {error} while running voxelfile", {
{"error", {{"message", *error}}}
});
}
return error ? 1 : 0;
}