forked from cms-patatrack/pixeltrack-standalone
-
Notifications
You must be signed in to change notification settings - Fork 0
/
ProductBase.h
93 lines (76 loc) · 3.18 KB
/
ProductBase.h
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
#ifndef CUDADataFormats_Common_ProductBase_h
#define CUDADataFormats_Common_ProductBase_h
#include <atomic>
#include <memory>
#include "CUDACore/SharedStreamPtr.h"
#include "CUDACore/SharedEventPtr.h"
namespace cms {
namespace cuda {
namespace impl {
class ScopedContextBase;
}
/**
* Base class for all instantiations of CUDA<T> to hold the
* non-T-dependent members.
*/
class ProductBase {
public:
ProductBase() = default; // Needed only for ROOT dictionary generation
~ProductBase();
ProductBase(const ProductBase&) = delete;
ProductBase& operator=(const ProductBase&) = delete;
ProductBase(ProductBase&& other)
: stream_{std::move(other.stream_)},
event_{std::move(other.event_)},
mayReuseStream_{other.mayReuseStream_.load()},
device_{other.device_} {}
ProductBase& operator=(ProductBase&& other) {
stream_ = std::move(other.stream_);
event_ = std::move(other.event_);
mayReuseStream_ = other.mayReuseStream_.load();
device_ = other.device_;
return *this;
}
bool isValid() const { return stream_.get() != nullptr; }
bool isAvailable() const;
int device() const { return device_; }
// cudaStream_t is a pointer to a thread-safe object, for which a
// mutable access is needed even if the cms::cuda::ScopedContext itself
// would be const. Therefore it is ok to return a non-const
// pointer from a const method here.
cudaStream_t stream() const { return stream_.get(); }
// cudaEvent_t is a pointer to a thread-safe object, for which a
// mutable access is needed even if the cms::cuda::ScopedContext itself
// would be const. Therefore it is ok to return a non-const
// pointer from a const method here.
cudaEvent_t event() const { return event_.get(); }
protected:
explicit ProductBase(int device, SharedStreamPtr stream, SharedEventPtr event)
: stream_{std::move(stream)}, event_{std::move(event)}, device_{device} {}
private:
friend class impl::ScopedContextBase;
friend class ScopedContextProduce;
// The following function is intended to be used only from ScopedContext
const SharedStreamPtr& streamPtr() const { return stream_; }
bool mayReuseStream() const {
bool expected = true;
bool changed = mayReuseStream_.compare_exchange_strong(expected, false);
// If the current thread is the one flipping the flag, it may
// reuse the stream.
return changed;
}
// The cudaStream_t is really shared among edm::Event products, so
// using shared_ptr also here
SharedStreamPtr stream_; //!
// shared_ptr because of caching in cms::cuda::EventCache
SharedEventPtr event_; //!
// This flag tells whether the CUDA stream may be reused by a
// consumer or not. The goal is to have a "chain" of modules to
// queue their work to the same stream.
mutable std::atomic<bool> mayReuseStream_ = true; //!
// The CUDA device associated with this product
int device_ = -1; //!
};
} // namespace cuda
} // namespace cms
#endif