10 #include <arich/modules/arichUnpacker/ARICHUnpackerModule.h>
12 #include <arich/modules/arichUnpacker/ARICHRawDataHeader.h>
14 #include <framework/core/ModuleManager.h>
17 #include <framework/datastore/StoreArray.h>
18 #include <framework/datastore/StoreObjPtr.h>
21 #include <framework/logging/Logger.h>
24 #include <framework/dataobjects/EventMetaData.h>
25 #include <rawdata/dataobjects/RawARICH.h>
26 #include <arich/dataobjects/ARICHDigit.h>
27 #include <arich/dataobjects/ARICHInfo.h>
28 #include <arich/dataobjects/ARICHRawDigit.h>
56 setDescription(
"Raw data unpacker for ARICH");
57 setPropertyFlags(c_ParallelProcessingCertified);
59 addParam(
"bitMask", m_bitMask,
"hit bit mask (8 bits/channel, only used for unsuppresed format!)", (uint8_t)0xFF);
60 addParam(
"debug", m_debug,
"prints debug information", 0);
62 addParam(
"inputRawDataName", m_inputRawDataName,
"name of RawARICH store array",
string(
""));
63 addParam(
"outputDigitsName", m_outputDigitsName,
"name of ARICHDigit store array",
string(
""));
64 addParam(
"outputRawDigitsName", m_outputRawDigitsName,
"name of ARICHRawDigit store array",
string(
""));
65 addParam(
"outputarichinfoName", m_outputarichinfoName,
"name of ARICHInfo store array",
string(
""));
66 addParam(
"RawUnpackerMode", m_rawmode,
"Activate RawUnpacker mode", 0);
67 addParam(
"DisableUnpackerMode", m_disable_unpacker,
"Disable Regular Unpacker mode", 0);
71 ARICHUnpackerModule::~ARICHUnpackerModule()
75 void ARICHUnpackerModule::initialize()
82 digits.registerInDataStore();
92 void ARICHUnpackerModule::beginRun()
96 void ARICHUnpackerModule::event()
108 double vth_thscan = 0.0;
111 std::cout << std::endl <<
"------------------------" << std::endl;
112 std::cout <<
"Run: " << evtMetaData->getRun() <<
" Event: " << evtMetaData->getEvent() << std::endl;
113 std::cout <<
"------------------------" << std::endl << std::endl;
116 unsigned thscan_mode = 0;
120 for (
auto& raw : rawData) {
121 for (
int finesse = 0; finesse < 4; finesse++) {
122 const int* buffer = raw.GetDetectorBuffer(0, finesse);
123 int bufferSize = raw.GetDetectorNwords(0, finesse);
129 trgtype = raw.GetTRGType(0);
135 readHeader(buffer, ibyte, head);
137 if (m_debug > 1) printBits(buffer, bufferSize);
140 std::cout <<
"Merger header" << std::endl;
144 int type = (int)head.type;
145 int ver = (
int)head.version;
146 int boardid = (int)head.mergerID;
147 int febno = (
int)head.FEBSlot;
148 unsigned int length_all = (
unsigned int)head.length;
149 unsigned int mrg_evtno = (
unsigned int)head.trigger;
151 rawdigit->setCopperId(raw.GetNodeID(0));
152 rawdigit->setHslbId(finesse);
157 unsigned begin = ibyte;
159 while (ibyte < head.length) {
163 readFEHeader(buffer, ibyte, febHead);
164 if (febHead.thscan_mode) {thscan_mode++;}
165 if (m_debug) febHead.print();
167 if ( febHead.version != head.version || febHead.mergerID != head.mergerID
168 || febHead.trigger != head.trigger) {
169 B2ERROR(
"ARICHUnpackerModule: data in FEB header not consistent with data in merger HEADER " <<
LogVar(
"FEB ID",
170 (
unsigned)febHead.FEBSlot) <<
171 LogVar(
"merger ID", (
unsigned)head.mergerID));
break;
175 ibyte += ARICHFEB_HEADER_SIZE;
176 int dataLen = febHead.length - ARICHFEB_HEADER_SIZE;
178 febHead.FEBSlot += 1;
180 unsigned mergID = m_mergerMap->getMergerIDfromSN((
unsigned)head.mergerID);
182 if (mergID == 99) { B2ERROR(
"ARICHUnpackerModule: unknown merger number. Merger data will be skipped. " <<
LogVar(
"merger ID", mergID) <<
LogVar(
"Serial Number", (
unsigned)head.mergerID));
break;}
184 unsigned moduleID = m_mergerMap->getModuleID(mergID, (
unsigned)febHead.FEBSlot);
186 if (!moduleID) { B2ERROR(
"ARICHUnpackerModule: no merger to FEB mapping. Merger data will be skipped. " <<
LogVar(
"merger ID", mergID) <<
LogVar(
"Serial Number", (
unsigned)head.mergerID) <<
LogVar(
"FEB slot", (
unsigned)febHead.FEBSlot));
break;}
189 if (m_debug) std::cout <<
"Hit channels: " << std::endl;
190 if (febHead.type == 1) {
191 for (
int i = 0; i < dataLen / 2; i++) {
192 int shift = (3 - ibyte % 4) * 8;
193 uint8_t asicCh = buffer[ibyte / 4] >> shift;
195 shift = (3 - ibyte % 4) * 8;
196 uint8_t hitBitSet = buffer[ibyte / 4] >> shift;
197 if (m_debug && hitBitSet) std::cout <<
"ch: " << (unsigned)asicCh <<
" " << std::bitset<8>(hitBitSet) << std::endl;
199 digits.appendNew(moduleID, (
unsigned)asicCh, hitBitSet);
202 }
else if (febHead.type == 2) {
203 unsigned asicCh = 143;
204 for (
int i = 0; i < dataLen; i++) {
205 int shift = (3 - ibyte % 4) * 8;
206 uint8_t hitBitSet = buffer[ibyte / 4] >> shift;
208 if (hitBitSet & m_bitMask) {
209 digits.appendNew(moduleID, asicCh, hitBitSet);
214 }
else B2ERROR(
"ARICHUnpackerModule: Unknown data type" <<
LogVar(
"type", febHead.type));
218 if (ceil(ibyte / 4.) != (
unsigned)bufferSize)
219 B2WARNING(
"ARICHUnpackerModule: data buffer size mismatch " <<
LogVar(
"size from copper", bufferSize) <<
LogVar(
"size from merger",
226 if (thscan_mode == 0 && m_rawmode == 0)
continue;
230 while (m_ibyte < length_all) {
232 readFEHeader(buffer, m_ibyte, febHead);
233 int type_feb = febHead.type;
234 ver = febHead.version;
235 boardid = febHead.mergerID;
236 febno = febHead.FEBSlot;
238 vth_thscan = (febHead.vth * 0.0024) - 1.27;
239 unsigned int length = febHead.length;
240 int evtno = febHead.trigger;
241 unsigned int jbyte = 0;
242 std::stringstream ss;
243 ss <<
"type=" << type_feb <<
", ver=" << ver <<
" "
244 <<
", boardid=" << boardid <<
", febno=" << febno
245 <<
", length=" << length <<
", evtno=" << evtno <<
" ";
247 long long feb_trigno = 0;
248 for (
int i = 0; i < 10; i++) {
249 int val = calbyte(buffer);
252 feb_trigno |= (0xff & val) << (5 - i) * 8;
257 if (type_feb == 0x02) {
260 while (jbyte < length) {
261 int val = calbyte(buffer);
264 ss <<
"ch# " << ch <<
"(" << val <<
") ";
266 if (febno < 0 || febno > 6) {
267 B2ERROR(
"FEB is bad : " <<
LogVar(
"FEB no.", febno) <<
LogVar(
"hslb", finesse) <<
LogVar(
"type", type_feb) <<
LogVar(
"ver",
268 ver) <<
LogVar(
"boardid", boardid) <<
LogVar(
"febno", febno) <<
LogVar(
"length", length) <<
LogVar(
"evtno", evtno));
270 feb.push_back(ch, val);
275 }
else if (type_feb == 0x01) {
280 while (jbyte < length) {
281 int ch = calbyte(buffer);
283 int val = calbyte(buffer);
286 ss <<
"ch# " << ch <<
"(" << val <<
") ";
288 if (febno < 0 || febno > 6) {
289 B2ERROR(
"FEB is bad : " <<
LogVar(
"FEB no.", febno) <<
LogVar(
"hslb", finesse) <<
LogVar(
"type", type_feb) <<
LogVar(
"ver",
290 ver) <<
LogVar(
"boardid", boardid) <<
LogVar(
"febno", febno) <<
LogVar(
"length", length) <<
LogVar(
"evtno", evtno));
293 feb.push_back(ch, val);
297 rawdigit->addFEB(feb, type, ver, boardid, febno, length, evtno, feb_trigno);
298 if (m_debug && hasHit) {
408 arichinfo->settrgtype(trgtype);
409 if (vth_thscan > -1.27) { arichinfo->setvth_thscan(vth_thscan); }
410 arichinfo->setntrack(0);
411 arichinfo->setnexthit(0);
412 arichinfo->setnhit(0);
413 if (thscan_mode > 0 || m_rawmode != 0)
414 { arichinfo->setthscan_mode(
true); }
416 { arichinfo->setthscan_mode(
false); }
420 void ARICHUnpackerModule::readHeader(
const int* buffer,
unsigned& ibyte,
ARICHRawHeader& head)
426 for (
int i = 0; i < 4; i++) {
427 shift = (3 - ibyte % 4) * 8;
428 line1[3 - i] = buffer[ibyte / 4] >> shift;
432 head.type = line1[3];
433 head.version = line1[2];
434 head.mergerID = line1[1];
435 head.FEBSlot = line1[0];
438 unsigned char len[4];
439 for (
int i = 0; i < 4; i++) {
440 shift = (3 - ibyte % 4) * 8;
441 len[3 - i] = buffer[ibyte / 4] >> shift;
445 unsigned seu = len[2];
449 uint32_t* tmp = (uint32_t*)len;
452 for (
int i = 0; i < 6; i ++) {
453 head.SEU_FEB[i] = (seu & (1 << i)) != 0;
458 for (
int i = 0; i < 4; i++) {
459 shift = (3 - ibyte % 4) * 8;
460 trg[3 - i] = buffer[ibyte / 4] >> shift;
463 tmp = (uint32_t*)trg;
468 void ARICHUnpackerModule::readFEHeader(
const int* buffer,
unsigned& ibyte,
ARICHRawHeader& head)
474 for (
int i = 0; i < 4; i++) {
475 shift = (3 - ibyte % 4) * 8;
476 line1[3 - i] = buffer[ibyte / 4] >> shift;
480 head.type = line1[3];
481 head.version = line1[2];
482 head.mergerID = line1[1];
483 head.FEBSlot = line1[0];
486 unsigned char len[4];
487 for (
int i = 0; i < 4; i++) {
488 shift = (3 - ibyte % 4) * 8;
489 len[3 - i] = buffer[ibyte / 4] >> shift;
493 unsigned vth_info = len[3] * 256 + len[2];
494 if (vth_info >= 32768) { head.thscan_mode =
true; vth_info -= 32768; }
499 uint32_t* tmp = (uint32_t*)len;
504 for (
int i = 0; i < 4; i++) {
505 shift = (3 - ibyte % 4) * 8;
506 trg[3 - i] = buffer[ibyte / 4] >> shift;
509 tmp = (uint32_t*)trg;
514 void ARICHUnpackerModule::printBits(
const int* buffer,
int bufferSize)
516 for (
int i = 0; i < bufferSize; i++) {
517 std::cout << i <<
"-th word bitset: " << std::bitset<32>(*(buffer + i)) << std::endl;
522 void ARICHUnpackerModule::endRun()
526 void ARICHUnpackerModule::terminate()
Raw data unpacker for ARICH.
bool isRequired(const std::string &name="")
Ensure this array/object has been registered previously.
bool registerInDataStore(DataStore::EStoreFlags storeFlags=DataStore::c_WriteOut)
Register the object/array in the DataStore.
bool create(bool replace=false)
Create a default object in the data store.
Accessor to arrays stored in the data store.
T * appendNew()
Construct a new T object at the end of the array.
Type-safe access to single objects in the data store.
Class to store variables with their name which were sent to the logging service.
#define REG_MODULE(moduleName)
Register the given module (without 'Module' suffix) with the framework.
Abstract base class for different kinds of events.