Process the event and put new data products into it.
58 {
59 ldmx_log(debug) << "QIEEncoder: produce() starts! Event number: "
60 << event.getEventHeader().getEventNumber();
61
62 std::vector<trigscint::QIEStream> qie_outs;
63 int n_samp = QIEStream::NUM_SAMPLES;
64 std::vector<int> init_vec(n_samp, 0);
65 ldmx_log(debug) << "num samples = " << n_samp;
66
67
68
69 for (int i_q = 0; i_q < n_channels_; i_q++) {
70 QIEStream qie_out;
71 qie_out.setADC(init_vec);
72 qie_out.setTDC(init_vec);
73 qie_out.setCID(init_vec);
74 qie_out.setElectronicsID(i_q);
75
76 qie_outs.push_back(qie_out);
77 }
78
79 ldmx_log(debug) << "Looking up input collection " << input_collection_ << "_"
80 << input_pass_name_;
82 input_collection_, input_pass_name_)};
83 ldmx_log(debug) << "Got input collection" << input_collection_ << "_"
84 << input_pass_name_;
85
86 bool is_ci_dunsync = false;
87
88 bool is_ci_dskipped = false;
89 bool is_cr_c0malformed = false;
90 bool is_cr_c1malformed = false;
91
92 int first_cid = -1;
93 ldmx_log(debug) << "entering loop over digis ";
94 for (auto &digi : digis) {
95 int bar = digi.getChanID();
96 auto itr = channel_map_.find(bar);
97 if (itr == channel_map_.end()) {
98 ldmx_log(fatal) << "Couldn't find an entry for bar " << bar
99 << "; check the (choice of) channel map!. Exiting event "
100 << event.getEventHeader().getEventNumber();
101 return;
102 }
103 int idx = itr->second;
104
105 qie_outs.at(idx).setChannelID(bar);
106 qie_outs.at(idx).setElectronicsID(idx);
107 ldmx_log(debug) << "Channel " << bar << " elec ID "
108 << qie_outs.at(idx).getElectronicsID();
109 std::vector<int> l_etdcs;
110 std::vector<uint8_t> cids;
111 for (int i_s = 0; i_s < n_samp; i_s++) {
112 int tdc = digi.getTDC().at(i_s);
113 int cid = digi.getCID().at(i_s);
114 if (cids.size() > 0 && (cid % 4) != ((cids.back() + 1) % 4)) {
115
116
117 is_ci_dskipped = true;
118 }
119 if (verbose_) {
120 std::vector<uint8_t> adcs;
121 int adc = digi.getADC().at(i_s);
122 uint8_t mant = adc % 64;
123 uint8_t exp = adc / 64;
124 ldmx_log(debug) << "\tSample " << i_s << std::left << std::setw(6)
125 << " ADC " << adc << ",\texp " << unsigned(exp)
126 << " mant " << unsigned(mant) << ",\tTDC = " << tdc
127 << ", LE TDC = " << std::bitset<8>(tdc / 16)
128 << " and capID= " << cid;
129 adcs.push_back(64 * exp + mant);
130 ldmx_log(debug) << "Combined ADC: " << std::showbase
131 << std::bitset<8>(adcs.back()) << " and original adc "
132 << std::bitset<8>(adc) << std::dec;
133 }
134
135 tdc /= 16;
136 l_etdcs.push_back(tdc);
137 cids.push_back((uint8_t)cid);
138 }
139 if (first_cid == -1) {
140
141
142 first_cid = cids.back();
143 }
144 if (first_cid != cids.back()) {
145 is_ci_dunsync =
146 true;
147 }
148 qie_outs.at(idx).setADC(digi.getADC());
149 qie_outs.at(idx).setTDC(l_etdcs);
150 }
151 if (is_ci_dunsync) ldmx_log(debug) << "Found unsynced CIDs!";
152 if (is_ci_dskipped) ldmx_log(info) << "Found skipped CIDs!";
153
154
155
156
157
158
159
160
161
162
163 uint16_t trigger_id = event.getEventHeader().getEventNumber();
164 uint8_t random_checksum =
165 30;
166 uint8_t flags = 0;
167
168 flags |= (is_cr_c0malformed << QIEStream::CRC0_ERR_POS);
169 flags |= (is_cr_c1malformed << QIEStream::CRC1_ERR_POS);
170 flags |= (is_ci_dunsync << QIEStream::CID_UNSYNC_POS);
171 flags |= (is_ci_dskipped << QIEStream::CID_SKIP_POS);
172 ldmx_log(debug) << "FLAGS: " << std::bitset<8>(flags);
173
174 std::vector<uint8_t> out_word;
175 std::vector<uint8_t> trigger_i_dwords;
176 for (int i_w = QIEStream::TRIGID_LEN_BYTES - 1; i_w >= 0; i_w--) {
177
178 uint8_t t_i_dword = trigger_id >> i_w * 8;
179 trigger_i_dwords.push_back(t_i_dword);
180 out_word.push_back(t_i_dword);
181 }
182
183 out_word.push_back(flags);
184 out_word.push_back(random_checksum);
185
186 if (verbose_) {
187 std::cout << "header word ";
188 for (auto word : out_word) std::cout << std::bitset<8>(word) << " ";
189 std::cout << std::endl;
190 }
191
192
193
194 for (int i_s = 0; i_s < n_samp; i_s++) {
195 for (int i_q = 0; i_q < n_channels_; i_q++) {
196 out_word.push_back(qie_outs.at(i_q).getADC().at(i_s));
197 }
198 for (int i_q = 0; i_q < n_channels_; i_q++) {
199 out_word.push_back(qie_outs.at(i_q).getTDC().at(i_s));
200 }
201 }
202
203
204 if (verbose_) {
205 std::cout << "total word ";
206 int widx = 0;
207 int i_wstart =
208 std::max(std::max(QIEStream::ERROR_POS, QIEStream::CHECKSUM_POS),
209 QIEStream::TRIGID_POS + (QIEStream::TRIGID_LEN_BYTES)) +
210 1;
211 for (auto word : out_word) {
212 if ((widx - i_wstart) % n_channels_ == 0) {
213 int sample = (widx - i_wstart) / n_channels_;
214 if (sample % 2 == 0)
215 std::cout << "\n sample " << sample / 2 << " | ";
216 else
217 std::cout << "\n TDC: ";
218 }
219 std::cout << (unsigned)word << " ";
220
221
222 widx++;
223 }
224 std::cout << std::endl;
225 }
226
227 event.add(output_collection_, out_word);
228}
class for storing QIE output