DUNE-DAQ
DUNE Trigger and Data Acquisition software
Loading...
Searching...
No Matches
NP02ReadoutApplication.cpp
Go to the documentation of this file.
1
18#include "confmodel/Session.hpp"
19
23
26
29
32#include "confmodel/GeoId.hpp"
35#include "confmodel/Service.hpp"
36
43
55
56
58
59#include "logging/Logging.hpp"
60#include <fmt/core.h>
61
62#include <string>
63#include <vector>
64
65// using namespace dunedaq;
66// using namespace dunedaq::appmodel;
67
68namespace dunedaq {
69namespace appmodel {
70
71//-----------------------------------------------------------------------------
72void
73NP02ReadoutApplication::generate_modules(std::shared_ptr<appmodel::ConfigurationHelper> helper) const
74{
75
76 TLOG_DEBUG(6) << "Generating modules for application " << this->UID();
77
78 ConfigObjectFactory obj_fac(this);
79
80 //
81 // Extract basic configuration objects
82 //
83
84 // Data reader
85 auto reader_conf = get_data_reader();
86 if (reader_conf == 0) {
87 throw(BadConf(ERS_HERE, "No DataReaderModule configuration given"));
88 }
89 std::string reader_class = reader_conf->get_template_for();
90
91 // Link handler
92 auto dlh_conf = get_link_handler();
93 // What is template for?
94 auto dlh_class = dlh_conf->get_template_for();
95
96 auto tph_conf = get_tp_handler();
97 if (tph_conf==nullptr && get_tp_generation_enabled()) {
98 throw(BadConf(ERS_HERE, "TP generation is enabled but there is no TP data handler configuration"));
99 }
100
101 std::string tph_class = "";
102 if (tph_conf != nullptr && get_tp_generation_enabled()) {
103 tph_class = tph_conf->get_template_for();
104 }
105
106 //
107 // Process the queue rules looking for inputs to our DL/TP handler modules
108 //
109 const QueueDescriptor* dlh_reqinput_qdesc = nullptr;
110 const QueueDescriptor* tp_input_qdesc = nullptr;
111 // const QueueDescriptor* tpReqInputQDesc = nullptr;
112 const QueueDescriptor* fa_output_qdesc = nullptr;
113
114 for (auto rule : get_queue_rules()) {
115 auto destination_class = rule->get_destination_class();
116 auto data_type = rule->get_descriptor()->get_data_type();
117 // Why datahander here?
118 if (destination_class == "DataHandlerModule" || destination_class == dlh_class || destination_class == tph_class) {
119 if (data_type == "DataRequest") {
120 dlh_reqinput_qdesc = rule->get_descriptor();
121 } else if ((data_type == "TriggerPrimitive" || data_type == "TriggerPrimitiveVector") && get_tp_generation_enabled()) {
122 tp_input_qdesc = rule->get_descriptor();
123 }
124 } else if (destination_class == "FragmentAggregatorModule") {
125 fa_output_qdesc = rule->get_descriptor();
126 }
127 }
128
129 //
130 // Process the network rules looking for the Fragment Aggregator and TP handler data reuest inputs
131 //
132 const NetworkConnectionDescriptor* fa_net_desc = nullptr;
133 const NetworkConnectionDescriptor* tp_net_desc = nullptr;
134 const NetworkConnectionDescriptor* ta_net_desc = nullptr;
135 const NetworkConnectionDescriptor* ts_net_desc = nullptr;
136 for (auto rule : get_network_rules()) {
137 auto endpoint_class = rule->get_endpoint_class();
138 auto data_type = rule->get_descriptor()->get_data_type();
139
140 if (endpoint_class == "FragmentAggregatorModule") {
141 fa_net_desc = rule->get_descriptor();
142 } else if (data_type == "TPSet") {
143 tp_net_desc = rule->get_descriptor();
144 } else if (data_type == "TriggerActivity") {
145 ta_net_desc = rule->get_descriptor();
146 } else if (data_type == "TimeSync") {
147 ts_net_desc = rule->get_descriptor();
148 }
149 }
150
151 // Create here the Queue on which all data fragments are forwarded to the fragment aggregator
152 // and a container for the queues of data request to TP handler and DLH
153 if (fa_output_qdesc == nullptr) {
154 throw(BadConf(ERS_HERE, "No fragment output queue descriptor given"));
155 }
156 std::vector<const confmodel::Connection*> req_queues;
157 conffwk::ConfigObject frag_queue_obj = obj_fac.create_queue_obj(fa_output_qdesc);
158
159 //
160 // Get the callback descriptor
161 //
162 const DataMoveCallbackDescriptor* raw_data_callback_desc = get_callback_desc();
163
164 if (raw_data_callback_desc == nullptr) {
165 throw(BadConf(ERS_HERE, "No Raw Data Callback descriptor given"));
166 }
167
168 //
169 // Scan Detector 2 DAQ connections to extract sender, receiver and stream information
170 //
171
172 std::vector<const confmodel::DaqModule*> modules;
173
174 // Loop over the detector to daq connections and generate one data reader per connection
175 // and the cooresponding datalink handlers
176
177 // Collect all streams
178 std::vector<std::pair<int16_t, const confmodel::DetectorStream*>> all_enabled_det_streams;
179 std::map<uint32_t, const appmodel::DataMoveCallbackConf*> callback_confs_by_sid;
180
181 std::vector<const conffwk::ConfigObject*> d2d_conn_objs;
182 uint16_t conn_idx = 0;
183
184
185 std::set<int16_t> numas;
186 for (auto d2d_conn : get_detector_connections()) {
187 uint16_t receiver_numa = 0;
188
189 // Are we sure?
190 if (helper->is_disabled(d2d_conn)) {
191 TLOG_DEBUG(7) << "Ignoring disabled DetectorToDaqConnection " << d2d_conn->UID();
192 continue;
193 }
194
195 d2d_conn_objs.push_back(&d2d_conn->config_object());
196
197 TLOG_DEBUG(6) << "Processing DetectorToDaqConnection " << d2d_conn->UID();
198 // get the readout groups and the interfaces and streams therein; 1 reaout group corresponds to 1 data reader module
199
200 if (d2d_conn->senders().empty()) {
201 throw(BadConf(ERS_HERE, "DetectorToDaqConnection does not contain sebders or receivers"));
202 }
203 if (d2d_conn->receiver() == nullptr) {
204 throw(BadConf(ERS_HERE, "DetectorToDaqConnection does not contain a receiver"));
205 }
206
207 // Loop over detector 2 daq connections to find senders and receivers
208 auto det_senders = d2d_conn->senders();
209 auto det_receiver = d2d_conn->receiver();
210
211
212 // Here I want to resolve the type of connection (network, felix, or?)
213 // Rules of engagement: if the receiver interface is network or felix, the receivers should be castable to the counterpart
214 if (reader_class == "DPDKReaderModule" || reader_class == "SocketReaderModule") {
215 if ((reader_class == "DPDKReaderModule" && !det_receiver->cast<appmodel::DPDKReceiver>()) ||
216 (reader_class == "SocketReaderModule" && !det_receiver->cast<appmodel::SocketReceiver>())) {
217 throw(BadConf(ERS_HERE, fmt::format("{} requires NWDetDataReceiver, found {} of class {}", reader_class, det_receiver->UID(), det_receiver->class_name())));
218 }
219
220 if (reader_class == "DPDKReaderModule") {
221 auto dpdk_reciever = det_receiver->cast<appmodel::DPDKReceiver>();
222 receiver_numa = (int16_t)dpdk_reciever->get_uses()->get_numa_id();
223 }
224
225 bool all_nw_senders = true;
226 for (auto s : det_senders) {
227 all_nw_senders &= (s->cast<appmodel::NWDetDataSender>() != nullptr);
228 }
229
230 // Ensure that all senders are compatible with receiver
231 if (!all_nw_senders) {
232 throw(BadConf(ERS_HERE, "Non-network DetDataSener found with NWreceiver"));
233 }
234 }
235
236 std::vector<const confmodel::DetectorStream*> enabled_det_streams;
237 // Loop over senders
238 for (auto stream : d2d_conn->streams()) {
239
240 // Are we sure?
241 if (helper->is_disabled(stream)) {
242 TLOG_DEBUG(7) << "Ignoring disabled DetectorStream " << stream->UID();
243 continue;
244 }
245
246 // loop over streams
247 all_enabled_det_streams.push_back(std::make_pair(receiver_numa, stream));
248 enabled_det_streams.push_back(stream);
249 numas.insert(receiver_numa);
250 }
251
252 }
253
254 //-----------------------------------------------------------------
255 //
256 // Create DataReaderModule object
257 //
258
259 //
260 // Instantiate DataReaderModule of type DPDKReaderModule
261 //
262
263 // Create the Data reader object
264
265 std::string reader_uid(fmt::format("datareader-{}-{}", this->UID(), std::to_string(conn_idx++)));
266 TLOG_DEBUG(6) << fmt::format("creating OKS configuration object for Data reader class {} with id {}", reader_class, reader_uid);
267 auto reader_obj = obj_fac.create(reader_class, reader_uid);
268
269 // Populate configuration and interfaces (leave output queues for later)
270 reader_obj.set_obj("configuration", &reader_conf->config_object());
271 reader_obj.set_objs("connections", d2d_conn_objs);
272
273 // Create the raw data callbacks
274 std::vector<const conffwk::ConfigObject*> raw_data_callback_objs;
275
276 // Create data queues
277 for (auto& [numa, ds] : all_enabled_det_streams) {
278 conffwk::ConfigObject callback_obj = obj_fac.create_callback_sid_obj(raw_data_callback_desc, ds->get_source_id());
279 const auto* callback_conf = obj_fac.get_dal<DataMoveCallbackConf>(callback_obj.UID());
280 raw_data_callback_objs.push_back(&callback_conf->config_object());
281 callback_confs_by_sid[ds->get_source_id()] = callback_conf;
282 }
283
284 reader_obj.set_objs("raw_data_callbacks", raw_data_callback_objs);
285
286 modules.push_back(obj_fac.get_dal<confmodel::DaqModule>(reader_obj.UID()));
287
288
289
290
291 //-----------------------------------------------------------------
292 //
293 // Prepare the tp handlers and related queues
294 //
295 std::vector<std::pair<uint32_t, const confmodel::Connection*>> tp_queues;
296
298
299 // Create TP handler object
300 auto tph_conf_obj = tph_conf->config_object();
301 auto tpsrc_ids = get_tp_source_ids();
302
303 if ((tpsrc_ids.size() % 3) > 0) {
304 throw(BadConf(ERS_HERE, fmt::format("number of TP source IDs must be a multiple of 3, current amount: {}", tpsrc_ids.size())));
305 }
306
307 for (auto sid : tpsrc_ids) {
308 conffwk::ConfigObject tp_queue_obj;
309 conffwk::ConfigObject tpreq_queue_obj;
310 std::string tp_uid("tphandler-" + std::to_string(sid->get_sid()));
311 auto tph_obj = obj_fac.create(tph_class, tp_uid);
312 tph_obj.set_by_val<uint32_t>("source_id", sid->get_sid());
313 tph_obj.set_by_val<uint32_t>("detector_id", 1); // 1 == kDAQ
314 tph_obj.set_by_val<bool>("post_processing_enabled", get_ta_generation_enabled());
315 tph_obj.set_obj("module_configuration", &tph_conf_obj);
316
317 // Create the TPs aggregator queue (from RawData Handlers to TP handlers)
318 tp_queue_obj = obj_fac.create_queue_sid_obj(tp_input_qdesc, sid->get_sid());
319 tp_queue_obj.set_by_val<uint32_t>("recv_timeout_ms", 50);
320 tp_queue_obj.set_by_val<uint32_t>("send_timeout_ms", 1);
321
322 tp_queues.push_back(std::make_pair(sid->get_sid(), obj_fac.get_dal<confmodel::Connection>(tp_queue_obj.UID())));
323 // Create tp data requests queue from Fragment Aggregator
324 tpreq_queue_obj = obj_fac.create_queue_sid_obj(dlh_reqinput_qdesc, sid->get_sid());
325 req_queues.push_back(obj_fac.get_dal<confmodel::Connection>(tpreq_queue_obj.UID()));
326
327 // Create the tp(set) publishing service
328 conffwk::ConfigObject tp_net_obj = obj_fac.create_net_obj(tp_net_desc, tp_uid);
329
330 // Create the ta(set) publishing service
331 conffwk::ConfigObject ta_net_obj = obj_fac.create_net_obj(ta_net_desc, tp_uid);
332
333 // Register queues with tp handler
334 tph_obj.set_objs("inputs", { &tp_queue_obj, &tpreq_queue_obj });
335 tph_obj.set_objs("outputs", { &tp_net_obj, &ta_net_obj, &frag_queue_obj });
336 modules.push_back(obj_fac.get_dal<confmodel::DaqModule>(tph_obj.UID()));
337 }
338 }
339
340 // Add output queueus of tps
341 std::vector<std::pair<uint32_t, const conffwk::ConfigObject*>> tp_queue_objs;
342 for (auto q : tp_queues) {
343 tp_queue_objs.push_back(std::make_pair(q.first, &q.second->config_object()));
344 }
345
346 //-----------------------------------------------------------------
347 //
348 // Create datalink handlers
349 //
350 // Recover the emulation flag
351
352 auto lb_conf = dlh_conf->get_latency_buffer();
353
354 std::map<int16_t, conffwk::ConfigObject> numa_dhlconf_map;
355 for ( int16_t numa : numas ) {
356 auto lb_confobj_numa = obj_fac.create(lb_conf->class_name(), fmt::format("{}-numa{}",lb_conf->UID(), numa));
357 lb_confobj_numa.set_by_val<uint32_t>("size", lb_conf->get_size());
358 lb_confobj_numa.set_by_val<bool>("numa_aware", lb_conf->get_numa_aware());
359 lb_confobj_numa.set_by_val<int16_t>("numa_node", numa);
360 lb_confobj_numa.set_by_val<bool>("intrinsic_allocator", lb_conf->get_intrinsic_allocator());
361 lb_confobj_numa.set_by_val<uint32_t>("alignment_size", lb_conf->get_alignment_size());
362 lb_confobj_numa.set_by_val<bool>("preallocation", lb_conf->get_preallocation());
363
364 auto dhl_confobj_numa = obj_fac.create(dlh_conf->class_name(), fmt::format("{}-numa{}",dlh_conf->UID(), numa));
365 dhl_confobj_numa.set_by_val<std::string>("template_for", dlh_conf->get_template_for());
366 dhl_confobj_numa.set_by_val<std::string>("input_data_type", dlh_conf->get_input_data_type());
367 dhl_confobj_numa.set_by_val<bool>("generate_timesync", dlh_conf->get_generate_timesync());
368 dhl_confobj_numa.set_by_val<uint64_t>("post_processing_delay_ticks", dlh_conf->get_post_processing_delay_ticks());
369 dhl_confobj_numa.set_by_val<std::string>("input_data_type", dlh_conf->get_input_data_type());
370 dhl_confobj_numa.set_obj("request_handler", &dlh_conf->get_request_handler()->config_object());
371 dhl_confobj_numa.set_obj("latency_buffer", &lb_confobj_numa);
372 dhl_confobj_numa.set_obj("data_processor", &dlh_conf->get_data_processor()->config_object());
373
374
375 numa_dhlconf_map[numa] = dhl_confobj_numa;
376
377 }
378
379 auto emulation_mode = reader_conf->get_emulation_mode();
380 for (auto& [numa, ds] : all_enabled_det_streams) {
381 uint32_t sid = ds->get_source_id();
382 TLOG_DEBUG(6) << fmt::format("Processing stream {}, id {}, det id {}", ds->UID(), ds->get_source_id(), ds->get_geo_id()->get_detector_id());
383 std::string uid(fmt::format("DLH-{}", sid));
384 TLOG_DEBUG(6) << fmt::format("creating OKS configuration object for Data Link Handler class {}, if {}", dlh_class, sid);
385 auto dlh_obj = obj_fac.create(dlh_class, uid);
386 dlh_obj.set_by_val<uint32_t>("source_id", sid);
387 dlh_obj.set_by_val<uint32_t>("detector_id", ds->get_geo_id()->get_detector_id());
388 dlh_obj.set_by_val<bool>("post_processing_enabled", get_tp_generation_enabled());
389 dlh_obj.set_by_val<bool>("emulation_mode", emulation_mode);
390 dlh_obj.set_obj("geo_id", &ds->get_geo_id()->config_object());
391 dlh_obj.set_obj("module_configuration", &numa_dhlconf_map[numa]);
392 dlh_obj.set_obj("raw_data_callback", &callback_confs_by_sid[sid]->config_object());
393
394 std::vector<const conffwk::ConfigObject*> dlh_ins, dlh_outs;
395
396 // Create request queue
397 conffwk::ConfigObject req_queue_obj = obj_fac.create_queue_sid_obj(dlh_reqinput_qdesc, ds);
398
399
400 // Add the requessts queue dal pointer to the outputs of the FragmentAggregatorModule
401 req_queues.push_back(obj_fac.get_dal<confmodel::Connection>(req_queue_obj.UID()));
402 dlh_ins.push_back(&req_queue_obj);
403 dlh_outs.push_back(&frag_queue_obj);
404
405
406 // Time Sync network connection
407 if (dlh_conf->get_generate_timesync()) {
408 // Add timestamp endpoint
409 conffwk::ConfigObject ts_net_obj = obj_fac.create_net_obj(ts_net_desc, std::to_string(sid));
410 dlh_outs.push_back(&ts_net_obj);
411 }
412
413 // here, we want to select which tp queues to add to the output, to separate mutiple detector elements
414 for (auto tpq : tp_queue_objs) {
415 if ((sid / 100) == (tpq.first / 10)) {
416 dlh_outs.push_back(tpq.second);
417 }
418 }
419 dlh_obj.set_objs("inputs", dlh_ins);
420 dlh_obj.set_objs("outputs", dlh_outs);
421
422 modules.push_back(obj_fac.get_dal<confmodel::DaqModule>(dlh_obj.UID()));
423 }
424
425
426 // Finally create Fragment Aggregator
427 auto aggregator_conf = get_fragment_aggregator();
428 if (aggregator_conf == 0) {
429 throw(BadConf(ERS_HERE, "No FragmentAggregatorModule configuration given"));
430 }
431 std::string faUid("fragmentaggregator-" + UID());
432 // conffwk::ConfigObject frag_aggr;
433 TLOG_DEBUG(7) << "creating OKS configuration object for Fragment Aggregator class ";
434 auto frag_aggr = obj_fac.create("FragmentAggregatorModule", faUid);
435 conffwk::ConfigObject fa_net_obj = obj_fac.create_net_obj(fa_net_desc);
436
437 // Process special Network rules!
438 // Looking for Fragment rules from DFAppplications in current Session
439 std::vector<conffwk::ConfigObject> fragOutObjs;
440 for (auto [uid, descriptor]:
441 helper->get_netdescriptors("Fragment", "DFApplication")) {
442 std::string dreqNetUid(descriptor->get_uid_base() + uid);
443 auto frag_conn = obj_fac.create("NetworkConnection", dreqNetUid);
444
445 frag_conn.set_by_val<std::string>("data_type", descriptor->get_data_type());
446 frag_conn.set_by_val<std::string>("connection_type", descriptor->get_connection_type());
447 // Override capacity, set to 2x expected number of Fragments
448 frag_conn.set_by_val<int>("capacity", all_enabled_det_streams.size() * 2);
449
450 auto serviceObj = descriptor->get_associated_service()->config_object();
451 frag_conn.set_obj("associated_service", &serviceObj);
452 fragOutObjs.push_back(frag_conn);
453 }
454
455 // Add output queueus of data requests and Fragments
456 std::vector<const conffwk::ConfigObject*> fa_output_objs;
457 for (auto& fNet : fragOutObjs) {
458 fa_output_objs.push_back(&fNet);
459 }
460
461 for (auto& q : req_queues) {
462 fa_output_objs.push_back(&q->config_object());
463 }
464
465 frag_aggr.set_obj("configuration", &aggregator_conf->config_object());
466 frag_aggr.set_objs("inputs", { &fa_net_obj, &frag_queue_obj });
467 frag_aggr.set_objs("outputs", fa_output_objs);
468
469 modules.push_back(obj_fac.get_dal<confmodel::DaqModule>(frag_aggr.UID()));
470
471 obj_fac.update_modules(modules);
472}
473
474
475}
476}
#define ERS_HERE
void update_modules(const std::vector< const confmodel::DaqModule * > &modules)
conffwk::ConfigObject create_queue_sid_obj(const QueueDescriptor *qdesc, uint32_t src_id) const
conffwk::ConfigObject create_callback_sid_obj(const DataMoveCallbackDescriptor *cdesc, uint32_t src_id) const
conffwk::ConfigObject create_net_obj(const NetworkConnectionDescriptor *ndesc, std::string uid) const
Helper function that gets a network connection config.
const T * get_dal(std::string uid) const
conffwk::ConfigObject create_queue_obj(const QueueDescriptor *qdesc, std::string uid="") const
conffwk::ConfigObject create(const std::string &class_name, const std::string &id) const
const dunedaq::confmodel::NetworkDevice * get_uses() const
Get "uses" relationship value.
void generate_modules(std::shared_ptr< appmodel::ConfigurationHelper >) const override
const dunedaq::appmodel::DataHandlerConf * get_tp_handler() const
Get "tp_handler" relationship value.
const std::vector< const dunedaq::confmodel::DetectorToDaqConnection * > & get_detector_connections() const
Get "detector_connections" relationship value. The list of detector channels to be read out by this r...
bool get_ta_generation_enabled() const
Get "ta_generation_enabled" attribute value.
const dunedaq::appmodel::FragmentAggregatorConf * get_fragment_aggregator() const
Get "fragment_aggregator" relationship value.
const std::vector< const dunedaq::appmodel::SourceIDConf * > & get_tp_source_ids() const
Get "tp_source_ids" relationship value.
const dunedaq::appmodel::DataHandlerConf * get_link_handler() const
Get "link_handler" relationship value.
const dunedaq::appmodel::DataMoveCallbackDescriptor * get_callback_desc() const
Get "callback_desc" relationship value.
const dunedaq::appmodel::DataReaderConf * get_data_reader() const
Get "data_reader" relationship value.
bool get_tp_generation_enabled() const
Get "tp_generation_enabled" attribute value.
const std::vector< const dunedaq::appmodel::NetworkConnectionRule * > & get_network_rules() const
Get "network_rules" relationship value.
const std::vector< const dunedaq::appmodel::QueueConnectionRule * > & get_queue_rules() const
Get "queue_rules" relationship value.
void set_by_val(const std::string &name, T value)
Set attribute value.
void set_objs(const std::string &name, const std::vector< const ConfigObject * > &o, bool skip_non_null_check=false)
Set relationship multi-value.
const std::string & UID() const noexcept
Return object identity.
void set_obj(const std::string &name, const ConfigObject *o, bool skip_non_null_check=false)
Set relationship single-value.
const ConfigObject & config_object() const
const std::string & UID() const noexcept
uint8_t get_numa_id() const
Get "numa_id" attribute value.
conffwk entry point
#define TLOG_DEBUG(lvl,...)
Definition Logging.hpp:112
Including Qt Headers.