File indexing completed on 2025-02-22 09:55:27
0001
0002
0003
0004
0005
0006
0007
0008
0009 #pragma once
0010
0011 #include "Acts/Material/MaterialSlab.hpp"
0012 #include "Acts/Utilities/TypeTraits.hpp"
0013 #include "ActsFatras/EventData/Particle.hpp"
0014
0015 #include <bitset>
0016 #include <tuple>
0017 #include <type_traits>
0018 #include <utility>
0019
0020 namespace ActsFatras {
0021 namespace detail {
0022
0023
0024
0025
0026 template <class T, class Tuple>
0027 struct TupleIndexOf;
0028 template <class T, class... Types>
0029 struct TupleIndexOf<T, std::tuple<T, Types...>> {
0030 static constexpr std::size_t value = 0u;
0031 };
0032 template <class T, class U, class... Types>
0033 struct TupleIndexOf<T, std::tuple<U, Types...>> {
0034 static constexpr std::size_t value =
0035 1u + TupleIndexOf<T, std::tuple<Types...>>::value;
0036 };
0037
0038
0039
0040
0041
0042
0043
0044
0045
0046
0047
0048
0049
0050
0051
0052
0053
0054
0055 template <template <typename> typename predicate_t, typename tuple_t,
0056 std::size_t kCounter, std::size_t... kIndices>
0057 struct TupleFilterImpl {
0058 static constexpr auto kIndex = kCounter - 1u;
0059 static constexpr bool kElementSelection =
0060 predicate_t<std::tuple_element_t<kIndex, tuple_t>>::value;
0061
0062 using SelectElement = typename TupleFilterImpl<predicate_t, tuple_t, kIndex,
0063 kIndex, kIndices...>::Type;
0064
0065 using SkipElement =
0066 typename TupleFilterImpl<predicate_t, tuple_t, kIndex, kIndices...>::Type;
0067
0068 using Type =
0069 std::conditional_t<kElementSelection, SelectElement, SkipElement>;
0070 };
0071 template <template <typename> typename predicate_t, typename tuple_t,
0072 std::size_t... kIndices>
0073 struct TupleFilterImpl<predicate_t, tuple_t, 0u, kIndices...> {
0074 using Type = std::index_sequence<kIndices...>;
0075 };
0076 template <template <typename> typename predicate_t, typename tuple_t>
0077 using TupleFilter = typename TupleFilterImpl<predicate_t, tuple_t,
0078 std::tuple_size_v<tuple_t>>::Type;
0079
0080
0081
0082
0083 template <typename process_t>
0084 class IsPointLikeProcess {
0085 struct MockUniformRandomBitGenerator {
0086 using result_type = unsigned int;
0087
0088 static constexpr result_type min() { return 0u; }
0089 static constexpr result_type max() { return 1u << 15u; }
0090 constexpr result_type operator()() { return 0u; }
0091 };
0092
0093 METHOD_TRAIT(generatePathLimits_method_t, generatePathLimits);
0094
0095 using scalar_pair_t = std::pair<Particle::Scalar, Particle::Scalar>;
0096
0097 public:
0098 static constexpr bool value = Acts::Concepts::has_method<
0099 const process_t, scalar_pair_t, generatePathLimits_method_t,
0100 MockUniformRandomBitGenerator&, const Particle&>;
0101 };
0102
0103 template <typename process_t>
0104 struct IsContinuousProcess {
0105 static constexpr bool value = !IsPointLikeProcess<process_t>::value;
0106 };
0107
0108 template <typename processes_t>
0109 using ContinuousIndices = TupleFilter<IsContinuousProcess, processes_t>;
0110 template <typename processes_t>
0111 using PointLikeIndices = TupleFilter<IsPointLikeProcess, processes_t>;
0112
0113 }
0114
0115
0116
0117
0118
0119
0120
0121
0122
0123
0124
0125
0126
0127
0128
0129
0130
0131
0132
0133
0134
0135
0136
0137
0138
0139
0140
0141
0142
0143
0144
0145
0146
0147
0148
0149
0150
0151
0152
0153
0154
0155
0156
0157
0158
0159
0160
0161
0162
0163
0164
0165
0166
0167
0168
0169
0170
0171
0172
0173
0174 template <typename... processes_t>
0175 class InteractionList {
0176 using Mask = std::bitset<sizeof...(processes_t)>;
0177 using Processes = std::tuple<processes_t...>;
0178 using ContinuousIndices = detail::ContinuousIndices<Processes>;
0179 using PointLikeIndices = detail::PointLikeIndices<Processes>;
0180
0181 public:
0182
0183 struct Selection {
0184 Particle::Scalar x0Limit =
0185 std::numeric_limits<Particle::Scalar>::infinity();
0186 Particle::Scalar l0Limit =
0187 std::numeric_limits<Particle::Scalar>::infinity();
0188 std::size_t x0Process = SIZE_MAX;
0189 std::size_t l0Process = SIZE_MAX;
0190 };
0191
0192
0193 void disable(std::size_t process) { m_mask.set(process); }
0194
0195
0196
0197
0198 template <typename process_t>
0199 void disable() {
0200 m_mask.set(detail::TupleIndexOf<process_t, Processes>::value);
0201 }
0202
0203
0204 template <std::size_t kProcess>
0205 std::tuple_element_t<kProcess, Processes>& get() {
0206 return std::get<kProcess>(m_processes);
0207 }
0208
0209
0210
0211
0212 template <typename process_t>
0213 process_t& get() {
0214 return std::get<process_t>(m_processes);
0215 }
0216
0217
0218
0219
0220
0221
0222
0223
0224
0225 template <typename generator_t>
0226 bool runContinuous(generator_t& rng, const Acts::MaterialSlab& slab,
0227 Particle& particle,
0228 std::vector<Particle>& generated) const {
0229 return runContinuousImpl(rng, slab, particle, generated,
0230 ContinuousIndices());
0231 }
0232
0233
0234
0235
0236
0237
0238
0239
0240 template <typename generator_t>
0241 Selection armPointLike(generator_t& rng, const Particle& particle) const {
0242 Selection selection;
0243 armPointLikeImpl(rng, particle, selection, PointLikeIndices());
0244 return selection;
0245 }
0246
0247
0248
0249
0250
0251
0252
0253
0254
0255
0256
0257
0258
0259 template <typename generator_t>
0260 bool runPointLike(generator_t& rng, std::size_t processIndex,
0261 Particle& particle,
0262 std::vector<Particle>& generated) const {
0263 return runPointLikeImpl(rng, processIndex, particle, generated,
0264 PointLikeIndices());
0265 }
0266
0267 private:
0268
0269 Mask m_mask;
0270 Processes m_processes;
0271
0272
0273
0274
0275 template <typename generator_t, std::size_t kI0, std::size_t... kIs>
0276 bool runContinuousImpl(generator_t& rng, const Acts::MaterialSlab& slab,
0277 Particle& particle, std::vector<Particle>& generated,
0278 std::index_sequence<kI0, kIs...> ) const {
0279 const auto& process = std::get<kI0>(m_processes);
0280
0281 if (!m_mask[kI0] && process(rng, slab, particle, generated)) {
0282
0283 return true;
0284 }
0285 return runContinuousImpl(rng, slab, particle, generated,
0286 std::index_sequence<kIs...>());
0287 }
0288 template <typename generator_t>
0289 bool runContinuousImpl(generator_t& ,
0290 const Acts::MaterialSlab& ,
0291 Particle& ,
0292 std::vector<Particle>& ,
0293 std::index_sequence<> ) const {
0294 return false;
0295 }
0296
0297
0298
0299
0300 template <typename generator_t, std::size_t kI0, std::size_t... kIs>
0301 void armPointLikeImpl(generator_t& rng, const Particle& particle,
0302 Selection& selection,
0303 std::index_sequence<kI0, kIs...> ) const {
0304
0305 if (!m_mask[kI0]) {
0306 auto [x0Limit, l0Limit] =
0307 std::get<kI0>(m_processes).generatePathLimits(rng, particle);
0308 if (x0Limit < selection.x0Limit) {
0309 selection.x0Limit = x0Limit;
0310 selection.x0Process = kI0;
0311 }
0312 if (l0Limit < selection.l0Limit) {
0313 selection.l0Limit = l0Limit;
0314 selection.l0Process = kI0;
0315 }
0316 }
0317
0318 armPointLikeImpl(rng, particle, selection, std::index_sequence<kIs...>());
0319 }
0320 template <typename generator_t>
0321 void armPointLikeImpl(generator_t& , const Particle& ,
0322 Selection& ,
0323 std::index_sequence<> ) const {}
0324
0325
0326
0327
0328
0329 template <typename generator_t, std::size_t kI0, std::size_t... kIs>
0330 bool runPointLikeImpl(generator_t& rng, std::size_t processIndex,
0331 Particle& particle, std::vector<Particle>& generated,
0332 std::index_sequence<kI0, kIs...> ) const {
0333 if (kI0 == processIndex) {
0334 if (m_mask[kI0]) {
0335
0336
0337 return false;
0338 }
0339 return std::get<kI0>(m_processes).run(rng, particle, generated);
0340 }
0341
0342 return runPointLikeImpl(rng, processIndex, particle, generated,
0343 std::index_sequence<kIs...>());
0344 }
0345 template <typename generator_t>
0346 bool runPointLikeImpl(generator_t& , std::size_t ,
0347 Particle& ,
0348 std::vector<Particle>& ,
0349 std::index_sequence<> ) const {
0350
0351
0352
0353 return false;
0354 }
0355 };
0356
0357 }