The Gaudi Framework  v30r3 (a5ef0a68)
RegistryEntry.cpp
Go to the documentation of this file.
1 //====================================================================
2 // RegistryEntry.cpp
3 //--------------------------------------------------------------------
4 //
5 // Package : DataSvc ( The LHCb Offline System)
6 //
7 // Description: implementation of the Transient data store
8 //
9 // Author : M.Frank
10 // History :
11 // +---------+----------------------------------------------+---------
12 // | Date | Comment | Who
13 // +---------+----------------------------------------------+---------
14 // | 29/10/98| Initial version | MF
15 // | 03/02/99| Protect dynamic_cast with try-catch clauses | MF
16 // +---------+----------------------------------------------+---------
17 //
18 //====================================================================
19 #define DATASVC_REGISTRYENTRY_CPP
20 
21 // STL include files
22 #include <algorithm>
23 
24 // Interfaces
27 
28 // Framework include files
29 #include "GaudiKernel/DataObject.h"
31 
32 // If you absolutely need optimization: switch off dynamic_cast.
33 // This improves access to the data store roughly by more than 10 %
34 // for balanced trees.
35 //
36 // M.Frank
37 //
38 #define CAST_REGENTRY( x, y ) dynamic_cast<x>( y )
39 //#define CAST_REGENTRY(x,y) (x)(y)
40 constexpr char SEPARATOR{'/'};
41 
44  : m_path( std::move( path ) ), m_pParent( parent )
45 {
46  std::string::size_type sep = m_path.rfind( SEPARATOR );
47  if ( m_path.front() != SEPARATOR ) m_path.insert( 0, 1, SEPARATOR );
48  if ( sep != std::string::npos ) m_path.erase( 0, sep );
50  addRef();
51 }
52 
55 {
57  if ( m_pObject ) {
58  if ( !m_isSoft ) m_pObject->setRegistry( nullptr );
59  m_pObject->release();
60  }
61  if ( m_pAddress ) {
62  if ( !m_isSoft ) m_pAddress->setRegistry( nullptr );
64  }
65 }
66 
69 {
70  unsigned long cnt = --m_refCount;
71  if ( !m_refCount ) delete this;
72  return cnt;
73 }
74 
77 {
78  m_pParent = pParent;
79  m_fullpath.clear();
81 }
82 
85 {
86  m_isSoft = true;
87  setObject( pObject );
88  // if ( 0 != m_pObject ) { // Useless: This justs sets my own address again...
89  // setAddress(m_pObject->address());
90  // }
91 }
92 
95 {
96  m_isSoft = true;
97  setAddress( pAddress );
98 }
99 
102 {
103  makeSoft( pObject );
104  m_isSoft = false;
105  if ( m_pObject ) m_pObject->setRegistry( this );
106  if ( m_pAddress ) m_pAddress->setRegistry( this );
107 }
108 
111 {
112  m_isSoft = false;
113  setAddress( pAddress );
114 }
115 
118 {
119  if ( pAddress ) {
120  pAddress->addRef();
121  pAddress->setRegistry( this );
122  }
123  if ( m_pAddress ) m_pAddress->release();
124  m_pAddress = pAddress;
125 }
126 
129 {
130  if ( pObject ) {
131  pObject->addRef();
132  if ( !isSoft() ) pObject->setRegistry( this );
133  }
134  if ( m_pObject ) m_pObject->release();
135  m_pObject = pObject;
136 }
137 
140 {
141  try {
142  RegistryEntry* pEntry = dynamic_cast<RegistryEntry*>( obj );
143  auto i = std::remove( std::begin( m_store ), std::end( m_store ), pEntry );
144  if ( i != std::end( m_store ) ) {
145  pEntry->release();
146  m_store.erase( i, std::end( m_store ) );
147  }
148  } catch ( ... ) {
149  }
150  return m_store.size();
151 }
152 
155 {
156  if ( nam.front() == SEPARATOR ) nam.remove_prefix( 1 );
157  auto i = std::find_if( m_store.begin(), m_store.end(), [&]( const auto* entry ) {
158  // skip leading SEPARATOR
159  return entry->name().compare( 1, std::string::npos, nam.data(), nam.size() ) == 0;
160  } );
161  // if the requested object is not present, this is an error....
162  if ( i == m_store.end() ) return StatusCode::FAILURE;
163  remove( *i );
164  return StatusCode::SUCCESS;
165 }
166 
169 {
170  if ( nam.front() != SEPARATOR ) nam.insert( 0, 1, SEPARATOR );
171  // if this object is already present, this is an error....
172  auto not_present =
173  std::none_of( std::begin( m_store ), std::end( m_store ), [&]( IRegistry* i ) { return nam == i->name(); } );
174  return not_present ? new RegistryEntry( std::move( nam ), this ) : nullptr;
175 }
176 
179 {
180  RegistryEntry* pEntry = CAST_REGENTRY( RegistryEntry*, obj );
181  return i_add( pEntry );
182 }
183 
186 {
187  // TODO: if this is the sole place where items are added to m_store,
188  // and we know here that they must be RegisteryEntry, can we
189  // drop the dynamic_cast every where else???
190  // TODO: if so, can we also change m_store to be std::vector<RegistryEntry*>
191  // instead
192  // TODO: if so, can we not make it std::vector<RegistryEntry> instead?
193  // TODO: if so, should make sure that a RegistryEntry can be std::move'ed
194  try {
195  pEntry->setDataSvc( m_pDataProviderSvc );
196  m_store.push_back( pEntry );
197  pEntry->setParent( this );
198  if ( !pEntry->isSoft() && pEntry->address() ) {
199  pEntry->address()->setRegistry( pEntry );
200  }
201  } catch ( ... ) {
202  }
203  return m_store.size();
204 }
205 
208 {
209  RegistryEntry* entry = i_create( std::move( name ) );
210  if ( !entry ) return StatusCode::FAILURE;
211  ( is_soft ) ? entry->makeSoft( pObject ) : entry->makeHard( pObject );
212  i_add( entry );
213  return StatusCode::SUCCESS;
214 }
215 
218 {
219  RegistryEntry* entry = i_create( std::move( name ) );
220  if ( !entry ) return StatusCode::FAILURE;
221  ( is_soft ) ? entry->makeSoft( pAddress ) : entry->makeHard( pAddress );
222  i_add( entry );
223  return StatusCode::SUCCESS;
224 }
225 
228 {
229  for ( auto& i : m_store ) {
230  RegistryEntry* entry = CAST_REGENTRY( RegistryEntry*, i );
231  if ( entry ) {
232  entry->deleteElements();
233  entry->release();
234  }
235  }
236  m_store.erase( m_store.begin(), m_store.end() );
237  return 0;
238 }
239 
242 {
243  auto i = std::find( m_store.begin(), m_store.end(), obj );
244  return ( i != m_store.end() ) ? ( *i ) : nullptr;
245 }
246 
249 {
250  if ( path.front() == SEPARATOR ) path.remove_prefix( 1 ); // strip leading '/', if present
251  while ( !path.empty() ) {
252  // check that the chars of path prior to / are the same as regEnt->name()
253  // (i.e. match { nam:"/Ab" path:"/Ab/C"}
254  // but not { nam:"/Abc" path:"/Ab/C"})
255  auto loc1 = path.find( SEPARATOR );
256  auto cpath = path.substr( 0, loc1 );
257  if ( loc1 != boost::string_ref::npos ) {
258  path.remove_prefix( loc1 + 1 );
259  } else {
260  path.clear();
261  }
263  [&]( const auto& reg ) { return cpath == boost::string_ref{reg->name()}.substr( 1 ); } );
264  if ( i != std::end( m_store ) ) {
265  RegistryEntry* regEnt = CAST_REGENTRY( RegistryEntry*, *i );
266  return path.empty() ? regEnt : regEnt->i_find( path );
267  }
268  // If this node is "/NodeA", this part allows to find "/NodeA/NodeB" as
269  // our "/NodeB" child.
270  if ( cpath != boost::string_ref{m_path}.substr( 1 ) ) break;
271  }
272  return nullptr;
273 }
274 
277 {
278  if ( key ) {
279  if ( key == m_pObject ) return const_cast<RegistryEntry*>( this );
280  // Look in the immediate level:
281  RegistryEntry* result = CAST_REGENTRY( RegistryEntry*, i_find( key->registry() ) );
282  if ( result ) return result;
283  // Go levels down
284  for ( const auto& i : m_store ) {
285  try {
286  const RegistryEntry* entry = CAST_REGENTRY( RegistryEntry*, i );
287  result = entry->i_find( key );
288  if ( result ) return result;
289  } catch ( ... ) {
290  }
291  }
292  }
293  return nullptr;
294 }
295 
296 // Traverse registry tree
298 {
299  bool go_down = pAgent->analyse( this, level );
300  StatusCode status;
301  if ( go_down ) {
302  for ( auto& i : m_store ) {
303  try {
304  RegistryEntry* entry = CAST_REGENTRY( RegistryEntry*, i );
305  entry->traverseTree( pAgent, level + 1 ).ignore();
306  } catch ( ... ) {
307  status = StatusCode::FAILURE;
308  }
309  }
310  }
311  return status;
312 }
313 
314 // Recursive helper to assemble the full path name of the entry
316 {
317  if ( m_pParent ) m_pParent->assemblePath( buffer );
318  buffer += m_path;
319 }
constexpr static const auto FAILURE
Definition: StatusCode.h:88
long deleteElements()
Delete all contained elements.
constexpr char SEPARATOR
const std::string & name() const override
Retrieve name of the entry.
T front(T...args)
StatusCode traverseTree(IDataStoreAgent *pAgent, int level=0)
traverse data tree
virtual bool analyse(IRegistry *pObject, int level)=0
Analyse the data object.
unsigned long addRef() override
IInterface implementation: Dereference the object.
T rfind(T...args)
RegistryEntry * m_pParent
Pointer to parent.
Definition: RegistryEntry.h:60
~RegistryEntry() override
Standard Destructor.
STL namespace.
void setRegistry(IRegistry *pRegistry)
Set pointer to Registry.
Definition: DataObject.h:71
unsigned long release() override
IInterface implementation: Reference the object.
#define CAST_REGENTRY(x, y)
virtual const name_type & name() const =0
Name of the directory (or key)
T end(T...args)
virtual unsigned long release()=0
release reference to object
bool isSoft() const
Is the link soft or hard.
T remove(T...args)
void setAddress(IOpaqueAddress *pAddress) override
Set/Update Opaque address.
IRegistry * registry() const
Get pointer to Registry.
Definition: DataObject.h:73
long i_add(RegistryEntry *entry)
Internal method to add entries.
STL class.
std::string m_fullpath
String containing full path of the object (volatile)
Definition: RegistryEntry.h:56
void assemblePath(std::string &buffer) const
The following entries serve two aspects: 1) They are faster for recursive calls, because they are non...
T push_back(T...args)
void setParent(RegistryEntry *pParent)
Set new parent pointer.
virtual unsigned long addRef()
Add reference to object.
Definition: DataObject.cpp:59
void setObject(DataObject *obj)
Set/Update object address.
This class is used for returning status codes from appropriate routines.
Definition: StatusCode.h:51
T erase(T...args)
IDataProviderSvc * m_pDataProviderSvc
Pointer to hosting transient store.
Definition: RegistryEntry.h:66
void setDataSvc(IDataProviderSvc *s)
Set the transient data store.
Definition: RegistryEntry.h:91
The IRegistry represents the entry door to the environment any data object residing in a transient da...
Definition: IRegistry.h:22
T clear(T...args)
IOpaqueAddress * address() const override
Retrieve opaque storage address.
IOpaqueAddress * m_pAddress
Pointer to opaque address (load info)
Definition: RegistryEntry.h:62
void makeSoft(DataObject *pObject)
Initialize link as soft link.
RegistryEntry * i_create(std::string name)
Internal method to create entries.
T move(T...args)
bool m_isSoft
Is the link soft or hard?
Definition: RegistryEntry.h:54
constexpr static const auto SUCCESS
Definition: StatusCode.h:87
Definition of an entry in the transient data store.
Definition: RegistryEntry.h:37
virtual unsigned long release()
release reference to object
Definition: DataObject.cpp:51
T insert(T...args)
DataObject * m_pObject
Pointer to object.
Definition: RegistryEntry.h:64
T find_if(T...args)
unsigned long m_refCount
Reference counter.
Definition: RegistryEntry.h:52
T size(T...args)
StatusCode add(std::string name, DataObject *pObject, bool is_soft=false)
Add entry to data store.
Generic data agent interface.
T begin(T...args)
T none_of(T...args)
const StatusCode & ignore() const
Ignore/check StatusCode.
Definition: StatusCode.h:165
Store m_store
Store of leaves.
Definition: RegistryEntry.h:68
RegistryEntry(std::string path, RegistryEntry *parent=nullptr)
Standard Constructor.
std::string m_path
Path name.
Definition: RegistryEntry.h:58
T substr(T...args)
StatusCode remove(boost::string_ref name)
Remove an entry from the store.
virtual void setRegistry(IRegistry *r)=0
Update directory pointer.
Opaque address interface definition.
A DataObject is the base class of any identifiable object on any data store.
Definition: DataObject.h:30
IRegistry * i_find(const IRegistry *pDirectory) const
Internal method to retrieve data directory.
T compare(T...args)
virtual unsigned long addRef()=0
Add reference to object.
void makeHard(DataObject *pObject)
Initialize link as hard link.