LDMX Software
trigscint::ZCCMDecoder Class Reference

Public Member Functions

 ZCCMDecoder (const std::string &name, framework::Process &process)
 
virtual ~ZCCMDecoder ()=default
 Default destructor, closes up boost archive and input stream.
 
void configure (framework::config::Parameters &ps) override
 Configure our converter based off the configuration parameters decoded from the passed python script.
 
void produce (framework::Event &event) override
 Process the event and put new data products into it.
 
void onProcessStart () override
 Callback for the EventProcessor to take any necessary action when the processing of events starts, such as creating histograms.
 
void onProcessEnd () override
 Callback for the EventProcessor to take any necessary action when the processing of events finishes, such as calculating job-summary quantities.
 
- Public Member Functions inherited from framework::Producer
 Producer (const std::string &name, Process &process)
 Class constructor.
 
virtual void process (Event &event) final
 Processing an event for a Producer is calling produce.
 
- Public Member Functions inherited from framework::EventProcessor
 DECLARE_FACTORY (EventProcessor, EventProcessor *, const std::string &, Process &)
 declare that we have a factory for this class
 
 EventProcessor (const std::string &name, Process &process)
 Class constructor.
 
virtual ~EventProcessor ()=default
 Class destructor.
 
virtual void beforeNewRun (ldmx::RunHeader &run_header)
 Callback for Producers to add parameters to the run header before conditions are initialized.
 
virtual void onNewRun (const ldmx::RunHeader &run_header)
 Callback for the EventProcessor to take any necessary action when the run being processed changes.
 
virtual void onFileOpen (EventFile &event_file)
 Callback for the EventProcessor to take any necessary action when a new event input ROOT file is opened.
 
virtual void onFileClose (EventFile &event_file)
 Callback for the EventProcessor to take any necessary action when a event input ROOT file is closed.
 
template<class T >
const T & getCondition (const std::string &condition_name)
 Access a conditions object for the current event.
 
TDirectory * getHistoDirectory ()
 Access/create a directory in the histogram file for this event processor to create histograms and analysis tuples.
 
void setStorageHint (framework::StorageControl::Hint hint)
 Mark the current event as having the given storage control hint from this module_.
 
void setStorageHint (framework::StorageControl::Hint hint, const std::string &purposeString)
 Mark the current event as having the given storage control hint from this module and the given purpose string.
 
int getLogFrequency () const
 Get the current logging frequency from the process.
 
int getRunNumber () const
 Get the run number from the process.
 
std::string getName () const
 Get the processor name.
 
void createHistograms (const std::vector< framework::config::Parameters > &histos)
 Internal function which is used to create histograms passed from the python configuration @parma histos vector of Parameters that configure histograms to create.
 

Private Attributes

std::string channel_map_file_name_
 the channel mapping
 
std::ifstream channel_map_file_
 
std::map< int, int > channel_map_
 
std::string module_map_file_name_
 the module mapping
 
std::ifstream module_map_file_
 
std::map< int, int > module_map_
 
std::string input_collection_
 
std::string output_collection_
 
std::string input_pass_name_
 
int number_of_lanes_ {14}
 
int n_channels_ {number_of_lanes_ * ZCCMOutput::NUM_CHAN_PER_LANE}
 
int n_samples_ {70}
 
bool is_real_data_ {true}
 
int modules_used_ [4] = {0}
 

Additional Inherited Members

- Protected Member Functions inherited from framework::EventProcessor
void abortEvent ()
 Abort the event immediately.
 
- Protected Attributes inherited from framework::EventProcessor
HistogramPool histograms_
 helper object for making and filling histograms
 
NtupleManagerntuple_ {NtupleManager::getInstance()}
 Manager for any ntuples.
 
logging::logger the_log_
 The logger for this EventProcessor.
 

Detailed Description

Definition at line 16 of file ZCCMDecoder.h.

Constructor & Destructor Documentation

◆ ZCCMDecoder()

trigscint::ZCCMDecoder::ZCCMDecoder ( const std::string & name,
framework::Process & process )
inline

Definition at line 18 of file ZCCMDecoder.h.

19 : Producer(name, process) {}
Producer(const std::string &name, Process &process)
Class constructor.
virtual void process(Event &event) final
Processing an event for a Producer is calling produce.

Member Function Documentation

◆ configure()

void trigscint::ZCCMDecoder::configure ( framework::config::Parameters & ps)
overridevirtual

Configure our converter based off the configuration parameters decoded from the passed python script.

Reimplemented from framework::EventProcessor.

Definition at line 9 of file ZCCMDecoder.cxx.

9 {
10 // Configure this instance of the encoder
11 output_collection_ = ps.get<std::string>("output_collection");
12 input_collection_ = ps.get<std::string>("input_collection");
13 input_pass_name_ = ps.get<std::string>("input_pass_name");
14 module_map_file_name_ = ps.get<std::string>("module_map_file");
15 channel_map_file_name_ = ps.get<std::string>("channel_map_file");
16 n_channels_ = ps.get<int>("number_channels");
17 n_samples_ = ps.get<int>("number_time_samples");
18 is_real_data_ = ps.get<bool>("is_real_data");
19 // daq_junk_words_len_bytes_ = ps.get<int>("daq_extra_header_bytes");
20 // daq_junk_tail_len_bits_ = ps.get<int>("daq_trailing_header_bits");
21
22 ldmx_log(debug) << "In configure, got parameters:" << "\noutput_collection = "
23 << output_collection_
24 << "\ninput_collection = " << input_collection_
25 << "\ninput_pass_name = " << input_pass_name_
26 << "\nmodule_map_file = " << module_map_file_name_
27 << "\nchannel_map_file = " << channel_map_file_name_
28 << "\nnumber_channels = " << n_channels_
29 << "\nnumber_time_samples = " << n_samples_
30 << "\nis_real_data = " << is_real_data_;
31
32 channel_map_file_.open(channel_map_file_name_, std::ios::in);
33 if (!channel_map_file_.is_open()) {
34 EXCEPTION_RAISE("BadMapFile", "The channel mapping file cannot be opened.");
35 return;
36 }
37 int n_chan_from_map = -1; // we will in effect count newlines
38 int ch_id, el_id;
39 while (!channel_map_file_.eof()) {
40 channel_map_file_ >> el_id >> ch_id;
41 // make the map based on electronics ID, to look up channel ID.
42 // the reason is, the elecID is given by the position of the word in
43 // the output. so these need to be strictly ordered.
44 // writing the right bar in the right position is easier if we can just
45 // read this map from beginning to end.
46 // barID can always be set, or looked up, as a property of the digi.
47
48 // here make the elecID the key (other way around when encoding)
49 channel_map_.insert(std::pair<int, int>(el_id, ch_id));
50 ldmx_log(debug) << el_id << " chID " << ch_id;
51 n_chan_from_map++; // will increment on last newline as well
52 }
53 channel_map_file_.close();
54 if (n_chan_from_map != n_channels_)
55 ldmx_log(fatal) << "The set number of channels: " << n_channels_
56 << " does not match the number obtained from the map :"
57 << n_chan_from_map;
58
59 // and similarly for the lane-->module mapping
60
61 module_map_file_.open(module_map_file_name_, std::ios::in);
62 if (!module_map_file_.is_open()) {
63 ldmx_log(fatal) << "The module mapping file cannot be opened.";
64 return;
65 }
66 int module_id, lane_id;
67 while (!module_map_file_.eof()) {
68 module_map_file_ >> lane_id >> module_id;
69 // make the map based on lane ID, to look up module ID.
70 // here make the lane the key (other way around when encoding)
71 module_map_.insert(std::pair<int, int>(lane_id, module_id));
72 ldmx_log(debug) << "lane " << lane_id << " --> module ID " << module_id;
73 modules_used_[module_id] = 1;
74 }
75 module_map_file_.close();
76 int n_lanes = n_channels_ / ZCCMOutput::NUM_CHAN_PER_LANE;
77 if (lane_id != n_lanes - 1)
78 ldmx_log(fatal) << "The set number of lanes " << n_lanes
79 << " seems not to match the number obtained from the map :"
80 << lane_id + 1;
81
82 return;
83}
const T & get(const std::string &name) const
Retrieve the parameter of the given name.
Definition Parameters.h:78
std::string channel_map_file_name_
the channel mapping
Definition ZCCMDecoder.h:40
std::string module_map_file_name_
the module mapping
Definition ZCCMDecoder.h:44

References channel_map_file_name_, framework::config::Parameters::get(), and module_map_file_name_.

◆ onProcessEnd()

void trigscint::ZCCMDecoder::onProcessEnd ( )
overridevirtual

Callback for the EventProcessor to take any necessary action when the processing of events finishes, such as calculating job-summary quantities.

Reimplemented from framework::EventProcessor.

Definition at line 321 of file ZCCMDecoder.cxx.

321 {
322 ldmx_log(debug) << "Process ends!";
323
324 return;
325}

◆ onProcessStart()

void trigscint::ZCCMDecoder::onProcessStart ( )
overridevirtual

Callback for the EventProcessor to take any necessary action when the processing of events starts, such as creating histograms.

Reimplemented from framework::EventProcessor.

Definition at line 315 of file ZCCMDecoder.cxx.

315 {
316 ldmx_log(debug) << "Process starts!";
317
318 return;
319}

◆ produce()

void trigscint::ZCCMDecoder::produce ( framework::Event & event)
overridevirtual

Process the event and put new data products into it.

Parameters
eventThe Event to process.

Implements framework::Producer.

Definition at line 85 of file ZCCMDecoder.cxx.

85 {
86 ldmx_log(debug) << "ZCCMDecoder: produce() starts! Event number: "
87 << event.getEventHeader().getEventNumber();
88
89 // this has to be configurable for now, to read real data
90 int n_samp = n_samples_;
91 ldmx_log(debug) << "num samples = " << n_samp;
92
93 ldmx_log(debug) << "Looking up input collection " << input_collection_ << "_"
94 << input_pass_name_;
95 const auto event_output{
96 event.getCollection<uint8_t>(input_collection_, input_pass_name_)};
97 ldmx_log(debug) << "Got input collection " << input_collection_ << "_"
98 << input_pass_name_;
99
100 /* -- PROCESS THE EVENT INFORMATION -- */
101 uint32_t time_stamp = 0;
102 for (int i_w = 0; i_w < ZCCMOutput::TIMESTAMP_LEN_BYTES; i_w++) {
103 int pos = ZCCMOutput::TIMESTAMP_POS + i_w;
104 uint8_t time_word = event_output.at(pos);
105 ldmx_log(debug) << "time stamp word at position " << pos
106 << " (with iW = " << i_w
107 << ") = " << std::bitset<8>(time_word);
108 time_stamp |= (time_word << i_w * 8); // shift by a byte at a time
109 }
110 // This time_stamp is local to the TS
111 event.getEventHeader().setIntParameter("TS Timestamp", time_stamp);
112
113 /* -- TS event header done; read the channel contents -- */
114
115 /*
116 in a loop over samples, loop over nlanes
117 get ADC, TDC, flags and lane nb
118 lane number uses a full byte
119 the flags word contains
120 - capID (2b)
121 - CE (1b, some channel alignment error flag)
122 - BC0 (most of the time 0, set to 1 with fixed frequency. if channels
123 are aligned, then BC0=1 should occur in the same time sample for all of
124 them)
125 - 4 trailing reserved 0's
126 */
127
128 std::map<int, std::vector<int>> adc_map;
129 std::map<int, std::vector<int>> tdc_map;
130 std::map<int, std::vector<int>> cid_map;
131 std::map<int, std::vector<int>> bc0_map;
132 std::map<int, std::vector<int>> ce_map;
133 // we need one output digi collection per pad
134 std::vector<std::vector<trigscint::TrigScintQIEDigis>> out_digis;
135 std::size_t n_modules =
136 std::ranges::count_if(modules_used_, [](int x) { return x != 0; });
137 // default construct (empty) vectors of digis separated into modules
138 out_digis.resize(n_modules);
139
140 // read in words from the output. line them up per channel and time sample.
141 // channels are in the electronics ordering
142 int word_length = ZCCMOutput::SAMPLE_WORD_LEN_BYTES;
143 int num_messages =
144 ((int)event_output.size() - ZCCMOutput::EVENTDATA_POS) / word_length;
145
146 // with one message per time sample per lane, we expect nSample*nLanes
147 // messages
148 int n_lanes = n_channels_ / ZCCMOutput::NUM_CHAN_PER_LANE;
149 int num_expected_messages = n_samp * n_lanes;
150 if (num_messages != num_expected_messages)
151 ldmx_log(warn) << "Unexpected stream length! Num messages is "
152 << num_messages << ", expect " << num_expected_messages;
153
154 // read in a lane stream at a time with a while loop
155 // operate on bytes, but i_word increments for each meassage i.e. each lane
156 int i_word = 0;
157 int start_lane = -1;
158 int sample_nb = -1;
159
160 while (i_word < num_messages) { // update the position to read the word from
161 uint word_pos = ZCCMOutput::EVENTDATA_POS + i_word * word_length;
162
163 uint16_t empty =
164 event_output.at(word_pos + ZCCMOutput::EMPTY_WORD_SAMPLE_WORD_POS);
165 uint8_t lane = event_output.at(word_pos + ZCCMOutput::LANE_SAMPLE_WORD_POS);
166 uint8_t flag =
167 event_output.at(word_pos + ZCCMOutput::FLAGS_SAMPLE_WORD_POS);
168
169 ldmx_log(debug) << "Start of message " << i_word << ".\n\tEmpty word is "
170 << std::bitset<16>(empty) << ", read at position "
171 << word_pos + ZCCMOutput::EMPTY_WORD_SAMPLE_WORD_POS
172 << ". \n\tFlag word is " << std::bitset<8>(flag)
173 << ", read at position "
174 << word_pos + ZCCMOutput::FLAGS_SAMPLE_WORD_POS
175 << ". \n\tLane is " << std::bitset<8>(lane)
176 << ", read at position "
177 << word_pos + ZCCMOutput::LANE_SAMPLE_WORD_POS << ".";
178
179 if (start_lane == -1) { // it is unset
180 start_lane = lane; // store the first one
181 ldmx_log(debug) << "Set time sample start lane to " << start_lane;
182 }
183 // extract the flag info
184 int cid{(flag >> ZCCMOutput::CAPID_POS_IN_FLAG) &
186 ldmx_log(trace) << "Got Cap ID " << cid;
187 bool bc0{static_cast<bool>((flag >> ZCCMOutput::BC0_POS_IN_FLAG) &
189 ldmx_log(trace) << "Got BC0 flag " << bc0;
190 bool ce{static_cast<bool>((flag >> ZCCMOutput::CE_POS_IN_FLAG) &
192 ldmx_log(trace) << "Got CE flag " << ce;
193 int empty_bits{(flag >> ZCCMOutput::EMPTY_FLAG_WORD_POS_IN_FLAG) &
195 if (empty_bits)
196 ldmx_log(fatal) << "Empty bits of flag not empty: " << empty_bits;
197
198 // TODO: check that BC0 bit is aligned across all lanes
199 // TODO: keep track of the CID cycling and flag when a CID is skipped
200 // bool is_cid_unsync{static_cast<bool>((flags >>
201 // ZCCMOutput::CID_UNSYNC_POS) &
202 // Mask8<ZCCMOutput::FLAG_SIZE_BITS>::M)};
203
204 // how do we know we're at the next time sample?
205 // 1. i_word has incremented by the number of lanes
206 // // i.e i_word % nLanes = 0
207 // 2. we're seeing the first lane again
208 // it is good, then, to check that these agree.
209
210 if (lane == start_lane) // we have cycled (or just started)
211 sample_nb++;
212
213 uint8_t cid_val = cid;
214
215 // get channel-by-channel info (ADC and TDC for this time sample)
216 for (int i_c = 0; i_c < ZCCMOutput::NUM_CHAN_PER_LANE; i_c++) {
217 if (i_word >= num_expected_messages) {
218 ldmx_log(fatal)
219 << "More words than expected! Breaking event data loop in sample "
220 << sample_nb << " at lane = " << int(lane) << ", channel nb "
221 << i_c;
222 break;
223 }
224 uint8_t adc_val =
225 event_output.at(word_pos + ZCCMOutput::ADC_SAMPLE_WORD_POS + i_c);
226 uint8_t tdc_val =
227 event_output.at(word_pos + ZCCMOutput::TDC_SAMPLE_WORD_POS + i_c);
228 // define a unique elecID for each channel in the readout, comprising of
229 // lane, module, channelNB then use that to look up bar ID in the the maps
230 int module = module_map_[lane];
231 int elec_id = 100 * lane + 10 * module + i_c; // numbering scheme: LLMC
232
233 ldmx_log(trace) << "got ADC value " << (unsigned)adc_val
234 << " and TDC value " << (unsigned)tdc_val
235 << " at channel idx " << i_c << " with elec id "
236 << elec_id;
237 if (adc_map.find(elec_id) == adc_map.end()) { // we have a new channel
238 std::vector<int> adcs(n_samp, 0);
239 adc_map.insert(std::pair<int, std::vector<int>>(elec_id, adcs));
240 }
241 adc_map[elec_id].at(sample_nb) = adc_val;
242 // perhaps overly prudent to check for TDC and CID as well, might tag them
243 // along on ADC
244 if (tdc_map.find(elec_id) == tdc_map.end()) { // we have a new channel
245 std::vector<int> tdcs(n_samp, 0);
246 tdc_map.insert(std::pair<int, std::vector<int>>(elec_id, tdcs));
247 }
248 tdc_map[elec_id].at(sample_nb) = tdc_val;
249 if (cid_map.find(elec_id) == cid_map.end()) { // we have a new channel
250 std::vector<int> cids(n_samp, 0);
251 cid_map.insert(std::pair<int, std::vector<int>>(elec_id, cids));
252 std::vector<int> bc0s(n_samp, 0);
253 bc0_map.insert(std::pair<int, std::vector<int>>(elec_id, bc0s));
254 std::vector<int> ces(n_samp, 0);
255 ce_map.insert(std::pair<int, std::vector<int>>(elec_id, ces));
256 }
257 cid_map[elec_id].at(sample_nb) = cid_val;
258 bc0_map[elec_id].at(sample_nb) = bc0;
259 ce_map[elec_id].at(sample_nb) = ce;
260 } // over channels
261 ldmx_log(debug) << "Done with lane " << int(lane) << " which we think is "
262 << i_word % n_lanes << " in sample " << sample_nb;
263
264 if (int(lane) != i_word % n_lanes)
265 ldmx_log(fatal) << "Lane ordering has been messed up! Expect lane "
266 << i_word % n_lanes << ", but we got lane " << int(lane)
267 << " in sample " << sample_nb;
268 // shift to next word
269 i_word++;
270 } // while event data words left in the stream
271
272 ldmx_log(debug) << "Done reading in header, ADC and TDC for event "
273 << event.getEventNumber();
274
275 // Reading step done.
276 /* -- make digis from all the info stored in maps -- */
277 for (std::map<int, std::vector<int>>::iterator itr = adc_map.begin();
278 itr != adc_map.end(); ++itr) {
279 TrigScintQIEDigis digi;
280 digi.setADC(itr->second);
281 if (channel_map_.find(itr->first) == channel_map_.end()) {
282 ldmx_log(fatal)
283 << "Couldn't find the bar ID corresponding to electronics ID "
284 << itr->first << "!! Skipping.";
285 continue;
286 }
287 int bar = channel_map_[itr->first];
288 digi.setElecID(itr->first);
289 digi.setChanID(bar);
290 int module = itr->first / 10 % 10;
291 digi.setModuleID(module);
292 int lane = itr->first / 100;
293 digi.setLaneID(lane);
294 digi.setTDC(tdc_map[itr->first]);
295 digi.setCID(cid_map[itr->first]);
296 digi.setBC0(bc0_map[itr->first]);
297 digi.setCE(ce_map[itr->first]);
298 if (bar == 0)
299 ldmx_log(debug) << "for bar 0, got time since spill "
300 << digi.getTimeSinceSpill();
301 out_digis[module].push_back(digi);
302 ldmx_log(debug) << "Iterator points to key " << itr->first
303 << " and mapped channel supposedly is " << bar;
304 ldmx_log(debug) << "Made digi with elecID = " << digi.getElecID()
305 << ", barID = " << digi.getChanID() << ", third adc value "
306 << digi.getADC().at(2) << " and third tdc "
307 << digi.getTDC().at(2);
308 }
309
310 // name them as ordered
311 for (uint i = 0; i < n_modules; i++)
312 event.add(output_collection_ + Form("%i", i + 1), out_digis[i]);
313}
void add(const std::string &collectionName, T &obj)
Adds an object to the event bus.
Definition Event.h:184
static const int TIMESTAMP_POS
for each time sample, every lane each sends a message: [ 1B ADC for 6 channels, 2B empty,...
Definition ZCCMOutput.h:102

References framework::Event::add(), trigscint::TrigScintQIEDigis::getADC(), trigscint::TrigScintQIEDigis::getChanID(), trigscint::TrigScintQIEDigis::getElecID(), trigscint::TrigScintQIEDigis::getTDC(), trigscint::TrigScintQIEDigis::getTimeSinceSpill(), trigscint::TrigScintQIEDigis::setADC(), trigscint::TrigScintQIEDigis::setBC0(), trigscint::TrigScintQIEDigis::setCE(), trigscint::TrigScintQIEDigis::setChanID(), trigscint::TrigScintQIEDigis::setCID(), trigscint::TrigScintQIEDigis::setElecID(), trigscint::TrigScintQIEDigis::setLaneID(), trigscint::TrigScintQIEDigis::setModuleID(), trigscint::TrigScintQIEDigis::setTDC(), and trigscint::ZCCMOutput::TIMESTAMP_POS.

Member Data Documentation

◆ channel_map_

std::map<int, int> trigscint::ZCCMDecoder::channel_map_
private

Definition at line 42 of file ZCCMDecoder.h.

◆ channel_map_file_

std::ifstream trigscint::ZCCMDecoder::channel_map_file_
private

Definition at line 41 of file ZCCMDecoder.h.

◆ channel_map_file_name_

std::string trigscint::ZCCMDecoder::channel_map_file_name_
private

the channel mapping

Definition at line 40 of file ZCCMDecoder.h.

Referenced by configure().

◆ input_collection_

std::string trigscint::ZCCMDecoder::input_collection_
private

Definition at line 49 of file ZCCMDecoder.h.

◆ input_pass_name_

std::string trigscint::ZCCMDecoder::input_pass_name_
private

Definition at line 51 of file ZCCMDecoder.h.

◆ is_real_data_

bool trigscint::ZCCMDecoder::is_real_data_ {true}
private

Definition at line 63 of file ZCCMDecoder.h.

63{true};

◆ module_map_

std::map<int, int> trigscint::ZCCMDecoder::module_map_
private

Definition at line 46 of file ZCCMDecoder.h.

◆ module_map_file_

std::ifstream trigscint::ZCCMDecoder::module_map_file_
private

Definition at line 45 of file ZCCMDecoder.h.

◆ module_map_file_name_

std::string trigscint::ZCCMDecoder::module_map_file_name_
private

the module mapping

Definition at line 44 of file ZCCMDecoder.h.

Referenced by configure().

◆ modules_used_

int trigscint::ZCCMDecoder::modules_used_[4] = {0}
private

Definition at line 65 of file ZCCMDecoder.h.

65{0};

◆ n_channels_

int trigscint::ZCCMDecoder::n_channels_ {number_of_lanes_ * ZCCMOutput::NUM_CHAN_PER_LANE}
private

Definition at line 59 of file ZCCMDecoder.h.

59{number_of_lanes_ * ZCCMOutput::NUM_CHAN_PER_LANE};

◆ n_samples_

int trigscint::ZCCMDecoder::n_samples_ {70}
private

Definition at line 61 of file ZCCMDecoder.h.

61{70};

◆ number_of_lanes_

int trigscint::ZCCMDecoder::number_of_lanes_ {14}
private

Definition at line 54 of file ZCCMDecoder.h.

54{14};

◆ output_collection_

std::string trigscint::ZCCMDecoder::output_collection_
private

Definition at line 50 of file ZCCMDecoder.h.


The documentation for this class was generated from the following files: