Project
Loading...
Searching...
No Matches
FrameworkGUIDebugger.cxx
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
14#include "DebugGUI/imgui.h"
15#include "DebugGUI/implot.h"
16#include "DebugGUI/imgui_extras.h"
24#include "PaletteHelpers.h"
25#include "FrameworkGUIState.h"
26#include "Framework/Signpost.h"
27#include <DebugGUI/icons_font_awesome.h>
28
29#include <fmt/format.h>
30
31#include <algorithm>
32#include <iostream>
33#include <set>
34#include <string>
35#include <cinttypes>
36#include <numeric>
37
39// Make sure we can use aggregated initialisers.
40#pragma GCC diagnostic push
41#pragma GCC diagnostic ignored "-Wpedantic"
42
43// Simplify debugging
44template class std::vector<o2::framework::DeviceMetricsInfo>;
45
46namespace o2::framework::gui
47{
48// Type erased information for the plotting
50 int mod;
51 size_t first;
52 size_t size;
53 const void* Y = nullptr;
54 const void* X = nullptr;
56 const char* legend = nullptr;
57 int axis = 0;
58};
59
60} // namespace o2::framework::gui
61
62template class std::vector<o2::framework::gui::MultiplotData>;
63
64namespace o2::framework::gui
65{
66
68{
69 switch (logLevel) {
70 case LogParsingHelpers::LogLevel::Info:
71 return PaletteHelpers::GREEN;
72 case LogParsingHelpers::LogLevel::Important:
73 return PaletteHelpers::GREEN;
74 case LogParsingHelpers::LogLevel::Debug:
75 return PaletteHelpers::WHITE;
76 case LogParsingHelpers::LogLevel::Alarm:
77 return PaletteHelpers::YELLOW;
78 case LogParsingHelpers::LogLevel::Warning:
79 return PaletteHelpers::DARK_YELLOW;
80 case LogParsingHelpers::LogLevel::Error:
81 return PaletteHelpers::RED;
82 case LogParsingHelpers::LogLevel::Critical:
83 return PaletteHelpers::RED;
84 case LogParsingHelpers::LogLevel::Fatal:
85 return PaletteHelpers::RED;
86 case LogParsingHelpers::LogLevel::Unknown:
87 return PaletteHelpers::DARK_RED;
88 default:
89 return PaletteHelpers::DARK_RED;
90 };
91}
92
93void displayHistory(const DeviceInfo& info, DeviceControl& control)
94{
95 if (info.history.empty()) {
96 return;
97 }
98 int startPos = info.historyPos;
99 const int historySize = info.history.size();
100
101 int triggerStartPos = startPos + 1 % historySize;
102 int triggerStopPos = startPos % historySize;
103
104 int j = startPos;
105 // We look for a stop trigger, so that we know where to stop the search for
106 // out start search. If no stop trigger is found, we search until the end
107 if (control.logStopTrigger[0]) {
108 while ((j % historySize) != ((startPos + 1) % historySize)) {
109 assert(j >= 0);
110 assert(j < historySize);
111 auto& line = info.history[j];
112 if (strstr(line.c_str(), control.logStopTrigger)) {
113 triggerStopPos = (j + 1) % historySize;
114 break;
115 }
116 // Wrap in case we end up below 0
117 j = (j == 0) ? historySize - 1 : j - 1;
118 }
119 }
120
121 // Look for the last instance of the start trigger before the
122 // last stop trigger.
123 j = startPos + 1;
124 if (control.logStartTrigger[0]) {
125 while ((j % historySize) != triggerStopPos) {
126 assert(historySize > j);
127 assert(historySize == 1000);
128 auto& line = info.history[j];
129 if (strstr(line.c_str(), control.logStartTrigger)) {
130 triggerStartPos = j;
131 }
132 j = (j + 1) % historySize;
133 }
134 }
135
136 // We start from the last trigger found. Eventually this is the first
137 // line in the ring buffer, if no trigger is specified.
138 size_t ji = triggerStartPos % historySize;
139 size_t je = triggerStopPos % historySize;
140 size_t iterations = 0;
141 while (historySize && ((ji % historySize) != je)) {
142 assert(iterations < historySize);
143 iterations++;
144 assert(historySize == 1000);
145 assert(ji < historySize);
146 auto& line = info.history[ji];
147 auto logLevel = info.historyLevel[ji];
148
149 // Skip empty lines
150 if (line.empty()) {
151 ji = (ji + 1) % historySize;
152 continue;
153 }
154 // Print matching lines. Notice we filter twice, once on input, to reduce the
155 // stream, a second time at display time, to avoid showing unrelevant
156 // messages from past.
157 if (strstr(line.c_str(), control.logFilter) != nullptr) {
158 if (logLevel >= control.logLevel) {
159 ImGui::TextColored(colorForLogLevel(logLevel), "%s", line.c_str());
160 }
161 }
162 ji = (ji + 1) % historySize;
163 }
164}
165
166struct HistoData {
167 int mod;
168 size_t first;
169 size_t size;
170 void* points = nullptr;
171 const size_t* time = nullptr;
172 char const* legend = nullptr;
173};
174
175enum struct MetricsDisplayStyle : int {
176 Lines = 0,
177 Histos = 1,
178 Sparks = 2,
179 Table = 3,
180 Stems = 4
181};
182
185 std::string label;
186};
187
189 int axis = 0; // The Axis to use for Y
190 bool visible = false;
191 bool selected = false;
192 std::string legend;
193 size_t legendHash = -1;
194};
195
201};
202
208
209// We use this to keep together all the kind of metrics
210// so that we can display driver and device metrics in the same plot
211// without an if.
213 gsl::span<DeviceMetricsInfo const> metrics[TOTAL_TYPES_OF_METRICS];
214 gsl::span<TopologyNodeInfo const> specs[TOTAL_TYPES_OF_METRICS];
215};
216
218 double startTime,
219 std::vector<MetricIndex>& visibleMetricsIndex,
220 std::vector<MetricDisplayState>& metricDisplayStates,
221 AllMetricsStore const& metricStore)
222{
223 static bool locked = false;
224 ImGui::Checkbox("Lock scrolling", &locked);
225 ImGui::BeginTable("##sparks table", 3, ImGuiTableFlags_Resizable | ImGuiTableFlags_ScrollY, ImVec2{-1, -1});
226 ImGui::TableSetupColumn("##close button", ImGuiTableColumnFlags_NoResize | ImGuiTableColumnFlags_WidthFixed, 25);
227 ImGui::TableSetupColumn("##plot name", ImGuiTableColumnFlags_WidthFixed, 200);
228 ImGui::TableSetupColumn("##plot", ImGuiTableColumnFlags_WidthFixed, 0);
229 ImGui::TableSetupScrollFreeze(2, 0);
230 for (size_t i = 0; i < visibleMetricsIndex.size(); ++i) {
231 auto& index = visibleMetricsIndex[i];
232 auto& metricsInfos = metricStore.metrics[index.storeIndex];
233 auto& metricsInfo = metricsInfos[index.deviceIndex];
234 auto& metric = metricsInfo.metrics[index.metricIndex];
235 auto& state = metricDisplayStates[index.stateIndex];
236
237 ImGui::TableNextColumn();
238 ImGui::PushID(index.stateIndex);
239 state.visible = !ImGui::Button("-");
240 ImGui::PopID();
241 ImGui::TableNextColumn();
242 ImGui::TextUnformatted(state.legend.c_str());
243 ImGui::TableNextColumn();
244 static ImPlotAxisFlags rtx_axis = ImPlotAxisFlags_NoTickLabels | ImPlotAxisFlags_NoLabel | ImPlotAxisFlags_NoTickMarks;
245 static ImPlotAxisFlags rty_axis = ImPlotAxisFlags_NoTickLabels | ImPlotAxisFlags_NoLabel | ImPlotAxisFlags_NoTickMarks;
246 ImGui::PushID(index.stateIndex);
248 data.mod = std::min(metric.filledMetrics, metricStorageSize(metric.type));
249 data.first = metric.pos - data.mod;
250 data.size = metric.filledMetrics;
251 data.legend = state.legend.c_str();
252
253 if (!locked) {
254 ImPlot::SetNextAxisLimits(ImAxis_X1, (startTime + ImGui::GetTime() - 100) * 1000, (startTime + ImGui::GetTime()) * 1000, ImGuiCond_Always);
255 ImPlot::SetNextAxisLimits(ImAxis_Y1, metricsInfo.min[index.metricIndex], metricsInfo.max[index.metricIndex] * 1.1, ImGuiCond_Always);
256 rty_axis |= ImPlotAxisFlags_LockMin;
257 }
258 if (ImPlot::BeginPlot("##sparks", ImVec2(700, 100), 0)) {
259 ImPlot::SetupAxes("time", "value", rtx_axis, rty_axis);
260 ImPlot::SetAxis(state.axis);
261 switch (metric.type) {
262 case MetricType::Enum: {
263 data.points = (void*)metricsInfo.enumMetrics[metric.storeIdx].data();
264 data.time = metricsInfo.enumTimestamps[metric.storeIdx].data();
265
266 auto getter = [](int idx, void* hData) -> ImPlotPoint {
267 auto histoData = reinterpret_cast<HistoData*>(hData);
268 size_t pos = (histoData->first + static_cast<size_t>(idx)) % histoData->mod;
269 assert(pos >= 0 && pos < metricStorageSize(MetricType::Enum));
270 return ImPlotPoint(histoData->time[pos], ((int8_t*)(histoData->points))[pos]);
271 };
272 ImPlot::PlotLineG("##plot", getter, &data, data.mod);
273 } break;
274 case MetricType::Int: {
275 data.points = (void*)metricsInfo.intMetrics[metric.storeIdx].data();
276 data.time = metricsInfo.intTimestamps[metric.storeIdx].data();
277
278 auto getter = [](int idx, void* hData) -> ImPlotPoint {
279 auto histoData = reinterpret_cast<HistoData*>(hData);
280 size_t pos = (histoData->first + static_cast<size_t>(idx)) % histoData->mod;
281 assert(pos >= 0 && pos < metricStorageSize(MetricType::Int));
282 return ImPlotPoint(histoData->time[pos], ((int*)(histoData->points))[pos]);
283 };
284 ImPlot::PlotLineG("##plot", getter, &data, data.mod);
285 } break;
286 case MetricType::Uint64: {
287 data.points = (void*)metricsInfo.uint64Metrics[metric.storeIdx].data();
288 data.time = metricsInfo.uint64Timestamps[metric.storeIdx].data();
289
290 auto getter = [](int idx, void* hData) -> ImPlotPoint {
291 auto histoData = reinterpret_cast<HistoData*>(hData);
292 size_t pos = (histoData->first + static_cast<size_t>(idx)) % histoData->mod;
293 assert(pos >= 0 && pos < metricStorageSize(MetricType::Uint64));
294 return ImPlotPoint(histoData->time[pos], ((uint64_t*)histoData->points)[pos]);
295 };
296 ImPlot::PlotLineG("##plot", getter, &data, data.mod, 0);
297 } break;
298 case MetricType::Float: {
299 data.points = (void*)metricsInfo.floatMetrics[metric.storeIdx].data();
300 data.time = metricsInfo.floatTimestamps[metric.storeIdx].data();
301
302 auto getter = [](int idx, void* hData) -> ImPlotPoint {
303 auto histoData = reinterpret_cast<HistoData*>(hData);
304 size_t pos = (histoData->first + static_cast<size_t>(idx)) % histoData->mod;
305 assert(pos >= 0 && pos < metricStorageSize(MetricType::Float));
306 return ImPlotPoint(histoData->time[pos], ((float*)histoData->points)[pos]);
307 };
308 ImPlot::PlotLineG("##plot", getter, &data, data.mod, 0);
309 } break;
310 default:
311 return;
312 break;
313 }
314 ImPlot::EndPlot();
315 }
316 ImGui::PopID();
317 }
318 ImGui::EndTable();
319}
320
321int formatSI(double value, char* buff, int size, void* user_data)
322{
323 if (value == 0.0) {
324 return snprintf(buff, size, "%.0f", value);
325 }
326 if (value < 10.0) {
327 return snprintf(buff, size, "%.2f", value);
328 }
329 if (value < 1000.0) {
330 return snprintf(buff, size, "%.0f", value);
331 }
332 if (value < 1000000.0) {
333 return snprintf(buff, size, "%.0f k", value / 1000.0);
334 }
335 if (value < 1000000000.0) {
336 return snprintf(buff, size, "%.0f M", value / 1000000.0);
337 }
338 if (value < 1000000000000.0) {
339 return snprintf(buff, size, "%.0f G", value / 1000000000.0);
340 }
341 return snprintf(buff, size, "%.0f T", value / 1000000000000.0);
342}
343
344int formatTimeSinceStart(double value, char* buff, int size, void* user_data)
345{
346 auto* startTime = (int64_t*)user_data;
347 if (value - *startTime < 0) {
348 buff[0] = '\0';
349 return 0;
350 }
351 int64_t seconds = (value - *startTime) / 1000;
352 int64_t minutes = seconds / 60;
353 return snprintf(buff, size, "%02" PRIi64 ":%02" PRIi64, minutes, seconds % 60);
354}
355
357 size_t rangeBegin, size_t rangeEnd, size_t bins, MetricsDisplayStyle displayType,
358 std::vector<MetricDisplayState>& state,
359 AllMetricsStore const& metricStore,
360 DriverInfo const& driverInfo)
361{
362 std::vector<void*> metricsToDisplay;
363 std::vector<const char*> deviceNames;
364 std::vector<MultiplotData> userData;
365#ifdef NDEBUG
366 for (size_t si = 0; si < TOTAL_TYPES_OF_METRICS; ++si) {
367 assert(metricsStore.metrics[si].size() == metricStore.specs[si].size());
368 }
369#endif
370 float maxValue[3] = {std::numeric_limits<float>::lowest(), std::numeric_limits<float>::lowest(), std::numeric_limits<float>::lowest()};
371 float minValue[3] = {0, 0, 0};
372 size_t maxDomain = std::numeric_limits<size_t>::lowest();
373 size_t minDomain = std::numeric_limits<size_t>::max();
374 size_t gmi = 0;
375
376 ImPlotAxisFlags axisFlags = 0;
377
378 for (size_t si = 0; si < TOTAL_TYPES_OF_METRICS; ++si) {
379 gsl::span<DeviceMetricsInfo const> metricsInfos = metricStore.metrics[si];
380 gsl::span<TopologyNodeInfo const> specs = metricStore.specs[si];
381 for (int di = 0; di < metricsInfos.size(); ++di) {
382 for (size_t mi = 0; mi < metricsInfos[di].metrics.size(); ++mi) {
383 if (state[gmi].visible == false) {
384 gmi++;
385 continue;
386 }
387 auto& metric = metricsInfos[di].metrics[mi];
388 deviceNames.push_back(specs[di].label.c_str());
390 data.size = metric.filledMetrics;
391 data.legend = state[gmi].legend.c_str();
392 data.type = metric.type;
393 data.axis = state[gmi].axis;
394 minValue[data.axis] = std::min(minValue[data.axis], metricsInfos[di].min[mi]);
395 maxValue[data.axis] = std::max(maxValue[data.axis], metricsInfos[di].max[mi]);
396 minDomain = std::min(minDomain, metricsInfos[di].minDomain[mi]);
397 maxDomain = std::max(maxDomain, metricsInfos[di].maxDomain[mi]);
398 axisFlags |= data.axis == 1 ? (ImPlotFlags_)ImPlotFlags_YAxis2 : ImPlotFlags_None;
399 axisFlags |= data.axis == 2 ? (ImPlotFlags_)ImPlotFlags_YAxis3 : ImPlotFlags_None;
400 switch (metric.type) {
401 case MetricType::Int: {
402 data.Y = metricsInfos[di].intMetrics[metric.storeIdx].data();
403 auto& timestamps = metricsInfos[di].intTimestamps[metric.storeIdx];
404 data.mod = std::min(metric.filledMetrics, timestamps.size());
405 data.first = metric.pos - data.mod;
406 data.X = timestamps.data();
407 } break;
408 case MetricType::Enum: {
409 data.Y = metricsInfos[di].enumMetrics[metric.storeIdx].data();
410 auto& timestamps = metricsInfos[di].enumTimestamps[metric.storeIdx];
411 data.mod = std::min(metric.filledMetrics, timestamps.size());
412 data.first = metric.pos - data.mod;
413 data.X = timestamps.data();
414 } break;
415 case MetricType::Uint64: {
416 data.Y = metricsInfos[di].uint64Metrics[metric.storeIdx].data();
417 auto& timestamps = metricsInfos[di].uint64Timestamps[metric.storeIdx];
418 data.mod = std::min(metric.filledMetrics, timestamps.size());
419 data.first = metric.pos - data.mod;
420 data.X = timestamps.data();
421 } break;
422 case MetricType::Float: {
423 data.Y = metricsInfos[di].floatMetrics[metric.storeIdx].data();
424 auto& timestamps = metricsInfos[di].floatTimestamps[metric.storeIdx];
425 data.mod = std::min(metric.filledMetrics, timestamps.size());
426 data.first = metric.pos - data.mod;
427 data.X = timestamps.data();
428 } break;
429 case MetricType::Unknown:
430 case MetricType::String: {
431 data.Y = nullptr;
432 data.type = MetricType::String;
433 } break;
434 }
435
436 userData.emplace_back(data);
437 gmi++;
438 }
439 }
440 }
441
442 maxDomain = std::max(minDomain + 1024, maxDomain);
443 for (size_t ai = 0; ai < 3; ++ai) {
444 maxValue[ai] = std::max(minValue[ai] + 1.f, maxValue[ai]);
445 }
446
447 // Nothing to show.
448 if (userData.empty()) {
449 return;
450 }
451 for (size_t ui = 0; ui < userData.size(); ++ui) {
452 metricsToDisplay.push_back(&(userData[ui]));
453 }
454
455 auto getterXY = [](int idx, void* hData) -> ImPlotPoint {
456 auto histoData = reinterpret_cast<const MultiplotData*>(hData);
457 size_t pos = (histoData->first + static_cast<size_t>(idx)) % histoData->mod;
458 double x = static_cast<const size_t*>(histoData->X)[pos];
459 double y = 0.;
460 if (histoData->type == MetricType::Int) {
461 y = static_cast<const int*>(histoData->Y)[pos];
462 } else if (histoData->type == MetricType::Uint64) {
463 y = static_cast<const uint64_t*>(histoData->Y)[pos];
464 } else if (histoData->type == MetricType::Float) {
465 y = static_cast<const float*>(histoData->Y)[pos];
466 } else if (histoData->type == MetricType::Enum) {
467 y = static_cast<const int8_t*>(histoData->Y)[pos];
468 }
469 auto point = ImPlotPoint{x, y};
470 return point;
471 };
472 static bool logScale = false;
473 ImGui::Checkbox("Log scale", &logScale);
474
475 ImPlot::SetNextAxisLimits(ImAxis_X1, minDomain, maxDomain, ImGuiCond_Once);
476
477 auto axisPadding = 0.;
478 if (displayType == MetricsDisplayStyle::Lines) {
479 axisPadding = 0.2;
480 }
481
482 ImPlot::SetNextAxisLimits(ImAxis_Y1, minValue[0] - (maxValue[0] - minValue[0]) * axisPadding,
483 maxValue[0] * (1. + axisPadding), ImGuiCond_Always);
484 ImPlot::SetNextAxisLimits(ImAxis_Y2, minValue[1] - (maxValue[1] - minValue[1]) * axisPadding,
485 maxValue[1] * (1. + axisPadding), ImGuiCond_Always);
486 ImPlot::SetNextAxisLimits(ImAxis_Y3, minValue[2] - (maxValue[2] - minValue[2]) * axisPadding,
487 maxValue[2] * (1. + axisPadding), ImGuiCond_Always);
488
489 switch (displayType) {
490 case MetricsDisplayStyle::Histos:
491 if (ImPlot::BeginPlot("##Some plot")) {
492 ImPlot::SetupAxes("time", "value");
493 ImPlot::SetupAxisFormat(ImAxis_Y1, formatSI, nullptr);
494 ImPlot::SetupAxisFormat(ImAxis_Y2, formatSI, nullptr);
495 ImPlot::SetupAxisFormat(ImAxis_Y3, formatSI, nullptr);
496 ImPlot::SetupAxisFormat(ImAxis_X1, formatTimeSinceStart, (void*)&driverInfo.startTimeMsFromEpoch);
497 for (size_t pi = 0; pi < metricsToDisplay.size(); ++pi) {
498 ImGui::PushID(pi);
499 auto data = (const MultiplotData*)metricsToDisplay[pi];
500 const char* label = ((MultiplotData*)metricsToDisplay[pi])->legend;
501 ImPlot::PlotBarsG(label, getterXY, metricsToDisplay[pi], data->mod, 1, 0);
502 ImGui::PopID();
503 }
504 ImPlot::EndPlot();
505 }
506
507 break;
508 case MetricsDisplayStyle::Lines: {
509 auto xAxisFlags = ImPlotAxisFlags_None;
510 auto yAxisFlags = ImPlotAxisFlags_LockMin;
511 // ImPlot::FitNextPlotAxes(true, true, true, true);
512 if (ImPlot::BeginPlot("##Some plot", {-1, -1}, axisFlags)) {
513 ImPlot::SetupAxes("time", "value", xAxisFlags, yAxisFlags);
514 ImPlot::SetupAxisFormat(ImAxis_Y1, formatSI, nullptr);
515 ImPlot::SetupAxisFormat(ImAxis_Y2, formatSI, nullptr);
516 ImPlot::SetupAxisFormat(ImAxis_Y3, formatSI, nullptr);
517 ImPlot::SetupAxisFormat(ImAxis_X1, formatTimeSinceStart, (void*)&driverInfo.startTimeMsFromEpoch);
518 for (size_t pi = 0; pi < metricsToDisplay.size(); ++pi) {
519 ImGui::PushID(pi);
520 auto data = (const MultiplotData*)metricsToDisplay[pi];
521 ImPlot::SetAxis(data->axis);
522 ImPlot::PlotLineG(data->legend, getterXY, metricsToDisplay[pi], data->mod, 0);
523 ImGui::PopID();
524 }
525 ImPlot::EndPlot();
526 }
527 } break;
528 case MetricsDisplayStyle::Stems:
529 if (ImPlot::BeginPlot("##Some plot")) {
530 ImPlot::SetupAxes("time", "value");
531 for (size_t pi = 0; pi < userData.size(); ++pi) {
532 auto data = reinterpret_cast<const MultiplotData*>(metricsToDisplay[pi]);
533 // FIXME: display a message for other metrics
534 if (data->type == MetricType::Uint64) {
535 ImGui::PushID(pi);
536 ImPlot::PlotScatterG(((MultiplotData*)metricsToDisplay[pi])->legend, getterXY, metricsToDisplay[pi], data->mod, 0);
537 ImGui::PopID();
538 }
539 }
540 ImPlot::EndPlot();
541 }
542 break;
543 default:
544 break;
545 }
546}
547
552
553void metricsTableRow(std::vector<MetricIndex> metricIndex,
554 AllMetricsStore const& metricsStore,
555 int row)
556{
557 ImGui::TableNextColumn();
558 ImGui::Text("%d", row);
559
560 for (auto index : metricIndex) {
561 auto& metricsInfos = metricsStore.metrics[index.storeIndex];
562 auto& metricsInfo = metricsInfos[index.deviceIndex];
563 auto& info = metricsInfos[index.deviceIndex].metrics[index.metricIndex];
564
565 ImGui::TableNextColumn();
566 if (info.filledMetrics <= row) {
567 ImGui::Text(" - ");
568 continue;
569 }
570 switch (info.type) {
571 case MetricType::Int: {
572 auto time = metricsInfo.intTimestamps[info.storeIdx][row];
573 ImGui::Text("%i, %" PRIu64, metricsInfo.intMetrics[info.storeIdx][row], (uint64_t)time);
574 } break;
575 case MetricType::Uint64: {
576 auto time = metricsInfo.uint64Timestamps[info.storeIdx][row];
577 ImGui::Text("%" PRIu64 ", %" PRIu64, metricsInfo.uint64Metrics[info.storeIdx][row], (uint64_t)time);
578 } break;
579 case MetricType::Float: {
580 auto time = metricsInfo.floatTimestamps[info.storeIdx][row];
581 ImGui::Text("%f, %" PRIu64, metricsInfo.floatMetrics[info.storeIdx][row], (uint64_t)time);
582 } break;
583 case MetricType::Enum: {
584 auto time = metricsInfo.enumTimestamps[info.storeIdx][row];
585 ImGui::Text("%i, %" PRIu64, metricsInfo.enumMetrics[info.storeIdx][row], (uint64_t)time);
586 } break;
587 case MetricType::String: {
588 auto time = metricsInfo.stringTimestamps[info.storeIdx][row];
589 ImGui::Text("%s, %" PRIu64, metricsInfo.stringMetrics[info.storeIdx][row].data, (uint64_t)time);
590 } break;
591 default:
592 break;
593 }
594 }
595}
596
597bool hasAll(const char* s, const char* q)
598{
599 /* base case: empty query */
600 if (*q == 0) {
601 return true;
602 }
603 do {
604 s = strchr(s, (int)*q);
605 if (s == nullptr) {
606 return false;
607 }
608 s++;
609 q++;
610 } while ((int)*q != 0);
611 return true;
612}
613
614void TextCenter(char const* text)
615{
616 float font_size = ImGui::GetFontSize() * strlen(text) / 2;
617 ImGui::Dummy(ImVec2(
618 ImGui::GetWindowSize().x / 2 -
619 font_size + (font_size / 2),
620 120));
621
622 ImGui::TextUnformatted(text);
623}
624
626 DriverInfo const& driverInfo,
627 std::vector<DeviceInfo> const& infos,
628 std::vector<DataProcessorInfo> const& metadata,
629 std::vector<DeviceControl>& controls,
630 AllMetricsStore const& metricsStore)
631{
632 if (state.bottomPaneVisible == false) {
633 return;
634 }
635 auto metricDisplayPos = 0;
636 static bool metricSelectorVisible = true;
637 static std::vector<MetricDisplayState> metricDisplayState;
638
639 // Calculate the full timestamp range for the selected metric
640 size_t minTime = -1;
641 size_t maxTime = 0;
642 constexpr size_t MAX_QUERY_SIZE = 256;
643 static char query[MAX_QUERY_SIZE];
644 static char lastSelectedQuery[MAX_QUERY_SIZE];
645
646 size_t totalMetrics = 0;
647 for (auto& metricsInfos : metricsStore.metrics) {
648 for (auto& metricInfo : metricsInfos) {
649 totalMetrics += metricInfo.metrics.size();
650 }
651 }
652
653 if (totalMetrics != metricDisplayState.size() || strcmp(lastSelectedQuery, query)) {
654 size_t gmi = 0;
655 std::vector<MetricDisplayState> newMetricDisplayStates;
656 newMetricDisplayStates.resize(totalMetrics);
657 static std::vector<int> order;
658 order.resize(metricDisplayState.size());
659 std::iota(order.begin(), order.end(), 0);
660 std::sort(order.begin(), order.end(), [](int a, int b) {
661 return metricDisplayState[a].legend < metricDisplayState[b].legend;
662 });
663
664 for (size_t si = 0; si < TOTAL_TYPES_OF_METRICS; ++si) {
665 auto& metricsInfos = metricsStore.metrics[si];
666 auto& specs = metricsStore.specs[si];
667 for (size_t di = 0; di < metricsInfos.size(); ++di) {
668 auto& metricInfo = metricsInfos[di];
669 auto& spec = specs[di];
670 for (size_t li = 0; li != metricInfo.metricLabels.size(); ++li) {
671 char const* metricLabel = metricInfo.metricLabels[li].label;
672 // find the equal range for the label
673 auto low = std::lower_bound(order.begin(), order.end(), spec.label, [](int a, std::string const& b) {
674 return metricDisplayState[a].legend < b;
675 });
676 auto up = std::upper_bound(low, order.end(), spec.label, [](std::string const& a, int b) {
677 return a < metricDisplayState[b].legend;
678 });
679 std::string legend = fmt::format("{}/{}", spec.label, metricLabel);
680 auto old = std::lower_bound(low, up, legend, [](int a, std::string const& b) {
681 return metricDisplayState[a].legend < b;
682 });
683 if (old != order.end() && metricDisplayState[*old].legend == legend) {
684 newMetricDisplayStates[gmi].visible = metricDisplayState[*old].visible;
685 newMetricDisplayStates[gmi].axis = metricDisplayState[*old].axis;
686 } else {
687 newMetricDisplayStates[gmi].visible = false;
688 }
689
690 newMetricDisplayStates[gmi].selected = hasAll(metricLabel, query);
691 newMetricDisplayStates[gmi].legend = legend;
692 gmi++;
693 }
694 }
695 }
696 metricDisplayState.swap(newMetricDisplayStates);
697 strcpy(lastSelectedQuery, query);
698 }
699
700 static std::vector<MetricIndex> selectedMetricIndex;
701
702 if (metricSelectorVisible) {
703 selectedMetricIndex.clear();
704 size_t gmi = 0;
705 for (size_t si = 0; si < TOTAL_TYPES_OF_METRICS; ++si) {
706 auto& metricsInfos = metricsStore.metrics[si];
707
708 for (size_t di = 0; di < metricsInfos.size(); ++di) {
709 auto& metricInfo = metricsInfos[di];
710 for (size_t li = 0; li != metricInfo.metricLabels.size(); ++li) {
711 auto& state = metricDisplayState[gmi];
712 if (state.selected) {
713 selectedMetricIndex.emplace_back(MetricIndex{si, di, li, gmi});
714 }
715 gmi++;
716 }
717 }
718 }
719
720 metricDisplayPos = ImGui::GetIO().DisplaySize.x / 4;
721 ImGui::SetNextWindowPos(ImVec2(0, ImGui::GetIO().DisplaySize.y - state.bottomPaneSize), 0);
722 ImGui::SetNextWindowSize(ImVec2(metricDisplayPos, state.bottomPaneSize), 0);
723 ImGui::Begin("Available metrics", nullptr, ImGuiWindowFlags_NoBringToFrontOnFocus | ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoCollapse | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoResize);
724
725 ImGui::Text(ICON_FA_FILTER);
726 ImGui::SameLine();
727 ImGui::InputText("##query-metrics", query, MAX_QUERY_SIZE);
728 size_t metricSize = 0;
729 for (auto deviceMetrics : metricsStore.metrics) {
730 metricSize += DeviceMetricsInfoHelpers::metricsStorageSize(deviceMetrics);
731 }
732 // ImGui::Text("Total memory used %zu MB", metricSize / 1000000);
733 ImGui::Text("%zu/%zu matching", selectedMetricIndex.size(), totalMetrics);
734
735 static const char* possibleAxis[] = {
736 "Y",
737 "Y1",
738 "Y2",
739 };
740 if (ImGui::BeginTable("##metrics-table", 3, ImGuiTableFlags_ScrollX | ImGuiTableFlags_ScrollY, ImVec2{-1, -1})) {
741 ImGui::TableSetupColumn("##close button", ImGuiTableColumnFlags_NoResize | ImGuiTableColumnFlags_WidthFixed, 20);
742 ImGui::TableSetupColumn("##axis kind", ImGuiTableColumnFlags_NoResize | ImGuiTableColumnFlags_WidthFixed, 30);
743 ImGui::TableSetupScrollFreeze(1, 0);
744 ImGuiListClipper clipper;
745 clipper.Begin(selectedMetricIndex.size());
746 while (clipper.Step()) {
747 for (size_t i = clipper.DisplayStart; i < clipper.DisplayEnd; ++i) {
748 auto& index = selectedMetricIndex[i];
749 auto& metricsInfos = metricsStore.metrics[index.storeIndex];
750 auto& nodes = metricsStore.specs[index.storeIndex];
751 auto& metricInfo = metricsInfos[index.deviceIndex];
752 auto& node = nodes[index.deviceIndex];
753 auto& label = metricInfo.metricLabels[index.metricIndex];
754 ImGui::PushID(index.stateIndex);
755 ImGui::TableNextRow();
756 ImGui::TableNextColumn();
757 ImGui::Checkbox("##checkbox", &metricDisplayState[index.stateIndex].visible);
758 ImGui::TableNextColumn();
759 if (metricDisplayState[index.stateIndex].visible) {
760 if (ImGui::BeginCombo("##Select style", possibleAxis[metricDisplayState[index.stateIndex].axis], ImGuiComboFlags_NoArrowButton)) {
761 for (int n = 0; n < IM_ARRAYSIZE(possibleAxis); n++) {
762 bool is_selected = (metricDisplayState[index.stateIndex].axis == n);
763 if (ImGui::Selectable(possibleAxis[n], is_selected)) {
764 metricDisplayState[index.stateIndex].axis = n;
765 }
766 if (is_selected) {
767 ImGui::SetItemDefaultFocus(); // You may set the initial focus when opening the combo (scrolling + for keyboard navigation support)
768 }
769 }
770 ImGui::EndCombo();
771 }
772 }
773 ImGui::TableNextColumn();
774 ImGui::Text("%s/%s", node.label.c_str(), label.label);
775 ImGui::PopID();
776 }
777 }
778 ImGui::EndTable();
779 }
780 ImGui::End();
781 }
782 ImGui::SetNextWindowPos(ImVec2(metricDisplayPos, ImGui::GetIO().DisplaySize.y - state.bottomPaneSize), 0);
783 ImGui::SetNextWindowSize(ImVec2(ImGui::GetIO().DisplaySize.x - metricDisplayPos, state.bottomPaneSize), 0);
784
785 ImGui::Begin("Devices", nullptr, ImGuiWindowFlags_NoBringToFrontOnFocus | ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoCollapse | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoResize);
786
787 if (!metricSelectorVisible) {
788 metricSelectorVisible = ImGui::Button(ICON_FA_CHEVRON_RIGHT "Show metric selector");
789 } else {
790 metricSelectorVisible = !ImGui::Button(ICON_FA_CHEVRON_LEFT "Hide metric selector");
791 }
792 static char const* plotStyles[] = {
793 "lines",
794 "histograms",
795 "sparks",
796 "table",
797 "stems"};
798 ImGui::SameLine();
799 static enum MetricsDisplayStyle currentStyle = MetricsDisplayStyle::Lines;
800 static char const* currentStyleStr = plotStyles[0];
801 ImGui::TextUnformatted("Metric display style:");
802 ImGui::SameLine();
803 ImGui::PushItemWidth(100);
804 if (ImGui::BeginCombo("##Select style", currentStyleStr)) {
805 for (int n = 0; n < IM_ARRAYSIZE(plotStyles); n++) {
806 bool is_selected = (currentStyleStr == plotStyles[n]); // You can store your selection however you want, outside or inside your objects
807 if (ImGui::Selectable(plotStyles[n], is_selected)) {
808 currentStyleStr = plotStyles[n];
809 currentStyle = (MetricsDisplayStyle)n;
810 }
811 if (is_selected) {
812 ImGui::SetItemDefaultFocus(); // You may set the initial focus when opening the combo (scrolling + for keyboard navigation support)
813 }
814 }
815 ImGui::EndCombo();
816 }
817 ImGui::PopItemWidth();
818
819 size_t gmi = 0;
820 int visibleMetrics = 0;
821 static std::vector<int> visibleDevicesIndex;
822 static std::vector<MetricIndex> visibleMetricsIndex;
823
824 visibleDevicesIndex.reserve(totalMetrics);
825 visibleDevicesIndex.clear();
826 visibleMetricsIndex.clear();
827
828 for (size_t si = 0; si < TOTAL_TYPES_OF_METRICS; ++si) {
829 auto& metricsInfos = metricsStore.metrics[si];
830 for (size_t di = 0; di < metricsInfos.size(); ++di) {
831 auto& metricInfo = metricsInfos[di];
832 bool deviceVisible = false;
833 for (size_t mi = 0; mi < metricInfo.metrics.size(); ++mi) {
834 auto& state = metricDisplayState[gmi];
835 if (state.visible) {
836 deviceVisible = true;
837 visibleMetrics++;
838 auto& metric = metricInfo.metrics[mi];
839 size_t const* timestamps = nullptr;
840 switch (metric.type) {
841 case MetricType::Int:
842 timestamps = metricInfo.intTimestamps[metric.storeIdx].data();
843 break;
844 case MetricType::Float:
845 timestamps = metricInfo.floatTimestamps[metric.storeIdx].data();
846 break;
847 case MetricType::String:
848 timestamps = metricInfo.stringTimestamps[metric.storeIdx].data();
849 break;
850 case MetricType::Uint64:
851 timestamps = metricInfo.uint64Timestamps[metric.storeIdx].data();
852 break;
853 case MetricType::Enum:
854 timestamps = metricInfo.enumTimestamps[metric.storeIdx].data();
855 break;
856 default:
857 throw std::runtime_error("Unknown metric type");
858 }
859
860 for (size_t ti = 0; ti != metricStorageSize(metric.type); ++ti) {
861 size_t minRangePos = (metric.pos + ti) % metricStorageSize(metric.type);
862 size_t curMinTime = timestamps[minRangePos];
863 if (curMinTime == 0) {
864 continue;
865 }
866 minTime = minTime < curMinTime ? minTime : curMinTime;
867 if (minTime != 0 && minTime != -1) {
868 break;
869 }
870 }
871 size_t maxRangePos = (size_t)(metric.pos) - 1 % metricStorageSize(metric.type);
872 size_t curMaxTime = timestamps[maxRangePos];
873 maxTime = std::max(maxTime, curMaxTime);
874 visibleMetricsIndex.push_back(MetricIndex{si, di, mi, gmi});
875 }
876 gmi++;
877 }
878 if (deviceVisible) {
879 visibleDevicesIndex.push_back(di);
880 }
881 }
882 }
883 if (visibleMetricsIndex.empty()) {
884 TextCenter("Please enable some metric.");
885 ImGui::End();
886 return;
887 };
888
889 switch (currentStyle) {
890 case MetricsDisplayStyle::Stems:
891 case MetricsDisplayStyle::Histos:
892 case MetricsDisplayStyle::Lines: {
893 displayDeviceMetrics("Metrics",
894 minTime, maxTime, 1024,
895 currentStyle, metricDisplayState, metricsStore, driverInfo);
896 } break;
897 case MetricsDisplayStyle::Sparks: {
898 displaySparks(state.startTime, visibleMetricsIndex, metricDisplayState, metricsStore);
899 } break;
900 case MetricsDisplayStyle::Table: {
901 static std::vector<float> visibleDevicesOffsets;
902 visibleDevicesOffsets.clear();
903 visibleDevicesOffsets.resize(visibleDevicesIndex.size());
904
905 size_t lastDevice = -1;
906 int visibleDeviceCount = -1;
908 for (auto index : visibleMetricsIndex) {
909 auto& metricsInfos = metricsStore.metrics[index.storeIndex];
910 if (lastDevice != index.deviceIndex) {
911 visibleDeviceCount++;
912 lastDevice = index.deviceIndex;
913 }
914 auto label = metricsInfos[index.deviceIndex].metricLabels[index.metricIndex].label;
915 visibleDevicesOffsets[visibleDeviceCount] += ImGui::CalcTextSize(label, nullptr, true).x;
916 }
917 // The Device name header.
918 if (ImGui::BeginTable("##metrics-table", visibleMetricsIndex.size() + 1, ImGuiTableFlags_Resizable | ImGuiTableFlags_ScrollY | ImGuiTableFlags_ScrollX, ImVec2{-1, -1})) {
919 ImGui::TableSetupColumn("##close button", ImGuiTableColumnFlags_NoResize | ImGuiTableColumnFlags_WidthFixed, 20);
920 for (int i = 0; i < visibleMetricsIndex.size(); ++i) {
921 ImGui::TableSetupColumn("##device-header", ImGuiTableColumnFlags_WidthFixed, 100);
922 }
923 ImGui::TableSetupScrollFreeze(1, 2);
924 ImGui::TableNextRow();
925 ImGui::TableNextColumn();
926 ImGui::TextUnformatted("");
927 visibleDeviceCount = -1;
928 lastDevice = -1;
929 for (auto index : visibleMetricsIndex) {
930 ImGui::TableNextColumn();
931 auto& devices = metricsStore.specs[index.storeIndex];
932 if (lastDevice == index.deviceIndex) {
933 continue;
934 }
935 visibleDeviceCount++;
936 lastDevice = index.deviceIndex;
937 auto& spec = devices[index.deviceIndex];
938
939 ImGui::TextUnformatted(spec.label.c_str());
940 }
941
942 // The metrics headers
943 ImGui::TableNextRow();
944 ImGui::TableNextColumn();
945 ImGui::TextUnformatted("#");
946 lastDevice = -1;
947 for (auto index : visibleMetricsIndex) {
948 ImGui::TableNextColumn();
949 auto& metricsInfos = metricsStore.metrics[index.storeIndex];
950 auto& metric = metricsInfos[index.deviceIndex];
951 auto label = metricsInfos[index.deviceIndex].metricLabels[index.metricIndex].label;
952 ImGui::Text("%s (%" PRIu64 ")", label, (uint64_t)metric.metrics[index.metricIndex].filledMetrics);
953 }
954
955 // Calculate which columns we want to see.
956 ImGuiListClipper clipper;
957 // For now
958 clipper.Begin(1024);
959
960 while (clipper.Step()) {
961 for (size_t i = clipper.DisplayStart; i < clipper.DisplayEnd; ++i) {
962 ImGui::TableNextRow();
963 metricsTableRow(visibleMetricsIndex, metricsStore, i);
964 }
965 }
966 ImGui::EndTable();
967 }
968 } break;
969 }
970 ImGui::End();
971}
972
974{
976 ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 4.);
977 if (info.active == false) {
978 ImGui::PushStyleColor(ImGuiCol_TitleBg, PaletteHelpers::DARK_RED);
979 ImGui::PushStyleColor(ImGuiCol_TitleBgActive, PaletteHelpers::RED);
980 ImGui::PushStyleColor(ImGuiCol_TitleBgCollapsed, PaletteHelpers::RED);
981 return;
982 }
983 switch (info.maxLogLevel) {
984 case LogLevel::Critical:
985 ImGui::PushStyleColor(ImGuiCol_TitleBg, PaletteHelpers::SHADED_RED);
986 ImGui::PushStyleColor(ImGuiCol_TitleBgActive, PaletteHelpers::RED);
987 ImGui::PushStyleColor(ImGuiCol_TitleBgCollapsed, PaletteHelpers::SHADED_RED);
988 break;
989 case LogLevel::Error:
990 ImGui::PushStyleColor(ImGuiCol_TitleBg, PaletteHelpers::SHADED_RED);
991 ImGui::PushStyleColor(ImGuiCol_TitleBgActive, PaletteHelpers::RED);
992 ImGui::PushStyleColor(ImGuiCol_TitleBgCollapsed, PaletteHelpers::SHADED_RED);
993 break;
994 case LogLevel::Alarm:
995 ImGui::PushStyleColor(ImGuiCol_TitleBg, PaletteHelpers::SHADED_YELLOW);
996 ImGui::PushStyleColor(ImGuiCol_TitleBgActive, PaletteHelpers::YELLOW);
997 ImGui::PushStyleColor(ImGuiCol_TitleBgCollapsed, PaletteHelpers::SHADED_YELLOW);
998 break;
999 case LogLevel::Warning:
1000 ImGui::PushStyleColor(ImGuiCol_TitleBg, PaletteHelpers::SHADED_YELLOW);
1001 ImGui::PushStyleColor(ImGuiCol_TitleBgActive, PaletteHelpers::YELLOW);
1002 ImGui::PushStyleColor(ImGuiCol_TitleBgCollapsed, PaletteHelpers::SHADED_YELLOW);
1003 break;
1004 case LogLevel::Info:
1005 ImGui::PushStyleColor(ImGuiCol_TitleBg, PaletteHelpers::SHADED_GREEN);
1006 ImGui::PushStyleColor(ImGuiCol_TitleBgActive, PaletteHelpers::GREEN);
1007 ImGui::PushStyleColor(ImGuiCol_TitleBgCollapsed, PaletteHelpers::SHADED_GREEN);
1008 break;
1009 default:
1010 ImGui::PushStyleColor(ImGuiCol_TitleBg, PaletteHelpers::SHADED_BLUE);
1011 ImGui::PushStyleColor(ImGuiCol_TitleBgActive, PaletteHelpers::BLUE);
1012 ImGui::PushStyleColor(ImGuiCol_TitleBgCollapsed, PaletteHelpers::SHADED_BLUE);
1013 break;
1014 }
1015}
1016
1018{
1019 ImGui::PopStyleColor(3);
1020 ImGui::PopStyleVar(1);
1021}
1022
1025void displayDriverInfo(DriverInfo const& driverInfo, DriverControl& driverControl)
1026{
1027 ImGui::Begin("Driver information");
1028 static int pid = getpid();
1029
1030 if (driverControl.state == DriverControlState::STEP) {
1031 driverControl.state = DriverControlState::PAUSE;
1032 }
1033 auto state = reinterpret_cast<int*>(&driverControl.state);
1034 ImGui::RadioButton("Play", state, static_cast<int>(DriverControlState::PLAY));
1035 ImGui::SameLine();
1036 ImGui::RadioButton("Pause", state, static_cast<int>(DriverControlState::PAUSE));
1037 ImGui::SameLine();
1038 ImGui::RadioButton("Step", state, static_cast<int>(DriverControlState::STEP));
1039
1040 ImGui::Columns();
1041
1042 ImGui::Text("PID: %d - Control port %d", pid, driverInfo.port);
1043 ImGui::Text("Frame cost (latency): %.1f(%.1f)ms", driverInfo.frameCost, driverInfo.frameLatency);
1044 ImGui::Text("State stack (depth %lu)", driverInfo.states.size());
1045 if (ImGui::Button("SIGCONT all children")) {
1046 kill(0, SIGCONT);
1047 }
1048 ImGui::SameLine();
1049 if (ImGui::Button("Debug driver")) {
1050 std::string pidStr = std::to_string(pid);
1051 setenv("O2DEBUGGEDPID", pidStr.c_str(), 1);
1052#ifdef __APPLE__
1053 std::string defaultAppleDebugCommand =
1054 "osascript -e 'tell application \"Terminal\"'"
1055 " -e 'activate'"
1056 " -e 'do script \"lldb -p \" & (system attribute \"O2DEBUGGEDPID\") & \"; exit\"'"
1057 " -e 'end tell'";
1058 setenv("O2DPLDEBUG", defaultAppleDebugCommand.c_str(), 0);
1059#else
1060 setenv("O2DPLDEBUG", "xterm -hold -e gdb attach $O2DEBUGGEDPID &", 0);
1061#endif
1062 int retVal = system(getenv("O2DPLDEBUG"));
1063 (void)retVal;
1064 }
1065
1066 ImGui::SameLine();
1067 if (ImGui::Button("Profile")) {
1068 std::string pidStr = std::to_string(pid);
1069 setenv("O2PROFILEDPID", pidStr.c_str(), 1);
1070#ifdef __APPLE__
1071 auto defaultAppleProfileCommand = fmt::format(
1072 "osascript -e 'tell application \"Terminal\"'"
1073 " -e 'activate'"
1074 " -e 'do script \"xcrun xctrace record --output dpl-profile-{0}.trace"
1075 " --instrument os_signpost --time-limit 30s --template Time\\\\ Profiler --attach {0} "
1076 " && open dpl-profile-{0}.trace && exit\"'"
1077 " -e 'end tell'",
1078 pid);
1079 std::cout << defaultAppleProfileCommand << std::endl;
1080 setenv("O2DPLPROFILE", defaultAppleProfileCommand.c_str(), 0);
1081#else
1082 setenv("O2DPLPROFILE", "xterm -hold -e perf record -a -g -p $O2PROFILEDPID > perf-$O2PROFILEDPID.data &", 0);
1083#endif
1084 int retVal = system(getenv("O2DPLPROFILE"));
1085 (void)retVal;
1086 }
1087
1088 // FIXME: this should really be a combo box
1089#ifdef __APPLE__
1090 if (ImGui::Button("Memory Profile")) {
1091 std::string pidStr = std::to_string(pid);
1092 setenv("O2PROFILEDPID", pidStr.c_str(), 1);
1093 auto defaultAppleProfileCommand = fmt::format(
1094 "osascript -e 'tell application \"Terminal\"'"
1095 " -e 'activate'"
1096 " -e 'do script \"xcrun xctrace record --output dpl-profile-{0}.trace"
1097 " --instrument os_signpost --time-limit 30s --template Allocations --attach {0} "
1098 " && open dpl-profile-{0}.trace && exit\"'"
1099 " -e 'end tell'",
1100 pid);
1101 std::cout << defaultAppleProfileCommand << std::endl;
1102 setenv("O2DPLPROFILE", defaultAppleProfileCommand.c_str(), 0);
1103 int retVal = system(getenv("O2DPLPROFILE"));
1104 (void)retVal;
1105 }
1106#endif
1107
1108 static bool selectedSignpost = false;
1109 if (ImGui::CollapsingHeader("Signposts", ImGuiTreeNodeFlags_DefaultOpen)) {
1110 if (ImGui::Checkbox("Driver", &selectedSignpost)) {
1111 if (selectedSignpost) {
1112 O2_LOG_ENABLE(driver);
1113 } else {
1114 O2_LOG_DISABLE(driver);
1115 }
1116 }
1117 }
1118
1119 for (size_t i = 0; i < driverInfo.states.size(); ++i) {
1120 ImGui::Text("#%lu: %s", i, DriverInfoHelper::stateToString(driverInfo.states[i]));
1121 }
1122
1123 ImGui::End();
1124}
1125
1126// FIXME: return empty function in case we were not built
1127// with GLFW support.
1129std::function<void(void)> getGUIDebugger(std::vector<DeviceInfo> const& infos,
1130 std::vector<DeviceSpec> const& devices,
1131 std::vector<DataProcessingStates> const& allStates,
1132 std::vector<DataProcessorInfo> const& metadata,
1133 std::vector<DeviceMetricsInfo> const& metricsInfos,
1134 DriverInfo const& driverInfo,
1135 std::vector<DeviceControl>& controls,
1136 DriverControl& driverControl)
1137{
1138 static gui::WorkspaceGUIState globalGUIState;
1139 gui::WorkspaceGUIState& guiState = globalGUIState;
1140 guiState.selectedMetric = -1;
1141 guiState.metricMaxRange = 0UL;
1142 guiState.metricMinRange = -1;
1143 // FIXME: this should probaly have a better mapping between our window state and
1144 guiState.devices.resize(infos.size());
1145 for (size_t i = 0; i < guiState.devices.size(); ++i) {
1146 gui::DeviceGUIState& state = guiState.devices[i];
1147 state.label = devices[i].id + "(" + std::to_string(infos[i].pid) + ")";
1148 }
1149 guiState.bottomPaneSize = 340;
1150 guiState.leftPaneSize = 200;
1151 guiState.rightPaneSize = 300;
1152
1153 // Show all the panes by default.
1154 guiState.bottomPaneVisible = true;
1155 guiState.leftPaneVisible = true;
1156 guiState.rightPaneVisible = true;
1157 timespec now;
1158 clock_gettime(CLOCK_REALTIME, &now);
1159 guiState.startTime = now.tv_sec - ImGui::GetTime();
1160 std::vector<TopologyNodeInfo> deviceNodesInfos;
1161 for (auto& device : devices) {
1162 deviceNodesInfos.push_back(TopologyNodeInfo{device.id});
1163 }
1164 std::vector<TopologyNodeInfo> driverNodesInfos;
1165 driverNodesInfos.push_back(TopologyNodeInfo{"driver"});
1166
1167 return [&infos, &devices, &allStates, &metadata, &controls, &metricsInfos, &driverInfo, &driverControl, deviceNodesInfos, driverNodesInfos]() {
1168 ImGuiStyle& style = ImGui::GetStyle();
1169 style.FrameRounding = 0.;
1170 style.WindowRounding = 0.;
1171 style.Colors[ImGuiCol_WindowBg] = ImVec4(0x1b / 255.f, 0x1b / 255.f, 0x1b / 255.f, 1.00f);
1172 style.Colors[ImGuiCol_ScrollbarBg] = ImVec4(0x1b / 255.f, 0x1b / 255.f, 0x1b / 255.f, 1.00f);
1173
1174 showTopologyNodeGraph(guiState, infos, devices, allStates, metadata, controls, metricsInfos);
1175
1176 AllMetricsStore metricsStore;
1177
1178 metricsStore.metrics[DEVICE_METRICS] = gsl::span(metricsInfos);
1179 metricsStore.metrics[DRIVER_METRICS] = gsl::span(&driverInfo.metrics, 1);
1180 metricsStore.specs[DEVICE_METRICS] = gsl::span(deviceNodesInfos);
1181 metricsStore.specs[DRIVER_METRICS] = gsl::span(driverNodesInfos);
1182 displayMetrics(guiState, driverInfo, infos, metadata, controls, metricsStore);
1183 displayDriverInfo(driverInfo, driverControl);
1184
1185 int windowPosStepping = (ImGui::GetIO().DisplaySize.y - 500) / guiState.devices.size();
1186
1187 for (size_t i = 0; i < guiState.devices.size(); ++i) {
1188 gui::DeviceGUIState& state = guiState.devices[i];
1189 assert(i < infos.size());
1190 assert(i < devices.size());
1191 const DeviceInfo& info = infos[i];
1192
1193 assert(controls.size() == devices.size());
1194 DeviceControl& control = controls[i];
1195
1197 if (control.logVisible) {
1198 ImGui::SetNextWindowPos(ImVec2(ImGui::GetIO().DisplaySize.x / 3 * 2, i * windowPosStepping), ImGuiCond_Once);
1199 ImGui::SetNextWindowSize(ImVec2(ImGui::GetIO().DisplaySize.x / 3, ImGui::GetIO().DisplaySize.y - 300),
1200 ImGuiCond_Once);
1201 ImGui::Begin(state.label.c_str(), &control.logVisible);
1202
1203 ImGui::InputText("Log filter", control.logFilter, sizeof(control.logFilter));
1204 ImGui::InputText("Log start trigger", control.logStartTrigger, sizeof(control.logStartTrigger));
1205 ImGui::InputText("Log stop trigger", control.logStopTrigger, sizeof(control.logStopTrigger));
1206 ImGui::Checkbox("Stop logging", &control.quiet);
1207 ImGui::SameLine();
1208 if (control.tracingFlags) {
1209 control.logLevel = LogParsingHelpers::LogLevel::Debug;
1210 }
1211 ImGui::Combo("Log level", reinterpret_cast<int*>(&control.logLevel), LogParsingHelpers::LOG_LEVELS,
1212 (int)LogParsingHelpers::LogLevel::Size, 5);
1213
1214 ImGui::Separator();
1215 ImGui::BeginChild("ScrollingRegion", ImVec2(0, -ImGui::GetTextLineHeightWithSpacing()), false,
1216 ImGuiWindowFlags_HorizontalScrollbar | ImGuiWindowFlags_NoMove);
1217 displayHistory(info, control);
1218 ImGui::EndChild();
1219 ImGui::End();
1220 }
1222 }
1223 };
1224}
1225
1226void updateMousePos(float x, float y)
1227{
1228 ImGuiIO& io = ImGui::GetIO();
1229 io.AddMousePosEvent(x, y);
1230}
1231
1232void updateMouseButton(bool clicked)
1233{
1234 ImGuiIO& io = ImGui::GetIO();
1235 io.AddMouseButtonEvent(0, clicked);
1236}
1237
1238void updateMouseWheel(int direction)
1239{
1240 ImGuiIO& io = ImGui::GetIO();
1241 if (direction > 0) {
1242 io.AddMouseWheelEvent(0, 1.0);
1243 } else {
1244 io.AddMouseWheelEvent(0, -1.0);
1245 }
1246}
1247
1248void updateWindowSize(int x, int y)
1249{
1250 ImGuiIO& io = ImGui::GetIO();
1251 io.DisplaySize = ImVec2(x, y);
1252}
1253
1254void keyEvent(char key, bool down)
1255{
1256 ImGuiIO& io = ImGui::GetIO();
1257 switch (key) {
1258 case 0:
1259 io.AddKeyEvent(ImGuiKey_Tab, down);
1260 break;
1261 case 1:
1262 io.AddKeyEvent(ImGuiKey_LeftArrow, down);
1263 break;
1264 case 2:
1265 io.AddKeyEvent(ImGuiKey_RightArrow, down);
1266 break;
1267 case 3:
1268 io.AddKeyEvent(ImGuiKey_UpArrow, down);
1269 break;
1270 case 4:
1271 io.AddKeyEvent(ImGuiKey_DownArrow, down);
1272 break;
1273 case 5:
1274 io.AddKeyEvent(ImGuiKey_PageUp, down);
1275 break;
1276 case 6:
1277 io.AddKeyEvent(ImGuiKey_PageDown, down);
1278 break;
1279 case 7:
1280 io.AddKeyEvent(ImGuiKey_Home, down);
1281 break;
1282 case 8:
1283 io.AddKeyEvent(ImGuiKey_End, down);
1284 break;
1285 case 10:
1286 io.AddKeyEvent(ImGuiKey_Delete, down);
1287 break;
1288 case 11:
1289 io.AddKeyEvent(ImGuiKey_Backspace, down);
1290 break;
1291 case 13:
1292 io.AddKeyEvent(ImGuiKey_Enter, down);
1293 break;
1294 case 14:
1295 io.AddKeyEvent(ImGuiKey_Escape, down);
1296 break;
1297 default:
1298 io.AddKeyEvent((ImGuiKey)key, down);
1299 }
1300}
1301
1302void charIn(char key)
1303{
1304 ImGuiIO& io = ImGui::GetIO();
1305 io.AddInputCharacter((unsigned short)key);
1306}
1307
1308} // namespace o2::framework::gui
benchmark::State & state
const auto bins
Definition PID.cxx:49
int16_t time
Definition RawEventData.h:4
int32_t i
int32_t retVal
uint16_t pos
Definition RawData.h:3
uint32_t j
Definition RawData.h:0
uint16_t pid
Definition RawData.h:2
#define O2_DECLARE_DYNAMIC_LOG(name)
Definition Signpost.h:483
#define O2_LOG_DISABLE(log)
Definition Signpost.h:489
#define O2_LOG_ENABLE(log)
Definition Signpost.h:488
StringRef key
GLdouble n
Definition glcorearb.h:1982
GLint GLenum GLint x
Definition glcorearb.h:403
GLsizeiptr size
Definition glcorearb.h:659
GLuint index
Definition glcorearb.h:781
GLsizei GLenum const void GLuint GLsizei GLfloat * metrics
Definition glcorearb.h:5500
GLboolean GLboolean GLboolean b
Definition glcorearb.h:1233
GLsizei const GLfloat * value
Definition glcorearb.h:819
GLboolean * data
Definition glcorearb.h:298
GLuint GLsizei const GLchar * label
Definition glcorearb.h:2519
typedef void(APIENTRYP PFNGLCULLFACEPROC)(GLenum mode)
GLboolean GLboolean GLboolean GLboolean a
Definition glcorearb.h:1233
State for the main GUI window.
void displayMetrics(gui::WorkspaceGUIState &state, DriverInfo const &driverInfo, std::vector< DeviceInfo > const &infos, std::vector< DataProcessorInfo > const &metadata, std::vector< DeviceControl > &controls, AllMetricsStore const &metricsStore)
void displayDeviceMetrics(const char *label, size_t rangeBegin, size_t rangeEnd, size_t bins, MetricsDisplayStyle displayType, std::vector< MetricDisplayState > &state, AllMetricsStore const &metricStore, DriverInfo const &driverInfo)
void showTopologyNodeGraph(WorkspaceGUIState &state, std::vector< DeviceInfo > const &infos, std::vector< DeviceSpec > const &specs, std::vector< DataProcessingStates > const &allStates, std::vector< DataProcessorInfo > const &metadata, std::vector< DeviceControl > &controls, std::vector< DeviceMetricsInfo > const &metricsInfos)
int formatSI(double value, char *buff, int size, void *user_data)
bool hasAll(const char *s, const char *q)
void updateMouseButton(bool clicked)
void keyEvent(char key, bool down)
void displayHistory(const DeviceInfo &info, DeviceControl &control)
std::function< void(void)> getGUIDebugger(std::vector< DeviceInfo > const &infos, std::vector< DeviceSpec > const &devices, std::vector< DataProcessingStates > const &allStates, std::vector< DataProcessorInfo > const &metadata, std::vector< DeviceMetricsInfo > const &metricsInfos, DriverInfo const &driverInfo, std::vector< DeviceControl > &controls, DriverControl &driverControl)
Helper to get the callback to draw the debug GUI.
void displaySparks(double startTime, std::vector< MetricIndex > &visibleMetricsIndex, std::vector< MetricDisplayState > &metricDisplayStates, AllMetricsStore const &metricStore)
void updateMousePos(float x, float y)
void updateWindowSize(int x, int y)
void updateMouseWheel(int direction)
void TextCenter(char const *text)
void displayDriverInfo(DriverInfo const &driverInfo, DriverControl &driverControl)
int formatTimeSinceStart(double value, char *buff, int size, void *user_data)
void pushWindowColorDueToStatus(const DeviceInfo &info)
ImVec4 colorForLogLevel(LogParsingHelpers::LogLevel logLevel)
void metricsTableRow(std::vector< MetricIndex > metricIndex, AllMetricsStore const &metricsStore, int row)
constexpr size_t metricStorageSize()
std::string to_string(gsl::span< T, Size > span)
Definition common.h:52
bool logVisible
wether the log window should be opened.
char logStartTrigger[MAX_USER_FILTER_SIZE]
Start printing log with the last occurence of this.
char logFilter[MAX_USER_FILTER_SIZE]
Lines in the log should match this to be displayed.
char logStopTrigger[MAX_USER_FILTER_SIZE]
Stop producing log with the first occurrence of this after the start.
int tracingFlags
What kind of events should run with the TRACE level.
LogParsingHelpers::LogLevel logLevel
Minimum log level for messages to appear.
bool quiet
wether we should be capturing device output.
std::vector< std::string > history
Definition DeviceInfo.h:56
bool active
Whether the device is active (running) or not.
Definition DeviceInfo.h:65
std::vector< LogParsingHelpers::LogLevel > historyLevel
Definition DeviceInfo.h:59
LogParsingHelpers::LogLevel maxLogLevel
The maximum log level ever seen by this device.
Definition DeviceInfo.h:47
size_t historyPos
The position inside the history circular buffer of this device.
Definition DeviceInfo.h:43
DriverControlState state
Current state of the state machine player.
LogLevel
Possible log levels for device log entries.
gsl::span< TopologyNodeInfo const > specs[TOTAL_TYPES_OF_METRICS]
gsl::span< DeviceMetricsInfo const > metrics[TOTAL_TYPES_OF_METRICS]
State for the Device specific inspector.
Information associated to a node in the topology.
std::vector< DeviceGUIState > devices
std::vector< int > row