Project
Loading...
Searching...
No Matches
DCS.h
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
15
16#ifndef TPC_DCSCalibData_H_
17#define TPC_DCSCalibData_H_
18
19#include <algorithm>
20#include <iterator>
21#include <string>
22#include <string_view>
23#include <vector>
24#include <numeric>
25#include <array>
26#include <unordered_map>
27#include <cstdlib>
28
29#include "Rtypes.h"
30
31#include "Framework/Logger.h"
32#include "DataFormatsTPC/Defs.h"
33
34using namespace o2::tpc;
35
36namespace o2::tpc::dcs
37{
38
39using DataType = float;
40
41using TimeStampType = uint64_t;
42
46template <typename T>
47struct DataPoint {
50
51 bool equalTime(const DataPoint& other) const { return time == other.time; }
52 bool operator<(const DataPoint& other) const { return time < other.time; }
53 bool operator<(const TimeStampType timeStamp) const { return time < timeStamp; }
54 DataPoint operator+(const DataPoint& other) const { return DataPoint{(time + other.time) / TimeStampType{2}, value + other.value}; }
55 DataPoint operator/(const DataType denom) const { return DataPoint{time, value / denom}; }
56
58};
59
63template <typename T>
66 uint32_t sensorNumber{};
67 std::vector<DPType> data;
68
69 void fill(const TimeStampType time, const T& value) { data.emplace_back(DPType{time, value}); }
70
71 void fill(const DPType& dataPoint) { data.emplace_back(dataPoint); }
72
73 void sort() { std::sort(data.begin(), data.end()); }
74
76 {
77 sort();
78 data.erase(
79 std::unique(data.begin(), data.end(),
80 [](const auto& dp1, const auto& dp2) {
81 return dp1.time == dp2.time;
82 }),
83 data.end());
84 data.shrink_to_fit();
85 }
86
87 void clear() { data.clear(); }
88
90 {
91 data.insert(data.end(), other.data.begin(), other.data.end());
92 }
93
97 const T& getValueForTime(const TimeStampType timeStamp) const
98 {
99 const auto i = std::upper_bound(data.begin(), data.end(), DPType{timeStamp, {}});
100 return (i == data.begin()) ? (*i).value : (*(i - 1)).value;
101 }
102
104 const T getAverageValueForTime(const TimeStampType timeStamp, const long range) const
105 {
106 return getAverageValueForTime(timeStamp, timeStamp, range);
107 }
108
110 const std::pair<T, long> getSumAndPoints(const TimeStampType from, const TimeStampType until, const long range) const
111 {
112 const auto iFrom = std::upper_bound(data.begin(), data.end(), DPType{from, {}});
113 const auto iUntil = (from == until) ? iFrom : std::upper_bound(data.begin(), data.end(), DPType{until, {}});
114 const auto distFrom = std::distance(data.begin(), iFrom);
115 const auto distUntil = std::distance(iUntil, data.end());
116 const auto nFrom = std::min(distFrom, range + 1);
117 const auto nUntil = std::min(distUntil, range);
118 const auto nPoints = std::distance(iFrom - nFrom, iUntil + nUntil);
119 return {std::accumulate(iFrom - nFrom, iUntil + nUntil, DPType{(*(iFrom - nFrom)).time, T{}}).value, nPoints};
120 }
121
122 const T getAverageValueForTime(const TimeStampType from, const TimeStampType until, const long range) const
123 {
124 const auto sumAndPoints = getSumAndPoints(from, until, range);
125 return sumAndPoints.first / static_cast<float>(sumAndPoints.second);
126 }
127
129};
130
131template <typename T>
132void doSortAndClean(std::vector<dcs::DataPointVector<T>>& dataVector)
133{
134 for (auto& data : dataVector) {
135 data.sortAndClean();
136 }
137}
138
139template <typename T>
140void doClear(std::vector<dcs::DataPointVector<T>>& dataVector)
141{
142 for (auto& data : dataVector) {
143 data.clear();
144 }
145}
146
147template <typename T>
148void doAppend(std::vector<dcs::DataPointVector<T>>& a, const std::vector<dcs::DataPointVector<T>>& b)
149{
150 if (a.size() != b.size()) {
151 LOGP(warning, "Trying to append std::vector<dcs::DataPointVector<T>>s of different size: {} != {}", a.size(), b.size());
152 }
153 for (size_t i = 0; i < a.size(); ++i) {
154 a[i].append(b[i]);
155 }
156}
157
158template <typename T>
159const T getAverageValueForTime(const std::vector<dcs::DataPointVector<T>>& dpVec, const TimeStampType from, const TimeStampType until, const long range)
160{
161 T ret{};
162 long nPoints{};
163
164 for (const auto& dps : dpVec) {
165 const auto sumAndPoints = dps.getSumAndPoints(from, until, range);
166 ret += sumAndPoints.first;
167 nPoints += sumAndPoints.second;
168 }
169 return (nPoints > 0) ? ret / static_cast<float>(nPoints) : T{};
170}
171
173// using RawDPsI = DataPointVector<int>;
174
179 struct Position {
180 float x;
181 float y;
182 };
183
184 Temperature() noexcept;
185
186 static constexpr int SensorsPerSide = 9;
187
188 static const std::unordered_map<std::string, int> SensorNameMap;
189
190 static constexpr std::array<Position, SensorsPerSide * SIDES> SensorPosition{{
191 {211.40f, 141.25f},
192 {82.70f, 227.22f},
193 {-102.40f, 232.72f},
194 {-228.03f, 112.45f},
195 {-246.96f, -60.43f},
196 {-150.34f, -205.04f},
197 {-16.63f, -253.71f},
198 {175.82f, -183.66f},
199 {252.74f, -27.68f},
200 {228.03f, 112.45f},
201 {102.40f, 232.72f},
202 {-71.15f, 244.09f},
203 {-211.40f, 141.25f},
204 {-252.74f, -27.68f},
205 {-175.82f, -183.66f},
206 {-16.63f, -253.71f},
207 {150.34f, -205.04f},
208 {252.74f, -27.68f},
209 }};
210
211 static constexpr auto& getSensorPosition(const size_t sensor) { return SensorPosition[sensor]; }
212
213 struct Stats {
214 DataType mean{};
215 DataType gradX{};
216 DataType gradY{};
217
218 Stats operator+(const Stats& other) const { return Stats{mean + other.mean, gradX + other.gradX, gradY + other.gradY}; }
219 Stats operator/(const DataType val) const { return Stats{mean / val, gradX / val, gradY / val}; }
220
222 };
224
227 std::vector<RawDPsF> raw;
228
229 const Stats& getStats(const Side s, const TimeStampType timeStamp) const
230 {
231 return (s == Side::A) ? statsA.getValueForTime(timeStamp) : statsC.getValueForTime(timeStamp);
232 }
233
235 {
236 return getAverageValueForTime(raw, 0, 9999999999999, 0);
237 }
238
239 void fill(std::string_view sensor, const TimeStampType time, const DataType temperature)
240 {
241 raw[SensorNameMap.at(sensor.data())].fill(time, temperature);
242 };
243
245 {
246 statsA.sortAndClean();
247 statsC.sortAndClean();
248 doSortAndClean(raw);
249 }
250
251 void clear()
252 {
253 doClear(raw);
254 statsA.clear();
255 statsC.clear();
256 }
257
259 {
260 statsA.append(other.statsA);
261 statsC.append(other.statsC);
262 doAppend(raw, other.raw);
263 }
264
266};
267
271struct HV {
272
273 HV()
274 noexcept;
275
276 // Exmple strings
277 // TPC_HV_A03_I_G1B_I
278 // TPC_HV_A03_O1_G1B_I
279 static constexpr size_t SidePos = 7;
280 static constexpr size_t SectorPos = 8;
281 static constexpr size_t ROCPos = 11;
282 static constexpr size_t GEMPos = 14;
283 static constexpr size_t ElectrodePos = 15;
284
285 enum class StackState : char {
286 NO_CONTROL = 2,
287 STBY_CONFIGURED = 3,
288 OFF = 4,
289 RAMPIG_DOWN = 7,
290 RAMPIG_UP = 8,
291 RAMPIG_DOWN_LOW = 9,
292 RAMPIG_UP_LOW = 10,
293 ON = 11,
294 ERROR = 13,
295 INTERMEDIATE = 14,
296 MIXED = 19,
297 INTERLOCK = 24,
298 ERROR_LOCAL = 25,
299 SOFT_INTERLOCK = 29,
300 };
301
302 static const std::unordered_map<StackState, std::string> StackStateNameMap;
303
305
306 std::vector<RawDPsF> voltages;
307 std::vector<RawDPsF> currents;
308 std::vector<RawDPsState> states;
309
310 static int getSector(std::string_view sensor)
311 {
312 const auto sideOffset = (sensor[SidePos] == 'A') ? 0 : SECTORSPERSIDE;
313 const auto sector = std::atoi(sensor.substr(SectorPos, 2).data());
314 return sector + sideOffset;
315 }
316
317 static GEMstack getStack(std::string_view sensor)
318 {
319 if (sensor[ROCPos] == 'I') {
320 return GEMstack::IROCgem;
321 }
322 const auto orocType = int(sensor[ROCPos + 1] - '0');
323 return static_cast<GEMstack>(orocType);
324 }
325
327 void fillUI(std::string_view sensor, const TimeStampType time, const DataType value)
328 {
329 const int sector = getSector(sensor);
330 const auto stack = getStack(sensor);
331 const auto rocOffset = int(stack != GEMstack::IROCgem);
332 const auto gem = int(sensor[GEMPos + rocOffset] - '0');
333 const bool isTop = sensor[ElectrodePos + rocOffset] == 'T';
334 // the counting is GEM1 top, bottom, GEM2 top, bottom, ...
335 const int electrode = 2 * (gem - 1) + !isTop;
336 const StackID stackID{sector, stack};
337 const int index = stackID.getIndex() * 2 * GEMSPERSTACK + electrode;
338
339 const auto type = sensor.back();
340 // LOGP(info, "Fill type: {}, index: {} (sec: {}, stack: {}, gem: {}, elec: {}), time: {}, value: {}", type, index, sector, stack, gem, electrode, time, value);
341 if (type == 'I') {
342 currents[index].fill(time, value);
343 } else if (type == 'U') {
344 voltages[index].fill(time, value);
345 }
346 }
347
349 void fillStatus(std::string_view sensor, const TimeStampType time, const uint32_t value)
350 {
351 const int sector = getSector(sensor);
352 const auto stack = getStack(sensor);
353 const StackID stackID{sector, stack};
354
355 // TODO: check value for validity
356 states[stackID.getIndex()].fill(time, static_cast<StackState>(value));
357 }
358
360 {
361 doSortAndClean(voltages);
362 doSortAndClean(currents);
364 }
365
366 void clear()
367 {
368 doClear(voltages);
369 doClear(currents);
371 }
372
373 void append(const HV& other)
374 {
375 doAppend(voltages, other.voltages);
376 doAppend(currents, other.currents);
377 doAppend(states, other.states);
378 }
379
381};
382
386struct Gas {
387 static constexpr size_t SensorPos = 4;
388 static constexpr size_t TypePosGC = 7;
389 static constexpr size_t TypePosAn = 15;
390
391 RawDPsF neon{};
392 RawDPsF co2{};
393 RawDPsF n2{};
394 RawDPsF argon{};
395 RawDPsF h2o{};
397 RawDPsF h2oSensor{};
398 RawDPsF o2Sensor{};
399
400 void fill(std::string_view sensor, const TimeStampType time, const DataType value)
401 {
402 if (sensor[SensorPos] == 'G') { // check if from GC
403 switch (sensor[TypePosGC]) {
404 case 'N': {
405 if (sensor[TypePosGC + 1] == 'E') {
406 neon.fill(time, value);
407 } else {
408 n2.fill(time, value);
409 }
410 break;
411 }
412 case 'A':
413 argon.fill(time, value);
414 break;
415 case 'C':
416 co2.fill(time, value);
417 break;
418 case 'O':
419 o2.fill(time, value);
420 break;
421 case 'W':
422 h2o.fill(time, value);
423 break;
424 default:
425 LOGP(warning, "Unknown gas sensor {}", sensor);
426 break;
427 }
428 } else { // otherwise dedicated sensor
429 switch (sensor[TypePosAn]) {
430 case 'H':
431 h2oSensor.fill(time, value);
432 break;
433 case 'O':
434 o2Sensor.fill(time, value);
435 break;
436 default:
437 LOGP(warning, "Unknown gas sensor {}", sensor);
438 break;
439 }
440 }
441 };
442
444 {
445 neon.sortAndClean();
446 co2.sortAndClean();
447 n2.sortAndClean();
448 argon.sortAndClean();
449 h2o.sortAndClean();
450 o2.sortAndClean();
451 h2oSensor.sortAndClean();
452 o2Sensor.sortAndClean();
453 }
454
455 void clear()
456 {
457 neon.clear();
458 co2.clear();
459 n2.clear();
460 argon.clear();
461 h2o.clear();
462 o2.clear();
463 h2oSensor.clear();
464 o2Sensor.clear();
465 }
466
467 void append(const Gas& other)
468 {
469 neon.append(other.neon);
470 co2.append(other.co2);
471 n2.append(other.n2);
472 argon.append(other.argon);
473 h2o.append(other.h2o);
474 o2.append(other.o2);
475 h2oSensor.append(other.h2oSensor);
476 o2Sensor.append(other.o2Sensor);
477 }
478
479 TimeStampType getMinTime() const;
480
481 TimeStampType getMaxTime() const;
482
484};
485
486} // namespace o2::tpc::dcs
487#endif
int16_t time
Definition RawEventData.h:4
int32_t i
uint32_t stack
Definition RawData.h:1
GLuint GLuint end
Definition glcorearb.h:469
GLenum array
Definition glcorearb.h:4274
GLuint index
Definition glcorearb.h:781
GLboolean GLboolean GLboolean b
Definition glcorearb.h:1233
GLenum GLint * range
Definition glcorearb.h:1899
GLsizei const GLfloat * value
Definition glcorearb.h:819
GLint GLint GLsizei GLint GLenum GLenum type
Definition glcorearb.h:275
GLboolean * data
Definition glcorearb.h:298
GLuint GLfloat * val
Definition glcorearb.h:1582
GLboolean GLboolean GLboolean GLboolean a
Definition glcorearb.h:1233
GLuint * states
Definition glcorearb.h:4932
void doAppend(std::vector< dcs::DataPointVector< T > > &a, const std::vector< dcs::DataPointVector< T > > &b)
Definition DCS.h:148
float DataType
Definition DCS.h:39
void doClear(std::vector< dcs::DataPointVector< T > > &dataVector)
Definition DCS.h:140
uint64_t TimeStampType
Definition DCS.h:41
const T getAverageValueForTime(const std::vector< dcs::DataPointVector< T > > &dpVec, const TimeStampType from, const TimeStampType until, const long range)
Definition DCS.h:159
void doSortAndClean(std::vector< dcs::DataPointVector< T > > &dataVector)
Definition DCS.h:132
Global TPC definitions and constants.
Definition SimTraits.h:167
GEMstack
TPC GEM stack types.
Definition Defs.h:53
constexpr unsigned char SECTORSPERSIDE
Definition Defs.h:40
Enum< T >::Iterator begin(Enum< T >)
Definition Defs.h:173
constexpr unsigned short GEMSPERSTACK
Definition Defs.h:58
constexpr unsigned char SIDES
Definition Defs.h:41
Side
TPC readout sidE.
Definition Defs.h:35
a couple of static helper functions to create timestamp values for CCDB queries or override obsolete ...
Defining DataPointCompositeObject explicitly as copiable.
GEM stack identification.
Definition Defs.h:77
std::vector< DPType > data
Definition DCS.h:67
void fill(const TimeStampType time, const T &value)
Definition DCS.h:69
void fill(const DPType &dataPoint)
Definition DCS.h:71
const T getAverageValueForTime(const TimeStampType from, const TimeStampType until, const long range) const
Definition DCS.h:122
void append(const DataPointVector< T > &other)
Definition DCS.h:89
ClassDefNV(DataPointVector, 1)
const std::pair< T, long > getSumAndPoints(const TimeStampType from, const TimeStampType until, const long range) const
calculate average value between from and unil, extending the range by +- range elements
Definition DCS.h:110
const T & getValueForTime(const TimeStampType timeStamp) const
Definition DCS.h:97
const T getAverageValueForTime(const TimeStampType timeStamp, const long range) const
calculate average value for timeStamp, extending the range by +- range elements
Definition DCS.h:104
bool equalTime(const DataPoint &other) const
Definition DCS.h:51
bool operator<(const TimeStampType timeStamp) const
Definition DCS.h:53
TimeStampType time
Definition DCS.h:48
DataPoint operator/(const DataType denom) const
Definition DCS.h:55
DataPoint operator+(const DataPoint &other) const
Definition DCS.h:54
bool operator<(const DataPoint &other) const
Definition DCS.h:52
ClassDefNV(DataPoint, 1)
void clear()
Definition DCS.h:455
void fill(std::string_view sensor, const TimeStampType time, const DataType value)
Definition DCS.h:400
void append(const Gas &other)
Definition DCS.h:467
void sortAndClean()
Definition DCS.h:443
void clear()
Definition DCS.h:366
void sortAndClean()
Definition DCS.h:359
std::vector< RawDPsF > voltages
voltages per GEM stack, counting is IROCs GEM1 top, bottom, GEM2 top, bottom, .. O1 ....
Definition DCS.h:306
static const std::unordered_map< StackState, std::string > StackStateNameMap
map state to string
Definition DCS.h:302
std::vector< RawDPsState > states
HV state per sector.
Definition DCS.h:308
std::vector< RawDPsF > currents
currents per GEM stack, counting is IROCs GEM1 top, bottom, GEM2 top, bottom, .. O1 ....
Definition DCS.h:307
static GEMstack getStack(std::string_view sensor)
Definition DCS.h:317
static int getSector(std::string_view sensor)
Definition DCS.h:310
void fillStatus(std::string_view sensor, const TimeStampType time, const uint32_t value)
Fill stack status information.
Definition DCS.h:349
void append(const HV &other)
Definition DCS.h:373
void fillUI(std::string_view sensor, const TimeStampType time, const DataType value)
Fill voltage and current information.
Definition DCS.h:327
Stats operator+(const Stats &other) const
Definition DCS.h:218
Stats operator/(const DataType val) const
Definition DCS.h:219
DataType getMeanTempRaw()
Definition DCS.h:234
StatsDPs statsA
statistics fit values per integration time A-Side
Definition DCS.h:225
ClassDefNV(Temperature, 1)
void append(const Temperature &other)
Definition DCS.h:258
StatsDPs statsC
statistics fit values per integration time C-Side
Definition DCS.h:226
static constexpr auto & getSensorPosition(const size_t sensor)
Definition DCS.h:211
std::vector< RawDPsF > raw
raw temperature values from DCS for
Definition DCS.h:227
void fill(std::string_view sensor, const TimeStampType time, const DataType temperature)
Definition DCS.h:239
const Stats & getStats(const Side s, const TimeStampType timeStamp) const
Definition DCS.h:229
VectorOfTObjectPtrs other