15#include <boost/tokenizer.hpp>
16#include <boost/lexical_cast.hpp>
26#include <fairlogger/Logger.h>
30RawReader::RawReader(
int region,
int link,
int run,
int sampaVersion)
31 : mUseRawInMode3(true),
32 mApplyChannelMask(false),
33 mCheckAdcClock(false),
37 mSampaVersion(sampaVersion),
39 mTimestampOfFirstData({0, 0, 0, 0, 0}),
42 mDataIterator(mData.end()),
44 mChannelMask(
nullptr),
45 mAdcError(std::make_shared<std::vector<std::tuple<short, short, short>>>()),
46 mEventSynchronizer(std::make_shared<RawReaderEventSync>())
54 for (
const auto&
f : *infiles) {
63 typedef boost::tokenizer<boost::char_separator<char>> tokenizer;
64 boost::char_separator<char> sep{
":"};
65 tokenizer tok{infile, sep};
66 if (std::distance(tok.begin(), tok.end()) < 3) {
67 LOG(error) <<
"Not enough arguments";
71 tokenizer::iterator it1 = tok.begin();
72 std::string
path = boost::lexical_cast<std::string>(*it1);
73 int run = std::stoi(
path.substr(
path.find(
"run") + 3, 6));
78 region = boost::lexical_cast<int>(*it1);
79 }
catch (boost::bad_lexical_cast) {
80 LOG(error) <<
"Please enter a region for " <<
path;
86 link = boost::lexical_cast<int>(*it1);
87 }
catch (boost::bad_lexical_cast) {
88 LOG(error) <<
"Please enter a link number for " <<
path;
92 int sampaVersion = -1;
93 if (++it1 == tok.end()) {
94 LOG(
debug) <<
"No SAMPA version was entered, assuming latest.";
97 sampaVersion = boost::lexical_cast<int>(*it1);
98 }
catch (boost::bad_lexical_cast) {
99 LOG(error) <<
"Please enter a valid SAMPA version for " <<
path;
118 if (mSampaVersion == -1) {
119 mSampaVersion = sampaVersion;
123 LOG(
debug) <<
"Run of RawReader is " << mRun <<
" and not " << run;
127 if (region != mRegion) {
128 LOG(
debug) <<
"Region of RawReader is " << mRegion <<
" and not " << region;
133 LOG(
debug) <<
"Link of RawReader is " << mLink <<
" and not " << link;
137 if (sampaVersion != mSampaVersion) {
138 LOG(
debug) <<
"SAMPA Version of RawReader is " << mSampaVersion <<
" and not " << sampaVersion;
143 if (!
file.is_open()) {
144 LOG(error) <<
"Can't read file " <<
path;
155 file.read((
char*)&
h,
sizeof(
h));
156 if (
h.reserved_01 != 0x0F ||
h.reserved_2() != 0x3fec2fec1fec0fec) {
157 LOG(error) <<
"Header does not look consistent";
163 if (
h.headerVersion == 1) {
165 auto ins = mEvents.insert(std::make_pair(
h.eventCount(), std::make_shared<std::vector<EventInfo>>()));
166 ins.first->second->push_back(eD);
168 if (
h.eventCount() == 0) {
169 LOG(
debug) <<
"Processing event 0 to find sync pattern for event synchronization";
174 mEventSynchronizer->addEventOffset(
176 (
h.timeStamp() - mTimestampOfFirstData[0]));
181 LOG(error) <<
"Header version " << (
int)
h.headerVersion <<
" not implemented.";
196 LOG(
debug) <<
"First event was not loaded (maybe important for decoding RAW GBT frames) loading first event.";
198 mTimestampOfFirstData.fill(0);
204 auto ev = mEvents.find(
event);
206 if (ev == mEvents.end()) {
211 for (
auto& eventInfo : *(ev->second)) {
212 switch (eventInfo.header.dataType) {
215 LOG(
debug) <<
"Data of readout mode 1 (RAW GBT frames)";
216 if (!decodeRawGBTFrames(eventInfo)) {
223 LOG(
debug) <<
"Data of readout mode 2 (decoded data)";
224 if (!decodePreprocessedData(eventInfo)) {
231 if (mUseRawInMode3) {
232 LOG(
debug) <<
"Data of readout mode 3 (decoding RAW GBT frames)";
233 if (!decodeRawGBTFrames(eventInfo)) {
237 LOG(
debug) <<
"Data of readout mode 3 (using decoded data)";
238 if (!decodePreprocessedData(eventInfo)) {
245 LOG(
debug) <<
"Readout mode not known";
251 mDataIterator = mData.begin();
257bool RawReader::decodePreprocessedData(EventInfo eventInfo)
259 std::ifstream
file(eventInfo.path);
260 if (!
file.is_open()) {
261 LOG(error) <<
"Can't read file " << eventInfo.path;
265 int nWords = eventInfo.header.nWords - 8;
266 uint32_t words[nWords];
267 LOG(
debug) <<
"reading " << nWords <<
" words from position " << eventInfo.posInFile <<
" in file " << eventInfo.path;
268 file.seekg(eventInfo.posInFile);
269 file.read((
char*)&words, nWords *
sizeof(words[0]));
273 std::array<char, 5> sampas;
274 std::array<char, 5> sampaChannelStart;
275 for (
char i = 0;
i < 5; ++
i) {
276 sampas[
i] = (
i == 4) ? 2 : (mRegion % 2) ?
i / 2 + 3 :
i / 2;
277 sampaChannelStart[
i] = (
i == 4) ?
278 ((mRegion % 2) ? 16 : 0)
283 std::array<uint32_t, 5>
ids;
284 std::array<bool, 5> writeValue;
285 writeValue.fill(
false);
286 std::array<std::array<uint16_t, 16>, 5> adcValues;
288 int indexStep = (eventInfo.header.dataType == 3) ? 8 : 4;
289 int offset = (eventInfo.header.dataType == 3) ? 4 : 0;
292 if (eventInfo.header.eventCount() > 0) {
293 LOG(
debug) <<
"Using offset [" << mTimestampOfFirstData[0] + mEventSynchronizer->getEventOffset(eventInfo.header.eventCount()) <<
"] "
294 <<
"instead of [" << eventInfo.header.timeStamp() <<
"] "
295 <<
" (diff = " << mTimestampOfFirstData[0] + mEventSynchronizer->getEventOffset(eventInfo.header.eventCount()) - eventInfo.header.timeStamp() <<
") ";
297 i_start = mTimestampOfFirstData[0] + mEventSynchronizer->getEventOffset(eventInfo.header.eventCount()) - eventInfo.header.timeStamp() - 1;
298 i_start *= indexStep;
301 LOG(
debug) <<
"Start index for Event " << eventInfo.header.eventCount() <<
" is " << i_start;
303 for (
long i = i_start;
i < nWords;
i =
i + indexStep) {
310 adcValues[4][((
ids[4] & 0x7) * 2) + 1] = (((
ids[4] >> 3) & 0x1) == 0) ? 0 : words[
i +
offset + 3] & 0x3FF;
311 adcValues[4][((
ids[4] & 0x7) * 2)] = (((
ids[4] >> 3) & 0x1) == 0) ? 0 : (words[
i +
offset + 3] >> 10) & 0x3FF;
312 adcValues[3][((
ids[3] & 0x7) * 2) + 1] = (((
ids[3] >> 3) & 0x1) == 0) ? 0 : (words[
i +
offset + 3] >> 20) & 0x3FF;
313 adcValues[3][((
ids[3] & 0x7) * 2)] = (((
ids[3] >> 3) & 0x1) == 0) ? 0 : ((words[
i +
offset + 2] & 0xFF) << 2) | ((words[
i +
offset + 3] >> 30) & 0x3);
314 adcValues[2][((
ids[2] & 0x7) * 2) + 1] = (((
ids[2] >> 3) & 0x1) == 0) ? 0 : (words[
i +
offset + 2] >> 8) & 0x3FF;
315 adcValues[2][((
ids[2] & 0x7) * 2)] = (((
ids[2] >> 3) & 0x1) == 0) ? 0 : (words[
i +
offset + 2] >> 18) & 0x3FF;
316 adcValues[1][((
ids[1] & 0x7) * 2) + 1] = (((
ids[1] >> 3) & 0x1) == 0) ? 0 : ((words[
i +
offset + 1] & 0x3F) << 4) | ((words[
i +
offset + 2] >> 28) & 0xF);
317 adcValues[1][((
ids[1] & 0x7) * 2)] = (((
ids[1] >> 3) & 0x1) == 0) ? 0 : (words[
i +
offset + 1] >> 6) & 0x3FF;
318 adcValues[0][((
ids[0] & 0x7) * 2) + 1] = (((
ids[0] >> 3) & 0x1) == 0) ? 0 : (words[
i +
offset + 1] >> 16) & 0x3FF;
319 adcValues[0][((
ids[0] & 0x7) * 2)] = (((
ids[0] >> 3) & 0x1) == 0) ? 0 : ((words[
i +
offset + 0] & 0xF) << 6) | ((words[
i +
offset + 1] >> 26) & 0x3F);
321 for (
char j = 0;
j < 5; ++
j) {
323 writeValue[
j] =
true;
325 if (mTimestampOfFirstData[
j] == 0) {
326 mTimestampOfFirstData[
j] = eventInfo.header.timeStamp() + 1 +
i / indexStep;
331 for (
char j = 0;
j < 5; ++
j) {
332 if (writeValue[
j] & (
ids[
j] == 0xF)) {
333 for (
int k = 0; k < 16; ++k) {
335 mRegion, mLink, sampas[
j], k + sampaChannelStart[
j]);
337 if (mApplyChannelMask &&
338 (mChannelMask !=
nullptr) &&
339 !mChannelMask->getValue(
CRU(mRegion), padPos.getPad(), padPos.getRow())) {
344 auto ins = mData.insert(std::make_pair(padPos, std::make_shared<std::vector<uint16_t>>()));
345 ins.first->second->push_back(adcValues[
j][k]);
353bool RawReader::decodeRawGBTFrames(EventInfo eventInfo)
355 std::ifstream
file(eventInfo.path);
356 if (!
file.is_open()) {
357 LOG(error) <<
"Can't read file " << eventInfo.path;
361 int nWords = eventInfo.header.nWords - 8;
362 uint32_t words[nWords];
363 LOG(
debug) <<
"reading " << nWords <<
" words from position " << eventInfo.posInFile <<
" in file " << eventInfo.path;
364 LOG(
debug) <<
"Header time stamp is " << eventInfo.header.timeStamp();
365 file.seekg(eventInfo.posInFile);
366 file.read((
char*)&words, nWords *
sizeof(words[0]));
370 std::array<char, 5> sampas;
371 std::array<char, 5> sampaChannelStart;
372 for (
char i = 0;
i < 5; ++
i) {
373 sampas[
i] = (
i == 4) ? 2 : (mRegion % 2) ?
i / 2 + 3 :
i / 2;
374 sampaChannelStart[
i] = (
i == 4) ?
375 ((mRegion % 2) ? 16 : 0)
380 std::array<SyncPatternMonitor, 5> syncMon{
386 std::array<AdcClockMonitor, 3> adcClockMon{
390 std::array<bool, 3> adcClockFound{
false,
false,
false};
392 std::array<short, 5> lastSyncPos;
393 std::array<std::list<uint16_t>, 5> adcValues;
395 uint64_t timebin = 0;
397 int indexStep = (eventInfo.header.dataType == 3) ? 8 : 4;
400 if (eventInfo.header.eventCount() > 0) {
401 LOG(
debug) <<
"Using offset [" << mTimestampOfFirstData[0] + mEventSynchronizer->getEventOffset(eventInfo.header.eventCount()) <<
"] "
402 <<
"instead of [" << eventInfo.header.timeStamp() <<
"] "
403 <<
" (diff = " << mTimestampOfFirstData[0] + mEventSynchronizer->getEventOffset(eventInfo.header.eventCount()) - eventInfo.header.timeStamp() <<
") ";
405 i_start = mTimestampOfFirstData[0] + mEventSynchronizer->getEventOffset(eventInfo.header.eventCount()) - eventInfo.header.timeStamp() - 1;
406 i_start *= indexStep;
409 LOG(
debug) <<
"Start index for Event " << eventInfo.header.eventCount() <<
" is " << i_start;
412 if (eventInfo.header.eventCount() > 0) {
413 frame.
setData(words[i_start - indexStep], words[i_start - indexStep + 1], words[i_start - indexStep + 2], words[i_start - indexStep + 3]);
417 for (
long i = i_start;
i < nWords;
i =
i + indexStep) {
419 lastSyncPos = mSyncPos;
421 frame.
setData(words[
i], words[
i + 1], words[
i + 2], words[
i + 3]);
423 if (syncMon[0].addSequence(frame.
getHalfWord(0, 0, 0), frame.
getHalfWord(0, 1, 0), frame.
getHalfWord(0, 2, 0), frame.
getHalfWord(0, 3, 0))) {
424 mSyncPos[0] = syncMon[0].getPosition();
426 if (syncMon[1].addSequence(frame.
getHalfWord(0, 0, 1), frame.
getHalfWord(0, 1, 1), frame.
getHalfWord(0, 2, 1), frame.
getHalfWord(0, 3, 1))) {
427 mSyncPos[1] = syncMon[1].getPosition();
429 if (syncMon[2].addSequence(frame.
getHalfWord(1, 0, 0), frame.
getHalfWord(1, 1, 0), frame.
getHalfWord(1, 2, 0), frame.
getHalfWord(1, 3, 0))) {
430 mSyncPos[2] = syncMon[2].getPosition();
432 if (syncMon[3].addSequence(frame.
getHalfWord(1, 0, 1), frame.
getHalfWord(1, 1, 1), frame.
getHalfWord(1, 2, 1), frame.
getHalfWord(1, 3, 1))) {
433 mSyncPos[3] = syncMon[3].getPosition();
436 mSyncPos[4] = syncMon[4].getPosition();
439 if (mCheckAdcClock) {
441 bool adcCheckErr0 = adcClockMon[0].addSequence(frame.
getAdcClock(0));
442 bool adcCheckErr1 = adcClockMon[1].addSequence(frame.
getAdcClock(1));
443 bool adcCheckErr2 = adcClockMon[2].addSequence(frame.
getAdcClock(2));
445 if (mSyncPos[0] >= 0) {
446 adcClockFound[0] = adcClockFound[0] | (!adcCheckErr0);
448 if (mSyncPos[2] >= 0) {
449 adcClockFound[1] = adcClockFound[1] | (!adcCheckErr1);
451 if (mSyncPos[4] >= 0) {
452 adcClockFound[2] = adcClockFound[2] | (!adcCheckErr2);
454 if (adcClockFound[0] & adcCheckErr0) {
455 adcClockFound[0] =
false;
456 LOG(
debug) <<
"ADC clock error of SAMPA " << ((mRegion % 2) ? 3 : 0) <<
" in frame [" <<
i / indexStep <<
"]";
457 mAdcError->emplace_back(std::make_tuple((mRegion % 2) ? 3 : 0,
i, mSyncPos[0]));
459 if (adcClockFound[1] & adcCheckErr1) {
460 adcClockFound[1] =
false;
461 LOG(
debug) <<
"ADC clock error of SAMPA " << ((mRegion % 2) ? 4 : 1) <<
" in frame [" <<
i / indexStep <<
"]";
462 mAdcError->emplace_back(std::make_tuple((mRegion % 2) ? 4 : 1,
i, mSyncPos[2]));
464 if (adcClockFound[2] & adcCheckErr2) {
465 adcClockFound[2] =
false;
466 LOG(
debug) <<
"ADC clock error of SAMPA " << 2 <<
" in frame [" <<
i / indexStep <<
"]";
467 mAdcError->emplace_back(std::make_tuple(2,
i, mSyncPos[4]));
473 for (
short iHalfSampa = 0; iHalfSampa < 5; ++iHalfSampa) {
474 if (mSyncPos[iHalfSampa] < 0) {
475 LOG(debug1) <<
"Sync pattern for half SAMPA " << iHalfSampa <<
" not yet found";
478 if (lastSyncPos[iHalfSampa] < 0) {
479 LOG(debug1) <<
"Sync pattern for half SAMPA " << iHalfSampa <<
" not yet found";
480 LOG(debug1) << lastFrame;
483 if (mTimestampOfFirstData[iHalfSampa] == 0) {
484 if (iHalfSampa == 0) {
485 LOG(
debug) <<
"Setting timestamp of first data to " << eventInfo.header.timeStamp() + 1 +
i / indexStep <<
" for half SAMPA " << iHalfSampa <<
" (" << 1 +
i / indexStep <<
")";
487 mTimestampOfFirstData[iHalfSampa] = eventInfo.header.timeStamp() + 1 +
i / indexStep;
490 switch (mSyncPos[iHalfSampa]) {
492 value1 = (frame.
getHalfWord(iHalfSampa / 2, 1, iHalfSampa % 2) << 5) | frame.
getHalfWord(iHalfSampa / 2, 0, iHalfSampa % 2);
493 value2 = (frame.
getHalfWord(iHalfSampa / 2, 3, iHalfSampa % 2) << 5) | frame.
getHalfWord(iHalfSampa / 2, 2, iHalfSampa % 2);
497 value1 = (lastFrame.
getHalfWord(iHalfSampa / 2, 2, iHalfSampa % 2) << 5) | lastFrame.
getHalfWord(iHalfSampa / 2, 1, iHalfSampa % 2);
498 value2 = (frame.
getHalfWord(iHalfSampa / 2, 0, iHalfSampa % 2) << 5) | lastFrame.
getHalfWord(iHalfSampa / 2, 3, iHalfSampa % 2);
502 value1 = (lastFrame.
getHalfWord(iHalfSampa / 2, 3, iHalfSampa % 2) << 5) | lastFrame.
getHalfWord(iHalfSampa / 2, 2, iHalfSampa % 2);
503 value2 = (frame.
getHalfWord(iHalfSampa / 2, 1, iHalfSampa % 2) << 5) | frame.
getHalfWord(iHalfSampa / 2, 0, iHalfSampa % 2);
507 value1 = (frame.
getHalfWord(iHalfSampa / 2, 0, iHalfSampa % 2) << 5) | lastFrame.
getHalfWord(iHalfSampa / 2, 3, iHalfSampa % 2);
508 value2 = (frame.
getHalfWord(iHalfSampa / 2, 2, iHalfSampa % 2) << 5) | frame.
getHalfWord(iHalfSampa / 2, 1, iHalfSampa % 2);
515 if (mSampaVersion == 1 || mSampaVersion == 2) {
516 adcValues[iHalfSampa].emplace_back(value1 ^ (1 << 9));
517 adcValues[iHalfSampa].emplace_back(value2 ^ (1 << 9));
519 adcValues[iHalfSampa].emplace_back(value1);
520 adcValues[iHalfSampa].emplace_back(value2);
524 for (
char j = 0;
j < 5; ++
j) {
525 if (adcValues[
j].
size() == 16) {
526 for (
int k = 0; k < 16; ++k) {
528 mRegion, mLink, sampas[
j], k + sampaChannelStart[
j]);
530 if (mApplyChannelMask &&
531 (mChannelMask !=
nullptr) &&
532 !mChannelMask->getValue(
CRU(mRegion), padPos.getPad(), padPos.getRow())) {
533 adcValues[
j].pop_front();
538 auto ins = mData.insert(std::make_pair(padPos, std::make_shared<std::vector<uint16_t>>()));
539 ins.first->second->push_back(adcValues[
j].front());
540 adcValues[
j].pop_front();
Class to monitor the ADC smapling clock contained in the GBT frame.
Class to monitor the data stream and detect synchronization patterns.
Class for time synchronization of RawReader instances.
Class to monitor the ADC smapling clock contained in the GBT frame.
GBTFrame class for the TPC.
short getAdcClock(short sampa) const
void setData(unsigned word3, unsigned word2, unsigned word1, unsigned word0)
short getHalfWord(const short sampa, const short halfword, const short chan=0) const
const PartitionInfo & getPartitionInfo(const unsigned char partition) const
const PadPos padPosRegion(const int cruNumber, const int fecInPartition, const int sampaOnFEC, const int channelOnSAMPA) const
static Mapper & instance(const std::string mappingDir="")
unsigned char getNumberOfFECs() const
int64_t getFirstEvent() const
bool addInputFile(std::string infile)
bool loadEvent(int64_t event)
Class to monitor the data stream and detect synchronization patterns.
GLuint GLsizei GLsizei * length
GLsizei const GLchar *const * path
Global TPC definitions and constants.
int posInFile
Position in data file.
Header header
Header of this evend.
std::string path
Path to data file.
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"