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