1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
|
#ifndef DataFormats_Common_RefCoreGet_h
#define DataFormats_Common_RefCoreGet_h
/*----------------------------------------------------------------------
RefCoreGet: Free function to get the pointer to a referenced product.
----------------------------------------------------------------------*/
#include "DataFormats/Common/interface/RefCore.h"
#include "DataFormats/Common/interface/WrapperBase.h"
#include "DataFormats/Common/interface/Wrapper.h"
#include <cassert>
#include <typeinfo>
namespace edm {
namespace refcore {
template <typename T>
inline T const* getProduct_(RefCore const& ref, const EDProductGetter* prodGetter) {
assert(!ref.isTransient());
WrapperBase const* product = ref.getProductPtr(typeid(T), prodGetter);
Wrapper<T> const* wrapper = static_cast<Wrapper<T> const*>(product);
if (wrapper == nullptr) {
ref.wrongTypeException(typeid(T), typeid(*product));
}
ref.setProductPtr(wrapper->product());
return wrapper->product();
}
} // namespace refcore
// Get the product using a RefCore from a RefProd.
// In this case the pointer cache in the RefCore
// is designed to hold a pointer to the container product.
template <typename T>
inline T const* getProduct(RefCore const& ref) {
T const* p = static_cast<T const*>(ref.productPtr());
if (p != nullptr)
return p;
if (ref.isTransient()) {
ref.nullPointerForTransientException(typeid(T));
}
auto productGetter = ref.productGetter();
if (nullptr == productGetter) {
p = static_cast<T const*>(ref.productPtr());
if (p != nullptr) {
//another thread updated the value since we checked
return p;
}
}
return refcore::getProduct_<T>(ref, productGetter);
}
namespace refcore {
template <typename T>
inline T const* getProductWithCoreFromRef_(RefCore const& ref, EDProductGetter const* prodGetter) {
WrapperBase const* product = ref.getProductPtr(typeid(T), prodGetter);
Wrapper<T> const* wrapper = static_cast<Wrapper<T> const*>(product);
if (wrapper == nullptr) {
ref.wrongTypeException(typeid(T), typeid(*product));
}
return wrapper->product();
}
} // namespace refcore
// Get the product using a RefCore from a Ref.
// In this case the pointer cache in the RefCore
// is designed to hold a pointer to an element in the container product.
template <typename T>
inline T const* getProductWithCoreFromRef(RefCore const& ref, EDProductGetter const* prodGetter) {
if (ref.isTransient()) {
ref.nullPointerForTransientException(typeid(T));
}
return refcore::getProductWithCoreFromRef_<T>(ref, prodGetter);
}
namespace refcore {
template <typename T>
inline T const* tryToGetProductWithCoreFromRef_(RefCore const& ref, EDProductGetter const* prodGetter) {
WrapperBase const* product = ref.tryToGetProductPtr(typeid(T), prodGetter);
if (product == nullptr) {
return nullptr;
}
Wrapper<T> const* wrapper = static_cast<Wrapper<T> const*>(product);
return wrapper->product();
}
} // namespace refcore
// get the product using a RefCore from a Ref
// In this case the pointer cache in the RefCore
// is for a pointer to an element in the container product.
// In this case we try only, which means we do not throw
// if we fail. This gives the calling function the
// chance to look in thinned containers.
template <typename T>
inline T const* tryToGetProductWithCoreFromRef(RefCore const& ref, EDProductGetter const* prodGetter) {
if (ref.isTransient()) {
ref.nullPointerForTransientException(typeid(T));
}
return refcore::tryToGetProductWithCoreFromRef_<T>(ref, prodGetter);
}
namespace refcore {
template <typename T>
inline std::tuple<T const*, unsigned int> getThinnedProduct_(RefCore const& ref,
unsigned int key,
EDProductGetter const* prodGetter) {
auto [product, thinnedKey] = ref.getThinnedProductPtr(typeid(T), key, prodGetter);
Wrapper<T> const* wrapper = static_cast<Wrapper<T> const*>(product);
return std::tuple(wrapper->product(), thinnedKey);
}
} // namespace refcore
template <typename T>
inline std::tuple<T const*, unsigned int> getThinnedProduct(RefCore const& ref,
unsigned int key,
EDProductGetter const* prodGetter) {
// The pointer to a thinned collection will never be cached
// T const* p = static_cast<T const*>(ref.productPtr());
// if (p != 0) return p;
if (ref.isTransient()) {
ref.nullPointerForTransientException(typeid(T));
}
return refcore::getThinnedProduct_<T>(ref, key, prodGetter);
}
} // namespace edm
#endif
|