Project
Loading...
Searching...
No Matches
FrameworkGUIDataRelayerUsage.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.
11#include "DebugGUI/imgui.h"
12#include <functional>
18#include "InspectorHelpers.h"
19#include "PaletteHelpers.h"
21#include <cstring>
22#include <cmath>
23
24static inline ImVec2 operator+(const ImVec2& lhs, const ImVec2& rhs) { return ImVec2(lhs.x + rhs.x, lhs.y + rhs.y); }
25static inline ImVec2 operator-(const ImVec2& lhs, const ImVec2& rhs) { return ImVec2(lhs.x - rhs.x, lhs.y - rhs.y); }
26
28{
29// This is to display the information in the data relayer
31 template <typename RECORD, typename ITEM>
32 static void draw(const char* /*name*/,
33 int& v,
34 ImVec2 const& sizeHint,
35 std::function<size_t()> const& getNumInputs,
36 std::function<size_t()> const& getNumRecords,
37 std::function<RECORD(size_t)> const& getRecord,
38 std::function<size_t(RECORD const&)> const& getNumItems,
39 std::function<ITEM const*(RECORD const&, size_t)> const& getItem,
40 std::function<int(ITEM const&)> const& getValue,
41 std::function<ImU32(int value)> const& getColor,
42 std::function<void(int row, int column)> const& describeCell)
43 {
44 float padding = 1;
45 // add slider to scroll between the grid display windows
46 size_t nw = getNumRecords() / WND;
47 ImGui::PushItemWidth(sizeHint.x);
48 ImGui::SliderInt("##window", &v, 1, nw, "wnd: %d", ImGuiSliderFlags_AlwaysClamp);
49 ImVec2 sliderMin = ImGui::GetItemRectMin();
50
51 constexpr float MAX_BOX_X_SIZE = 16.f;
52 constexpr float MAX_BOX_Y_SIZE = 16.f;
53
54 ImVec2 size = ImVec2(sizeHint.x, std::min(sizeHint.y, MAX_BOX_Y_SIZE * getNumItems(0) + 2));
55 ImU32 BORDER_COLOR = ImColor(200, 200, 200, 255);
56 ImU32 BACKGROUND_COLOR = ImColor(20, 20, 20, 255);
57 ImU32 BORDER_COLOR_A = ImColor(200, 200, 200, 0);
58 ImU32 BACKGROUND_COLOR_A = ImColor(0, 0, 0, 0);
59
60 ImDrawList* drawList = ImGui::GetWindowDrawList();
61 ImVec2 winPos = sliderMin;
62
63 // overlay activity indicator on the slider
64 auto xsz = size.x / nw;
65 drawList->AddRectFilled(
66 ImVec2{0., 0.} + winPos,
67 ImVec2{size.x, size.y} + winPos,
68 BACKGROUND_COLOR_A);
69 drawList->AddRect(
70 ImVec2{0. - 1, -1} + winPos,
71 ImVec2{size.x + 1, size.y - 1} + winPos,
72 BORDER_COLOR_A);
73
74 const static auto colorA = ImColor(ImVec4{0.945, 0.096, 0.278, 0.5});
75 const static auto colorE = ImColor(ImVec4{0, 0, 0, 0});
76
77 drawList->PrimReserve(nw * 6, nw * 4);
78 for (size_t iw = 0; iw < nw; ++iw) {
79 ImVec2 xOffset{iw * xsz + 2 * padding, 0};
80 ImVec2 xSize{xsz - 2 * padding, 0};
81 ImVec2 yOffset{0, 2 * padding};
82 ImVec2 ySize{0, 16 - 4 * padding};
83 bool active = 0;
84 for (size_t ir = iw; ir < ((iw + WND > getNumRecords()) ? getNumRecords() : iw + WND); ++ir) {
85 for (size_t i = 0; i < getNumItems(ir); ++i) {
86 active = getValue(*getItem(ir, i)) > 0;
87 if (active) {
88 break;
89 }
90 }
91 }
92 drawList->PrimRect(
93 xOffset + yOffset + winPos,
94 xOffset + xSize + yOffset + ySize + winPos,
95 active ? colorA : colorE);
96 }
97
98 // display the grid
99 size_t recordsWindow = v * WND;
100 auto boxSizeX = std::min(size.x / WND, MAX_BOX_X_SIZE);
101 auto numInputs = getNumInputs();
102 winPos = ImGui::GetCursorScreenPos() + ImVec2{0, 7};
103 ImGui::InvisibleButton("sensible area", ImVec2(size.x, size.y));
104 if (ImGui::IsItemHovered()) {
105 auto pos = ImGui::GetMousePos() - winPos;
106 auto slot = (v - 1) * WND + std::lround(std::trunc(pos.x / size.x * WND));
107 auto row = std::lround(std::trunc(pos.y / size.y * numInputs));
108 describeCell(row, slot);
109 }
110
111 drawList->AddRectFilled(
112 ImVec2(0., 0.) + winPos,
113 ImVec2{size.x, size.y} + winPos,
114 BACKGROUND_COLOR);
115 drawList->AddRect(
116 ImVec2(0. - 1, -1) + winPos,
117 ImVec2{size.x + 1, size.y - 1} + winPos,
118 BORDER_COLOR);
119
120 size_t totalRects = 0;
121 for (size_t ri = (v - 1) * WND; ri < recordsWindow; ri++) {
122 auto record = getRecord(ri);
123 totalRects += getNumItems(record);
124 }
125
126 drawList->PrimReserve(totalRects * 6, totalRects * 4);
127 for (size_t ri = (v - 1) * WND; ri < recordsWindow; ri++) {
128 auto record = getRecord(ri);
129 ImVec2 xOffset{((ri - (v - 1) * WND) * boxSizeX) + padding, 0};
130 ImVec2 xSize{boxSizeX - 2 * padding, 0};
131 auto me = getNumItems(record);
132 auto boxSizeY = std::min(size.y / me, MAX_BOX_Y_SIZE);
133 for (size_t mi = 0; mi < me; mi++) {
134 ImVec2 yOffSet{0, (mi * boxSizeY) + padding};
135 ImVec2 ySize{0, boxSizeY - 2 * padding};
136
137 drawList->PrimRect(
138 xOffset + yOffSet + winPos,
139 xOffset + xSize + yOffSet + ySize + winPos,
140 getColor(getValue(*getItem(record, mi))));
141 }
142 }
143
144 ImGui::SetCursorScreenPos(winPos + size);
145 }
146};
147
148void displayDataRelayer(DeviceMetricsInfo const& /*metrics*/,
149 DeviceInfo const& /*info*/,
150 DeviceSpec const& spec,
152 ImVec2 const& size,
153 int& v)
154{
155 auto getNumInputs = [&states]() -> size_t {
156 auto& inputsView = states.statesViews[(int)ProcessingStateId::DATA_QUERIES];
157 std::string_view inputs(states.statesBuffer.data() + inputsView.first, inputsView.size);
158 if (inputs.size() == 0) {
159 return 0;
160 }
161 // count the number of semi-colon separators to get number of inputs
162 int numInputs = std::count(inputs.begin(), inputs.end(), ';');
163 return numInputs;
164 };
165 auto getNumRecords = [&states]() -> size_t {
166 auto& view = states.statesViews[(int)ProcessingStateId::DATA_RELAYER_BASE];
167 if (view.size == 0) {
168 return 0;
169 }
170 // The first number is the size of the pipeline
171 int numRecords = strtoul(states.statesBuffer.data() + view.first, nullptr, 10);
172 return numRecords;
173 };
174 auto getRecord = [](size_t i) -> int {
175 return i;
176 };
177 auto getNumItems = [&states](int record) -> int {
178 auto& view = states.statesViews[(int)ProcessingStateId::DATA_RELAYER_BASE + record];
179 if (view.size == 0) {
180 return 0;
181 }
182 char const* beginData = strchr(states.statesBuffer.data() + view.first, ' ') + 1;
183 // The number of elements is given by the size of the state, minus the header
184 int size = view.size - (beginData - (states.statesBuffer.data() + view.first));
185 return size;
186 };
187 auto getItem = [&states](int const& record, size_t i) -> int8_t const* {
188 static int8_t const zero = '0';
189 static int8_t const error = '4';
190 char const *buffer = states.statesBuffer.data();
191 auto& view = states.statesViews[(int)ProcessingStateId::DATA_RELAYER_BASE + record];
192 if (view.size == 0) {
193 return &zero;
194 }
195 char const* const beginData = strchr(buffer + view.first, ' ') + 1;
196 // Protect against buffer overflows
197 if ((size_t)view.size <= beginData - buffer + i - view.first) {
198 return &error;
199 }
200 return (int8_t const*)beginData + i; };
201 auto getValue = [](int8_t const& item) -> int { return item - '0'; };
202 auto getColor = [](int value) {
203 static const ImU32 SLOT_EMPTY = ImColor(70, 70, 70, 255);
204 static const ImU32 SLOT_FULL = ImColor(PaletteHelpers::RED);
205 static const ImU32 SLOT_DISPATCHED = ImColor(PaletteHelpers::YELLOW);
206 static const ImU32 SLOT_DONE = ImColor(PaletteHelpers::GREEN);
207 static const ImU32 SLOT_ERROR = ImColor(0xfe, 0x43, 0x65, 255);
208 switch (value) {
209 case 0:
210 return SLOT_EMPTY;
211 case 1:
212 return SLOT_FULL;
213 case 2:
214 return SLOT_DISPATCHED;
215 case 3:
216 return SLOT_DONE;
217 }
218 return SLOT_ERROR;
219 };
220 auto describeCell = [&states, &spec](int row, int slot) -> void {
221 ImGui::BeginTooltip();
222
223 // display the input (origin/descr/subspec)
224 auto& inputsView = states.statesViews[(int)ProcessingStateId::DATA_QUERIES];
225 std::string_view inputs(states.statesBuffer.data() + inputsView.first, inputsView.size);
226 auto beginInputs = inputs.begin();
227 auto endInputs = beginInputs + inputsView.size;
228 char const* input = beginInputs;
229 size_t i = 0;
230 while (input != endInputs) {
231 auto end = std::find(input, endInputs, ';');
232 if ((end - input) == 0) {
233 continue;
234 }
235 if (i == (size_t)row) {
236 ImGui::Text("%d %.*s (%s)", row, int(end - input), input, InspectorHelpers::getLifeTimeStr(spec.inputs[i].matcher.lifetime).c_str());
237 break;
238 }
239 ++i;
240 input = end + 1;
241 }
242
243 // display context variables
244 ImGui::Text("Input query matched values for slot: %d", slot);
245 auto& view = states.statesViews[(short)ProcessingStateId::CONTEXT_VARIABLES_BASE + (short)slot];
246 auto begin = view.first;
247 for (size_t vi = 0; vi < data_matcher::MAX_MATCHING_VARIABLE; ++vi) {
248 std::string_view state(states.statesBuffer.data() + begin, view.size);
249 // find the semi-colon, which separates entries in the variable list
250 auto pos = state.find(';');
251 std::string_view value = state.substr(0, pos);
252 // Do not display empty values.
253 if (value.empty()) {
254 begin += 1;
255 continue;
256 }
257 switch (vi) {
259 ImGui::Text("$%zu (startTime): %.*s", vi, (int)value.size(), value.data());
260 break;
262 ImGui::Text("$%zu (tfCounter): %.*s", vi, (int)value.size(), value.data());
263 break;
265 ImGui::Text("$%zu (firstTForbit): %.*s", vi, (int)value.size(), value.data());
266 break;
267 default:
268 ImGui::Text("$%zu: %.*s", vi, (int)value.size(), value.data());
269 }
270 begin += pos + 1;
271 }
272 ImGui::EndTooltip();
273 };
274
275 if (getNumRecords()) {
276 HeatMapHelper::draw<int, int8_t>("DataRelayer",
277 v,
278 size,
279 getNumInputs,
280 getNumRecords,
281 getRecord,
282 getNumItems,
283 getItem,
284 getValue,
285 getColor,
286 describeCell);
287 }
288}
289
290} // namespace o2::framework::gui
benchmark::State & state
uint16_t padding
int32_t i
uint16_t pos
Definition RawData.h:3
GLuint buffer
Definition glcorearb.h:655
GLsizeiptr size
Definition glcorearb.h:659
GLuint GLuint end
Definition glcorearb.h:469
const GLdouble * v
Definition glcorearb.h:832
GLsizei const GLfloat * value
Definition glcorearb.h:819
GLuint * states
Definition glcorearb.h:4932
@ FIRSTTFORBIT_POS
The DataHeader::tfCounter associated to the timeslice.
@ TFCOUNTER_POS
The DataProcessingHeader::startTime associated to the timeslice.
State for the main GUI window.
void displayDataRelayer(DeviceMetricsInfo const &, DeviceInfo const &, DeviceSpec const &spec, DataProcessingStates const &states, ImVec2 const &size, int &v)
View of the DataRelayer metrics for a given DeviceInfo.
D const SVectorGPU< T, D > & rhs
Definition SMatrixGPU.h:191
Vertex< T > operator-(const Vertex< T > &a, const Vertex< T > &b)
Definition Vertex.h:98
BinCenterView< AxisIterator > operator+(BinCenterView< AxisIterator > lhs, int n)
double getValue(DPVAL dp)
std::vector< InputRoute > inputs
Definition DeviceSpec.h:62
static const std::string getLifeTimeStr(Lifetime lifetime)
static void draw(const char *, int &v, ImVec2 const &sizeHint, std::function< size_t()> const &getNumInputs, std::function< size_t()> const &getNumRecords, std::function< RECORD(size_t)> const &getRecord, std::function< size_t(RECORD const &)> const &getNumItems, std::function< ITEM const *(RECORD const &, size_t)> const &getItem, std::function< int(ITEM const &)> const &getValue, std::function< ImU32(int value)> const &getColor, std::function< void(int row, int column)> const &describeCell)
o2::InteractionRecord ir(0, 0)
std::vector< int > row