Class Reference for E1039 Core & Analysis Software
MainDaqParser.cc
Go to the documentation of this file.
1 #include <iomanip>
2 #include <sstream>
3 #include <TSystem.h>
4 #include <TDatime.h>
5 #include <phool/PHTimer2.h>
6 #include "CodaInputManager.h"
7 #include "MainDaqParser.h"
8 using namespace std;
9 
10 // In the current design, all event-word handlers (i.e. functions) should print out all error messages needed, so that their caller should print nothing but just check the returned value.
11 
12 const std::vector<std::string> MainDaqParser::LIST_TIMERS = {
13  "open_file" ,
14  "open_to_end" ,
15  "parse_one_spill" ,
16  "next_coda_event" ,
17  "process_coda_physics" ,
18  "process_coda_prestart",
19  "process_coda_end" ,
20  "process_coda_fee" ,
21  "process_phys_flush" ,
22  "process_phys_slow" ,
23  "process_phys_prestart",
24  "process_phys_bos" ,
25  "process_phys_eos" ,
26  "pack_one_spill_data" ,
27  "map_chan"
28 };
29 
31  : m_file_size_min(32768)
32  , m_sec_wait(15)
33  , m_n_wait(40)
34  , m_timer_sp_input (new PHTimer2("timer_sp_input"))
35  , m_timer_sp_decode(new PHTimer2("timer_sp_decode"))
36  , m_timer_sp_map (new PHTimer2("timer_sp_map"))
37  , m_use_local_spill_id(false)
38  , m_force_local_spill_id(false)
39 {
40  coda = new CodaInputManager();
41  list_sd = new SpillDataMap();
42  list_ed = new EventDataMap();
43  sd_now = 0; // Will be just a pointer to one object in "list_sd"
44  list_ed_now = new EventDataMap();
45 
46  for (auto it = LIST_TIMERS.begin(); it != LIST_TIMERS.end(); it++) {
47  string name = *it;
48  m_timers[name] = new PHTimer2(name);
49  }
50 }
51 
53 {
54  if (coda ) delete coda;
55  if (list_sd ) delete list_sd;
56  if (list_ed ) delete list_ed;
57  if (list_ed_now) delete list_ed_now;
58  for (auto it = m_timers.begin(); it != m_timers.end(); it++) delete it->second;
59  if (m_timer_sp_input ) delete m_timer_sp_input;
60  if (m_timer_sp_decode) delete m_timer_sp_decode;
61  if (m_timer_sp_map ) delete m_timer_sp_map;
62 }
63 
64 int MainDaqParser::OpenCodaFile(const std::string fname, const long file_size_min, const int sec_wait, const int n_wait)
65 {
66  m_file_size_min = file_size_min;
67  m_sec_wait = sec_wait;
68  m_n_wait = n_wait;
69  dec_par.timeStart = time(NULL);
70  dec_par.fn_in = fname;
71 
72  m_timers["open_file"]->restart();
73  int ret = coda->OpenFile(fname, file_size_min, sec_wait, n_wait);
74  m_timers["open_file"]->stop();
75  m_timers["open_to_end"]->restart();
76  return ret;
77 }
78 
80 {
81  static EventDataMap::iterator it = list_ed_now->begin();
82  if (it == list_ed_now->end()) { // No event in the event buffer. Try to find new ones.
83  list_ed_now->clear();
84  while (list_ed_now->size() == 0 && ! coda->IsEnded()) {
85  m_timers["parse_one_spill"]->restart();
86  ParseOneSpill();
87  m_timers["parse_one_spill"]->stop();
88  }
89  it = list_ed_now->begin();
90  }
91  if (it != list_ed_now->end()) { // Event available. Return it.
92  rd = &run_data;
93  sd = sd_now;
94  ed = &it->second;
95  it++;
96  return true;
97  } else { // No event available. End.
98  rd = 0;
99  sd = 0;
100  ed = 0;
101  return false;
102  }
103 }
104 
105 int MainDaqParser::ParseOneSpill()
106 {
107  static bool call_1st = true;
108  if (call_1st) call_1st = false;
109  else if (dec_par.time_wait > 0) {
110  cout << "...sleep (" << dec_par.time_wait << ") to pretend waiting for next spill..." << endl;
111  for (int ii = dec_par.time_wait; ii > 0; ii--) sleep(1); // Looped to accept the online-monitor connection.
112  cout << "...done." << endl;
113  }
114 
115  if (dec_par.verb>=1) {
116  cout << "MainDaqParser::ParseOneSpill():";
117  ProcInfo_t info;
118  int ret = gSystem->GetProcInfo(&info);
119  if (ret != 0) {
120  cout << " ret=" << ret;
121  } else {
122  TDatime dati;
123  cout << " " << dati.AsSQLString() << " CpuSys=" << info.fCpuSys << " CpuUser=" << info.fCpuUser << " MemResident=" << info.fMemResident << " MemVirtual=" << info.fMemVirtual;
124  }
125  cout << endl;
126  }
127 
128  dec_par.at_bos = false;
129  int* event_words = 0;
130  while (true) {
131  m_timers["next_coda_event"]->restart();
132  m_timer_sp_input->restart();
133  bool found_event = coda->NextCodaEvent(dec_par.event_count, event_words);
134  m_timers["next_coda_event"]->stop();
135  m_timer_sp_input->stop();
136  if (! found_event) break;
137 
138  m_timer_sp_decode->restart();
139 
140  int evt_type_id = event_words[1];
141  if (dec_par.verb > 3) cout << "NextCodaEvent(): " << dec_par.event_count << " 0x" << hex << evt_type_id << dec << endl;
142 
143  // The last 4 hex digits will be 0x01cc for Prestart, Go Event,
144  // and End Event, and 0x10cc for Physics Events
145  int ret = 0;
146  switch (evt_type_id & 0xFFFF) {
147  case PHYSICS_EVENT:
148  m_timers["process_coda_physics"]->restart();
149  ret = ProcessCodaPhysics(event_words);
150  m_timers["process_coda_physics"]->stop();
151  break;
152  case CODA_EVENT:
153  switch (evt_type_id) {
154  case PRESTART_EVENT:
155  m_timers["process_coda_prestart"]->restart();
156  ret = ProcessCodaPrestart(event_words);
157  m_timers["process_coda_prestart"]->stop();
158  break;
159  case GO_EVENT: // do nothing
160  break;
161  case END_EVENT:
162  m_timers["process_coda_end"]->restart();
163  ret = ProcessCodaEnd(event_words);
164  m_timers["process_coda_end"]->stop();
165  coda->ForceEnd();
166  if (dec_par.verb>3) cout << "END_EVENT." << endl;
167  break;
168  default:
169  cerr << "!!ERROR!! Uncovered Coda event type: " << evt_type_id << ". Exit." << endl;
170  return false;
171  }
172  break;
173  case FEE_PREFIX:
174  m_timers["process_coda_fee"]->restart();
175  ret = ProcessCodaFee(event_words);
176  m_timers["process_coda_fee"]->stop();
177  break;
178  case 0: // Special case which requires waiting and retrying. Purpose?? Still needed??
179  cout << "Case '0' @ event count " << dec_par.event_count << "." << endl;
180  ret = coda->OpenFile(dec_par.fn_in, m_file_size_min, m_sec_wait, m_n_wait);
181  if (ret == 0) {
182  ret = coda->JumpCodaEvent(dec_par.event_count, event_words, dec_par.event_count - 1) ? 0 : 2;
183  }
184  break;
185  default: // If no match to any given case, print it and exit.
186  cerr << "!!ERROR!! Uncovered Coda event type: " << evt_type_id << ". Exit." << endl;
187  return false;
188  }
189  m_timer_sp_decode->stop();
190  if (ret != 0) {
191  cout << "WARNING: ParseOneSpill(): ret = " << ret << endl;
192  break;
193  }
194  if (dec_par.at_bos) break;
195  }
196  return 0;
197 }
198 
203 {
204  m_timers["open_to_end"]->stop();
205  coda->CloseFile();
206  dec_par.timeEnd = time(NULL);
207  if (dec_par.verb>0) {
208  cout << "\nMainDaqParser: Statistics:\n"
209  << " Phys events: all = " << run_data.n_phys_evt << "\n"
210  << " Flush events: all = " << run_data.n_flush_evt << "\n"
211  << " v1495 events: all = " << run_data.n_v1495 << ", 0xd1ad = " << run_data.n_v1495_d1ad << ", 0xd2ad = " << run_data.n_v1495_d2ad << ", 0xd3ad = " << run_data.n_v1495_d3ad << "\n"
212  << " Real events: all = " << run_data.n_evt_all << ", decoded = " << run_data.n_evt_dec << "\n"
213  << " TDC hits: total = " << run_data.n_hit << ", bad = " << run_data.n_hit_bad << "\n"
214  << " v1495 hits: total = " << run_data.n_t_hit << ", bad = " << run_data.n_t_hit_bad << "\n"
215  << " Real decoding time: " << (dec_par.timeEnd - dec_par.timeStart) << endl;
216 
217  cout << "MainDaqParser: Timer:\n";
218  for (auto it = LIST_TIMERS.begin(); it != LIST_TIMERS.end(); it++) {
219  m_timers[*it]->print_stat();
220  }
221  }
222  if (dec_par.is_online) {
223  dec_err.PrintData();
224  dec_err.InitData();
225  }
226 
227  return 0;
228 }
229 
235 int MainDaqParser::ProcessCodaPrestart(int* words)
236 {
237  // int evLength = words[0];
238  // int prestartCode = words[1];
239  run_data.utime_b = words[2];
240  dec_par.runID = run_data.run_id = words[3];
241  int run_type = words[4];
242 
243  if (dec_par.verb>0) {
244  cout << "ProcessCodaPrestart: run " << dec_par.runID << ", utime " << run_data.utime_b << ", type " << run_type << ", sampling " << dec_par.sampling << endl;
245  }
246 
248  coda->SetRunNumber(dec_par.runID);
249 
250  return 0;
251 }
252 
258 int MainDaqParser::ProcessCodaEnd(int* words)
259 {
260  run_data.utime_e = words[2];
261  if (dec_par.verb>0) cout << " ProcessCodaEnd " << run_data.utime_e << endl;
262  return 0;
263 }
264 
265 int MainDaqParser::ProcessCodaFee(int* words)
266 {
267  if (words[1] != FEE_EVENT) {
268  cerr << "!!ERROR!! Not FEE_EVENT in case of FEE_PREFIX: " << words[1] << ". Exit." << endl;
269  return 1;
270  }
271  if (words[0] > 8) { // Process only if there is content
272  if (words[3] == (int)0xe906f011) {
273  return ProcessCodaFeeBoard(words);
274  } else if (words[3] == (int)0xe906f012) {
275  if (dec_par.verb) cout << "feePrescale: event " << dec_par.event_count << endl;
276  return ProcessCodaFeePrescale(words);
277  } else if (words[3] == (int)0xe906f005) {
278  return ProcessCodaFeeV1495(words);
279  }
280  }
281  return 0;
282 }
283 
284 int MainDaqParser::ProcessCodaFeeBoard(int* words)
285 {
286  if (dec_par.verb > 1) cout << "CodaFeeBoard: event " << dec_par.event_count << endl;
287 
288  FeeData data;
289  int size = words[0];
290  data.roc = words[2];
291  int i_wd = 3;
292  while (i_wd < size)
293  {
294  // words[i_wd] = 0xe906f011 // ignore this flag
295  i_wd++;
296  // This word contains only the boardID
297  data.board = get_hex_bits (words[i_wd], 7, 4);
298  data.hard = get_hex_bits (words[i_wd], 1, 2);
299  i_wd++;
300  // This is the TDC registry word
301  data.falling_enabled = get_hex_bit (words[i_wd], 2);
302  data.segmentation = get_hex_bit (words[i_wd], 1);
303  if ( data.segmentation == 0 ) data.segmentation = 512;
304  else if ( data.segmentation == 2 ) data.segmentation = 1024;
305  else if ( data.segmentation == 6 ) data.segmentation = 2048;
306 
307  i_wd++;
308  // Another TDC registry word
309  data.multihit_elim_enabled = get_bin_bit (words[i_wd], 23);
310  data.updating_enabled = get_bin_bit (words[i_wd], 22);
311 
312  data.elim_window = 0;
313  if (data.multihit_elim_enabled) {
314  data.elim_window = get_bin_bits (words[i_wd], 21, 6);
315  data.elim_window = 4 * (4 + data.elim_window);
316  }
317 
318  i_wd++;
319  data.lowLimit = get_hex_bits(words[i_wd], 2, 3);
320  data.highLimit = get_hex_bits(words[i_wd], 6, 3);
321  data.selectWindow = (get_hex_bit (words[i_wd], 7) == 8 ? 1 : 0);
322  i_wd++;
323 
324  run_data.fee.push_back(data);
325  if (dec_par.verb > 1) {
326  cout << " feeE " << data.roc << " " << data.board << " " << data.hard
327  << " " << data.falling_enabled << " " << data.segmentation
328  << " " << data.multihit_elim_enabled << " " << data.updating_enabled
329  << " " << data.elim_window << " " << data.selectWindow << " "
330  << data.lowLimit << " " << data.highLimit << endl;
331  }
332  }
333 
334  run_data.n_fee_event++;
335  return 0;
336 }
337 
338 int MainDaqParser::ProcessCodaFeePrescale(int* words)
339 {
340  if (dec_par.verb > 1) cout << "CodaFeePrescale: event " << dec_par.event_count << endl;
341  int feeTriggerBits = words[4];
343  for (int ii = 0; ii < 10; ii++) {
344  run_data.trig_bit[ii] = get_bin_bit (feeTriggerBits, ii);
345  }
346  for (int ii = 0; ii < 5; ii++) run_data.fpga_enabled[ii] = get_bin_bit (feeTriggerBits, ii );
347  for (int ii = 0; ii < 5; ii++) run_data. nim_enabled[ii] = get_bin_bit (feeTriggerBits, ii+5);
348 
350  for (int ii = 0; ii < 8; ii++) {
351  run_data.prescale[ii] = words[ii + 5];
352  }
353  for (int ii = 0; ii < 5; ii++) run_data.fpga_prescale[ii] = words[ii + 5];
354  for (int ii = 0; ii < 3; ii++) run_data. nim_prescale[ii] = words[ii + 10];
355 
356  run_data.n_fee_prescale++;
357  if (dec_par.verb > 1) {
358  cout << " feeP ";
359  for (int ii = 0; ii < 10; ii++) cout << " " << run_data.trig_bit[ii];
360  cout << "\n feeP ";
361  for (int ii = 0; ii < 8; ii++) cout << " " << run_data.prescale[ii];
362  cout << endl;
363  }
364  return 0;
365 }
366 
367 // Words: 0=size 1=0x00840100 2=ROC 3=0xe906f005 4=N 5=v1 v2 v3 v4 v5
368 int MainDaqParser::ProcessCodaFeeV1495(int* words)
369 {
370  if (dec_par.verb > 0) cout << "CodaFeeV1495: event " << dec_par.event_count << endl;
371  //int size = words[0];
372  //data.roc = words[2];
373  int num = words[4];
374  if (run_data.v1495_id[0] != 0) {
375  if (dec_par.verb > 0) cout << " Multiple events. Overwriting." << endl;
376  memset(run_data.v1495_id, 0, sizeof(run_data.v1495_id));
377  }
378  run_data.v1495_id[0] = num;
379  for (int ii = 0; ii < num; ii++) {
380  int value = words[5+ii];
381  run_data.v1495_id[ii+1] = value;
382  if (dec_par.verb > 0) cout << " feeV1495\t" << ii << "\t" << value << "\n";
383  }
384  return 0;
385 }
386 
397 int MainDaqParser::ProcessCodaPhysics(int* words)
398 {
399  run_data.n_phys_evt++;
400  int eventCode = words[1];
401  dec_par.codaID = words[4];
402  int ret = 0;
403  switch (get_hex_bits(eventCode, 7, 4) ) {
404  case FLUSH_EVENTS:
406  //PrintWords(words, 0, evLength);
407  //PrintCodaEventSummary(words);
408  if (! dec_par.has_1st_bos) break;
409  run_data.n_flush_evt++;
410  m_timers["process_phys_flush"]->restart();
411  if (dec_par.verb > 10) PrintCodaEventSummary(words);
412  if (dec_par.verb > 20) PrintWords(words, 0, words[0]);
413  ret = ProcessPhysStdAndFlush(words, FLUSH_EVENTS);
414  m_timers["process_phys_flush"]->stop();
415  if (dec_err.GetFlushError()) run_data.n_flush_evt_bad++;
416  break;
417  case SLOW_CONTROL:
418  m_timers["process_phys_slow"]->restart();
419  ret = ProcessPhysSlow(words);
420  m_timers["process_phys_slow"]->stop();
421  break;
422  case PRESTART_INFO:
423  m_timers["process_phys_prestart"]->restart();
424  ret = ProcessPhysPrestart(words);
425  m_timers["process_phys_prestart"]->stop();
426  break;
427  case STANDARD_PHYSICS:
428  //if (! dec_par.has_1st_bos) break;
429  //dec_par.eventIDstd++;
430  //SetEventInfo(&(*list_ed)[dec_par.eventIDstd].event, dec_par.eventIDstd);
431  //ret = ProcessPhysStdAndFlush(words, STANDARD_PHYSICS);
432  break;
433  case SPILL_COUNTER:
434  ret = ProcessPhysSpillCounter(words);
435  break;
436  case RUN_DESCRIPTOR:
437  ret = ProcessPhysRunDesc(words);
438  break;
439  case BEGIN_SPILL:
440  m_timers["process_phys_bos"]->restart();
441  ret = ProcessPhysBOSEOS(words, BEGIN_SPILL);
442  m_timers["process_phys_bos"]->stop();
443  break;
444  case END_SPILL:
445  m_timers["process_phys_eos"]->restart();
446  ret = ProcessPhysBOSEOS(words, END_SPILL);
447  m_timers["process_phys_eos"]->stop();
448  break;
449  default: // Awaiting further event types
450  ret = -1;
451  cout << "Unknown event code: " << eventCode << ". Ignore." << endl;
452  break;
453  }
454  if (ret != 0) run_data.n_phys_evt_bad++;
455  return ret;
456 }
457 
458 int MainDaqParser::ProcessPhysRunDesc(int* words)
459 {
460  string desc = "";
461  int evLength = words[0];
462  // Assemble the descriptor from the hex
463  for (int i_wd = 4; i_wd < (evLength + 1); i_wd++) {
464  unsigned int number = words[i_wd];
465  for (int ii = 0; ii < 4; ii++) {
466  char word = (char)(number & 0xFF);
467  if (word == 0x22) word = 0x27; // replace " with '.
468  desc += word;
469  number = number >> 8;
470  }
471  }
472  run_data.run_desc = desc;
473  run_data.n_run_desc++;
474  if (dec_par.verb) cout << " run desc: " << desc.length() << " chars." << endl;
475  return 0;
476 }
477 
478 // Called in ProcessCodaPhysics() on PRESTART_INFO
479 // ProcessPhysEvent() is called in main() on PHYSICS_EVENT
480 // Never called as long as checking run 28700??
481 int MainDaqParser::ProcessPhysPrestart(int* words)
482 {
483  int evLength = words[0];
484  vector<string> list_line;
485  string line = "";
486  for (int i_wd = 4; i_wd < (evLength + 1); i_wd++) {
487  unsigned int number = words[i_wd];
488  for (int ii = 0; ii < 4; ii++) {
489  if ((number & 0xFF) == '\n') {
490  list_line.push_back(line);
491  line = "";
492  } else {
493  line += (char)(number & 0xFF);
494  }
495  number = number >> 8;
496  }
497  }
498  if (line.length() > 0) list_line.push_back(line);
499  else cerr << "Unexpectedly line.length() == 0." << endl;
500  for (unsigned int ii = 0; ii < list_line.size(); ii++) {
501  cout << " pre " << list_line[ii] << endl;
502  }
503  return 0;
504 }
505 
517 int MainDaqParser::ProcessPhysSlow(int* words)
518 {
519  if (dec_par.verb) cout << "Slow Cont @ coda " << dec_par.event_count << ": ";
520  int evLength = words[0];
521  dec_par.targPos_slow = 0;
522 
523  vector<string> list_line;
524  string line = "";
525  for (int i_wd = 4; i_wd < (evLength + 1); i_wd++) {
526  unsigned int number = words[i_wd];
527  for (int ii = 0; ii < 4; ii++) {
528  if ((number & 0xFF) == '\n') {
529  list_line.push_back(line);
530  line = "";
531  } else {
532  line += (char)(number & 0xFF);
533  }
534  number = number >> 8;
535  }
536  }
537  if (line.length() > 0) list_line.push_back(line);
538  else cout << "WARNING: Unexpectedly line.length() == 0." << endl;
539 
540  std::vector<SlowControlData> list_data; //< temporary list
541  for (unsigned int ii = 0; ii < list_line.size(); ii++) {
542  istringstream iss(list_line[ii]);
543  string ts, name, value, type;
544  iss >> ts >> name >> value >> type;
545  if (name == "TARGPOS_CONTROL" ) dec_par.targPos_slow = atoi(value.c_str());
546  else if (name == "local_spillcount") dec_par.spillID_slow = atoi(value.c_str());
547  //if (dec_par.verb) cout << " slow [" << ts << "] [" << name << "] [" << value << "] [" << type <<"]\n";
548  SlowControlData data;
549  data.ts = ts;
550  data.name = name;
551  data.value = value;
552  data.type = type;
553  list_data.push_back(data);
554  // Kenichi found on 2018-08-30 that the past (and current) decoder always miss the entry with name=U:TODB25
555  // since it value includes ' '. He believe it has to be fixed at the slowcontrol side (i.e. removing ' ').
556  }
557  for (unsigned int ii = 0; ii < list_data.size(); ii++) {
558  unsigned int spill_id = dec_par.spillID_slow - 1; // default value
559  SlowControlData* data = &list_data[ii];
560  if ( data->type == "target" &&
561  (data->name == "TARGPOS_CONTROL" ||
562  data->name.substr(0, 4) == "FREQ" ||
563  data->name.substr(0, 4) == "PROX" ||
564  data->name.substr(0, 5) == "TARG_" ) ) {
565  spill_id = dec_par.spillID_slow; // No "- 1" according to the original decoder.
566  }
567  SpillData* spill_data = &(*list_sd)[spill_id];
568  spill_data->spill_id = spill_id;
569  spill_data->list_slow_cont.push_back(*data); // put into the global list
570  spill_data->n_slow++;
571  }
572  if (dec_par.verb) cout << " spill " << dec_par.spillID_slow << ", target " << (short)dec_par.targPos_slow << endl;
573  // In the past decoder, almost all variables obtained here are inserted into
574  // the Beam, HV, Environment and Target tables according to "type".
575  return 0;
576 }
577 
582 int MainDaqParser::ProcessPhysSpillCounter(int* words)
583 {
584  int n_wd = words[0];
585  int i_wd_sc = 0;
586  char spillNum[12];
587  for (int i_wd = 4; i_wd < n_wd + 1; i_wd++) {
588  int number = words[i_wd];
589  for (int i_byte = 0; i_byte < 4; i_byte++) {
590  spillNum[i_wd_sc++] = (char)(number & 0xFF);
591  number = number >> 8;
592  }
593  }
594  spillNum[i_wd_sc] = '\0';
595 
599  dec_par.spillID_cntr = atoi(spillNum);
600  if (dec_par.verb) {
601  cout << "Spill Counter @ coda = " << dec_par.event_count << ": spill = " << dec_par.spillID_cntr << endl;
602  }
603  run_data.n_spill++;
604  return 0;
605 }
606 
607 int MainDaqParser::ProcessPhysBOSEOS(int* words, const int event_type)
608 {
609  int spill_type = (event_type == BEGIN_SPILL ? TYPE_BOS : TYPE_EOS);
610  string spill_type_str = (event_type == BEGIN_SPILL ? "BOS" : "EOS" );
611  if (spill_type == TYPE_BOS) {
612  dec_par.has_1st_bos = true;
613  dec_par.at_bos = true;
614  if (PackOneSpillData() != 0) {
615  cout << "Error in PackOneSpillData. Exiting..." << endl;
616  return 1;
617  }
618 
621  static int spillID_local = 0;
622  if (m_force_local_spill_id ||
623  (m_use_local_spill_id && dec_par.spillID_cntr == 0)) dec_par.spillID_cntr = ++spillID_local;
624 
628  SpillData* data = &(*list_sd)[dec_par.spillID];
629  data->spill_id = dec_par.spillID;
630  data->spill_id_slow = dec_par.spillID_slow;
631  data->run_id = dec_par.runID;
632  data->targ_pos = dec_par.targPos;
633 
636  dec_par.turn_id_max = 0;
637  } else { // EOS
638  m_timer_sp_input ->reset();
639  m_timer_sp_decode->reset();
640  }
641 
642  if (dec_par.verb) {
643  cout << spill_type_str << " @ coda " << dec_par.event_count << ": spill " << dec_par.spillID << "." << endl;
644  }
645  dec_par.spillType = spill_type;
646 
647  int idx = 7;
648  int evLength = words[0];
649  while (idx < evLength && (words[idx] & 0xfffff000) != 0xe906f000) {
650  int rocEvLength = words[idx];
651  int idx_roc_end = idx + rocEvLength; // inclusive endpoint
652  if ( (rocEvLength + idx) > evLength) {
653  cout << "Word limit error: " << rocEvLength << " + " << idx << " > " << evLength << endl;
654  return 1;
655  }
656  idx++;
657  int rocID = get_hex_bits (words[idx], 7, 4);
658  dec_par.rocID = rocID;
659  idx += 3;
660  int codaEvVmeTime = words[idx];
661  idx++;
662  if (rocID == 2) {
663  SpillData* data = &(*list_sd)[dec_par.spillID];
664  if (spill_type == TYPE_BOS) {
666  data->bos_vme_time = codaEvVmeTime;
667  data->n_bos_spill++;
668  } else {
670  data->eos_vme_time = codaEvVmeTime;
671  data->n_eos_spill++;
672  }
673  if (dec_par.verb > 2) cout << " " << spill_type_str << " spill: " << dec_par.spillID << " " << dec_par.runID << " " << dec_par.event_count << " " << (short)dec_par.targPos << " " << codaEvVmeTime << endl;
674  }
676  if (spill_type != TYPE_BOS && rocID == 25) idx = idx_roc_end + 1;
677 
680  if (rocEvLength <= 6) idx = idx_roc_end + 1;
681 
682  while (idx <= idx_roc_end) {
683  int e906flag = words[idx];
684  //cout << " " << spill_type_str << " " << idx << " 0x" << hex << e906flag << dec << endl;
685  idx++;
686  idx = ProcessBoardData (words, idx, idx_roc_end, e906flag, event_type);
687  if (idx == IDX_SKIP_EVENT) return 0;
688  else if (idx == IDX_SKIP_ROC ) idx = idx_roc_end + 1;
689  }
690  }
691  return 0;
692 }
693 
701 int MainDaqParser::ProcessPhysStdAndFlush(int* words, const int event_type)
702 {
703  const bool print_event = false;
704  int evLength = words[0];
705  //int codaEvVmeTime = 0;
706 
707  dec_err.SetFlushError(false);
708 
709  //PrintWords(words, 0, evLength+1);
710  if (print_event) cout << "\nEvent code = 0x" << hex << words[1] << dec << " (" << evLength << ")";
711 
712  int ret = 0;
713  int idx = 7; // the 1st word of ROC data.
714  while (idx < evLength) { // Loop over ROCs
715  int rocEvLength = words[idx];
716  //KN: In run 23751, ROC 2 often (always?) says "rocEvLength = 6" but
717  // the 1st board (0xe906f00f) says "boardEvLength = 512 or 1024".
718  // We should skip such a case. This problem is not seen in run 23930.
719  if (rocEvLength <= 6) { // Was "< 5" in the previous version.
720  idx += rocEvLength + 1;
721  continue; // Move to next ROC
722  } else if (get_hex_bits(rocEvLength, 7, 4) != 0) {
723  dec_err.SetFlushError(true);
724  cerr << "ERROR: rocEvLength != 0x0000****." << endl;
725  ret = -1;
726  break;
727  }
728 
729  // evLength is all the words summing all ROCs, and rocEvlLength is just for this ROC only.
730  // then, of course, evLength should be greater than rocEvLength
731  int idx_roc_end = idx + rocEvLength + 1; // this index itself is _not_ inside this ROC
732  if (idx_roc_end > evLength + 1) {
733  dec_err.SetFlushError(true);
734  cerr << "ERROR: ROC Event Length exceeds event length\n"
735  << " Event: " << dec_par.event_count << ", EventLength = " << evLength
736  << ", position = " << idx << ", rocEvLength = " << words[idx] << endl;
737  ret = -1;
738  break;
739  }
740 
741  idx++; // go to next position to get ROCID
742  int rocID = get_hex_bits (words[idx], 5, 2);
743  if (rocID > 32) {
744  dec_err.SetFlushError(true);
745  cerr << "ERROR: rocID > 32." << endl;
746  ret = -1;
747  break;
748  }
749  dec_par.rocID = rocID;
750  if (print_event) cout << "\n ROC " << setw(2) << rocID << " (" << setw(3) << rocEvLength << ") |";
751 
752  idx += 3; // move from "rocID" to "vmeTime"
753  //int vmeTime = words[idx];
754  //if (rocID == 2) codaEvVmeTime = vmeTime;
755 
756  idx++; // move to the 1st word of board data
757  while (idx < idx_roc_end) { // Loop over boards
758  int e906flag = words[idx];
759  if (get_hex_bits(e906flag, 7, 4) != (int)0xe906) {
760  cerr << "ERROR: Not e906flag (0x" << hex << e906flag << dec << ") " << event_type << endl;
761  if (dec_par.verb > 1) {
762  cout << " At idx = " << idx << " / " << evLength << ".\n";
763  PrintWords(words, 0, idx + 20);
764  }
765  idx = idx_roc_end; // try to move to the next ROC
766  break;
767  }
768  if (e906flag == (int)0xe906c0da) { // i.e. end of this ROC
769  idx++;
770  break;
771  }
772  idx++; // Move to the 1st word of board data
773  int board_id = get_hex_bits(words[idx], 6, 1);
774  int n_wd_bd = get_hex_bits(words[idx], 3, 4);
775  if (print_event) cout << hex << " " << (e906flag&0xFFFF) << "@" << board_id << dec << "(" << n_wd_bd << ")";
776  idx = ProcessBoardData(words, idx, idx_roc_end, e906flag, event_type);
777  if (idx == IDX_SKIP_EVENT) {
778  if (dec_par.verb > 1) cout << "ERROR: ProcessBoardData() returned IDX_SKIP_EVENT @ 0x" << hex << e906flag << dec << ", board " << board_id << ",roc " << rocID << ", coda " << dec_par.event_count << endl;
779  return 0;
780  } else if (idx == IDX_SKIP_ROC) {
781  if (dec_par.verb > 1) cout << "ERROR: ProcessBoardData() returned IDX_SKIP_ROC @ 0x" << hex << e906flag << dec << ", board " << board_id << ",roc " << rocID << ", coda " << dec_par.event_count << endl;
782  idx = idx_roc_end;
783  }
784  }
785  if (idx != idx_roc_end) {
786  cout << "ERROR: idx(" << idx << ") != idx_roc_end(" << idx_roc_end << ") @ roc " << rocID << ", coda " << dec_par.event_count << endl;
787  return 0;
788  }
789  }
790  if (print_event) cout << endl;
791 
793 
794  return ret;
795 }
796 
811 int MainDaqParser::ProcessBoardData (int* words, int idx, int idx_roc_end, int e906flag, const int event_type)
812 {
813  if (e906flag == (int)0xE906F003) { idx = ProcessBoardScaler (words, idx);
814  } else if (e906flag == (int)0xE906F005) {
815  if (event_type == FLUSH_EVENTS) idx = ProcessBoardV1495TDC (words, idx);
816  else idx = ProcessBoardStdV1495TDC(words, idx);
817  } else if (e906flag == (int)0xE906F018) { idx = ProcessBoardJyTDC2 (words, idx, idx_roc_end);
818  } else if (e906flag == (int)0xE906F019) { return IDX_SKIP_ROC;
819  } else if (e906flag == (int)0xe906f01b) { idx = ProcessBoardFeeQIE (words, idx);
820  } else if (e906flag == (int)0xE906F014) {
821  if (event_type == FLUSH_EVENTS) idx = ProcessBoardTriggerCount (words, idx);
822  else idx = ProcessBoardStdTriggerCount(words, idx);
823  } else if (e906flag == (int)0xE906F00F) {
824  if (event_type == FLUSH_EVENTS) idx = ProcessBoardTriggerBit (words, idx, idx_roc_end);
825  else idx = ProcessBoardStdTriggerBit(words, idx);
826  } else if (e906flag == (int)0xE906F010) { idx = ProcessBoardStdJyTDC2 (words, idx, idx_roc_end);
827  } else if (e906flag == (int)0xe906f013) { idx = ProcessBoardStdFeeQIE (words, idx);
828  } else {
829  cerr << "ERROR: Unknown board flag (0x" << hex << e906flag << dec << ") at coda " << dec_par.event_count << ", roc " << (int)dec_par.rocID << ", idx " << idx << endl;
830  PrintWords(words, idx-10, idx+40);
831  return IDX_SKIP_ROC; // Temporary solution. Skip all boards and move to the next ROC.
832  Abort("Unexpected board type.");
833  }
834  return idx;
835 }
836 
837 int MainDaqParser::ProcessBoardScaler (int* words, int idx)
838 {
839  int boardID = (0xFFFF & words[idx]);
840  idx++;
841 
842  SpillData* spill_data = &(*list_sd)[dec_par.spillID];
843  spill_data->n_scaler++;
844  for (int ii = 0; ii < 32; ii++) {
845  unsigned int value = words[idx];
846  idx++;
847 
848  ScalerData data;
849  data.type = dec_par.spillType;
850  data.coda = dec_par.event_count;
851  data.roc = dec_par.rocID;
852  data.board = boardID;
853  data.chan = ii;
854  data.value = value;
855  //if (! dec_par.map_scaler.Find(data.roc, data.board, data.chan, data.name)) {
856  if (! dec_par.chan_map_scaler.Find(data.roc, data.board, data.chan, data.name)) {
857  if (dec_par.verb > 3) cout << " Unmapped Scaler: " << data.roc << " " << data.board << " " << data.chan << "\n";
858  continue;
859  }
860  if (dec_par.verb > 3) cout << " scaler " << dec_par.spillID << " " << data.type << " " << data.name << " " << data.value << "\n";
861  spill_data->list_scaler.push_back(data);
862  }
863 
864  return idx;
865 }
866 
867 int MainDaqParser::ProcessBoardTriggerBit (int* words, int j, int idx_roc_end)
868 {
869  int n_words = words[j]; // N of words including this word itself (with an exception).
870  if (n_words == 0) return j+1; // Exception: n_words = 0 (not 1) in case of no event.
871  j++; // Move to the 1st event word.
872  n_words--; // Now it means N of trigger-bit & event-ID words.
873  //PrintWords(words, j-5, j+n_words+5);
874  if (n_words % 2 != 0) {
875  PrintWords(words, j-10, j+20);
876  Abort("N of trigger words % 2 != 0.");
877  }
878  int n_evt = n_words / 2; // because one event contains 1 trigger-bits & 1 event ID.
879  for (int i_evt = 0; i_evt < n_evt; i_evt++) {
880  int triggerBits = words[j + 2*i_evt ];
881  int evt_id = words[j + 2*i_evt + 1];
882  EventData* ed = &(*list_ed)[evt_id];
883  ed->n_trig_b++;
884  EventInfo* evt = &ed->event;
885  SetEventInfo(evt, evt_id);
886  evt->trigger_bits = triggerBits;
887 
888  int trigger[10];
889  for (int i = 0; i < 10; i++) trigger[i] = get_bin_bit (triggerBits, i);
890  for (int i = 0; i < 5; i++) {
891  // This function doesn't support the bit order for old runs (# < 4923).
892  evt->MATRIX[i] = trigger[i ];
893  evt->NIM [i] = trigger[i+5];
894  }
895  }
896 
897  return j + n_words;
898 }
899 
900 int MainDaqParser::ProcessBoardTriggerCount (int* words, int j)
901 {
902  int n_words = words[j]; // N of words including this word itself (with an exception).
903  if (n_words == 0) return j+1; // Exception: n_words = 0 (not 1) in case of no event.
904  j++; // Move to the 1st event word.
905  n_words--; // Now it means N of trigger-bit & event-ID words.
906 
907  if (n_words % 11 != 0) {
908  PrintWords(words, j-10, j+20);
909  Abort("N of trigger-count words % 11 != 0.");
910  }
911  int n_evt = n_words / 11; // because one event contains 10 counters & 1 event ID.
912  for (int i_evt = 0; i_evt < n_evt; i_evt++) {
913  int idx_evt = j + 11*i_evt; // The 1st index of this event
914  int evt_id = words[idx_evt + 10];
915  EventData* ed = &(*list_ed)[evt_id];
916  ed->n_trig_c++;
917  EventInfo* evt = &ed->event;
918  SetEventInfo(evt, evt_id);
919  for (int i_type = 0; i_type < 5; i_type++) {
920  evt->RawMATRIX [i_type] = words[idx_evt + i_type ];
921  evt->AfterInhMATRIX[i_type] = words[idx_evt + i_type + 5];
922  }
923  }
924 
925  return j + n_words;
926 }
927 
940 int MainDaqParser::ProcessBoardFeeQIE (int* words, int idx)
941 {
942  while (words[idx] == (int)0xe906e906) idx++; // Still necessary??
943 
944  // int boardID = get_hex_bits (words[idx], 7, 4);
945  int n_wd = get_hex_bits (words[idx], 3, 4);
946  int idx_end = idx + n_wd + 1; // It points to the 1st word of the next board.
947  idx++;
948  if (n_wd == 0) return idx;
949 
950  while (words[idx] == (int)0xe906e906) { idx++; idx_end++; }
951 
952  idx++; // skip the number-of-words word for this TDC
953 
954  // One physics event contains 49 words:
955  // * 1 word for the number of filled words (44 or 39)
956  // * [5 words for spill header] ... appear only in 1st event in spill
957  // * 8 words for 4 presums,
958  // * 2 words for trigger counts
959  // * 2 words for turn ID
960  // * 1 words for rf ID
961  // * 25 words for rf intensities
962  // * 4 or 9 words for padding with "0"
963  // * 1 word for event ID
964  // The number of physics events per Coda event is 7 in most cases, but can be less
965  // in last Coda event per spill. Such case is caught by "idx < idx_end".
966  for (int i_evt = 0; i_evt < 7 && idx < idx_end; i_evt++) {
967  int idx_evt = idx; // Increment idx_evt (not idx) and update idx at loop end.
968  int eventID = words[idx_evt + 48];
969 
970  int n_wd_evt = get_hex_bits(words[idx_evt], 7, 4);
971  if (n_wd_evt == 44) idx_evt += 5; // skip the spill header
972  else if (n_wd_evt != 39) {
974  //cerr << "!! QIE: Unexpected N of words: " << dec_par.event_count << " " << i_evt << " " << n_wd_evt << " (" << n_wd << ")." << endl;
975  return idx_end;
976  }
977 
978  idx_evt++; // Move to the 1st word of data block
979 
980  unsigned int sums_vals[4]; // QIE records four intensity sums (called "presum")
981  for (int i_sum = 0; i_sum < 4 ; i_sum++) {
982  sums_vals[i_sum] = get_hex_bits(words[idx_evt],7,4)*65536 + get_hex_bits(words[idx_evt+1],7,4);
983  idx_evt += 2;
984  }
985  //while (words[idx_evt] == (int)0xe906e906) { idx_evt++; idx_end++; }
986 
987  unsigned int triggerCount = words[idx_evt] | ( get_hex_bits (words[idx_evt+1], 7, 4) );
988  idx_evt += 2;
989  //while (words[idx_evt] == (int)0xe906e906) { idx_evt++; idx_end++; }
990 
991  unsigned int turnOnset = words[idx_evt] | ( get_hex_bits (words[idx_evt+1], 7, 4) );
992  if (turnOnset > dec_par.turn_id_max) dec_par.turn_id_max = turnOnset;
993  idx_evt += 2;
994  //while (words[idx_evt] == (int)0xe906e906) { idx_evt++; idx_end++; }
995 
996  unsigned int rfOnset = get_hex_bits(words[idx_evt],7,4);
997  idx_evt++;
998 
999  unsigned int rf_vals[25];
1000  for (int i_rf = 0; i_rf < 25; i_rf++) { // RF-12...RF+12
1001  if (words[idx_evt] == (int)0xe906e906) Abort("Unexpected 0xe906e906 in QIE.");
1002  rf_vals[i_rf] = ( get_hex_bits(words[idx_evt],7,4) );
1003  idx_evt++;
1004  }
1005 
1006  EventData* ed = &(*list_ed)[eventID];
1007  ed->n_qie++;
1008  EventInfo* evt = &ed->event;
1009  //if (ed->n_qie == 2) {
1010  // cout << "QIE#1 " << dec_par.event_count << " " << i_evt << " " << evt->eventID << " " << evt->spillID << " "
1011  // << evt->triggerCount << " " << evt->turnOnset << " " << evt->rfOnset << endl;
1012  //}
1013  SetEventInfo(evt, eventID);
1014 
1015  for (int ii = 0; ii < 4; ii++) evt->sums[ii] = sums_vals[ii];
1016  evt->triggerCount = triggerCount;
1017  evt->turnOnset = turnOnset;
1018  evt->rfOnset = rfOnset;
1019  for (int ii = 0; ii < 25 ; ii++) evt->rf[ii+4] = rf_vals[ii]; // We are now missing first 4 bins and last 4 bins. YC, 2016/05/27
1020 
1021  idx += 49;
1022  }
1023 
1024  if (idx != idx_end) {
1025  cout << idx << " != " << idx_end;
1026  Abort("idx != idx_end in feeQIE().");
1027  }
1028  return idx_end;
1029 }
1030 
1037 int MainDaqParser::ProcessBoardV1495TDC (int* words, int idx)
1038 {
1039  int idx_begin = idx;
1040  if (dec_par.verb>3) {
1041  cout << "ProcessBoardV1495TDC(): idx = " << idx << endl;
1042  PrintWords(words, idx-5, idx+20);
1043  }
1044 
1045  int boardID = words[idx++]; // was get_hex_bits (words[idx], 3, 4);
1046  int n_wd_fpga = words[idx++]; // N of words taken from FPGA buffer.
1047  if (n_wd_fpga == 0) return idx; // No event data. Just finish.
1048  if (words[idx] == (int)0xe9060bad) {
1050  return idx+1;
1051  }
1052  int idx_end = idx + n_wd_fpga; // Exclusive endpoint. To be incremented in the while loop.
1053 
1054  int i_evt = 0;
1055  vector<int> list_chan;
1056  vector<int> list_time;
1057  while (idx < idx_end) {
1058  if (get_hex_bits(words[idx], 7, 1) == 1) { // event header
1059  idx_end += 2; // one stop word and one event-ID word (which was added by CPU, not FPGA).
1060  int word_stop = words[idx+1];
1061  int evt_id = words[idx+2];
1064  int evt_id_fpga = (words[idx+3]<<15) + words[idx+4];
1065  if (evt_id != evt_id_fpga) {
1066  cerr << "!! EventID mismatch @ v1495: " << evt_id << "@CPU vs " << evt_id_fpga << "@FPGA at event " << dec_par.event_count << ":" << i_evt << " board 0x" << hex << boardID << dec << endl;
1067  }
1068 
1069  EventData* ed = &(*list_ed)[evt_id];
1070  ed->n_v1495++;
1071  EventInfo* evt = &ed->event;
1072  SetEventInfo(evt, evt_id);
1073 
1074  if (word_stop == (int)0xd2ad) {
1075  evt->flag_v1495 |= 0x2;
1076  run_data.n_v1495_d2ad++;
1077  } else if (word_stop == (int)0xd3ad) {
1078  evt->flag_v1495 |= 0x4;
1079  run_data.n_v1495_d3ad++;
1080  } else if (get_hex_bits(word_stop, 3, 1) != 1) {
1081  evt->flag_v1495 |= 0x8;
1082  //cerr << " !! v1495: bad stop word: " << word_stop << endl;
1083  } else { // OK. Insert hits.
1084  int time_stop = get_hex_bits(word_stop, 2, 3);
1085  for (unsigned int ii = 0; ii < list_chan.size(); ii++) {
1086  HitData hit;
1087  hit.event = evt_id;
1088  hit.id = ++dec_par.hitID;
1089  hit.roc = dec_par.rocID;
1090  hit.board = boardID;
1091  hit.chan = list_chan[ii];
1092  hit.time = time_stop - list_time[ii];
1093  ed->list_hit_trig.push_back(hit);
1094  }
1095  }
1096  list_chan.clear();
1097  list_time.clear();
1098  run_data.n_v1495++;
1099  idx += 5;
1100  i_evt++;
1101  } else { // start signal
1103  list_chan.push_back( get_hex_bits (words[idx], 3, 2) );
1104  list_time.push_back( get_hex_bits (words[idx], 1, 2) );
1105  idx++;
1106  }
1107  }
1108  if (idx != idx_end) {
1109  cout << "ProcessBoardV1495TDC(): idx != idx_end (" << idx << " != " << idx_end << ") @ coda=" << dec_par.event_count << " roc=" << dec_par.rocID << " board=" << hex << boardID << dec << "\n"
1110  << " idx_begin=" << idx_begin << " n_wd_fpga=" << n_wd_fpga << endl;
1111  PrintWords(words, idx_begin, idx);
1112  return IDX_SKIP_ROC;
1113  //exit(1);
1114  //Abort("idx != idx_end in ProcessBoardV1495TDC.");
1115  }
1116  return idx_end;
1117 }
1118 
1140 int MainDaqParser::ProcessBoardJyTDC2 (int* words, int idx_begin, int idx_roc_end)
1141 {
1142  int hex7 = get_hex_bits (words[idx_begin], 7, 1);
1143  int hex54 = get_hex_bits (words[idx_begin], 5, 2);
1144  int boardID = get_hex_bits (words[idx_begin], 6, 3);
1145  int nWordsBoard = get_hex_bits (words[idx_begin], 3, 4);
1146  int roc = (int)dec_par.rocID;
1147  //if (roc < 12 || roc > 30) Abort("rocID out of 12...30.");
1148  if (nWordsBoard == 0) {
1149  if (dec_par.verb > 5) cerr << "ERROR: nWordsBoard==0: coda " << dec_par.event_count << ", roc " << (int)dec_par.rocID << ".\n";
1150  return idx_begin + 1;
1151  }
1152 
1153  if (hex7 != 0 || hex54 != 0) {
1154  // According to Kun on 2017-Jan-01, the board sometimes gets to output only
1155  // "0x8*******" or "0x9*******". This "if" condition is strict enough to
1156  // catch this error. Shifter has to reset VME crate to clear this error.
1157  if (dec_par.verb > 5) cerr << "ERROR: WORD_ONLY89: coda " << dec_par.event_count << ", roc " << (int)dec_par.rocID << ".\n";
1159  return idx_begin + 1;
1160  }
1161 
1163  int idx_events_begin = idx_begin + 2;
1164  int idx_events_end = idx_events_begin + nWordsBoard - 1; // "-1" to exclude the word "X".
1165  if (words[idx_begin + 1] == (int)0xe906e906) { // Dummy word *could* exist. Skip it
1166  idx_events_begin++;
1167  idx_events_end ++;
1168  }
1169  if (idx_events_end > idx_roc_end) {
1170  if (dec_par.verb > 5) {
1171  cerr << "ERROR: WORD_OVERFLOW: coda " << dec_par.event_count << ", roc " << (int)dec_par.rocID << ", event_end " << idx_events_end << " vs roc_end " << idx_roc_end << ".\n";
1172  PrintWords(words, idx_events_begin, idx_events_end);
1173  }
1175  return IDX_SKIP_ROC;
1176  }
1177 
1178  int i_evt = 0;
1179  bool header_found = false;
1180  double time_header = 0;
1181  vector<int > list_chan;
1182  vector<double> list_time;
1183  for (int idx = idx_events_begin; idx < idx_events_end; idx++) {
1184  if (get_hex_bits(words[idx], 7, 1) == 8) { // header = stop hit
1185  if (header_found) { // Not seen in run 23930, but seen in run 23751.
1186  if (dec_par.verb > 5) cerr << "ERROR: MULTIPLE_HEADER: coda " << dec_par.event_count << ", roc " << (int)dec_par.rocID << ".\n";
1188  }
1189  int word = words[idx];
1190  if (get_bin_bit(word, 16) == 0) Abort("Stop signal is not rising. Not supported.");
1191  double fine = 4.0 - get_hex_bit (word, 0) * 4.0 / 9.0;
1192  double rough = 4.0 * get_hex_bits(word, 3, 3);
1193  time_header = fine + rough;
1194  header_found = true;
1195  } else if (get_hex_bits(words[idx], 7, 1) == 0 &&
1196  get_hex_bits(words[idx], 6, 7) != 0 ) { // event ID
1197  if (! header_found) { // Not seen in run 23930, but seen in run 23751.
1198  if (dec_par.verb > 5) cerr << "ERROR: EVT_ID_ONLY: coda " << dec_par.event_count << ", roc " << (int)dec_par.rocID << ".\n";
1199  dec_err.AddTdcError(dec_par.event_count, roc, DecoError::EVT_ID_ONLY); // eventID without stop word
1200  run_data.n_hit_bad++;
1201  continue;
1202  }
1203  int evt_id = words[idx];
1204  EventData* ed = &(*list_ed)[evt_id];
1205  ed->n_tdc++;
1206  EventInfo* evt = &ed->event;
1207  SetEventInfo(evt, evt_id);
1208 
1210  for (unsigned int ii = 0; ii < list_chan.size(); ii++) {
1211  HitData hit;
1212  hit.event = evt_id;
1213  hit.id = ++dec_par.hitID;
1214  hit.roc = roc;
1215  hit.board = boardID;
1216  hit.chan = list_chan[ii];
1217  hit.time = list_time[ii];
1218  ed->list_hit.push_back(hit);
1219  //cout << " HIT " << dec_par.spillID << " " << evt_id << " " << (int)dec_par.rocID << " " << boardID << " " << list_chan[ii] << " " << list_time[ii] << endl;
1220  }
1221  i_evt++;
1222  header_found = false;
1223  list_chan.clear();
1224  list_time.clear();
1225  } else { // start hit
1226  if (! header_found) { // Not seen in run 23930, but seen in run 23751.
1227  if (dec_par.verb > 5) cerr << "ERROR: START_WO_STOP: coda " << dec_par.event_count << ", roc " << (int)dec_par.rocID << ".\n";
1228  dec_err.AddTdcError(dec_par.event_count, roc, DecoError::START_WO_STOP); // Start without stop word
1229  run_data.n_hit_bad++;
1230  continue;
1231  }
1232  int word = words[idx];
1233  if (get_bin_bit(word, 16) == 0) { // Not seen in run 23930, but seen in run 23751.
1234  if (dec_par.verb > 5) cerr << "ERROR: START_NOT_RISE: coda " << dec_par.event_count << ", roc " << (int)dec_par.rocID << ".\n";
1235  dec_err.AddTdcError(dec_par.event_count, roc, DecoError::START_NOT_RISE); // Start signal is not rising
1236  run_data.n_hit_bad++;
1237  }
1238  double fine = 4.0 - get_hex_bit (word, 0) * 4.0 / 9.0;
1239  double rough = 4.0 * get_hex_bits(word, 3, 3);
1240  double time = time_header - (fine + rough);
1241  if (time < 0) time += 4096; // This constant 4096 is correct???
1242  list_time.push_back(time);
1243 
1244  int cable_chan = get_hex_bit (word, 6);
1245  int cable_id = get_bin_bits(word, 29, 2);
1246  int chan = cable_chan + 16*cable_id;
1247  list_chan.push_back(chan);
1248  }
1249  }
1250  if (header_found) { // Not seen in run 23930, but seen in run 23751.
1251  if (dec_par.verb > 5) cerr << "ERROR: DIRTY_FINISH: coda " << dec_par.event_count << ", roc " << (int)dec_par.rocID << ".\n";
1252  dec_err.AddTdcError(dec_par.event_count, roc, DecoError::DIRTY_FINISH); // Not finished cleanly
1253  }
1254 
1255  return idx_events_end;
1256 }
1257 
1262 int MainDaqParser::ProcessBoardStdTriggerBit (int* words, int idx)
1263 {
1264 // int evt_id = words[idx];
1265 // int triggerBits = words[idx+1];
1266 // EventData* ed = &(*list_ed)[dec_par.eventIDstd];
1267 // ed->n_trig_b++;
1268 // EventInfo* evt = &ed->event;
1269 // evt->trigger_bits = triggerBits;
1270 // for (int i = 0; i < 5; i++) {
1271 // // This function doesn't support the bit order for old runs (# < 4923).
1272 // evt->MATRIX[i] = get_bin_bit (triggerBits, i );
1273 // evt->NIM [i] = get_bin_bit (triggerBits, i+5);
1274 // }
1275  return idx + 2;
1276 }
1277 
1278 int MainDaqParser::ProcessBoardStdTriggerCount (int* words, int idx)
1279 {
1280  EventData* ed = &(*list_ed)[dec_par.eventIDstd];
1281  ed->n_trig_c++;
1282  EventInfo* evt = &ed->event;
1283 
1284  for (int ii = 0; ii < 5; ii++) {
1285  evt->RawMATRIX [ii] = words[idx + ii ];
1286  evt->AfterInhMATRIX[ii] = words[idx + ii + 5];
1287  }
1288 
1289  return idx + 10;
1290 }
1291 
1297 int MainDaqParser::ProcessBoardStdFeeQIE (int* words, int idx)
1298 {
1299  if (dec_par.runID < 22400) Abort("StdFeeQIE does not support run < 22400.");
1300 
1301  while (words[idx] == (int)0xe906e906) idx++; // Still necessary??
1302 
1303  // int boardID = get_hex_bits (words[idx], 7, 4);
1304  int n_wd = get_hex_bits (words[idx], 3, 4);
1305  int idx_end = idx + n_wd + 1; // It points to the 1st word of the next board.
1306  idx++;
1307  if (n_wd == 0) return idx;
1308 
1309  while (words[idx] == (int)0xe906e906) { idx++; idx_end++; }
1310 
1311  int n_wd2 = get_hex_bits(words[idx], 7, 4);
1312  if (n_wd2 == 0x2C) idx += 5;
1313  idx++; // skip the number-of-words word for this TDC
1314 
1315  EventData* ed = &(*list_ed)[dec_par.eventIDstd];
1316  ed->n_qie++;
1317  EventInfo* evt = &ed->event;
1318 
1319  // QIE records four intensity sums (called "presum")
1320  for (int i_sum = 0; i_sum < 4 ; i_sum++) {
1321  evt->sums[i_sum] = get_hex_bits(words[idx],7,4)*65536 + get_hex_bits(words[idx+1],7,4);
1322  idx += 2;
1323  }
1324  //while (words[idx_evt] == (int)0xe906e906) { idx_evt++; idx_end++; }
1325 
1326  evt->triggerCount = words[idx] | ( get_hex_bits (words[idx+1], 7, 4) );
1327  idx += 2;
1328  //while (words[idx_evt] == (int)0xe906e906) { idx_evt++; idx_end++; }
1329 
1330  evt->turnOnset = words[idx] | ( get_hex_bits (words[idx+1], 7, 4) );
1332  idx += 2;
1333  //while (words[idx_evt] == (int)0xe906e906) { idx_evt++; idx_end++; }
1334 
1335  evt->rfOnset = get_hex_bits(words[idx],7,4);
1336  idx++;
1337 
1338  for (int i_rf = 0; i_rf < 25; i_rf++) { // RF-12...RF+12
1339  if (words[idx] == (int)0xe906e906) Abort("Unexpected 0xe906e906 in QIE.");
1340  evt->rf[i_rf+4] = get_hex_bits(words[idx],7,4);
1341  idx++;
1342  }
1343 
1344  if (idx != idx_end) {
1345  cout << idx << " != " << idx_end;
1346  Abort("idx != idx_end in StdFeeQIE().");
1347  }
1348  return idx_end;
1349 }
1350 
1354 int MainDaqParser::ProcessBoardStdV1495TDC (int* words, int idx)
1355 {
1356  EventData* ed = &(*list_ed)[dec_par.eventIDstd];
1357  ed->n_v1495++;
1358  EventInfo* evt = &ed->event;
1359 
1360  int boardID = words[idx++]; // was get_hex_bits (words[idx], 3, 4);
1361 
1362  // Check to see if there is an error in next word
1363  // If so, don't submit the information from this board for this event
1364  bool ignore = (get_bin_bit(words[idx + 1], 12) != 1);
1365 
1366  if ((words[idx] & 0xFFFF) == (int)0xd1ad) {
1367  evt->flag_v1495 |= 0x1;
1368  run_data.n_v1495_d1ad++;
1369  }
1370  if ((words[idx + 1] & 0xFFFF) == (int)0xd2ad) {
1371  evt->flag_v1495 |= 0x2;
1372  run_data.n_v1495_d2ad++;
1373  } else if ((words[idx + 1] & 0xFFFF) == (int)0xd3ad) {
1374  evt->flag_v1495 |= 0x4;
1375  run_data.n_v1495_d3ad++;
1376  }
1377  run_data.n_v1495++;
1378 
1379  // Contains 0x000000xx where xx is number of channel entries
1380  int numChannels = (words[idx] & 0x0000FFFF);
1381  if (numChannels > 255) {
1382  //printf("number of Channels stated in v1495 readout exceeds 255. Skipping...\n");
1383  return idx + 2;
1384  }
1385  idx++;
1386 
1387  // Next contains 0x00001xxx where xxx is the stop time
1388  unsigned int stopTime = (words[idx] & 0x00000FFF);
1389  idx++;
1390 
1391  if (! ignore) {
1392  HitData hit;
1393  hit.event = dec_par.eventIDstd;
1394  hit.roc = dec_par.rocID;
1395  hit.board = boardID;
1396 
1397  for (int kk = 0; kk < numChannels; kk++) {
1398  unsigned int channel = get_hex_bits(words[idx+kk], 3, 2);
1399  unsigned int channelTime = get_hex_bits(words[idx+kk], 1, 2);
1400  double tdcTime = stopTime - channelTime;
1401  hit.id = ++dec_par.hitID;
1402  hit.chan = channel;
1403  hit.time = tdcTime;
1404  ed->list_hit_trig.push_back(hit);
1405  }
1406  }
1407 
1408  return idx + numChannels;
1409 }
1410 
1417 int MainDaqParser::ProcessBoardStdJyTDC2 (int* words, int idx_begin, int idx_roc_end)
1418 {
1419  int boardID = get_hex_bits (words[idx_begin], 6, 3);
1420  int nWordsBoard = get_hex_bits (words[idx_begin], 3, 4);
1421  if (nWordsBoard == 0) return idx_begin + 1;
1422  int idx_end = idx_begin + nWordsBoard;
1423  if (idx_end > idx_roc_end) {
1424  cerr << "WARNING: Word overflow. Skip ROC (" << dec_par.rocID << ")" << endl;
1425  return IDX_SKIP_ROC;
1426  }
1427 
1428  int idx = idx_begin + 1;
1429  if (words[idx] == (int)0xe906e906) idx++; // Dummy word *could* exist. Skip it
1430 
1431  int trigger_clock_cycle = get_hex_bit(words[idx], 0);
1432  double trigger_rough = 4.0 * get_hex_bits(words[idx], 3, 3);
1433  double trigger_fine;
1434  if (get_bin_bit(words[idx], 16)) { // rising-edge mode
1435  trigger_fine = 4.0 - trigger_clock_cycle * 4.0 / 9.0;
1436  } else {
1437  trigger_fine = 0.5 * trigger_clock_cycle;
1438  }
1439  double trigger_time = trigger_rough + trigger_fine;
1440  idx++;
1441 
1442  EventData* ed = &(*list_ed)[dec_par.eventIDstd];
1443  ed->n_tdc++;
1444  HitData hit;
1445  hit.event = dec_par.eventIDstd;
1446  hit.roc = (int)dec_par.rocID;
1447  hit.board = boardID;
1448 
1449  while (idx < idx_end) {
1450  bool rising = get_bin_bit(words[idx], 16);
1451  if (rising && (words[idx] & 0x80000000) == 0) {
1452  double fine = 4.0 - get_hex_bit (words[idx], 0) * 4.0 / 9.0;
1453  double rough = 4.0 * get_hex_bits(words[idx], 3, 3);
1454  double time = trigger_time - (fine + rough);
1455  if (time < 0) time += 4096;
1456 
1457  int cable_chan = get_hex_bit (words[idx], 6);
1458  int cable_id = get_bin_bits(words[idx], 29, 2);
1459  int chan = cable_chan + 16*cable_id;
1460 
1461  hit.id = ++dec_par.hitID;
1462  hit.chan = chan;
1463  hit.time = time;
1464  ed->list_hit.push_back(hit);
1465  }
1466  idx++;
1467  }
1468 
1469  return idx_end;
1470 }
1471 
1472 int MainDaqParser::PackOneSpillData()
1473 {
1474  if (dec_par.verb > 2) cout << "PackOneSpillData(): n=" << list_ed->size() << endl;
1475  m_timers["pack_one_spill_data"]->restart();
1476  m_timer_sp_map->reset_and_start();
1477 
1478  if (dec_par.is_online) {
1481  }
1482 
1483  sd_now = &(*list_sd)[dec_par.spillID];
1484  run_data.n_evt_all += list_ed->size();
1485 
1488  for (EventDataMap::iterator it = list_ed->begin(); it != list_ed->end(); ) {
1489  unsigned int evt_id = it->first;
1490  EventData* ed = &it->second;
1491  EventInfo* event = &ed->event;
1492  if (evt_id == 0 || // 1st event? bad anyway
1493  (dec_par.sampling > 0 && evt_id % dec_par.sampling != 0) || // sampled out
1494  (dec_par.turn_id_max > 360000 && event->turnOnset == 0 && event->NIM[2])) { // NIM3 after spill end
1495  it = list_ed->erase(it);
1496  continue;
1497  }
1498  it++;
1499  run_data.n_evt_dec++;
1500 
1501  unsigned int n_taiwan = ed->list_hit .size();
1502  unsigned int n_v1495 = ed->list_hit_trig.size();
1503  run_data.n_hit += n_taiwan;
1504  run_data.n_t_hit += n_v1495;
1505 
1508  m_timers["map_chan"]->restart();
1509  for (unsigned int ih = 0; ih < n_taiwan; ih++) {
1510  HitData* hd = &ed->list_hit[ih];
1511  if (! dec_par.chan_map_taiwan.Find(hd->roc, hd->board, hd->chan, hd->det, hd->ele)) {
1512  if (dec_par.verb > 3) cout << " Unmapped Taiwan: " << hd->roc << " " << hd->board << " " << hd->chan << "\n";
1513  }
1514  }
1515  for (unsigned int ih = 0; ih < n_v1495; ih++) {
1516  HitData* hd = &ed->list_hit_trig[ih];
1517  if (! dec_par.chan_map_v1495.Find(hd->roc, hd->board, hd->chan, hd->det, hd->ele, hd->lvl)) {
1518  if (dec_par.verb > 3) cout << " Unmapped v1495: " << hd->roc << " " << hd->board << " " << hd->chan << "\n";
1519  }
1520  }
1521  m_timers["map_chan"]->stop();
1522  }
1523 
1524  EventDataMap* ptr = list_ed_now;
1525  list_ed_now = list_ed;
1526  list_ed = ptr;
1527 
1528  m_timers["pack_one_spill_data"]->stop();
1529  m_timer_sp_map->stop();
1530 
1531  sd_now->time_input = m_timer_sp_input ->get_accumulated_time();
1532  sd_now->time_decode = m_timer_sp_decode->get_accumulated_time();
1533  sd_now->time_map = m_timer_sp_map ->get_accumulated_time();
1534 
1535  return 0;
1536 }
1537 
1538 void MainDaqParser::SetEventInfo(EventInfo* evt, const int eventID)
1539 {
1540  evt->runID = dec_par.runID;
1541  evt->eventID = eventID;
1542  evt->spillID = dec_par.spillID;
1543  if (evt->codaEventID == 0) {
1545  } else if ((unsigned int)evt->codaEventID != dec_par.event_count) {
1546  if (dec_par.verb > 10) cout << " CodaEventID Mismatch @ eventID " << eventID << ", rocID " << dec_par.rocID << ": " << evt->codaEventID << " vs " << dec_par.event_count << "\n";
1547  if ((unsigned int)evt->codaEventID > dec_par.event_count) dec_par.event_count = evt->codaEventID;
1548  }
1549 }
int get_hex_bit(unsigned int hexNum, int numBitFromRight)
int get_bin_bit(unsigned int binNum, int numBitFromRight)
void PrintCodaEventSummary(int *words)
void PrintWords(int *words, int idx_begin, int idx_end, int idx_atte)
int get_hex_bits(unsigned int hexNum, int numBitFromRight, int numBits)
void Abort(const char *message)
int get_bin_bits(unsigned int binNum, int numBitFromRight, int numBits)
@ FEE_PREFIX
@ FLUSH_EVENTS
@ SLOW_CONTROL
@ END_EVENT
@ PHYSICS_EVENT
@ END_SPILL
@ CODA_EVENT
@ BEGIN_SPILL
@ PRESTART_INFO
@ GO_EVENT
@ RUN_DESCRIPTOR
@ FEE_EVENT
@ PRESTART_EVENT
@ STANDARD_PHYSICS
@ SPILL_COUNTER
std::map< unsigned int, EventData > EventDataMap
Definition: DecoData.h:224
std::map< unsigned int, SpillData > SpillDataMap
Definition: DecoData.h:166
#define NULL
Definition: Pdb.h:9
bool Find(const short roc, const short board, const short chan, std::string &name)
bool Find(const short roc, const short board, const short chan, short &det, short &ele)
bool Find(const short roc, const int board, const short chan, short &det, short &ele, short &lvl)
int OpenFile(const std::string fname, const long file_size_min=0, const int sec_wait=10, const int n_wait=0)
bool JumpCodaEvent(unsigned int &event_count, int *&event_words, const unsigned int n_evt)
bool NextCodaEvent(unsigned int &event_count, int *&event_words)
void SetRunNumber(const int run)
void AggregateData()
Definition: DecoError.cc:48
void PrintData(std::ostream &os=std::cout)
Definition: DecoError.cc:56
void CountFlush()
Definition: DecoError.cc:35
void InitData()
Definition: DecoError.cc:24
void AddTdcError(const int event, const int roc, const TdcError_t type)
Definition: DecoError.cc:41
void SetFlushError(const bool val)
Definition: DecoError.h:44
bool GetFlushError()
Definition: DecoError.h:45
@ V1495_0BAD
Definition: DecoError.h:25
@ WORD_ONLY89
Definition: DecoError.h:17
@ EVT_ID_ONLY
Definition: DecoError.h:20
@ START_NOT_RISE
Definition: DecoError.h:22
@ DIRTY_FINISH
Definition: DecoError.h:23
@ START_WO_STOP
Definition: DecoError.h:21
@ WORD_OVERFLOW
Definition: DecoError.h:18
@ MULTIPLE_HEADER
Definition: DecoError.h:19
void SetID(const int run_id, const int spill_id)
Definition: DecoError.cc:18
bool NextPhysicsEvent(EventData *&ed, SpillData *&sd, RunData *&rd)
DecoError dec_err
Definition: MainDaqParser.h:97
DecoParam dec_par
Definition: MainDaqParser.h:96
int OpenCodaFile(const std::string fname, const long file_size_min=32768, const int sec_wait=15, const int n_wait=40)
Class to measure the time spent by a code block.
Definition: PHTimer2.h:17
double get_accumulated_time() const
Definition: PHTimer2.h:34
void reset()
Definition: PHTimer2.cc:26
void reset_and_start()
Definition: PHTimer2.cc:48
void restart()
Definition: PHTimer2.cc:42
void stop()
Definition: PHTimer2.cc:33
unsigned short int runID
Definition: DecoParam.h:32
bool has_1st_bos
Set 'true' at the 1st BOS event.
Definition: DecoParam.h:47
int spillID_slow
Definition: DecoParam.h:36
int codaID
current Coda event ID
Definition: DecoParam.h:41
int time_wait
waiting time in second to pretend the online data flow.
Definition: DecoParam.h:23
long int hitID
current hit ID, commonly used by Hit and TriggerHit.
Definition: DecoParam.h:45
int InitMapper()
Definition: DecoParam.cc:32
int spillID_cntr
Definition: DecoParam.h:35
bool at_bos
Set 'true' at BOS, which triggers parsing the data in one spill.
Definition: DecoParam.h:48
bool is_online
Definition: DecoParam.h:20
int spillID
Definition: DecoParam.h:34
time_t timeEnd
Definition: DecoParam.h:54
ChanMapV1495 chan_map_v1495
Definition: DecoParam.h:26
short targPos_slow
Definition: DecoParam.h:38
short spillType
current spill type
Definition: DecoParam.h:42
short rocID
current ROC ID
Definition: DecoParam.h:43
short targPos
Definition: DecoParam.h:37
time_t timeStart
Definition: DecoParam.h:54
std::string fn_in
Definition: DecoParam.h:18
ChanMapTaiwan chan_map_taiwan
Definition: DecoParam.h:25
int verb
Verbosity. 0 = error, 1 = warning, 2 = info, 3 = debug, 4 = insane.
Definition: DecoParam.h:22
ChanMapScaler chan_map_scaler
Definition: DecoParam.h:27
int sampling
Definition: DecoParam.h:21
unsigned int event_count
current event count
Definition: DecoParam.h:40
int eventIDstd
current event ID of standard physics events
Definition: DecoParam.h:44
unsigned int turn_id_max
Max turnOnset in a spill. Used to drop NIM3 events that came after the beam stops....
Definition: DecoParam.h:51
unsigned int n_tdc
Definition: DecoData.h:213
HitDataList list_hit
Definition: DecoData.h:218
unsigned int n_qie
Definition: DecoData.h:211
EventInfo event
Definition: DecoData.h:217
unsigned int n_trig_c
Definition: DecoData.h:215
unsigned int n_v1495
Definition: DecoData.h:212
unsigned int n_trig_b
Definition: DecoData.h:214
HitDataList list_hit_trig
Definition: DecoData.h:219
int RawMATRIX[5]
Definition: DecoData.h:194
int runID
Definition: DecoData.h:190
unsigned int rf[33]
Definition: DecoData.h:203
unsigned int rfOnset
Definition: DecoData.h:202
int eventID
Definition: DecoData.h:188
unsigned int sums[4]
Definition: DecoData.h:199
int MATRIX[5]
Definition: DecoData.h:197
unsigned int triggerCount
Definition: DecoData.h:200
unsigned int turnOnset
Definition: DecoData.h:201
int codaEventID
Definition: DecoData.h:189
short flag_v1495
Definition: DecoData.h:204
int AfterInhMATRIX[5]
Definition: DecoData.h:195
int trigger_bits
Definition: DecoData.h:198
int spillID
Definition: DecoData.h:191
int NIM[5]
Definition: DecoData.h:196
unsigned int lowLimit
Definition: DecoData.h:63
unsigned int roc
Definition: DecoData.h:55
unsigned int board
Definition: DecoData.h:56
unsigned int updating_enabled
Definition: DecoData.h:61
unsigned int segmentation
Definition: DecoData.h:59
unsigned int falling_enabled
Definition: DecoData.h:58
unsigned int highLimit
Definition: DecoData.h:64
unsigned int hard
Definition: DecoData.h:57
unsigned int elim_window
Definition: DecoData.h:62
unsigned int selectWindow
Definition: DecoData.h:65
unsigned int multihit_elim_enabled
Definition: DecoData.h:60
short roc
Definition: DecoData.h:175
double time
Definition: DecoData.h:181
int id
Definition: DecoData.h:174
short chan
Definition: DecoData.h:177
int board
Definition: DecoData.h:176
int event
Definition: DecoData.h:173
short ele
Definition: DecoData.h:179
short lvl
Definition: DecoData.h:180
short det
Definition: DecoData.h:178
int utime_b
Definition: DecoData.h:74
int v1495_id[10]
Definition: DecoData.h:84
int n_v1495
Definition: DecoData.h:102
int prescale[8]
Definition: DecoData.h:83
int run_id
Definition: DecoData.h:73
int n_fee_prescale
Definition: DecoData.h:89
int n_t_hit
Definition: DecoData.h:99
int n_phys_evt
Definition: DecoData.h:94
int trig_bit[10]
Definition: DecoData.h:82
int n_phys_evt_bad
Definition: DecoData.h:95
int n_hit_bad
Definition: DecoData.h:100
int n_hit
Definition: DecoData.h:98
FeeDataList fee
Definition: DecoData.h:85
int n_spill
Definition: DecoData.h:91
int fpga_enabled[5]
Definition: DecoData.h:77
int n_evt_dec
Definition: DecoData.h:93
int n_v1495_d3ad
Definition: DecoData.h:105
int n_t_hit_bad
Definition: DecoData.h:101
int n_flush_evt_bad
Definition: DecoData.h:97
int n_v1495_d2ad
Definition: DecoData.h:104
int utime_e
Definition: DecoData.h:75
int n_flush_evt
Definition: DecoData.h:96
int n_fee_event
Definition: DecoData.h:88
int n_v1495_d1ad
Definition: DecoData.h:103
int n_run_desc
Definition: DecoData.h:90
int fpga_prescale[5]
Definition: DecoData.h:79
int n_evt_all
Definition: DecoData.h:92
std::string run_desc
Definition: DecoData.h:86
short type
Definition: DecoData.h:128
std::string name
Definition: DecoData.h:134
short chan
Definition: DecoData.h:132
int coda
Definition: DecoData.h:129
short board
Definition: DecoData.h:131
int value
Definition: DecoData.h:133
short roc
Definition: DecoData.h:130
std::string name
Definition: DecoData.h:118
std::string type
Definition: DecoData.h:120
std::string ts
Definition: DecoData.h:117
std::string value
Definition: DecoData.h:119
double time_input
In msec.
Definition: DecoData.h:156
unsigned int eos_vme_time
Definition: DecoData.h:149
ScalerDataList list_scaler
Definition: DecoData.h:161
unsigned int bos_coda_id
Definition: DecoData.h:146
unsigned int n_bos_spill
Definition: DecoData.h:151
unsigned int n_scaler
Definition: DecoData.h:154
unsigned int spill_id
Definition: DecoData.h:142
unsigned int n_eos_spill
Definition: DecoData.h:152
double time_map
In msec.
Definition: DecoData.h:158
unsigned int eos_coda_id
Definition: DecoData.h:148
double time_decode
In msec.
Definition: DecoData.h:157
unsigned int bos_vme_time
Definition: DecoData.h:147
unsigned int n_slow
Definition: DecoData.h:153
SlowControlDataList list_slow_cont
Definition: DecoData.h:160