QwAnalysis
QwFactory.h
Go to the documentation of this file.
1 #ifndef __QWFACTORY__
2 #define __QWFACTORY__
3 
4 // System headers
5 #include <cxxabi.h>
6 #include <cstdlib>
7 using std::type_info;
8 
9 // Qweak headers
10 #include "QwLog.h"
11 
12 // Forward declarations
13 class VQwSubsystem;
14 class VQwDataElement;
15 
16 // Exceptions
19 };
20 
21 
22 /**
23  * \class VQwFactory
24  * \ingroup QwAnalysis
25  * \brief Pure virtual factory
26  *
27  * In order to enable the instantiation of types based on run-time
28  * information, we generate a map of type factories by type name.
29  * This map is filled automatically when the executable is loaded, and
30  * contains concrete factories derived from this pure virtual base class.
31  */
32 template <class base_t>
33 class VQwFactory {
34 
35  public:
36 
37  /// Default virtual destructor
38  virtual ~VQwFactory() { }
39 
40  /// Create an object of type with name
41  static base_t* Create(const std::string& type, const std::string& name) {
42  return GetFactory(type)->Create(name);
43  }
44 
45  /// Dynamic cast of object into type
46  static base_t* Cast(base_t* base, const std::string& type) {
47  return GetFactory(type)->Cast(base);
48  }
49 
50  /// Test whether object inherits from type
51  static bool InheritsFrom(base_t* base, const std::string& type) {
52  return (Cast(base,type) != 0);
53  }
54 
55  /// Creation of objects by type (pure virtual)
56  virtual base_t* Create(const std::string& name) const = 0;
57  /// Dynamic cast to type (pure virtual)
58  virtual base_t* Cast(base_t* base) const = 0;
59 
60  protected:
61 
62  /// Map from string to concrete type factories
63  static std::map<std::string,VQwFactory*>& GetRegisteredTypes() {
64  static std::map<std::string,VQwFactory<base_t>*> theMap;
65  return theMap;
66  }
67  /// List available type factories
68  static void ListRegisteredTypes() {
69  typename std::map<std::string,VQwFactory<base_t>*>::iterator type;
70  for (type = GetRegisteredTypes().begin();
71  type != GetRegisteredTypes().end(); type++ )
72  QwMessage << type->first << QwLog::endl;
73  }
74 
75  /// Get a concrete type factory by string
76  static VQwFactory* GetFactory(const std::string& type) {
77  if (GetRegisteredTypes().find(type) != GetRegisteredTypes().end())
78  return GetRegisteredTypes()[type];
79  else {
80  QwError << "Type " << type << " is not registered!" << QwLog::endl;
81  QwMessage << "Available types:" << QwLog::endl;
83  QwWarning << "To register this type, add the following line to the top "
84  << "of the source file:" << QwLog::endl;
85  QwWarning << " RegisterSomethingFactory(" << type << ");" << QwLog::endl;
86  QwWarning << "Ensure that the dynamic library contains the factory object."
87  << QwLog::endl;
89  }
90  }
91 
92 }; // class VQwFactory
93 
94 
95 /**
96  * \class QwFactory
97  * \ingroup QwAnalysis
98  * \brief Concrete templated type factory
99  *
100  * This class represents concrete instances of the virtual VQwFactory
101  * from which it inherits. Each concrete factory can create types with
102  * a given name.
103  */
104 template <class base_t, class type_t>
105 class QwFactory: public VQwFactory<base_t> {
106 
107  public:
108 
109  /// Constructor which stores type name in list of registered types
110  QwFactory(const std::string& type) {
112  }
113 
114  /// Concrete type creation
115  base_t* Create(const std::string& name) const {
116  return new type_t(name);
117  }
118 
119  /// Dynamic cast of type
120  type_t* Cast(base_t* base) const {
121  return dynamic_cast<type_t*>(base);
122  }
123 
124 }; // class QwFactory
125 
126 /// Factory type with functionality for subsystems
128 /// Factory type with functionality for data elements
130 
131 // TODO It would be nice to be able to define something like
132 // template <class type_t>
133 // typedef class QwFactory<VQwSubsystem,type_t> QwSubsystemFactory;
134 // to be able to write e.g.
135 // QwSubsystemFactory<QwLumi>
136 //
137 // This seems to be functionality that is in C++0x...
138 // typedef <class type_t> using QwSubsystemFactory = QwFactory<VQwSubsystem,type_t>;
139 //
140 // For now the following will do as well, but adds a layer of inheritance
141 template <class subsystem_t>
142 class QwSubsystemFactory: public QwFactory<VQwSubsystem,subsystem_t> { };
143 template <class dataelement_t>
144 class QwDataElementFactory: public QwFactory<VQwDataElement,dataelement_t> { };
145 
146 
147 /// Polymorphic copy constructor virtual base class
148 template <class base_t>
150 
151  public:
152 
153  /// Virtual destructor
154  virtual ~VQwCloneable() { }
155 
156  /// Get demangled name of this class
157  std::string GetClassName() const {
158  int status;
159  const type_info& ti = typeid(*this);
160  char* name = abi::__cxa_demangle(ti.name(), 0, 0, &status);
161  std::string str(name);
162  free(name);
163  return str;
164  }
165 
166  /// Abstract clone method when no derived method is defined
167  virtual base_t* Clone() const {
168  QwError << "Clone() is not implemented for class " << GetClassName() << "!" << QwLog::endl;
169  QwMessage << "Modify the class definition of " << GetClassName() << " to:" << QwLog::endl;
170  QwMessage << " class " << GetClassName() << ": "
171  << "public MQwSomethingCloneable<" << GetClassName() << ">" << QwLog::endl;
172  return 0;
173  }
174 
175  /// Virtual factory getter
176  virtual const VQwFactory<base_t>* Factory() const { return 0; }
177 
178 }; // class VQwCloneable
179 
180 
181 /// Polymorphic copy construction by curiously recurring template pattern (mix-in)
182 /// We have lost covariancy: clone will have the base type, not the derived type...
183 template <class base_t, class type_t>
184 class MQwCloneable: virtual public VQwCloneable<base_t> {
185 
186  public:
187 
188  /// Virtual destructor
189  virtual ~MQwCloneable() { };
190 
191  /// Concrete clone method
192  virtual base_t* Clone() const {
193  return new type_t(static_cast<const type_t&>(*this));
194  }
195 
196  /// Factory getter
197  const VQwFactory<base_t>* Factory() const { return fFactory; }
198 
199  /// Object creation
200  static base_t* Create(const std::string& name) {
201  if (fFactory) return fFactory->Create(name);
202  else return 0;
203  }
204 
205  /// Object dynamic cast
206  static type_t* Cast(type_t* type) {
207  if (fFactory) return fFactory->Cast(type);
208  else return 0;
209  }
210 
211  private:
212 
213  /// Pointer to the factory of this type
215 
216 }; // class MQwCloneable
217 
218 /// Mix-in factory functionality for subsystems
220 template <class subsystem_t>
221 class MQwSubsystemCloneable: public MQwCloneable<VQwSubsystem,subsystem_t> { };
222 /// Mix-in factory functionality for data elements
224 template <class dataelement_t>
225 class MQwDataElementCloneable: public MQwCloneable<VQwDataElement,dataelement_t> { };
226 
227 
228 /// Macros to create and register the subsystem factory of type A
229 /// Note: a call to this macro should be followed by a semi-colon!
230 #define RegisterSubsystemFactory(A) template<> const VQwSubsystemFactory* MQwCloneable<VQwSubsystem,A>::fFactory = new QwFactory<VQwSubsystem,A>(#A)
231 
232 /// Macros to create and register the data element factory of type A
233 /// Note: a call to this macro should be followed by a semi-colon!
234 #define RegisterDataElementFactory(A) template<> const VQwDataElementFactory* MQwCloneable<VQwDataElement,A>::fFactory = new QwFactory<VQwDataElement,A>(#A)
235 
236 
237 #endif // __QWFACTORY__
#define QwMessage
Predefined log drain for regular messages.
Definition: QwLog.h:50
virtual base_t * Clone() const
Abstract clone method when no derived method is defined.
Definition: QwFactory.h:167
virtual ~MQwCloneable()
Virtual destructor.
Definition: QwFactory.h:189
std::string GetClassName() const
Get demangled name of this class.
Definition: QwFactory.h:157
class VQwCloneable< VQwSubsystem > VQwSubsystemCloneable
Mix-in factory functionality for subsystems.
Definition: QwFactory.h:219
virtual ~VQwFactory()
Default virtual destructor.
Definition: QwFactory.h:38
const VQwFactory< base_t > * Factory() const
Factory getter.
Definition: QwFactory.h:197
type_t * Cast(base_t *base) const
Dynamic cast of type.
Definition: QwFactory.h:120
static base_t * Cast(base_t *base, const std::string &type)
Dynamic cast of object into type.
Definition: QwFactory.h:46
Pure virtual factory.
Definition: QwFactory.h:33
virtual const VQwFactory< base_t > * Factory() const
Virtual factory getter.
Definition: QwFactory.h:176
The pure virtual base class of all data elements.
A logfile class, based on an identical class in the Hermes analyzer.
static std::map< std::string, VQwFactory * > & GetRegisteredTypes()
Map from string to concrete type factories.
Definition: QwFactory.h:63
QwFactory(const std::string &type)
Constructor which stores type name in list of registered types.
Definition: QwFactory.h:110
The pure virtual base class of all subsystems.
Definition: VQwSubsystem.h:59
class VQwCloneable< VQwDataElement > VQwDataElementCloneable
Mix-in factory functionality for data elements.
Definition: QwFactory.h:223
static base_t * Create(const std::string &type, const std::string &name)
Create an object of type with name.
Definition: QwFactory.h:41
static VQwFactory * GetFactory(const std::string &type)
Get a concrete type factory by string.
Definition: QwFactory.h:76
virtual ~VQwCloneable()
Virtual destructor.
Definition: QwFactory.h:154
Concrete templated type factory.
Definition: QwFactory.h:105
static void ListRegisteredTypes()
List available type factories.
Definition: QwFactory.h:68
static std::ostream & endl(std::ostream &)
End of the line.
Definition: QwLog.cc:299
static base_t * Create(const std::string &name)
Object creation.
Definition: QwFactory.h:200
Polymorphic copy constructor virtual base class.
Definition: QwFactory.h:149
#define QwWarning
Predefined log drain for warnings.
Definition: QwLog.h:45
class VQwFactory< VQwSubsystem > VQwSubsystemFactory
Factory type with functionality for subsystems.
Definition: QwFactory.h:127
static type_t * Cast(type_t *type)
Object dynamic cast.
Definition: QwFactory.h:206
static bool InheritsFrom(base_t *base, const std::string &type)
Test whether object inherits from type.
Definition: QwFactory.h:51
base_t * Create(const std::string &name) const
Concrete type creation.
Definition: QwFactory.h:115
class VQwFactory< VQwDataElement > VQwDataElementFactory
Factory type with functionality for data elements.
Definition: QwFactory.h:129
virtual base_t * Clone() const
Concrete clone method.
Definition: QwFactory.h:192
#define QwError
Predefined log drain for errors.
Definition: QwLog.h:40
static const VQwFactory< base_t > * fFactory
Pointer to the factory of this type.
Definition: QwFactory.h:214