DD4hep  1.31.0
Detector Description Toolkit for High Energy Physics
Plugins.h
Go to the documentation of this file.
1 //==========================================================================
2 // AIDA Detector description implementation
3 //--------------------------------------------------------------------------
4 // Copyright (C) Organisation europeenne pour la Recherche nucleaire (CERN)
5 // All rights reserved.
6 //
7 // For the licensing terms see $DD4hepINSTALL/LICENSE.
8 // For the list of contributors see $DD4hepINSTALL/doc/CREDITS.
9 //
10 // Author : M.Frank
11 //
12 //==========================================================================
13 #ifndef DD4HEP_PLUGINS_H
14 #define DD4HEP_PLUGINS_H
15 
16 // Framework include files
17 #include <DD4hep/config.h>
18 
19 // ROOT include files
20 #ifndef __CINT__
21 #include <string>
22 #include <typeinfo>
23 
24 #if __cplusplus >= 201703
25 #include <any>
26 inline bool any_has_value(std::any a){ return a.has_value(); }
27 #else
28 # include <boost/any.hpp>
29 namespace std {
30  using boost::any;
31  using boost::any_cast;
32  using boost::bad_any_cast;
33  inline bool any_has_value(std::any a){ return !a.empty(); }
34 } // namespace std
35 #endif
36 
37 #ifndef DD4HEP_PARSERS_NO_ROOT
38 #include <RVersion.h>
39 #endif
40 
42 namespace dd4hep {
43 
44  class Detector;
45  class NamedObject;
46  template <typename T> class Handle;
47 
50  typedef std::string str_t;
51 
52  template <typename T> static T* ptr(const T* _p) { return (T*)_p; }
53  template <typename T> static T& ref(const T* _p) { return *(T*)_p; }
54  template <typename T> static T val(const T* _p) { return T(*_p); }
55  template <typename T> static T value(const void* _p) { return (T)_p; }
56  static const char* value(const void* _p) { return (const char*)(_p); }
57  };
58  template <> inline int PluginFactoryBase::value<int>(const void* _p) { return *(int*)(_p); }
59  template <> inline long PluginFactoryBase::value<long>(const void* _p) { return *(long*)(_p); }
60  template <> inline std::string PluginFactoryBase::value<std::string>(const void* _p) { return *(std::string*)(_p); }
61  template <> inline const std::string& PluginFactoryBase::value<const std::string&>(const void* _p) { return *(std::string*)(_p); }
62 
64 
73  struct PluginDebug {
74  int m_debug;
76  PluginDebug(int dbg = 2) noexcept(false);
78  ~PluginDebug() noexcept(false);
80  std::string missingFactory(const std::string& name) const;
81  };
82 
84  class PluginService {
85  private:
86  public:
87  typedef std::any stub_t;
88  template <typename R, typename... Args> static R Create(const std::string& id, Args... args) {
89  typedef R(*func)(Args...);
90  std::any f;
91  try {
92 #if DD4HEP_PLUGINSVC_VERSION==1
93  union { void* ptr; func fcn; } fptr;
94  f = getCreator(id,typeid(R(Args...)));
95  fptr.ptr = std::any_cast<void*>(f);
96  if ( fptr.ptr )
97  return (*fptr.fcn)(std::forward<Args>(args)...);
98 #elif DD4HEP_PLUGINSVC_VERSION==2
99  f = getCreator(id,typeid(R(Args...)));
100  if ( std::any_cast<func>(f) )
101  return std::any_cast<func>(f)(std::forward<Args>(args)...);
102 #endif
103  }
104  catch(const std::bad_any_cast& e) {
105  print_bad_cast(id, f, typeid(R(Args...)), e.what());
106  }
107  return 0;
108  }
109  template <typename FUNCTION> static std::any function(FUNCTION func) {
110 #if DD4HEP_PLUGINSVC_VERSION==1
111  union { void* ptr; FUNCTION fcn; } fptr;
112  fptr.fcn = func;
113  return std::any(fptr.ptr);
114 #elif DD4HEP_PLUGINSVC_VERSION==2
115  return std::any(func);
116 #endif
117  }
118  static bool debug();
119  static bool setDebug(bool new_value);
120  static stub_t getCreator(const std::string& id, const std::type_info& info);
121  static void addFactory(const std::string& id,
122  stub_t&& func,
123  const std::type_info& signature_type,
124  const std::type_info& return_type);
125  static void print_bad_cast(const std::string& id, const stub_t& stub, const std::type_info& signature, const char* msg);
126  };
127 
129  template <typename DD4HEP_SIGNATURE> class PluginRegistry {
130  public:
132  typedef DD4HEP_SIGNATURE signature_t;
133  template <typename R, typename... Args> static void add(const std::string& id, R(*func)(Args...)) {
134  svc_t::addFactory(id,svc_t::function(func),typeid(R(Args...)),typeid(R));
135  }
136  };
137 } /* End namespace dd4hep */
138 
139 namespace {
141  template <typename P, typename S> class Factory {};
142 }
143 
144 #define DD4HEP_FACTORY_CALL(type,name,signature) dd4hep::PluginRegistry<signature>::add(name,Factory<type,signature>::call)
145 #define DD4HEP_IMPLEMENT_PLUGIN_REGISTRY(X,Y)
146 
147 #define DD4HEP_OPEN_PLUGIN(ns,name) namespace ns { namespace { struct name {}; } } namespace dd4hep
148 #define DD4HEP_PLUGINSVC_CNAME(name, serial) name##_dict_##serial
149 #define DD4HEP_PLUGINSVC_FACTORY(type,name,signature,serial) \
150  namespace { \
151  struct DD4HEP_PLUGINSVC_CNAME(__typeFactory__,serial) { \
152  DD4HEP_PLUGINSVC_CNAME(__typeFactory__,serial)() { \
153  DD4HEP_FACTORY_CALL(type,#name,signature); }}; \
154  static const DD4HEP_PLUGINSVC_CNAME(__typeFactory__,serial) \
155  DD4HEP_PLUGINSVC_CNAME(s____typeFactory__,serial); \
156  }
157 
158 
159 #define DD4HEP_PLUGIN_FACTORY_ARGS_0(R) \
160  template <typename P> class Factory<P, R()> \
161  : public dd4hep::PluginFactoryBase { \
162  public: \
163  static R call(); \
164  }; \
165  template <typename P> inline R Factory<P,R()>::call()
166 
167 #define DD4HEP_PLUGIN_FACTORY_ARGS_1(R,A0) \
168  template <typename P> class Factory<P, R(A0)> \
169  : public dd4hep::PluginFactoryBase { \
170  public: \
171  static R call(A0 a0); \
172  }; \
173  template <typename P> inline R Factory<P,R(A0)>::call(A0 a0)
174 
175 #define DD4HEP_PLUGIN_FACTORY_ARGS_2(R,A0,A1) \
176  template <typename P> class Factory<P, R(A0,A1)> \
177  : public dd4hep::PluginFactoryBase { \
178  public: \
179  static R call(A0 a0,A1 a1); \
180  }; \
181  template <typename P> inline R Factory<P,R(A0,A1)>::call(A0 a0, A1 a1)
182 
183 #define DD4HEP_PLUGIN_FACTORY_ARGS_3(R,A0,A1,A2) \
184  template <typename P> class Factory<P, R(A0,A1,A2)> \
185  : public dd4hep::PluginFactoryBase { \
186  public: \
187  static R call(A0 a0,A1 a1,A2 a2); \
188  }; \
189  template <typename P> inline R Factory<P,R(A0,A1,A2)>::call(A0 a0, A1 a1, A2 a2)
190 
191 #define DD4HEP_PLUGIN_FACTORY_ARGS_4(R,A0,A1,A2,A3) \
192  template <typename P> class Factory<P, R(A0,A1,A2,A3)> \
193  : public dd4hep::PluginFactoryBase { \
194  public: \
195  static R call(A0 a0,A1 a1,A2 a2, A3 a3); \
196  }; \
197  template <typename P> inline R Factory<P,R(A0,A1,A2,A3)>::call(A0 a0, A1 a1, A2 a2, A3 a3)
198 
199 #define DD4HEP_PLUGIN_FACTORY_ARGS_5(R,A0,A1,A2,A3,A4) \
200  template <typename P> class Factory<P, R(A0,A1,A2,A3,A4)> \
201  : public dd4hep::PluginFactoryBase { \
202  public: \
203  static R call(A0 a0,A1 a1,A2 a2, A3 a3, A4 a4); \
204  }; \
205  template <typename P> inline R Factory<P,R(A0,A1,A2,A3,A4)>::call(A0 a0, A1 a1, A2 a2, A3 a3, A4 a4)
206 
207 #endif /* __CINT__ */
208 #endif // DD4HEP_PLUGINS_H
std::any_has_value
bool any_has_value(std::any a)
Definition: Plugins.h:33
dd4hep::PluginRegistry::signature_t
DD4HEP_SIGNATURE signature_t
Definition: Plugins.h:132
config.h
dd4hep::PluginService::stub_t
std::any stub_t
Definition: Plugins.h:87
dd4hep::PluginRegistry
Factory template for the plugin mechanism.
Definition: Plugins.h:129
dd4hep::info
std::size_t info(const std::string &src, const std::string &msg)
Definition: RootDictionary.h:65
dd4hep::PluginDebug::PluginDebug
PluginDebug(int dbg=2) noexcept(false)
Default constructor.
Definition: Plugins.cpp:137
dd4hep::debug
std::size_t debug(const std::string &src, const std::string &msg)
Definition: RootDictionary.h:64
dd4hep::PluginRegistry::svc_t
PluginService svc_t
Definition: Plugins.h:131
dd4hep::PluginService
Factory template for the plugin mechanism.
Definition: Plugins.h:84
dd4hep::PluginFactoryBase::str_t
std::string str_t
Definition: Plugins.h:50
dd4hep::PluginFactoryBase::ptr
static T * ptr(const T *_p)
Definition: Plugins.h:52
dd4hep::PluginFactoryBase
Factory base class implementing some utilities.
Definition: Plugins.h:49
dd4hep::PluginDebug
Helper to debug plugin manager calls.
Definition: Plugins.h:73
Gaudi::PluginService::v1::Details::getCreator
GAUDIPS_API void * getCreator(const std::string &id, const std::string &type)
Definition: PluginServiceV1.cpp:116
dd4hep::PluginRegistry::add
static void add(const std::string &id, R(*func)(Args...))
Definition: Plugins.h:133
dd4hep::PluginFactoryBase::value
static const char * value(const void *_p)
Definition: Plugins.h:56
dd4hep::PluginDebug::missingFactory
std::string missingFactory(const std::string &name) const
Helper to check factory existence.
Definition: Plugins.cpp:147
std
Definition: Plugins.h:29
dd4hep::PluginService::Create
static R Create(const std::string &id, Args... args)
Definition: Plugins.h:88
dd4hep
Namespace for the AIDA detector description toolkit.
Definition: AlignmentsCalib.h:28
dd4hep::PluginFactoryBase::val
static T val(const T *_p)
Definition: Plugins.h:54
_p
#define _p(x)
Definition: IoStreams.cpp:94
dd4hep::PluginDebug::m_debug
int m_debug
Definition: Plugins.h:74
dd4hep::PluginFactoryBase::value
static T value(const void *_p)
Definition: Plugins.h:55
dd4hep::PluginFactoryBase::ref
static T & ref(const T *_p)
Definition: Plugins.h:53