1 // Dear emacs, this is -*- c++ -*-
3 Copyright (C) 2002-2017 CERN for the benefit of the ATLAS collaboration
5 #ifndef ATHLINKS_ELEMENTLINK_ICC
6 #define ATHLINKS_ELEMENTLINK_ICC
12 #include "xAODRootAccessInterfaces/TVirtualEvent.h"
13 #include "xAODRootAccessInterfaces/TActiveEvent.h"
15 ////////////////////////////////////////////////////////////////////////////////
17 // Implementation of the constructors
20 template< typename STORABLE >
21 ElementLink< STORABLE >::ElementLink()
23 m_container( 0 ), m_element( 0 ), m_elementCached( false ),
24 m_event( xAOD::TActiveEvent::event() ) {
28 template< typename STORABLE >
29 ElementLink< STORABLE >::ElementLink( xAOD::TVirtualEvent* event )
31 m_container( 0 ), m_element( 0 ), m_elementCached( false ),
36 template< typename STORABLE >
37 ElementLink< STORABLE >::ElementLink( const ID_type& key, index_type index,
38 xAOD::TVirtualEvent* event )
39 : ElementLinkBase( 0, index ),
40 m_container( 0 ), m_element( 0 ), m_elementCached( false ),
43 // Get the active event if the user didn't specify one explicitly:
45 m_event = xAOD::TActiveEvent::event();
48 // Translate the id into a hashed key:
49 m_persKey = m_event->getHash( key );
52 template< typename STORABLE >
53 ElementLink< STORABLE >::ElementLink( sgkey_t id, index_type index,
54 xAOD::TVirtualEvent* event )
55 : ElementLinkBase( id, index ),
56 m_container( 0 ), m_element( 0 ), m_elementCached( false ),
59 // Get the active event if the user didn't specify one explicitly:
61 m_event = xAOD::TActiveEvent::event();
65 template< typename STORABLE >
66 ElementLink< STORABLE >::ElementLink( const ID_type& key, index_type index,
68 xAOD::TVirtualEvent* event )
69 : ElementLinkBase( 0, index ),
70 m_container( 0 ), m_element( element ), m_elementCached( true ),
73 // Get the active event if the user didn't specify one explicitly:
75 m_event = xAOD::TActiveEvent::event();
78 // Translate the id into a hashed key:
79 m_persKey = m_event->getHash( key );
82 template< typename STORABLE >
83 ElementLink< STORABLE >::ElementLink( sgkey_t id, index_type index,
85 xAOD::TVirtualEvent* event )
86 : ElementLinkBase( id, index ),
87 m_container( 0 ), m_element( element ), m_elementCached( true ),
90 // Get the active event if the user didn't specify one explicitly:
92 m_event = xAOD::TActiveEvent::event();
96 template< typename STORABLE >
97 ElementLink< STORABLE >::ElementLink( BaseConstReference container,
99 xAOD::TVirtualEvent* event )
100 : ElementLinkBase( 0, index ),
101 m_container( &container ), m_element( 0 ), m_elementCached( false ),
104 // Get the active event if the user didn't specify one explicitly:
106 m_event = xAOD::TActiveEvent::event();
109 // Find the identifier of the container that we received:
113 template< typename STORABLE >
114 ElementLink< STORABLE >::ElementLink( ElementType element,
115 BaseConstReference container,
116 xAOD::TVirtualEvent* event )
118 m_container( &container ), m_element( element ), m_elementCached( true ),
121 // Get the active event if the user didn't specify one explicitly:
123 m_event = xAOD::TActiveEvent::event();
126 // Find the identifier of the container that we received:
128 // Find the index of the specified object within the container:
132 template< typename STORABLE >
133 ElementLink< STORABLE >::ElementLink( const ElementLink& parent )
134 : ElementLinkBase( parent ),
135 m_container( parent.m_container ), m_element( parent.m_element ),
136 m_elementCached( parent.m_elementCached ), m_event( parent.m_event ) {
141 ////////////////////////////////////////////////////////////////////////////////
143 template< typename STORABLE >
144 ElementLink< STORABLE >&
145 ElementLink< STORABLE >::operator=( const ElementLink< STORABLE >& rhs ) {
147 // Copy the contents of the base class:
148 ElementLinkBase::operator=( rhs );
150 // Copy the contents of this class:
151 m_container = rhs.m_container;
152 m_element = rhs.m_element;
153 m_elementCached = rhs.m_elementCached;
154 m_event = rhs.m_event;
156 // Return this object:
160 ////////////////////////////////////////////////////////////////////////////////
162 // Implementation of the setter functions
165 /// This function can be used to set up an existing element link to point at
166 /// an indexed element of a specific object in memory. The link needs to ask
167 /// the event for the ID of the received object, so the function can take a
170 /// @param container The container that the link should point to
171 /// @param index The index of the element inside the container that the link
172 /// needs to point to
173 /// @returns A dummy value. Failure is signalled by an exception.
175 template< typename STORABLE >
176 bool ElementLink< STORABLE >::toIndexedElement( BaseConstReference container,
179 // Remember the specified parameters:
180 m_container = &container;
183 // Find the identifier of this object:
186 // The object is no longer cached:
188 m_elementCached = false;
193 /// This is probably the slowest function of the class. It first needs to look
194 /// up the ID of the received container inside the event, and then needs to
195 /// find the index of the object within this container.
197 /// Avoid using if at all possible!
199 /// @param container The container that the link should point to
200 /// @param element The element within the container that the link should
202 /// @returns A dymmy value. Failure is signalled by an exception.
204 template< typename STORABLE >
205 bool ElementLink< STORABLE >::toContainedElement( BaseConstReference container,
206 ElementType element ) {
208 // Remember the specified parameters:
209 m_container = &container;
212 // Find the identifier of this object:
214 // Find the index of the element inside the object:
217 // Assume that the object is correctly cached:
218 m_elementCached = true;
223 /// This function can be used to set the link to point to a new element inside
224 /// the container that it already points to. The function is slow, as it needs
225 /// to find the index of the element inside the container.
227 /// Avoid using if at all possible!
229 /// @param element The element within the container that the link should
231 /// @returns A dymmy value. Failure is signalled by an exception.
233 template< typename STORABLE >
234 bool ElementLink< STORABLE >::setElement( ElementType element ) {
236 // Remember the specified parameter:
239 // Find the index of the element inside the object:
242 // Assume that the object is correctly cached:
243 m_elementCached = true;
248 /// This function tells the object to start using a new container. It should
249 /// be used together with the setElement function most of the time. It is
250 /// however a slow function, so please avoid it if possible.
252 /// @param data The container that the link should point to
253 /// @param replace <code>true</code> if the element pointer will be replaced
254 /// @returns A dymmy value. Failure is signalled by an exception.
256 template< typename STORABLE >
257 bool ElementLink< STORABLE >::setStorableObject( BaseConstReference data,
260 // Remember the specified parameter:
263 // Find the identifier of the container:
266 // Do different things based on the second argument:
268 // Remove the cached element:
270 m_elementCached = false;
273 // Try to find the cached element inside the container:
280 /// This is probably the most preferred way of setting an existing ElementLink
281 /// to point to a new element. It doesn't need to do any costly lookups.
283 /// @param key The user-readable key of the container
284 /// @param index The index of the element inside its container
285 /// @param event An alternative event object to use
287 template< typename STORABLE >
289 ElementLink< STORABLE >::resetWithKeyAndIndex( const ID_type& key,
291 xAOD::TVirtualEvent* event ) {
293 // Remember the new event object:
296 m_event = xAOD::TActiveEvent::event();
299 // Update the persistent variables:
300 setPersKey( m_event->getHash( key ) );
301 setPersIndex( index );
306 /// This is the fastest way of setting an existing ElementLink to point to a
307 /// new element. It's just a bit more inconvenient than the function receiving
308 /// a string identifier for the container.
310 /// @param id A hashed identifier for the container
311 /// @param index The index of the element inside its container
312 /// @param event An alternative event object to use
314 template< typename STORABLE >
316 ElementLink< STORABLE >::resetWithKeyAndIndex( sgkey_t id,
318 xAOD::TVirtualEvent* event ) {
320 // Remember the new event object:
323 m_event = xAOD::TActiveEvent::event();
326 // Update the persistent variables:
328 setPersIndex( index );
333 template< typename STORABLE >
334 void ElementLink< STORABLE >::reset() {
336 // Reset the persistent variables:
344 ////////////////////////////////////////////////////////////////////////////////
346 ////////////////////////////////////////////////////////////////////////////////
348 // Implementation of the accessor functions and operators
351 template< typename STORABLE >
352 typename ElementLink< STORABLE >::BaseConstPointer
353 ElementLink< STORABLE >::getStorableObjectPointer() const {
355 // Trigger the retrieval of the container/element:
357 // Return the pointer, not caring if it's a null pointer or not:
361 template< typename STORABLE >
362 typename ElementLink< STORABLE >::BaseConstReference
363 ElementLink< STORABLE >::getStorableObjectRef() const {
365 // Trigger the retrieval of the container/element:
367 // Check if a container is now available:
368 if( ! m_container ) {
369 throw std::runtime_error( "ElementLink::getStorableObjectRef "
370 "No container is available" );
373 // Return a reference to the container:
377 template< typename STORABLE >
378 const typename ElementLink< STORABLE >::ID_type&
379 ElementLink< STORABLE >::dataID() const {
381 return m_event->getName( persKey() );
384 template< typename STORABLE >
385 bool ElementLink< STORABLE >::operator==( const ElementLink& rhs ) const {
387 return ( ( key() == rhs.key() ) && ( index() == rhs.index() ) );
390 template< typename STORABLE >
391 bool ElementLink< STORABLE >::operator!=( const ElementLink& rhs ) const {
393 return ( ! ( *this == rhs ) );
396 template< typename STORABLE >
397 typename ElementLink< STORABLE >::ElementConstReference
398 ElementLink< STORABLE >::operator*() const {
400 // Access the pointer:
401 ElementConstPointer ptr = cptr();
402 // Check if it could be retrieved:
404 throw std::runtime_error( "ElementLink::operator*() "
405 "Element not available" );
408 // Return the reference:
412 /// This is the only function in the ElementLink code that tries to retrieve
413 /// information from the event.
415 /// @returns A constant pointer to the element, or a null pointer if
418 template< typename STORABLE >
419 typename ElementLink< STORABLE >::ElementConstPointer
420 ElementLink< STORABLE >::cptr() const {
422 /// Default key, calculated just once:
423 static const uint32_t DEFAULT_KEY = ( xAOD::TVirtualEvent::DEFAULT_KEY &
424 xAOD::TVirtualEvent::KEY_MASK );
426 // If the object is already cached, return right away:
427 if( m_elementCached ) return &m_element;
428 // If we can't retrieve it, sad day for us:
429 if( ( ! m_event ) || ( ! persKey() ) ||
430 ( persKey() == DEFAULT_KEY ) ) return 0;
432 // The object should be cached from now on:
433 m_elementCached = true;
435 // If the container is cached at least, then things are relatively simple:
437 typename STORABLE::size_type index =
438 static_cast< typename STORABLE::size_type >( persIndex() );
439 m_element = ( *m_container )[ index ];
443 // Ask the Event object for the container:
444 if( ! m_event->retrieve( m_container, persKey() ) ) {
446 m_elementCached = false;
450 // Get the correct element out of the container:
451 typename STORABLE::size_type index =
452 static_cast< typename STORABLE::size_type >( persIndex() );
453 m_element = ( *m_container )[ index ];
455 // Return a pointer to the element:
459 template< typename STORABLE >
460 bool ElementLink< STORABLE >::isValid() const {
462 // If we already have the element cached, the answer is simple:
463 if( m_elementCached && m_element ) {
467 /// Default key, calculated just once:
468 static const uint32_t DEFAULT_KEY = ( xAOD::TVirtualEvent::DEFAULT_KEY &
469 xAOD::TVirtualEvent::KEY_MASK );
471 // Start with the simple checks:
472 if( ( ! m_event ) || ( ! persKey() ) ||
473 ( persKey() == DEFAULT_KEY ) ) return false;
475 // Check whether the container can be accessed:
476 if( ! m_event->retrieve( m_container, persKey(), true ) ) {
480 // Check that the container is large enough:
481 if( m_container->size() <=
482 static_cast< typename STORABLE::size_type >( persIndex() ) ) {
486 // Apparently things should work well:
491 ////////////////////////////////////////////////////////////////////////////////
493 template< typename STORABLE >
494 bool ElementLink< STORABLE >::toTransient() {
498 m_elementCached = false;
503 /// This function is used internally to find the hashed identifier for a
504 /// container that was given to the ElementLink by reference. It throws an
505 /// exception if the object can't be found in the event.
507 template< typename STORABLE >
508 void ElementLink< STORABLE >::findIdentifier() {
510 // Ask the event for the identifier of this object:
511 const sgkey_t key = m_event->getKey( m_container );
513 throw std::runtime_error( "ElementLink::findIdentifier: Couldn't find "
514 "the received object in the current event" );
517 // Remember this key:
523 /// This function is used internally to find the index of a given element inside
524 /// the container. It has to do a linear search, so it's quite slow...
526 template< typename STORABLE >
527 void ElementLink< STORABLE >::findIndex() {
529 // Check that we have a valid container:
530 if( ! m_container ) {
531 throw std::runtime_error( "ElementLink::findIndex: No valid container" );
534 // Look for the element insied the container:
535 const_iterator el_itr = m_container->begin();
536 const_iterator el_end = m_container->end();
537 for( index_type i = 0; el_itr != el_end; ++el_itr, ++i ) {
539 // Check if this is the element that we're looking for:
540 if( *el_itr != m_element ) continue;
542 // If yes, we're finished:
547 // If we didn't find the element in the container, that's a problem...
548 throw std::runtime_error( "ElementLink::findIndex: Couldn't find element "
549 "in the current container" );
554 template <typename STORABLE>
555 bool operator< (const ElementLink<STORABLE> &lhs,
556 const ElementLink<STORABLE> &rhs)
558 return std::make_pair(lhs.persKey(), lhs.index()) < std::make_pair(rhs.persKey(), rhs.index());
561 template <typename STORABLE>
562 bool operator> (const ElementLink<STORABLE> &lhs,
563 const ElementLink<STORABLE> &rhs)
568 #endif // ATHLINKS_ELEMENTLINK_ICC