-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathspectraprofiler.cpp
470 lines (452 loc) · 19 KB
/
spectraprofiler.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
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
// std
#include <cassert>
#include <chrono>
#include <condition_variable>
#include <cstdlib>
#include <ctime>
#include <exception>
#include <forward_list>
#include <fstream>
#include <functional>
#include <future>
#include <iostream>
#include <memory>
#include <mutex>
#include <sstream>
#include <string>
#include <vector>
// json
#include <nlohmann/json.hpp>
// OpenCV
#include <opencv2/core/opengl.hpp>
#include <opencv2/highgui.hpp>
#include <opencv2/imgproc.hpp>
#include <opencv2/cvconfig.h>
#ifdef HAVE_CUDA
#include <opencv2/cudaimgproc.hpp>
#else
#pragma message("Missing CUDA support in OpenCV, make sure to adjust configuration file accordingly")
#endif
// IFF SDK
#include <iff.h>
//#define IMAGE_MONO
constexpr bool ENABLE_VSYNC = true; //prevents tearing, but may increase latency
constexpr bool WINDOW_FULLSCREEN = false;
constexpr int MAX_WINDOW_WIDTH = 1280;
constexpr int MAX_WINDOW_HEIGHT = 1024;
constexpr char CONFIG_FILENAME[] = "spectraprofiler.json";
constexpr auto EV_STEP = 1.f / 3;
#ifdef _WIN32
const std::string PYTHON3_EXEC = "python";
#else
const std::string PYTHON3_EXEC = "python3";
#endif
namespace
{
using wb_promise = std::promise<nlohmann::json>;
std::unique_ptr<wb_promise> wb_ptr;
void wb_handler(const char* const params)
{
const auto j = nlohmann::json::parse(params);
wb_ptr->set_value(j["cam"]["wb"]);
}
using written_promise = std::promise<bool>;
std::unique_ptr<written_promise> written_ptr;
void written_handler(const char* const callback_data)
{
const auto j = nlohmann::json::parse(callback_data);
written_ptr->set_value(j["success"].get<bool>());
}
std::string iso8601_timestamp()
{
const auto t = std::chrono::system_clock::to_time_t(std::chrono::system_clock::now());
std::tm tp;
#ifdef _WIN32
gmtime_s(&tp, &t);
#else
gmtime_r(&t, &tp);
#endif
char timestamp[] = "19700101T000000Z";
std::strftime(timestamp, sizeof(timestamp), "%Y%m%dT%H%M%SZ", &tp);
return timestamp;
}
template<typename T>
std::string to_string_safe(const T number)
{
std::ostringstream result;
result.imbue(std::locale::classic());
result << number;
return result.str();
}
}
int main()
{
nlohmann::json config;
try
{
config = nlohmann::json::parse(std::ifstream(CONFIG_FILENAME), nullptr, true, true);
}
catch(const std::exception& e)
{
std::cerr << "Invalid configuration provided: " << e.what() << "\n";
return EXIT_FAILURE;
}
const auto it_chains = config.find("chains");
if(it_chains == config.end())
{
std::cerr << "Invalid configuration provided: missing `chains` section\n";
return EXIT_FAILURE;
}
if(!it_chains->is_array())
{
std::cerr << "Invalid configuration provided: section `chains` must be an array\n";
return EXIT_FAILURE;
}
if(it_chains->empty())
{
std::cerr << "Invalid configuration provided: section `chains` must not be empty\n";
return EXIT_FAILURE;
}
const auto it_iff = config.find("IFF");
if(it_iff == config.end())
{
std::cerr << "Invalid configuration provided: missing `IFF` section\n";
return EXIT_FAILURE;
}
iff_initialize(it_iff->dump().c_str());
std::vector<iff_chain_handle_t> chain_handles;
for(const auto& chain_config : *it_chains)
{
const auto chain_handle = iff_create_chain(chain_config.dump().c_str(),
[](const char* const element_name, const int error_code)
{
std::ostringstream message;
message << "Chain element `" << element_name << "` reported an error: " << error_code;
iff_log(IFF_LOG_LEVEL_ERROR, message.str().c_str());
});
chain_handles.push_back(chain_handle);
}
const auto total_chains = chain_handles.size();
size_t grid_x = 1;
size_t grid_y = 1;
while(grid_x * grid_y < total_chains)
{
++grid_x;
if(grid_x * grid_y >= total_chains)
{
break;
}
++grid_y;
}
#ifdef HAVE_CUDA
using Mat = cv::cuda::GpuMat;
namespace imgproc = cv::cuda;
#else
using Mat = cv::Mat;
namespace imgproc = cv;
#endif
using exporter_t = std::function<void(const void*, size_t, const iff_image_metadata*)>;
auto export_callbacks = std::vector<exporter_t>(total_chains);
auto buffer_mutexes = std::vector<std::mutex>(total_chains);
auto pending_buffers = std::vector<std::unique_ptr<Mat>>(total_chains);
auto unused_buffer_lists = std::vector<std::forward_list<std::unique_ptr<Mat>>>(total_chains);
std::condition_variable render_cond;
std::mutex render_mutex;
bool render_requested = false;
for(size_t i = 0; i < total_chains; ++i)
{
for(int j = 0; j < 3; ++j) //one buffer currently rendering, one buffer already filled, one buffer currently filling
{
unused_buffer_lists[i].emplace_front(new Mat());
}
export_callbacks[i] = [&, i](const void* const data, const size_t size, const iff_image_metadata* const metadata)
{
#ifdef IMAGE_MONO
const auto pitch = metadata->width * size_t{1} + metadata->padding;
#else
const auto pitch = metadata->width * size_t{4} + metadata->padding;
#endif
if(size < pitch * metadata->height)
{
std::ostringstream message;
message << "Ignoring invalid buffer: " << metadata->width << "x" << metadata->height << "+" << metadata->padding << " " << size << " bytes";
iff_log(IFF_LOG_LEVEL_WARNING, message.str().c_str());
return;
}
#ifdef IMAGE_MONO
const Mat src_image(cv::Size(metadata->width, metadata->height), CV_8UC1, const_cast<void*>(data), pitch);
#else
const Mat src_image(cv::Size(metadata->width, metadata->height), CV_8UC4, const_cast<void*>(data), pitch);
#endif
auto& unused_buffer_list = unused_buffer_lists[i];
std::unique_ptr<Mat> pending_buffer;
{
std::lock_guard<std::mutex> buffer_lock(buffer_mutexes[i]);
assert(!unused_buffer_list.empty());
pending_buffer = std::move(unused_buffer_list.front());
unused_buffer_list.pop_front();
}
#ifdef IMAGE_MONO
imgproc::cvtColor(src_image, *pending_buffer, cv::COLOR_GRAY2BGRA);
#else
imgproc::cvtColor(src_image, *pending_buffer, cv::COLOR_RGBA2BGRA); //OpenCV requires BGR channel order
#endif
{
std::lock_guard<std::mutex> buffer_lock(buffer_mutexes[i]);
pending_buffers[i].swap(pending_buffer);
if(pending_buffer)
{
unused_buffer_list.push_front(std::move(pending_buffer));
}
}
{
std::lock_guard<std::mutex> render_lock(render_mutex);
render_requested = true;
}
render_cond.notify_one();
};
const auto& chain_handle = chain_handles[i];
iff_set_export_callback(chain_handle, "exporter",
[](const void* const data, const size_t size, iff_image_metadata* const metadata, void* const private_data)
{
const auto export_function = reinterpret_cast<const exporter_t*>(private_data);
(*export_function)(data, size, metadata);
},
&export_callbacks[i]);
iff_set_callback(chain_handle, "writer/frame_written_callback", written_handler);
iff_execute(chain_handle, nlohmann::json{{"exporter", {{"command", "on"}}}}.dump().c_str());
}
const std::string window_name = "IFF SDK Spectra Profiler";
cv::namedWindow(window_name, cv::WINDOW_NORMAL | cv::WINDOW_OPENGL);
cv::setWindowProperty(window_name, cv::WND_PROP_VSYNC, ENABLE_VSYNC ? 1 : 0);
if(WINDOW_FULLSCREEN)
{
cv::setWindowProperty(window_name, cv::WND_PROP_FULLSCREEN, cv::WINDOW_FULLSCREEN);
}
using renderer_t = std::function<void()>;
renderer_t render_callback = [&]()
{
static auto textures = std::vector<cv::ogl::Texture2D>(total_chains);
#ifdef HAVE_CUDA
static auto buffers = std::vector<cv::ogl::Buffer> (total_chains);
static auto gpumats = std::vector<cv::cuda::GpuMat> (total_chains);
#endif
{
std::lock_guard<std::mutex> render_lock(render_mutex);
render_requested = false;
}
for(size_t i = 0; i < total_chains; ++i)
{
std::unique_ptr<Mat> pending_buffer;
{
std::lock_guard<std::mutex> buffer_lock(buffer_mutexes[i]);
pending_buffer = std::move(pending_buffers[i]);
}
if(pending_buffer)
{
#ifdef HAVE_CUDA
// When copying from cuda::GpuMat to ogl::Texture2D OpenCV creates temporary ogl::Buffer
// and registers it with CUDA (cudaGraphicsGLRegisterBuffer function) each time.
// This seems to be unnecessary and problematic (fails after some time) on Windows
// (at least when NVDEC function cuvidMapVideoFrame is also used in another thread).
// Instead create ogl::Buffer and map it as cuda::GpuMat once - this is faster and more stable.
// Source for the fact that mapping can be done just once:
// https://web.archive.org/web/20180611003604/https://github.com/nvpro-samples/gl_cuda_interop_pingpong_st#but-how-do-i-read-and-write-to-gl_texture_3d-in-cuda-and-what-will-it-cost-me
// Error message without this optimization:
// OpenCV(4.6.0) Error: Gpu API call (unknown error) in `anonymous-namespace'::CudaResource::registerBuffer, file ...\opencv-4.6.0\modules\core\src\opengl.cpp, line 176
if(buffers[i].size() != pending_buffer->size())
{
buffers[i].create(pending_buffer->size(), pending_buffer->type(), cv::ogl::Buffer::Target::PIXEL_UNPACK_BUFFER);
gpumats[i] = buffers[i].mapDevice();
buffers[i].unmapDevice();
}
pending_buffer->copyTo(gpumats[i]);
textures[i].copyFrom(buffers[i]);
#else
textures[i].copyFrom(*pending_buffer);
#endif
std::lock_guard<std::mutex> buffer_lock(buffer_mutexes[i]);
unused_buffer_lists[i].push_front(std::move(pending_buffer));
}
}
for(size_t y = 0; y < grid_y; ++y)
{
for(size_t x = 0; x < grid_x; ++x)
{
const auto i = grid_x * y + x;
if(i >= total_chains)
{
break;
}
// image aspect ratio might not be preserved
cv::ogl::render(textures[i], { x * 1. / grid_x, y * 1. / grid_y, 1. / grid_x, 1. / grid_y });
}
}
};
cv::setOpenGlDrawCallback(window_name,
[](void* const private_data)
{
const auto render_function = reinterpret_cast<const renderer_t*>(private_data);
(*render_function)();
},
&render_callback);
iff_log(IFF_LOG_LEVEL_INFO, "Press Esc to terminate the program");
bool size_set = WINDOW_FULLSCREEN;
bool rendering = true;
float cur_ev = 0;
while(true)
{
const auto keycode = cv::pollKey();
if(keycode != -1)
{
if((keycode & 0xff) == 27)
{
iff_log(IFF_LOG_LEVEL_INFO, "Esc key was pressed, stopping the program");
break;
}
else if((keycode & 0xff) == 8)
{
iff_log(IFF_LOG_LEVEL_INFO, "Backspace key was pressed, disabling acquisition");
for(const auto chain_handle : chain_handles)
{
iff_execute(chain_handle, nlohmann::json{{"exporter", {{"command", "off"}}}}.dump().c_str());
}
}
else if((keycode & 0xff) == 13)
{
iff_log(IFF_LOG_LEVEL_INFO, "Enter key was pressed, enabling acquisition");
for(const auto chain_handle : chain_handles)
{
iff_execute(chain_handle, nlohmann::json{{"exporter", {{"command", "on"}}}}.dump().c_str());
}
}
else if((keycode & 0xff) == 32)
{
if(rendering)
{
iff_log(IFF_LOG_LEVEL_INFO, "Space key was pressed, pausing rendering");
rendering = false;
}
else
{
iff_log(IFF_LOG_LEVEL_INFO, "Space key was pressed, resuming rendering");
rendering = true;
}
}
else if((keycode & 0xff) == 49)
{
cur_ev -= EV_STEP;
std::ostringstream message;
message << "'1' was pressed, decreasing exposure by " << EV_STEP << " EV, setting `ev_correction` to: " << cur_ev;
iff_log(IFF_LOG_LEVEL_INFO, message.str().c_str());
for(const auto chain_handle : chain_handles)
{
iff_set_params(chain_handle, nlohmann::json{{"autoctrl", {{"ev_correction", cur_ev}}}}.dump().c_str());
}
}
else if((keycode & 0xff) == 50)
{
cur_ev += EV_STEP;
std::ostringstream message;
message << "'2' was pressed, increasing exposure by " << EV_STEP << " EV, setting `ev_correction` to: " << cur_ev;
iff_log(IFF_LOG_LEVEL_INFO, message.str().c_str());
for(const auto chain_handle : chain_handles)
{
iff_set_params(chain_handle, nlohmann::json{{"autoctrl", {{"ev_correction", cur_ev}}}}.dump().c_str());
}
}
else if((keycode & 0xff) == 9)
{
iff_log(IFF_LOG_LEVEL_INFO, "Tab key was pressed, creating color profile");
for(const auto chain_handle : chain_handles)
{
wb_ptr.reset(new wb_promise);
iff_get_params(chain_handle, R"({"cam": {"params": ["wb"]}})", wb_handler);
const auto wb = wb_ptr->get_future().get();
wb_ptr.reset();
const auto dcp_output_dir = iso8601_timestamp();
written_ptr.reset(new written_promise);
iff_execute(chain_handle, nlohmann::json{{"writer", {{"command", "on"}, {"args", {{"frames_count", 1}, {"subdirectory", dcp_output_dir}}}}}}.dump().c_str());
const auto written = written_ptr->get_future().get();
written_ptr.reset();
if(written)
{
const auto result = std::system((PYTHON3_EXEC +
" coloric.py" +
" -c coloric.json" +
" -r " + to_string_safe(wb["r"].get<double>()) +
" -g " + to_string_safe((wb["g1"].get<double>() + wb["g2"].get<double>()) / 2) +
" -b " + to_string_safe(wb["b"].get<double>()) +
" -i " + dcp_output_dir + "/chart.tif" +
" -o " + dcp_output_dir
).c_str());
if(EXIT_SUCCESS == result)
{
std::ostringstream message;
message << "Color profile successfully written to: " << dcp_output_dir << "/color_profile.dcp";
iff_log(IFF_LOG_LEVEL_INFO, message.str().c_str());
}
else
{
iff_log(IFF_LOG_LEVEL_ERROR, "Failed to create a color profile!");
}
}
else
{
iff_log(IFF_LOG_LEVEL_ERROR, "Failed to write TIFF file!");
}
}
}
else
{
std::ostringstream message;
message << "Key press ignored, code: " << keycode;
iff_log(IFF_LOG_LEVEL_DEBUG, message.str().c_str());
}
}
if(rendering)
{
{
std::unique_lock<std::mutex> render_lock(render_mutex);
render_cond.wait_for(render_lock, std::chrono::seconds(1), [&](){ return render_requested; });
}
if(!size_set)
{
// try to preserve image aspect ratio by sizing the window accordingly
// (assuming all chains produce images with the same aspect ratio)
cv::Size size;
{
std::lock_guard<std::mutex> buffer_lock(buffer_mutexes[0]);
if(pending_buffers[0])
{
size = pending_buffers[0]->size();
}
}
if(!size.empty())
{
size.width *= static_cast<int>(grid_x);
size.height *= static_cast<int>(grid_y);
if(size.width > MAX_WINDOW_WIDTH)
{
size.height = static_cast<cv::Size::value_type>(MAX_WINDOW_WIDTH / size.aspectRatio());
size.width = MAX_WINDOW_WIDTH;
}
if(size.height > MAX_WINDOW_HEIGHT)
{
size.width = static_cast<cv::Size::value_type>(MAX_WINDOW_HEIGHT * size.aspectRatio());
size.height = MAX_WINDOW_HEIGHT;
}
cv::resizeWindow(window_name, size);
size_set = true;
}
}
cv::updateWindow(window_name);
}
}
for(const auto chain_handle : chain_handles)
{
iff_release_chain(chain_handle);
}
iff_finalize();
return EXIT_SUCCESS;
}