-
Notifications
You must be signed in to change notification settings - Fork 43
/
Copy pathSBSDecodeF1TDCModule.cxx
319 lines (280 loc) · 11.3 KB
/
SBSDecodeF1TDCModule.cxx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
/* Uncomment this to turn on high resolution mode */
/* #define USE_HIRES */
/////////////////////////////////////////////////////////////////////
//
// SBSDecodeF1TDCModule
// JLab F1 TDC Module
//
/////////////////////////////////////////////////////////////////////
//
// - 2021/02/24 Juan Carlos Cornejo
// * Modify it to support both high and low res in the same class
// * Module 3204 is high res, and 6401 is low res
// - Version 1 - 2017/12/13 (Marco Carmignotto)
// * Tested for High Resolution only
// * Work with multiple modules in the same bank
//
// - Version 0 - 2017/11/30 (Marco Carmignotto)
// * Multihits
//
// - Version -1 (Robert Michaels)
//
/////////////////////////////////////////////////////////////////////
#include "SBSDecodeF1TDCModule.h"
#include "VmeModule.h"
#include "THaEvData.h"
#include "TMath.h"
#include <iostream>
#include <string>
#include <sstream>
using namespace std;
namespace Decoder {
// const Int_t NTDCCHAN = 32;
const Int_t MAXHIT = 20;
Module::TypeIter_t SBSDecodeF1TDCHighResModule::fgThisType =
DoRegister( ModuleType( "Decoder::SBSDecodeF1TDCHighResModule" , 3204 ));
Module::TypeIter_t SBSDecodeF1TDCLowResModule::fgThisType =
DoRegister( ModuleType( "Decoder::SBSDecodeF1TDCLowResModule" , 6401 ));
SBSDecodeF1TDCModule::SBSDecodeF1TDCModule(Int_t crate, Int_t slot) : VmeModule(crate, slot), IsInit(false) {
fDebugFile = nullptr;
}
SBSDecodeF1TDCModule::~SBSDecodeF1TDCModule() = default;
void SBSDecodeF1TDCModule::CommonInit() {
//fTdcData = new Int_t[NTDCCHAN*MAXHIT];
//fNumHits = new Int_t[NTDCCHAN*MAXHIT];
fDebugFile=nullptr;
//fDebugFile = new std::ofstream("hcal_tdc_test_decoder.log");
Clear();
IsInit = kTRUE;
nF1=0;
F1slots.resize(50);
for(int & F1slot : F1slots) {
F1slot = 0;
}
//F1slots = new Int_t[50];
//memset(F1slots, 0, 50*sizeof(Int_t));
fWdcntMask=0;
}
Bool_t SBSDecodeF1TDCModule::IsSlot(UInt_t rdata)
{
fHeaderMask = 0x07ffffff;
if (fDebugFile)
*fDebugFile << "is SBSDecodeF1TDC slot ? "<<hex<<fHeader
<<" "<<fHeaderMask<< " " << (rdata & fHeaderMask) <<" "<<rdata<<dec<<endl;
return ((rdata != 0xffffffff) & ((rdata & fHeaderMask)==fHeader));
}
Int_t SBSDecodeF1TDCModule::GetData(Int_t chan, Int_t hit) const
{
Int_t idx = chan*MAXHIT + hit;
if (idx < 0 || idx > Int_t(MAXHIT*fNumChan*nF1)) return 0;
return fTdcData[idx];
}
Int_t SBSDecodeF1TDCModule::GetNumHits(Int_t chan) {
if (chan < 0 || chan > Int_t(nF1*fNumChan)) return -1;
return fNumHits[chan];
}
void SBSDecodeF1TDCModule::Clear(const Option_t* opt) {
VmeModule::Clear(opt);
//memset(fTdcData, 0, fNumChan*MAXHIT*sizeof(Int_t));
//memset(fNumHits, 0, fNumChan*sizeof(Int_t));
// std::fill(fTdcData.begin(), fTdcData.end(), 0);
// std::fill(fNumHits.begin(), fNumHits.end(), 0);
for(int & datum : fTdcData) datum=0;
for(int & numhit : fNumHits) numhit=0;
}
UInt_t SBSDecodeF1TDCModule::LoadSlot( THaSlotData *sldat, const UInt_t *evbuffer, UInt_t pos, UInt_t len) {
// the 4-arg version of LoadSlot. Let it call the 3-arg version.
// I'm not sure we need both (historical)
return LoadSlot(sldat, evbuffer+pos, evbuffer+pos+len);
}
UInt_t SBSDecodeF1TDCModule::LoadSlot(THaSlotData *sldat, const UInt_t *evbuffer, const UInt_t *pstop) {
// This is the 3-arg version of LoadSlot
// Note, this increments evbuffer
if (fDebugFile) *fDebugFile << "SBSDecodeF1TDCModule:: loadslot "<<endl;
fWordsSeen = 0;
// CAUTION: this routine re-numbers the channels
// compared to the labelled numbering scheme on the board itself.
// According to the labelling and internal numbering scheme,
// the F1 module has odd numbered channels on one connector
// and even numbered channels on the other.
// However we usually put neighboring blocks/wires into the same
// cable, connector etc.
// => hana therefore uses a numbering scheme different from the module
//
// In normal resolution mode, the scheme is:
// connector 1: ch 0 - 15
// conncetor 2: ch 16 - 31
// connector 33: ch 32 - 47
// connector 34: ch 48 - 63
//
// In high-resolution mode, only two connectors are used since
// two adjacent channels are internally combined and read out as the
// internally-even numbered channels.
// this is kind of inconvenient for the rest of the software
// => hana therefore uses a numbering scheme different from the module
// connector 1: unused
// connector 2: ch 0 - 15
// connector 33: unused
// connector 34: ch 16 - 31
//
// In both modes:
// it is assumed that we only get data from one single trigger
// if the F1 is run in multiblock mode (buffered mode)
// this might not be the case anymore - but this will be interesting anyhow
// triggertime and eventnumber are not yet read out, they will again
// be useful when multiblock mode (buffered mode) is used
const UInt_t F1_HIT_OFLW = 1<<24; // bad
const UInt_t F1_OUT_OFLW = 1<<25; // bad
const UInt_t F1_RES_LOCK = 1<<26; // good
const UInt_t DATA_CHK = F1_HIT_OFLW | F1_OUT_OFLW | F1_RES_LOCK;
const UInt_t DATA_MARKER = 1<<23;
// look at all the data
const UInt_t *loc = evbuffer;
Int_t fDebug=2;
if(fDebug > 1 && fDebugFile!=0) *fDebugFile<< "Debug of (My Private) SBSDecodeF1TDC data, fResol = "<<fResol<<" model num "<<fModelNum<<endl;
// To account for multiple slots
Int_t lastSlot = 0;
Int_t idxSlot = -1;
trigTime = 0;
Int_t raw_cor = 0;
// Expect the F1 first word to be an event counter
// The second word should be 0xf1daffff
// Then there should be header, trailer and data words
// header/trailer words have 0 in bit 23
// data words have 1 in bit 23
if(fDebug > 1 && fDebugFile!=0) *fDebugFile<< " loc = " << loc << " pstop " << pstop << " logic " << (loc <= pstop)<< " *loc " << *loc << endl;
while ( loc <= pstop && *loc != 0xda0000ff ) {
Int_t checkf1slot = ((*loc)&0xf8000000)>>27;
if(fDebug > 1 && fDebugFile!=0) *fDebugFile << " check slot = " << checkf1slot << endl;
if(fDebug > 1 && fDebugFile!=0) *fDebugFile<< " data = " << hex << *loc << " " << ( (*loc) & DATA_MARKER )<< endl;
if ( !( (*loc) & DATA_MARKER ) && checkf1slot>0 ) {
// header/trailer word, to be ignored (except for trigTime)
// trigTime = ((*loc)>>7)&0x1FF;
//Trig Time is the bitwise AND of header word with FF80
// FF80 = 1111 1111 1000 0000 This looks correct, we get bits 15:7 inclusive and nothing else.
trigTime = ((*loc)&0xFF80);
//ALSO: why are we using a signed integral type to hold trigtime here?
//Anyway, let's try setting the 7 LSBs of trigTime to 1 to match how rollover detection/correction was done during GEp-III:
// 0111 1111 = 0x007F
//trigTime |= 0x007F;
Int_t chn = (*loc)&0x7;
Int_t chip = ((*loc>>3))&0x7;
Int_t ixor = ((*loc>>6))&0x1;
//Int_t ievnum = ((*loc>>16))&0x3F;
Int_t itrigFIFOoverflow = ((*loc>>2))&0x1;
if(fDebug > 1 && fDebugFile!=0)
*fDebugFile<< "[" << (loc-evbuffer) << "] header/trailer 0x"
<<hex<<*loc<<dec<< " chn = " << chn << " chip = " << chip << " ixor = " << ixor << " Trig FIFO overflow = " << itrigFIFOoverflow <<endl;
if (itrigFIFOoverflow==1) {
if(fDebug > 1 && fDebugFile!=0) *fDebugFile << "\tTrigger Overflow error chip = " << dec << chip << " chan = " << chn << hex<< endl;
}
} else if((*loc)!=0xf1daffff && checkf1slot>0 ) {
if (fDebug > 1 && fDebugFile!=0)
*fDebugFile<< "[" << (loc-evbuffer) << "] data 0x"
<<hex<<*loc<<dec<<endl;
Int_t chn = ((*loc)>>16) & 0x3f; // internal channel number
Int_t chan;
if (IsHiResolution()) {
// drop last bit for channel renumbering
chan=(chn >> 1);
} else {
// do the reordering of the channels, for contiguous groups
// odd numbered TDC channels from the board -> +16
chan = (chn & 0x20) + 16*(chn & 0x01) + ((chn & 0x1e)>>1);
}
Int_t f1slot = ((*loc)&0xf8000000)>>27;
//FIXME: cross-check slot number here
// Make sure we have the current slot considered - important for multiple modules
if(fDebugFile) {
*fDebugFile << "f1slot=" << f1slot << ", lastSlot=" << lastSlot << std::endl;
}
if(f1slot != lastSlot && f1slot!=30) { // we may have a new slot here
Bool_t newslot = 1;
for(Int_t k=0; k<nF1; k++) {
if(f1slot == F1slots[k]) newslot=0; // we saw this slot before
}
if(newslot) { // increase size of vectors
F1slots[nF1]=f1slot;
nF1++;
fNumHits.resize(fNumChan*nF1);
fTdcData.resize(fNumChan*nF1*MAXHIT);
}
lastSlot = f1slot;
// Updating idxSlot
for(Int_t k=0; k<nF1; k++) {
if(f1slot == F1slots[k]) idxSlot = k;
}
}
//cout << "f1slot: " << f1slot << " ";//endl;
if (f1slot!=30 && ((*loc) & DATA_CHK) != F1_RES_LOCK ) {
cout << "\tWarning: F1 TDC " << hex << (*loc) << dec;
cout << "\tSlot (Ch) = " << f1slot << "(" << chan << ")";
if ( (*loc) & F1_HIT_OFLW ) {
cout << "\tHit-FIFO overflow";
}
if ( (*loc) & F1_OUT_OFLW ) {
cout << "\tOutput FIFO overflow";
}
if ( ! ((*loc) & F1_RES_LOCK ) ) {
cout << "\tResolution lock failure!";
}
cout << endl;
}
Int_t raw= (*loc) & 0xffff;
if(fDebug > 1 && fDebugFile!=0) {
*fDebugFile<<" int_chn chan data "<<dec<<chn<<" "<<chan
<<" 0x"<<hex<<raw<<dec<<endl;
}
// For now, only load data when our slot number matches that of
// the base class
// Does this line have the potential to cause shenanigans/data loss?
if((Int_t)fSlot == f1slot) {
raw_cor = raw;
sldat->loadData("tdc",chan,raw_cor,trigTime); //
}
if((*loc)!=0xf1daffff && nF1>0 && f1slot!=30) { // Make sure memory is allocated to save data
Int_t chSlot = idxSlot*fNumChan + chan;
Int_t idx = chSlot*MAXHIT + fNumHits[chSlot]; // multiple hits, multiple slots
if (idx >= 0 && idx < Int_t(MAXHIT*nF1*fNumChan) && fNumHits[chSlot]<MAXHIT) fTdcData[idx] = raw;
if(fNumHits[chSlot]>=MAXHIT) cout << "F1TDC warning: more than " << MAXHIT << " hits in channel " << chan << " / slot " << f1slot << ". Not taking last hit(s), although counting them." << endl;
(fNumHits[chSlot])++;
//cout << "chSlot=" << chSlot << " - lastSlot=" << lastSlot << " - nF1=" << nF1 << " -- ch=" << chan << " - word=0x" << hex << (*loc) << dec << " - raw=" << raw << endl;
}
fWordsSeen++;
}
loc++;
}
return fWordsSeen;
}
SBSDecodeF1TDCLowResModule::SBSDecodeF1TDCLowResModule(Int_t crate, Int_t slot) :
SBSDecodeF1TDCModule(crate,slot)
{
SBSDecodeF1TDCLowResModule::Init();
}
SBSDecodeF1TDCHighResModule::SBSDecodeF1TDCHighResModule(Int_t crate, Int_t slot) :
SBSDecodeF1TDCModule(crate,slot)
{
SBSDecodeF1TDCHighResModule::Init();
}
void SBSDecodeF1TDCLowResModule::Init() {
fName = "My private F1 TDC 6401 (LoRes 64 Channels)";
SetResolution(0);
fNumChan = 64;
CommonInit();
};
void SBSDecodeF1TDCHighResModule::Init() {
fName = "My private F1 TDC 3204 (HighRes 32 Channels)";
SetResolution(1);
fNumChan = 32;
CommonInit();
};
SBSDecodeF1TDCLowResModule::~SBSDecodeF1TDCLowResModule()
{
}
SBSDecodeF1TDCHighResModule::~SBSDecodeF1TDCHighResModule()
{
}
}
ClassImp(Decoder::SBSDecodeF1TDCModule)
ClassImp(Decoder::SBSDecodeF1TDCLowResModule)
ClassImp(Decoder::SBSDecodeF1TDCHighResModule)