LDMX Software
ZCCMDecoder.cxx
1#include "TrigScint/ZCCMDecoder.h"
2
3#include <TMath.h>
4
5#include <bitset>
6
7namespace trigscint {
8
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}
84
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) {
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}
314
316 ldmx_log(debug) << "Process starts!";
317
318 return;
319}
320
322 ldmx_log(debug) << "Process ends!";
323
324 return;
325}
326
327} // namespace trigscint
328
#define DECLARE_PRODUCER(CLASS)
Macro which allows the framework to construct a producer given its name during configuration.
Implements an event buffer system for storing event data.
Definition Event.h:42
void add(const std::string &collectionName, T &obj)
Adds an object to the event bus.
Definition Event.h:184
Class encapsulating parameters for configuring a processor.
Definition Parameters.h:29
const T & get(const std::string &name) const
Retrieve the parameter of the given name.
Definition Parameters.h:78
class for storing QIE output
void setLaneID(const int laneid)
Store the lane (fiber) ID.
uint32_t getTimeSinceSpill() const
Store the event time since spill counter.
std::vector< int > getADC() const
Get ADCs of all time samples.
void setCID(const std::vector< int > cid)
Store cids of all time samples.
void setTDC(const std::vector< int > tdc)
Store tdcs of all time samples.
void setBC0(const std::vector< int > bc0)
Store bc0s of all time samples.
void setChanID(const int chanid)
Store the channel ID.
int getElecID() const
Get electronics ID.
int getChanID() const
Get channel ID.
std::vector< int > getTDC() const
Get tdcs of all time samples.
void setModuleID(const int moduleid)
Store the module ID (pad number-1)
void setCE(const std::vector< int > ce)
Store ces of all time samples.
void setElecID(const int elecid)
Store the electronics ID.
void setADC(const std::vector< int > adc)
Store adcs of all time samples.
std::string channel_map_file_name_
the channel mapping
Definition ZCCMDecoder.h:40
void produce(framework::Event &event) override
Process the event and put new data products into it.
void configure(framework::config::Parameters &ps) override
Configure our converter based off the configuration parameters decoded from the passed python script.
void onProcessEnd() override
Callback for the EventProcessor to take any necessary action when the processing of events finishes,...
void onProcessStart() override
Callback for the EventProcessor to take any necessary action when the processing of events starts,...
std::string module_map_file_name_
the module mapping
Definition ZCCMDecoder.h:44
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