CMS 3D CMS Logo

ProductRegistryHelper.cc
Go to the documentation of this file.
1 /*----------------------------------------------------------------------
2 
3 ----------------------------------------------------------------------*/
4 
13 
14 #include <vector>
15 #include <typeindex>
16 
17 namespace edm {
19 
21 
22  namespace {
23  void throwProducesWithoutAbility(const char* transitionName, std::string const& productTypeName) {
25  << "Module declares it can produce a product of type \'" << productTypeName << "\'\nin a " << transitionName
26  << ", but does not have the ability to produce in " << transitionName << "s.\n"
27  << "You must add a template parameter of type " << transitionName << "Producer\n"
28  << "or " << transitionName << "Producer to the EDProducer or EDFilter base class\n"
29  << "of the module. Or you could remove the call to the function \'produces\'\n"
30  << "(Note legacy modules are not ever allowed to produce in Runs or Lumis)\n";
31  }
32  } // namespace
33 
34  void ProductRegistryHelper::addToRegistry(TypeLabelList::const_iterator const& iBegin,
35  TypeLabelList::const_iterator const& iEnd,
36  ModuleDescription const& iDesc,
37  ProductRegistry& iReg,
38  ProductRegistryHelper* iProd,
39  bool iIsListener) {
40  std::vector<std::string> missingDictionaries;
41  std::vector<std::string> producedTypes;
42  std::set<std::tuple<BranchType, std::type_index, std::string>> registeredProducts;
43 
44  for (TypeLabelList::const_iterator p = iBegin; p != iEnd; ++p) {
45  if (p->transition_ == Transition::BeginRun && not iProd->hasAbilityToProduceInBeginRuns()) {
46  throwProducesWithoutAbility("BeginRun", p->typeID_.userClassName());
47  } else if (p->transition_ == Transition::EndRun && not iProd->hasAbilityToProduceInEndRuns()) {
48  throwProducesWithoutAbility("EndRun", p->typeID_.userClassName());
49  } else if (p->transition_ == Transition::BeginLuminosityBlock && not iProd->hasAbilityToProduceInBeginLumis()) {
50  throwProducesWithoutAbility("BeginLuminosityBlock", p->typeID_.userClassName());
51  } else if (p->transition_ == Transition::EndLuminosityBlock && not iProd->hasAbilityToProduceInEndLumis()) {
52  throwProducesWithoutAbility("EndLuminosityBlock", p->typeID_.userClassName());
53  } else if (p->transition_ == Transition::BeginProcessBlock &&
55  throwProducesWithoutAbility("BeginProcessBlock", p->typeID_.userClassName());
56  } else if (p->transition_ == Transition::EndProcessBlock && not iProd->hasAbilityToProduceInEndProcessBlocks()) {
57  throwProducesWithoutAbility("EndProcessBlock", p->typeID_.userClassName());
58  }
59  if (!checkDictionary(missingDictionaries, p->typeID_)) {
60  checkDictionaryOfWrappedType(missingDictionaries, p->typeID_);
61  producedTypes.emplace_back(p->typeID_.className());
62  continue;
63  }
64  auto branchType = convertToBranchType(p->transition_);
65  if (branchType != InEvent) {
66  std::tuple<BranchType, std::type_index, std::string> entry{
67  branchType, p->typeID_.typeInfo(), p->productInstanceName_};
68  if (registeredProducts.end() != registeredProducts.find(entry)) {
69  //ignore registration of items if in both begin and end transitions for now
70  // This is to work around ExternalLHEProducer
71  continue;
72  } else {
73  registeredProducts.insert(entry);
74  }
75  }
76 
77  TypeWithDict type(p->typeID_.typeInfo());
78  BranchDescription pdesc(branchType,
79  iDesc.moduleLabel(),
80  iDesc.processName(),
81  p->typeID_.userClassName(),
82  p->typeID_.friendlyClassName(),
83  p->productInstanceName_,
84  iDesc.moduleName(),
85  iDesc.parameterSetID(),
86  type,
87  true,
88  isEndTransition(p->transition_));
89  if (p->aliasType_ == TypeLabelItem::AliasType::kSwitchAlias) {
90  if (p->branchAlias_.empty()) {
92  << "Branch alias type has been set to SwitchAlias, but the alias content is empty.\n"
93  << "Please report this error to the FWCore developers";
94  }
95  pdesc.setSwitchAliasModuleLabel(p->branchAlias_);
96  }
97  if (p->isTransform_) {
98  pdesc.setOnDemand(true);
99  pdesc.setIsTransform(true);
100  }
101  setIsMergeable(pdesc);
102 
103  if (pdesc.transient()) {
104  if (!checkDictionary(missingDictionaries, pdesc.wrappedName(), pdesc.wrappedType())) {
105  // It is should be impossible to get here, because the only way to
106  // make it transient is in the line that causes the wrapped dictionary
107  // to be created. Just to be safe I leave this check here ...
108  producedTypes.emplace_back(pdesc.className());
109  continue;
110  }
111  } else {
112  // also check constituents of wrapped types if it is not transient
113  if (!checkClassDictionaries(missingDictionaries, pdesc.wrappedName(), pdesc.wrappedType())) {
114  producedTypes.emplace_back(pdesc.className());
115  continue;
116  }
117  }
118  if (!p->branchAlias_.empty())
119  pdesc.insertBranchAlias(p->branchAlias_);
120  iReg.addProduct(pdesc, iIsListener);
121  }
122 
123  if (!missingDictionaries.empty()) {
124  std::string context("Calling ProductRegistryHelper::addToRegistry, checking dictionaries for produced types");
125  throwMissingDictionariesException(missingDictionaries, context, producedTypes);
126  }
127  }
128 } // namespace edm
std::string_view transitionName(GlobalContext::Transition)
void throwMissingDictionariesException(std::vector< std::string > &missingDictionaries, std::string const &context)
virtual ~ProductRegistryHelper() noexcept(false)
std::string const & moduleName() const
static void addToRegistry(TypeLabelList::const_iterator const &iBegin, TypeLabelList::const_iterator const &iEnd, ModuleDescription const &iDesc, ProductRegistry &iReg, ProductRegistryHelper *iProd, bool iIsListener=false)
virtual bool hasAbilityToProduceInEndRuns() const
constexpr bool isEndTransition(Transition iValue)
Definition: Transition.h:37
virtual bool hasAbilityToProduceInBeginLumis() const
ParameterSetID const & parameterSetID() const
std::vector< TypeLabelItem > TypeLabelList
bool checkDictionary(std::vector< std::string > &missingDictionaries, TypeID const &typeID)
virtual bool hasAbilityToProduceInEndProcessBlocks() const
bool checkClassDictionaries(std::vector< std::string > &missingDictionaries, TypeID const &typeID)
TypeLabelList const & typeLabelList() const
used by the fwk to register the list of products of this module
std::string const & processName() const
HLT enums.
virtual bool hasAbilityToProduceInBeginProcessBlocks() const
virtual bool hasAbilityToProduceInBeginRuns() const
bool checkDictionaryOfWrappedType(std::vector< std::string > &missingDictionaries, TypeID const &unwrappedTypeID)
virtual bool hasAbilityToProduceInEndLumis() const
std::string const & moduleLabel() const
void addProduct(BranchDescription const &productdesc, bool iFromListener=false)
void setIsMergeable(BranchDescription &)
constexpr BranchType convertToBranchType(Transition iValue)
Definition: Transition.h:26