QwAnalysis
MQwF1TDC.cc
Go to the documentation of this file.
1 /**********************************************************\
2 * File: MQwF1TDC.C *
3 * *
4 * Author: P. M. King, Rakitha Beminiwattha, *
5 * J. H. Lee *
6 * Time-stamp: <2010-04-12 12:12> *
7 \**********************************************************/
8 
9 #include "MQwF1TDC.h"
10 #include "QwColor.h"
11 #include "QwLog.h"
12 
13 #include <math.h>
14 
15 
16 const UInt_t MQwF1TDC::kF1Mask_SlotNumber = 0xf8000000;
17 const UInt_t MQwF1TDC::kF1Mask_ResolutionLockFlag = 0x04000000;
18 const UInt_t MQwF1TDC::kF1Mask_OutputFIFOFlag = 0x02000000;
19 const UInt_t MQwF1TDC::kF1Mask_HitFIFOFlag = 0x01000000;
20 
21 const UInt_t MQwF1TDC::kF1Mask_HeaderFlag = 0x00800000;
22 
23 const UInt_t MQwF1TDC::kF1Mask_FakeDataFlag = 0x00400000;
24 const UInt_t MQwF1TDC::kF1Mask_ChannelNumber = 0x003f0000;
25 const UInt_t MQwF1TDC::kF1Mask_ChipAddress = 0x00380000;
26 const UInt_t MQwF1TDC::kF1Mask_ChannelAddress = 0x00070000;
27 
28 const UInt_t MQwF1TDC::kF1Mask_Dataword = 0x0000ffff;
29 
30 const UInt_t MQwF1TDC::kF1Mask_HeaderTrigFIFOFlag = 0x00400000;
31 const UInt_t MQwF1TDC::kF1Mask_HeaderEventNumber = 0x003f0000;
32 const UInt_t MQwF1TDC::kF1Mask_HeaderTriggerTime = 0x0000ff80;
33 const UInt_t MQwF1TDC::kF1Mask_HeaderXorSetupFlag = 0x00000040;
34 const UInt_t MQwF1TDC::kF1Mask_HeaderChannelNumber = 0x0000003f;
35 const UInt_t MQwF1TDC::kF1Mask_HeaderChipAddress = 0x00000038;
36 const UInt_t MQwF1TDC::kF1Mask_HeaderChannelAddress = 0x00000007;
37 
38 
39 
41 {
42  fF1ROCNumber = 0;
43  fF1SlotNumber = 0;
44 
45  fF1HeaderFlag = kFALSE;
46 
47  fF1HitFIFOFlag = kFALSE;
48  fF1OutputFIFOFlag = kFALSE;
49  fF1ResolutionLockFlag = kFALSE;
50 
51  fF1FakeDataFlag = kFALSE;
52  fF1ChannelNumber = 0;
53  fF1Dataword = 0;
54 
55 
56  fF1HeaderTrigFIFOFlag = kFALSE;
59  fF1HeaderXorSetupFlag = kFALSE;
60 
62 
63  // This initial fF1MaxChannelsPerModule 64
64  // is used to "resize" a vector in RegisterSlotNumber() function
65  // in each Subsystem, before one can access the real F1TDC
66  // configuration from CODA buffer. And it is a constant value (64)
67  // and is totally independent upon the real F1TDC configuration.
68  // The real maximum channel number can be access via F1TDContainer
69  // class of each subsystem.
70 
71  // However, Qweak uses only the Normal Resolution configuration.
72  // Thus, it is always 64 channels we uses.
73  // If someone wants to use the High Resolution Mode of F1TDC,
74  // it would be better to change this number to 32 by hand.
75  // Friday, September 3 13:50:49 EDT 2010, jhlee
76 
77  fF1OverFlowEntryFlag = kFALSE;
78  fF1ValidDataSlotFlag = kFALSE;
79 }
80 
82 
83 
84 void MQwF1TDC::DecodeTDCWord(UInt_t &word, const UInt_t roc_id)
85 {
86 
87  fF1ROCNumber = roc_id;
88  fF1SlotNumber = (word & kF1Mask_SlotNumber)>>27;
89 
90  if( fF1SlotNumber>=1 && fF1SlotNumber<=21 ) fF1ValidDataSlotFlag = kTRUE;
91  else fF1ValidDataSlotFlag = kFALSE;
92 
93 
94  fF1HeaderFlag = ((word & kF1Mask_HeaderFlag)==0);
95  // TRUE if the mask bit IS NOT set
96 
97  // These three flags should be TRUE if their mask bit IS set
98  fF1HitFIFOFlag = ((word & kF1Mask_HitFIFOFlag )!=0);
99  fF1OutputFIFOFlag = ((word & kF1Mask_OutputFIFOFlag )!=0);
101 
102  if (fF1HeaderFlag){
103  // This is a header word.
104  fF1Dataword = 0;
112  }
113  else {
114  // This is a data word.
115  fF1FakeDataFlag = ((word & kF1Mask_FakeDataFlag)!=0); // This flag should be TRUE if their mask bit IS set
116  fF1ChannelNumber = ( word & kF1Mask_ChannelNumber )>>16;
117  fF1ChipAddress = ( word & kF1Mask_ChipAddress )>>19;
118  fF1ChannelAddress = ( word & kF1Mask_ChannelAddress )>>16;
119  fF1Dataword = ( word & kF1Mask_Dataword );
120 
121  if(fF1Dataword == 65535) fF1OverFlowEntryFlag = kTRUE;
122  else fF1OverFlowEntryFlag = kFALSE;
123  // skip to record overflow dataword entry (65535, 0xFFFF)
126  // std::cout << "fake flag " << fF1FakeDataFlag
127  // << " channel: " << fF1ChannelNumber
128  // << " raw time: " << fF1Dataword
129  // << std::endl;
130  }
131  return;
132 }
133 
134 std::ostream& operator<< (std::ostream& os, const MQwF1TDC &f1tdc)
135 {
136  if(f1tdc.fF1HeaderFlag) {
137  os << "<<<< Header:";
138  }
139  else {
140  os << ">>>> DATA :";
141  }
142 
143  os << " Ch" << std::setw(3) << f1tdc.fF1ChannelNumber;
144  os << "[" << f1tdc.fF1ChipAddress;
145  os << "," << f1tdc.fF1ChannelAddress;
146  os << "]";
147 
148  if(f1tdc.fF1HeaderFlag) {
149  os << " Xor " << f1tdc.fF1HeaderXorSetupFlag
150  << " tOF " << f1tdc.fF1HeaderTrigFIFOFlag;
151  }
152  else {
153  os << " - DATA "
154  << f1tdc.fF1FakeDataFlag
155  << " - ";
156  }
157 
158  os << "(hitOF,outOF,resLK)("
159  << f1tdc.fF1HitFIFOFlag
160  << f1tdc.fF1OutputFIFOFlag
161  << f1tdc.fF1ResolutionLockFlag
162  << ")";
163  os << " ROC" << std::setw(2) << f1tdc.fF1ROCNumber;
164 
165  Int_t slot = 0;
166  slot = f1tdc.fF1SlotNumber;
167 
168  os << " Slot" << std::setw(2) << slot;
169 
170  if(f1tdc.fF1HeaderFlag) {
171  os << " EvtN" << std::setw(2) << f1tdc.fF1HeaderEventNumber;
172  os << " TriT" << std::setw(4) << f1tdc.fF1HeaderTriggerTime;
173  }
174  else {
175  os << " RawT " << std::setw(10) << f1tdc.fF1Dataword;
176  }
177 
178  if(slot == 0) {
179  os << ": a filler word";
180  }
181  if( (not f1tdc.fF1HeaderFlag) and (f1tdc.fF1FakeDataFlag) ){
182  os << ": --> fake data";
183  }
184 
185  // os << std::endl;
186  return os;
187 }
188 
189 
190 
191 void MQwF1TDC::Print(Bool_t flag)
192 {
193  if (! this) return; // do nothing if this is a null object
194 
195  if(flag) {
196  std::cout << *this << std::endl;
197  }
198  return;
199 }
200 
201 
202 void MQwF1TDC::PrintTDCHeader(Bool_t flag)
203 {
204  if (! this) return; // do nothing if this is a null object
205 
206  if(flag) {
207  std::cout << *this << std::endl;
208  }
209  return;
210 }
211 
212 
213 void MQwF1TDC::PrintTDCData(Bool_t flag)
214 {
215  if (! this) return; // do nothing if this is a null object
216 
217  if(flag) {
218  std::cout << *this << std::endl;
219  }
220  return;
221 }
222 
223 
224 // Double_t MQwF1TDC::SubtractReference(Double_t rawtime, Double_t reftime)
225 // {
226 // // Note that this produces the opposite sign of the corrected time
227 // // as compared to the UInt_t version from revision 423 shown above.
228 // //
229 // // For Region 3, according to the UInt_t version from revision 423,
230 // // the internal parameters should be:
231 // // fMinDiff = -30000
232 // // fMaxDiff = 30000
233 // // fOffset = 64495
234 // // fTimeShift = 8929
235 // //
236 // Double_t real_time = rawtime - reftime;
237 
238 // if( real_time < fMinDiff ) {
239 // real_time += fOffset;
240 // }
241 // else if( real_time > fMaxDiff) {
242 // real_time -= fOffset;
243 // }
244 // real_time = real_time + fTimeShift;
245 // return real_time;
246 // }
247 
248 
249 
250 // Double_t MQwF1TDC::ActualTimeDifference(Double_t raw_time, Double_t ref_time)
251 // {
252 
253 // Double_t trigger_window = 12896.0;
254 // Double_t time_offset = 65341.0;
255 
256 // Double_t time_condition = 0.0;
257 // Double_t local_time_difference = 0.0;
258 // Double_t actual_time_difference = 0.0;
259 
260 // local_time_difference = raw_time - ref_time;
261 
262 // if(local_time_difference == 0.0) {
263 // // raw_time is the same as ref_time
264 // actual_time_difference = local_time_difference;
265 // }
266 // else {
267 // time_condition = fabs(local_time_difference);
268 // // maximum value is trigger_window -1,
269 // // thus 12896-1 = 12895 - 0 = 12985
270 // if(time_condition < trigger_window) {
271 // // there is no ROLLEVENT within trigger window
272 // actual_time_difference = local_time_difference;
273 // }
274 // else {
275 // // there is an ROLLOVER event within trigger window
276 // if (local_time_difference > 0.0) {
277 // // ref_time is in after ROLLOVER event
278 // actual_time_difference = local_time_difference - time_offset;
279 // }
280 // else {
281 // // we already excluded local_time_diffrence == 0 case.
282 // // ref_time is in before ROLLOVER event
283 // actual_time_difference = local_time_difference + time_offset;
284 // }
285 
286 // }
287 // }
288 // return actual_time_difference;
289 // }
290 
291 
292 
293 void MQwF1TDC::PrintResolutionLockStatus(const UInt_t roc_id)
294 {
295  if (not fF1ResolutionLockFlag) {
296  QwWarning << "F1TDC board RESOULTION LOCK FAIL at Ch "
297  << GetTDCChannelNumber() << " ROC " << roc_id
298  << " Slot " << GetTDCSlotNumber() << QwLog::endl;
299  }
300  return;
301 }
302 
303 
304 
305 void MQwF1TDC::PrintHitFIFOStatus(const UInt_t roc_id)
306 {
307  if (fF1HitFIFOFlag) {
308  QwWarning << "F1TDC board HIT FIFO FULL at Ch "
309  << GetTDCChannelNumber() << " ROC " << roc_id
310  << " Slot " << GetTDCSlotNumber() << QwLog::endl;
311  }
312  return;
313 }
314 
315 
316 
317 void MQwF1TDC::PrintOutputFIFOStatus(const UInt_t roc_id)
318 {
319  if (fF1OutputFIFOFlag) {
320  QwWarning << "F1TDC board OUTPUT FIFO FULL at Ch "
321  << GetTDCChannelNumber() << " ROC " << roc_id
322  << " Slot " << GetTDCSlotNumber() << QwLog::endl;
323  }
324  return;
325 }
326 
327 
329 {
330  // fF1ValidDataSlotFlag = TRUE,
331  // fF1ResolutionFlag = TRUE,
332  // fF1HeaderFlag = FALSE,
333  // fF1OverFlowEntry = FALSE,
334  // fF1FakeDataWord = FALSE, then it is a valid data word.
336  // if( fF1ValidDataSlotFlag && fF1ResolutionLockFlag && !fF1HeaderFlag)
337  return kTRUE;
338  else
339  return kFALSE;
340 }
341 
342 
343 // Bool_t MQwF1TDC::CheckDataIntegrity(const UInt_t roc_id, UInt_t *buffer, UInt_t num_words)
344 // {
345 // UInt_t reference_trig_time = 0;
346 // UInt_t reference_event_num = 0;
347 
348 // const Int_t valid_trigger_time_offset = 1;
349 
350 // Bool_t event_ok_flag = kFALSE;
351 // Bool_t trig_time_ok_flag = kFALSE;
352 // Bool_t data_integrity_flag = kFALSE;
353 
354 // Bool_t temp_print_flag = false;
355 
356 // for (UInt_t i=0; i<num_words ; i++) {
357 
358 // DecodeTDCWord(buffer[i], roc_id);
359 
360 
361 // // We use the multiblock data transfer for F1TDC, thus
362 // // we must get the event number and the trigger time from the first buffer
363 // // (buffer[0]), and these valuse can be used to check "data" integrity
364 // // over all F1TDCs
365 // if ( i==0 ) {//;
366 // if ( IsHeaderword() ) {//;;
367 // reference_event_num = GetTDCEventNumber();
368 // reference_trig_time = GetTDCTriggerTime();
369 // PrintTDCHeader(temp_print_flag);
370 // }//;;
371 // else {//;;
372 // QwWarning << QwColor(Qw::kRed)
373 // << "The first word of F1TDC must be header word. "
374 // << "Something wrong into this CODA stream.\n";
375 // QwWarning << QwLog::endl;
376 // return false;
377 // }//;;
378 // }//;
379 // else {//;
380 // if ( IsHeaderword() ) {//;;
381 // // Check date integrity, if it is fail, we skip this whole buffer to do further process
382 // event_ok_flag = ( reference_event_num == GetTDCHeaderEventNumber() );
383 // trig_time_ok_flag = abs( reference_trig_time - GetTDCHeaderTriggerTime() ) <= valid_trigger_time_offset;
384 // PrintTDCHeader(temp_print_flag);
385 
386 // // If SEU exists, Xor Setup Register bit is changing from 1 to 0, I think.
387 // // Thus, it sets the trigger time 0 and the event number 0.
388 // // For example,
389 // // Ch 8 Xor 0 tOF 0(hitOF,outOF,resLK)(001) ROC 9 Slot 10 EvtN 0 TriT 0
390 // // And it is the source of the trigger time and the event number differences
391 // // within the same ROC. In the CheckDataIntegrity routine, I decided
392 // // to skip such a event.
393 // // Sunday, August 8 03:42:48 EDT 2010, jhlee
394 
395 // if (IsHeaderXorSetup()) {//;;;
396 // // Trigger Time difference of up to 1 count among the chips is acceptable
397 // // For the Trigger Time, this assumes that an external SYNC_RESET signal has
398 // // been successfully applied at the start of the run
399 // if (not trig_time_ok_flag) {
400 // if(temp_print_flag){
401 // QwWarning << QwColor(Qw::kBlue)
402 // << "There is the no SYNC_RESET on the F1TDC board at"
403 // << " Ch " << fF1ChannelNumber
404 // << " ROC " << fF1ROCNumber
405 // << " Slot " << fF1SlotNumber <<"\n";
406 // QwWarning << QwColor(Qw::kBlue)
407 // << "This event is excluded from the ROOT data stream.\n";
408 // QwWarning << QwColor(Qw::kRed)
409 // << "Please, send an email to (a) Qweak DAQ expert(s) if you have time.\n";
410 // QwWarning << QwLog::endl;
411 // }
412 // }
413 // // Any difference in the Event Number among the chips indicates a serious error
414 // // that requires a reset of the board.
415 
416 // if (not event_ok_flag) {
417 // if(temp_print_flag){
418 // QwWarning << QwColor(Qw::kRed)
419 // << "There is the Event Number Mismatch issue on the F1TDC board at"
420 // << " ROC " << fF1ROCNumber
421 // << " Slot " << fF1SlotNumber << "\n";
422 // QwWarning << QwColor(Qw::kRed)
423 // << "This event is excluded from the ROOT data stream.\n";
424 // QwWarning << QwColor(Qw::kRed)
425 // << "Please, send an email to (a) Qweak DAQ expert(s) if you have time.\n";
426 // QwWarning << QwLog::endl;
427 // }
428 // }
429 
430 // data_integrity_flag = (event_ok_flag) && (trig_time_ok_flag) && IsNotHeaderTrigFIFO() ;
431 
432 // if (not data_integrity_flag) return data_integrity_flag; //false
433 // // we stop check data, because all other next buffers
434 // // is useless and we don't need to check them in order to save some time.
435 // } //;;;
436 // else {//;;;
437 
438 // if (temp_print_flag) {
439 
440 // // I don't include the SEU in the CheckDataIntegrity.
441 // // At this moment, I have no idea how I handle during data processes.
442 // // Sunday, August 8 04:02:17 EDT 2010, jhlee
443 
444 // QwWarning << QwColor(Qw::kRed)
445 // << "There is the Single Event Upset (SEU) on the F1TDC board at"
446 // << " ROC " << fF1ROCNumber
447 // << " Slot " << fF1SlotNumber << "\n";
448 // QwWarning << QwColor(Qw::kRed)
449 // << "Please, send an email to (a) Qweak DAQ expert(s) if you have time.\n";
450 // QwWarning << QwLog::endl;
451 // }
452 // }//;;;
453 // }//;;
454 // else { //;; // dataword
455 // if(!fF1OverFlowEntryFlag) PrintTDCData(temp_print_flag);
456 // }//;;
457 // }//;
458 // }//for (UInt_t i=0; i<num_words ; i++) {
459 
460 
461 // return (data_integrity_flag);
462 
463 // }
464 
static const UInt_t kF1Mask_HeaderChannelNumber
Definition: MQwF1TDC.h:139
UInt_t fF1ChannelAddress
Definition: MQwF1TDC.h:158
Bool_t fF1HeaderFlag
Definition: MQwF1TDC.h:148
static const UInt_t kF1Mask_HeaderTriggerTime
Definition: MQwF1TDC.h:132
static const UInt_t kF1Mask_ChannelNumber
Definition: MQwF1TDC.h:116
UInt_t fF1ChannelNumber
Definition: MQwF1TDC.h:156
Bool_t fF1ResolutionLockFlag
Definition: MQwF1TDC.h:152
Bool_t fF1ValidDataSlotFlag
Definition: MQwF1TDC.h:171
static const UInt_t kF1Mask_SlotNumber
Definition: MQwF1TDC.h:87
std::ostream & operator<<(std::ostream &out, const QwColor &color)
Output stream operator which uses the enum-to-escape-code mapping.
Definition: QwColor.h:153
static const UInt_t kF1Mask_OutputFIFOFlag
Definition: MQwF1TDC.h:97
Bool_t fF1HeaderTrigFIFOFlag
Definition: MQwF1TDC.h:163
static const UInt_t kF1Mask_FakeDataFlag
Definition: MQwF1TDC.h:112
~MQwF1TDC()
Definition: MQwF1TDC.cc:81
void PrintOutputFIFOStatus(const UInt_t roc_id)
Definition: MQwF1TDC.cc:317
MQwF1TDC()
Definition: MQwF1TDC.cc:40
static const UInt_t kF1Mask_HeaderChipAddress
Definition: MQwF1TDC.h:140
static const UInt_t kF1Mask_ResolutionLockFlag
Definition: MQwF1TDC.h:96
static const UInt_t kF1Mask_HeaderEventNumber
Definition: MQwF1TDC.h:129
static const UInt_t kF1Mask_HitFIFOFlag
Definition: MQwF1TDC.h:98
UInt_t fF1HeaderEventNumber
Definition: MQwF1TDC.h:164
static const UInt_t kF1Mask_HeaderTrigFIFOFlag
Definition: MQwF1TDC.h:126
Bool_t fF1HitFIFOFlag
Definition: MQwF1TDC.h:150
static const UInt_t kF1Mask_ChipAddress
Definition: MQwF1TDC.h:117
void PrintTDCHeader(Bool_t flag)
Definition: MQwF1TDC.cc:202
Bool_t fF1OverFlowEntryFlag
Definition: MQwF1TDC.h:170
const UInt_t & GetTDCSlotNumber() const
Definition: MQwF1TDC.h:44
void PrintResolutionLockStatus(const UInt_t roc_id)
Definition: MQwF1TDC.cc:293
static const UInt_t kF1Mask_HeaderFlag
Definition: MQwF1TDC.h:101
UInt_t fF1ChipAddress
Definition: MQwF1TDC.h:157
Bool_t fF1FakeDataFlag
Definition: MQwF1TDC.h:155
static const UInt_t kF1Mask_ChannelAddress
Definition: MQwF1TDC.h:118
const UInt_t & GetTDCChannelNumber() const
Definition: MQwF1TDC.h:45
A logfile class, based on an identical class in the Hermes analyzer.
Bool_t IsValidDataword() const
Definition: MQwF1TDC.cc:328
UInt_t fF1MaxChannelsPerModule
Definition: MQwF1TDC.h:169
void PrintHitFIFOStatus(const UInt_t roc_id)
Definition: MQwF1TDC.cc:305
UInt_t fF1SlotNumber
Definition: MQwF1TDC.h:146
static const UInt_t kF1Mask_HeaderChannelAddress
Definition: MQwF1TDC.h:141
void Print(Bool_t flag)
Definition: MQwF1TDC.cc:191
static std::ostream & endl(std::ostream &)
End of the line.
Definition: QwLog.cc:299
static const UInt_t kF1Mask_Dataword
Definition: MQwF1TDC.h:121
void PrintTDCData(Bool_t flag)
Definition: MQwF1TDC.cc:213
UInt_t fF1Dataword
Definition: MQwF1TDC.h:159
Bool_t fF1OutputFIFOFlag
Definition: MQwF1TDC.h:151
#define QwWarning
Predefined log drain for warnings.
Definition: QwLog.h:45
UInt_t fF1ROCNumber
Definition: MQwF1TDC.h:145
static const UInt_t kF1Mask_HeaderXorSetupFlag
Definition: MQwF1TDC.h:135
void DecodeTDCWord(UInt_t &word, const UInt_t roc_id)
Definition: MQwF1TDC.cc:84
UInt_t fF1HeaderTriggerTime
Definition: MQwF1TDC.h:165
Bool_t fF1HeaderXorSetupFlag
Definition: MQwF1TDC.h:166