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