LCOV - code coverage report
Current view: top level - appmodel/src - ReadoutApplication.cpp (source / functions) Coverage Total Hit
Test: code.result Lines: 0.0 % 202 0
Test Date: 2026-03-29 15:29:34 Functions: 0.0 % 10 0

            Line data    Source code
       1              : /**
       2              :  * @file generate_modules.cpp
       3              :  *
       4              :  * Implementation of ReadoutApplication's generate_modules dal method
       5              :  *
       6              :  * This is part of the DUNE DAQ Software Suite, copyright 2023.
       7              :  * Licensing/copyright details are in the COPYING file that you should have
       8              :  * received with this code.
       9              :  */
      10              : 
      11              : 
      12              : 
      13              : #include "appmodel/ConfigurationHelper.hpp"
      14              : #include "ConfigObjectFactory.hpp"
      15              : #include "appmodel/ReadoutApplication.hpp"
      16              : #include "conffwk/Configuration.hpp"
      17              : #include "confmodel/DetDataReceiver.hpp"
      18              : #include "confmodel/DetDataSender.hpp"
      19              : #include "confmodel/DetectorStream.hpp"
      20              : #include "confmodel/DetectorToDaqConnection.hpp"
      21              : #include "confmodel/Session.hpp"
      22              : 
      23              : #include "appmodel/NWDetDataReceiver.hpp"
      24              : #include "appmodel/NWDetDataSender.hpp"
      25              : #include "appmodel/DPDKReceiver.hpp"
      26              : 
      27              : #include "appmodel/FelixDataReceiver.hpp"
      28              : #include "appmodel/FelixDataSender.hpp"
      29              : 
      30              : #include "appmodel/SocketReceiver.hpp"
      31              : #include "confmodel/QueueWithSourceId.hpp"
      32              : 
      33              : #include "confmodel/Connection.hpp"
      34              : #include "confmodel/GeoId.hpp"
      35              : #include "confmodel/NetworkConnection.hpp"
      36              : #include "confmodel/ResourceSet.hpp"
      37              : #include "confmodel/Service.hpp"
      38              : 
      39              : #include "appmodel/SourceIDConf.hpp"
      40              : #include "appmodel/DataMoveCallbackConf.hpp"
      41              : #include "appmodel/DataReaderModule.hpp"
      42              : #include "appmodel/DataReaderConf.hpp"
      43              : #include "appmodel/DataRecorderModule.hpp"
      44              : #include "appmodel/DataRecorderConf.hpp"
      45              : 
      46              : #include "appmodel/DataHandlerModule.hpp"
      47              : #include "appmodel/DataHandlerConf.hpp"
      48              : #include "appmodel/FragmentAggregatorModule.hpp"
      49              : #include "appmodel/FragmentAggregatorConf.hpp"
      50              : #include "appmodel/NetworkConnectionDescriptor.hpp"
      51              : #include "appmodel/NetworkConnectionRule.hpp"
      52              : #include "appmodel/QueueConnectionRule.hpp"
      53              : #include "appmodel/QueueDescriptor.hpp"
      54              : #include "appmodel/RequestHandler.hpp"
      55              : 
      56              : 
      57              : 
      58              : #include "appmodel/appmodelIssues.hpp"
      59              : 
      60              : #include "logging/Logging.hpp"
      61              : #include <fmt/core.h>
      62              : 
      63              : #include <string>
      64              : #include <vector>
      65              : 
      66              : // using namespace dunedaq;
      67              : // using namespace dunedaq::appmodel;
      68              : 
      69              : namespace dunedaq {
      70              : namespace appmodel {
      71              : 
      72              : //-----------------------------------------------------------------------------
      73              : 
      74              : std::vector<const confmodel::Resource*>
      75            0 : ReadoutApplication::contained_resources() const {
      76            0 :   return to_resources(get_detector_connections());
      77              : }
      78              : 
      79              : void
      80            0 : ReadoutApplication::generate_modules(std::shared_ptr<ConfigurationHelper> helper) const {
      81              : 
      82            0 :   TLOG_DEBUG(6) << "Generating modules for application " << this->UID();
      83              : 
      84            0 :   ConfigObjectFactory obj_fac(this);
      85              :   // conffwk::Configuration& confdb = this->configuration();
      86              : 
      87              :   //
      88              :   // Extract basic configuration objects
      89              :   //
      90              : 
      91              :   // Data reader
      92            0 :   auto reader_conf = get_data_reader();
      93            0 :   if (reader_conf == 0) {
      94              :     throw(BadConf(ERS_HERE, "No DataReaderModule configuration given"));
      95              :   }
      96            0 :   std::string reader_class = reader_conf->get_template_for();
      97              : 
      98              :   // Link handler
      99            0 :   auto dlh_conf = get_link_handler();
     100              :   // What is template for?
     101            0 :   auto dlh_class = dlh_conf->get_template_for();
     102              : 
     103            0 :   auto tph_conf = get_tp_handler();
     104            0 :   if (tph_conf==nullptr && get_tp_generation_enabled()) {
     105            0 :     throw(BadConf(ERS_HERE, "TP generation is enabled but there is no TP data handler configuration"));
     106              :   }
     107              : 
     108            0 :   std::string tph_class = "";
     109            0 :   if (tph_conf != nullptr && get_tp_generation_enabled()) {
     110            0 :     tph_class = tph_conf->get_template_for();
     111              :   }
     112              : 
     113              :   //
     114              :   // Process the queue rules looking for inputs to our DL/TP handler modules
     115              :   //
     116            0 :   const QueueDescriptor* dlh_reqinput_qdesc = nullptr;
     117            0 :   const QueueDescriptor* tp_input_qdesc = nullptr;
     118              :   // const QueueDescriptor* tpReqInputQDesc = nullptr;
     119            0 :   const QueueDescriptor* fa_output_qdesc = nullptr;
     120              : 
     121            0 :   for (auto rule : get_queue_rules()) {
     122            0 :     auto destination_class = rule->get_destination_class();
     123            0 :     auto data_type = rule->get_descriptor()->get_data_type();
     124              :     // Why datahander here? It is the base class for several DataHandler types (e.g. FDDataHandlerModule, SNBDataHandlerModule)
     125            0 :     if (destination_class == "DataHandlerModule" || destination_class == dlh_class || destination_class == tph_class) {
     126            0 :       if (data_type == "DataRequest") {
     127            0 :         dlh_reqinput_qdesc = rule->get_descriptor();
     128            0 :       } else if ((data_type == "TriggerPrimitive" || data_type == "TriggerPrimitiveVector") && get_tp_generation_enabled()) {
     129            0 :         tp_input_qdesc = rule->get_descriptor();
     130              :       }
     131            0 :     } else if (destination_class == "FragmentAggregatorModule") {
     132            0 :       fa_output_qdesc = rule->get_descriptor();
     133              :     }
     134            0 :   }
     135              : 
     136            0 :   if (dlh_reqinput_qdesc == nullptr) {
     137            0 :     throw(BadConf(ERS_HERE, "No data link handler request input queue descriptor given"));
     138              :   }
     139              : 
     140              :   //
     141              :   // Process the network rules looking for the Fragment Aggregator and TP handler data reuest inputs
     142              :   //
     143            0 :   const NetworkConnectionDescriptor* fa_net_desc = nullptr;
     144            0 :   const NetworkConnectionDescriptor* tp_net_desc = nullptr;
     145            0 :   const NetworkConnectionDescriptor* ta_net_desc = nullptr;
     146            0 :   const NetworkConnectionDescriptor* ts_net_desc = nullptr;
     147            0 :   for (auto rule : get_network_rules()) {
     148            0 :     auto endpoint_class = rule->get_endpoint_class();
     149            0 :     auto data_type = rule->get_descriptor()->get_data_type();
     150              : 
     151            0 :     if (endpoint_class == "FragmentAggregatorModule") {
     152            0 :       fa_net_desc = rule->get_descriptor();
     153            0 :     } else if (data_type == "TPSet") {
     154            0 :       tp_net_desc = rule->get_descriptor();
     155            0 :     } else if (data_type == "TriggerActivity") {
     156            0 :       ta_net_desc = rule->get_descriptor();
     157            0 :     } else if (data_type == "TimeSync") {
     158            0 :       ts_net_desc = rule->get_descriptor();
     159              :     }
     160            0 :   }
     161              : 
     162            0 :   if (fa_net_desc == nullptr) {
     163            0 :     throw(BadConf(ERS_HERE, "No Fragment Aggregator network descriptor given"));
     164              :   }
     165            0 :   if (ts_net_desc == nullptr && dlh_conf->get_generate_timesync()) {
     166            0 :     throw(BadConf(ERS_HERE, "No Time Sync network descriptor given but time sync generation is enabled"));
     167              :   }
     168              : 
     169              :   //
     170              :   // Get the callback descriptor
     171              :   //
     172            0 :   const DataMoveCallbackDescriptor* raw_data_callback_desc = get_callback_desc();
     173              : 
     174            0 :   if (raw_data_callback_desc == nullptr) {
     175              :     throw(BadConf(ERS_HERE, "No Raw Data Callback descriptor given"));
     176              :   }
     177              : 
     178              :   // Create here the Queue on which all data fragments are forwarded to the fragment aggregator
     179              :   // and a container for the queues of data request to TP handler and DLH
     180            0 :   if (fa_output_qdesc == nullptr) {
     181            0 :     throw(BadConf(ERS_HERE, "No fragment output queue descriptor given"));
     182              :   }
     183            0 :   std::vector<const confmodel::Connection*> req_queues;
     184            0 :   conffwk::ConfigObject frag_queue_obj = obj_fac.create_queue_obj(fa_output_qdesc);
     185              : 
     186              :   //
     187              :   // Scan Detector 2 DAQ connections to extract sender, receiver and stream information
     188              :   //
     189              : 
     190            0 :   std::vector<const confmodel::DaqModule*> modules;
     191              : 
     192              :   // Loop over the detector to daq connections and generate one data reader per connection
     193              :   // and the cooresponding datalink handlers
     194              : 
     195              :   // Collect all streams
     196            0 :   std::vector<const confmodel::DetectorStream*> all_enabled_det_streams;
     197            0 :   std::map<uint32_t, const appmodel::DataMoveCallbackConf*> callback_confs_by_sid;
     198              : 
     199              :   // std::vector<const conffwk::ConfigObject*> d2d_conn_objs;
     200            0 :   uint16_t conn_idx = 0;
     201              : 
     202            0 :   for (auto d2d_conn : get_detector_connections()) {
     203            0 :     if (helper->is_disabled(d2d_conn)) {
     204            0 :       TLOG_DEBUG(7) << "Ignoring disabled DetectorToDaqConnection " << d2d_conn->UID();
     205            0 :       continue;
     206            0 :     }
     207              : 
     208            0 :     TLOG_DEBUG(6) << "Processing DetectorToDaqConnection " << d2d_conn->UID();
     209              : 
     210              :     // Are these tests necessary? Schema does not allow 0 cardinality
     211              :     // for these relationships!!  TODO
     212            0 :     if (d2d_conn->senders().empty()) {
     213            0 :       throw(BadConf(ERS_HERE, "DetectorToDaqConnection does not contain senders"));
     214              :     }
     215            0 :     if (d2d_conn->receiver() == nullptr) {
     216            0 :       throw(BadConf(ERS_HERE, "DetectorToDaqConnection does not contain a receiver"));
     217              :     }
     218              : 
     219              :     // Find senders and receiver
     220            0 :     auto det_senders = d2d_conn->senders();
     221            0 :     auto det_receiver = d2d_conn->receiver();
     222              : 
     223            0 :     std::vector<const confmodel::DetectorStream*> enabled_det_streams;
     224              :     // Loop over streams
     225            0 :     for (auto stream : d2d_conn->streams()) {
     226              : 
     227              :       // Are we sure?
     228            0 :       if (helper->is_disabled(stream)) {
     229            0 :         TLOG_DEBUG(7) << "Ignoring disabled DetectorStream " << stream->UID();
     230            0 :         continue;
     231            0 :       }
     232              : 
     233            0 :       all_enabled_det_streams.push_back(stream);
     234            0 :       enabled_det_streams.push_back(stream);
     235            0 :     }
     236              : 
     237              : 
     238              :     // Here I want to resolve the type of connection (network, felix, or?)
     239              :     // Rules of engagement: if the receiver interface is network or felix, the receivers should be castable to the counterpart
     240            0 :     if (reader_class == "DPDKReaderModule" || reader_class == "SocketReaderModule") {
     241            0 :       if (!d2d_conn->castable("NetworkDetectorToDaqConnection")) {
     242            0 :         throw(BadConf(ERS_HERE, fmt::format("{} requires NetworkDetectorToDaqConnection, found {} of class {}", reader_class, d2d_conn->UID(), d2d_conn->class_name())));
     243              :       }
     244            0 :       if ((reader_class == "DPDKReaderModule" && !det_receiver->cast<appmodel::DPDKReceiver>()) ||
     245            0 :           (reader_class == "SocketReaderModule" && !det_receiver->cast<appmodel::SocketReceiver>())) {
     246            0 :         throw(BadConf(ERS_HERE, fmt::format("{} requires NWDetDataReceiver, found {} of class {}", reader_class, det_receiver->UID(), det_receiver->class_name())));
     247              :       }
     248              :     }
     249            0 :     else if (reader_class == "FelixReaderModule") {
     250            0 :       if (!d2d_conn->castable("FelixDetectorToDaqConnection")) {
     251            0 :         throw(BadConf(ERS_HERE, fmt::format("{} requires FelixDetectorToDaqConnection, found {} of class {}", reader_class, d2d_conn->UID(), d2d_conn->class_name())));
     252              :       }
     253            0 :       if (!det_receiver->cast<appmodel::FelixDataReceiver>()) {
     254            0 :         throw(BadConf(ERS_HERE, fmt::format("FelixReaderModule requires FelixDataReceiver, found {} of class {}", det_receiver->UID(), det_receiver->class_name())));
     255              :       }
     256              :     }
     257              :   // }
     258              : 
     259              :   //-----------------------------------------------------------------
     260              :   //
     261              :   // Create DataReaderModule object
     262              :   //
     263              : 
     264              :   //
     265              :   // Instantiate DataReaderModule of type DPDKReaderModule
     266              :   //
     267              : 
     268              :    // Create the Data reader object
     269              : 
     270            0 :     std::string reader_uid(fmt::format("datareader-{}-{}", this->UID(), std::to_string(conn_idx++)));
     271            0 :     TLOG_DEBUG(6) << fmt::format("creating OKS configuration object for Data reader class {} with id {}", reader_class, reader_uid);
     272            0 :     auto reader_obj = obj_fac.create(reader_class, reader_uid);
     273              : 
     274              :     // Populate configuration and interfaces (leave output queues for later)
     275            0 :     reader_obj.set_obj("configuration", &reader_conf->config_object());
     276            0 :     reader_obj.set_objs("connections", {&d2d_conn->config_object()});
     277              : 
     278              :     // Create the raw data callbacks
     279            0 :     std::vector<const conffwk::ConfigObject*> raw_data_callback_objs;
     280              : 
     281              :     // Create data queues
     282            0 :     for (auto ds : enabled_det_streams) {
     283            0 :       conffwk::ConfigObject callback_obj = obj_fac.create_callback_sid_obj(raw_data_callback_desc, ds->get_source_id());
     284            0 :       const auto* callback_conf = obj_fac.get_dal<DataMoveCallbackConf>(callback_obj.UID());
     285            0 :       raw_data_callback_objs.push_back(&callback_conf->config_object());
     286            0 :       callback_confs_by_sid[ds->get_source_id()] = callback_conf;
     287            0 :     }
     288              : 
     289            0 :     reader_obj.set_objs("raw_data_callbacks", raw_data_callback_objs);
     290              : 
     291            0 :     modules.push_back(obj_fac.get_dal<confmodel::DaqModule>(reader_obj.UID()));
     292            0 :   }
     293              : 
     294              : 
     295              :   //-----------------------------------------------------------------
     296              :   //
     297              :   // Prepare the tp handlers and related queues
     298              :   //
     299            0 :   std::vector<const confmodel::Connection*> tp_queues;
     300            0 :   if (get_tp_generation_enabled()) {
     301            0 :     if (tp_input_qdesc == nullptr) {
     302            0 :         throw(BadConf(ERS_HERE, "TP generation is enabled but no TP input queue descriptor given"));
     303              :     }
     304            0 :     if (tp_net_desc == nullptr) {
     305            0 :         throw(BadConf(ERS_HERE, "TP generation is enabled but no TPSet network descriptor given"));
     306              :     }
     307            0 :     if (ta_net_desc == nullptr) {
     308            0 :       throw(BadConf(ERS_HERE, "TP generation is enabled but no TriggerActivity network descriptor given"));
     309              :     }
     310              :     // Create TP handler object
     311            0 :     auto tph_conf_obj = tph_conf->config_object();
     312            0 :     auto tpsrc_ids = get_tp_source_ids();
     313              : 
     314            0 :     for (auto sid : tpsrc_ids) {
     315            0 :       conffwk::ConfigObject tp_queue_obj;
     316            0 :       conffwk::ConfigObject tpreq_queue_obj;
     317            0 :       std::string tp_uid("tphandler-" + std::to_string(sid->get_sid()));
     318            0 :       auto tph_obj = obj_fac.create(tph_class, tp_uid);
     319            0 :       tph_obj.set_by_val<uint32_t>("source_id", sid->get_sid());
     320            0 :       tph_obj.set_by_val<uint32_t>("detector_id", 1); // 1 == kDAQ
     321            0 :       tph_obj.set_by_val<bool>("post_processing_enabled", get_ta_generation_enabled());
     322            0 :       tph_obj.set_obj("module_configuration", &tph_conf_obj);
     323              : 
     324              :       // Create the TPs aggregator queue (from RawData Handlers to TP handlers)
     325            0 :       tp_queue_obj = obj_fac.create_queue_sid_obj(tp_input_qdesc, sid->get_sid());
     326            0 :       tp_queue_obj.set_by_val<uint32_t>("recv_timeout_ms", 50);
     327            0 :       tp_queue_obj.set_by_val<uint32_t>("send_timeout_ms", 1);
     328              : 
     329            0 :       tp_queues.push_back(obj_fac.get_dal<confmodel::Connection>(tp_queue_obj.UID()));
     330              :       // Create tp data requests queue from Fragment Aggregator
     331            0 :       tpreq_queue_obj = obj_fac.create_queue_sid_obj(dlh_reqinput_qdesc, sid->get_sid());
     332            0 :       req_queues.push_back(obj_fac.get_dal<confmodel::Connection>(tpreq_queue_obj.UID()));
     333              : 
     334              :       // Create the tp(set) publishing service
     335            0 :       conffwk::ConfigObject tp_net_obj = obj_fac.create_net_obj(tp_net_desc, tp_uid);
     336              : 
     337              :       // Create the ta(set) publishing service
     338            0 :       conffwk::ConfigObject ta_net_obj = obj_fac.create_net_obj(ta_net_desc, tp_uid);
     339              : 
     340              :       // Register queues with tp hankder
     341            0 :       tph_obj.set_objs("inputs", { &tp_queue_obj, &tpreq_queue_obj });
     342            0 :       tph_obj.set_objs("outputs", { &tp_net_obj, &ta_net_obj, &frag_queue_obj });
     343              : 
     344            0 :       modules.push_back(obj_fac.get_dal<confmodel::DaqModule>(tph_obj.UID()));
     345            0 :     }
     346            0 :   }
     347              : 
     348              :     // Add output queueus of tps
     349            0 :   std::vector<const conffwk::ConfigObject*> tp_queue_objs;
     350            0 :   for (auto q : tp_queues) {
     351            0 :     tp_queue_objs.push_back(&q->config_object());
     352              :   }
     353              : 
     354              :   //-----------------------------------------------------------------
     355              :   //
     356              :   // Create datalink handlers
     357              :   //
     358              :   // Recover the emulation flag
     359            0 :   auto emulation_mode = reader_conf->get_emulation_mode();
     360            0 :   for (auto ds : all_enabled_det_streams) {
     361              : 
     362            0 :     uint32_t sid = ds->get_source_id();
     363            0 :     TLOG_DEBUG(6) << fmt::format("Processing stream {}, id {}, det id {}", ds->UID(), ds->get_source_id(), ds->get_geo_id()->get_detector_id());
     364            0 :     std::string uid(fmt::format("DLH-{}", sid));
     365            0 :     TLOG_DEBUG(6) << fmt::format("creating OKS configuration object for Data Link Handler class {}, if {}", dlh_class, sid);
     366            0 :     auto dlh_obj = obj_fac.create(dlh_class, uid);
     367            0 :     dlh_obj.set_by_val<uint32_t>("source_id", sid);
     368            0 :     dlh_obj.set_by_val<uint32_t>("detector_id", ds->get_geo_id()->get_detector_id());
     369            0 :     dlh_obj.set_by_val<bool>("post_processing_enabled", get_tp_generation_enabled());
     370            0 :     dlh_obj.set_by_val<bool>("emulation_mode", emulation_mode);
     371            0 :     dlh_obj.set_obj("geo_id", &ds->get_geo_id()->config_object());
     372            0 :     dlh_obj.set_obj("module_configuration", &dlh_conf->config_object());
     373            0 :     dlh_obj.set_obj("raw_data_callback", &callback_confs_by_sid[sid]->config_object());
     374              : 
     375            0 :     std::vector<const conffwk::ConfigObject*> dlh_ins, dlh_outs;
     376              : 
     377              :     // Create request queue
     378            0 :     conffwk::ConfigObject req_queue_obj = obj_fac.create_queue_sid_obj(dlh_reqinput_qdesc, ds);
     379              : 
     380              : 
     381              :     // Add the requessts queue dal pointer to the outputs of the FragmentAggregatorModule
     382            0 :     req_queues.push_back(obj_fac.get_dal<confmodel::Connection>(req_queue_obj.UID()));
     383            0 :     dlh_ins.push_back(&req_queue_obj);
     384            0 :     dlh_outs.push_back(&frag_queue_obj);
     385              : 
     386              : 
     387              :     // Time Sync network connection
     388            0 :     if (dlh_conf->get_generate_timesync()) {
     389              :       // Add timestamp endpoint
     390            0 :       conffwk::ConfigObject ts_net_obj = obj_fac.create_net_obj(ts_net_desc, std::to_string(sid));
     391            0 :       dlh_outs.push_back(&ts_net_obj);
     392            0 :     }
     393              : 
     394            0 :     for (auto tpq : tp_queue_objs) {
     395            0 :       dlh_outs.push_back(tpq);
     396              :     }
     397            0 :     dlh_obj.set_objs("inputs", dlh_ins);
     398            0 :     dlh_obj.set_objs("outputs", dlh_outs);
     399              : 
     400            0 :     modules.push_back(obj_fac.get_dal<confmodel::DaqModule>(dlh_obj.UID()));
     401            0 :   }
     402              : 
     403              :   // Finally create Fragment Aggregator
     404            0 :   auto aggregator_conf = get_fragment_aggregator();
     405            0 :   if (aggregator_conf == 0) {
     406              :     throw(BadConf(ERS_HERE, "No FragmentAggregatorModule configuration given"));
     407              :   }
     408            0 :   std::string faUid("fragmentaggregator-" + UID());
     409              :   // conffwk::ConfigObject frag_aggr;
     410            0 :   TLOG_DEBUG(7) << "creating OKS configuration object for Fragment Aggregator class ";
     411            0 :   auto frag_aggr = obj_fac.create("FragmentAggregatorModule", faUid);
     412            0 :   conffwk::ConfigObject fa_net_obj = obj_fac.create_net_obj(fa_net_desc);
     413              : 
     414              :   // Process special Network rules!
     415              :   // Looking for Fragment rules from DFAppplications in current Session
     416            0 :   std::vector<conffwk::ConfigObject> fragOutObjs;
     417            0 :   for (auto [uid, descriptor]:
     418            0 :          helper->get_netdescriptors("Fragment", "DFApplication")) {
     419            0 :     std::string dreqNetUid(descriptor->get_uid_base() + uid);
     420            0 :     auto frag_conn = obj_fac.create("NetworkConnection", dreqNetUid);
     421              : 
     422            0 :     frag_conn.set_by_val<std::string>("data_type", descriptor->get_data_type());
     423            0 :     frag_conn.set_by_val<std::string>("connection_type", descriptor->get_connection_type());
     424              :     // Override capacity, set to 2x expected number of Fragments
     425            0 :     frag_conn.set_by_val<int>("capacity", all_enabled_det_streams.size() * 2);
     426              : 
     427            0 :     auto serviceObj = descriptor->get_associated_service()->config_object();
     428            0 :     frag_conn.set_obj("associated_service", &serviceObj);
     429            0 :     fragOutObjs.push_back(frag_conn);
     430            0 :   }    
     431              : 
     432              :   // Add output queueus of data requests and Fragments
     433            0 :   std::vector<const conffwk::ConfigObject*> fa_output_objs;
     434            0 :   for (auto& fNet : fragOutObjs) {
     435            0 :     fa_output_objs.push_back(&fNet);
     436              :   }
     437              : 
     438            0 :   for (auto& q : req_queues) {
     439            0 :     fa_output_objs.push_back(&q->config_object());
     440              :   }
     441              : 
     442            0 :   frag_aggr.set_obj("configuration", &aggregator_conf->config_object());
     443            0 :   frag_aggr.set_objs("inputs", { &fa_net_obj, &frag_queue_obj });
     444            0 :   frag_aggr.set_objs("outputs", fa_output_objs);
     445            0 :   modules.push_back(obj_fac.get_dal<confmodel::DaqModule>(frag_aggr.UID()));
     446              : 
     447            0 :   obj_fac.update_modules(modules);
     448            0 : }
     449              : 
     450              : } // namespace appmodel
     451              : } // namespace dunedaq
        

Generated by: LCOV version 2.0-1