File indexing completed on 2025-04-04 08:46:10
0001
0002
0003
0004
0005
0006
0007
0008 #pragma once
0009
0010 #include <cmath>
0011 #include <numeric>
0012 #include <vector>
0013
0014 #include "corecel/cont/Range.hh"
0015 #include "corecel/cont/Span.hh"
0016 #include "corecel/math/Integrator.hh"
0017 #include "celeritas/Types.hh"
0018 #include "celeritas/mat/IsotopeView.hh"
0019 #include "celeritas/neutron/data/NeutronInelasticData.hh"
0020 #include "celeritas/phys/AtomicNumber.hh"
0021
0022 #include "../CascadeOptions.hh"
0023
0024 namespace celeritas
0025 {
0026 namespace detail
0027 {
0028
0029
0030
0031
0032 class NuclearZoneBuilder
0033 {
0034 public:
0035
0036
0037 using AtomicMassNumber = AtomicNumber;
0038 using MevMass = units::MevMass;
0039 using Energy = units::MevEnergy;
0040 using ComponentVec = std::vector<ZoneComponent>;
0041 using Data = HostVal<NuclearZoneData>;
0042
0043
0044 public:
0045
0046 inline NuclearZoneBuilder(CascadeOptions const& options,
0047 NeutronInelasticScalars const& scalars,
0048 Data* data);
0049
0050
0051 inline void operator()(IsotopeView const& target);
0052
0053 private:
0054
0055
0056 struct ZoneDensity
0057 {
0058 real_type radius;
0059 real_type integral;
0060 };
0061 using VecZoneDensity = std::vector<ZoneDensity>;
0062
0063
0064
0065
0066 CascadeOptions const& options_;
0067
0068 real_type skin_depth_;
0069 MevMass neutron_mass_;
0070 MevMass proton_mass_;
0071
0072 CollectionBuilder<ZoneComponent> components_;
0073 CollectionBuilder<NuclearZones, MemSpace::host, IsotopeId> zones_;
0074
0075
0076
0077
0078 inline ComponentVec calc_zone_components(IsotopeView const& target) const;
0079
0080
0081 VecZoneDensity calc_zones_light(AtomicMassNumber a) const;
0082
0083
0084 VecZoneDensity calc_zones_small(AtomicMassNumber a) const;
0085
0086
0087 VecZoneDensity calc_zones_heavy(AtomicMassNumber a) const;
0088
0089
0090 inline real_type calc_nuclear_radius(AtomicMassNumber a) const;
0091 };
0092
0093
0094
0095
0096
0097
0098
0099 NuclearZoneBuilder::NuclearZoneBuilder(CascadeOptions const& options,
0100 NeutronInelasticScalars const& scalars,
0101 Data* data)
0102 : options_(options)
0103 , skin_depth_(0.611207 * options.radius_scale)
0104 , neutron_mass_(scalars.neutron_mass)
0105 , proton_mass_(scalars.proton_mass)
0106 , components_(&data->components)
0107 , zones_(&data->zones)
0108 {
0109 CELER_EXPECT(options_);
0110 CELER_EXPECT(data);
0111 }
0112
0113
0114
0115
0116
0117 void NuclearZoneBuilder::operator()(IsotopeView const& target)
0118 {
0119 auto comp = this->calc_zone_components(target);
0120
0121 NuclearZones nucl_zones;
0122 nucl_zones.zones = components_.insert_back(comp.begin(), comp.end());
0123 zones_.push_back(nucl_zones);
0124 }
0125
0126
0127
0128
0129
0130
0131
0132
0133
0134
0135 auto NuclearZoneBuilder::calc_zone_components(IsotopeView const& target) const
0136 -> ComponentVec
0137 {
0138 using A = AtomicMassNumber;
0139
0140
0141 A const a = target.atomic_mass_number();
0142
0143
0144 VecZoneDensity zone_dens;
0145 if (a < A{5})
0146 {
0147 zone_dens = this->calc_zones_light(a);
0148 }
0149 else if (a < A{12})
0150 {
0151 zone_dens = this->calc_zones_small(a);
0152 }
0153 else
0154 {
0155 zone_dens = this->calc_zones_heavy(a);
0156 }
0157 CELER_ASSERT(!zone_dens.empty());
0158
0159
0160 constexpr real_type four_thirds_pi = 4 * constants::pi / real_type{3};
0161
0162 ComponentVec components(zone_dens.size());
0163 real_type prev_volume{0};
0164 for (auto i : range(components.size()))
0165 {
0166 real_type volume = four_thirds_pi * ipow<3>(zone_dens[i].radius);
0167
0168
0169 components[i].radius = zone_dens[i].radius;
0170 components[i].volume = volume - prev_volume;
0171 prev_volume = volume;
0172 }
0173
0174
0175 int num_protons = target.atomic_number().get();
0176 int num_nucleons[] = {num_protons, a.get() - num_protons};
0177 real_type mass[] = {proton_mass_.value(), neutron_mass_.value()};
0178 real_type dm[] = {target.proton_loss_energy().value(),
0179 target.neutron_loss_energy().value()};
0180 static_assert(std::size(mass) == ZoneComponent::NucleonArray::size());
0181 static_assert(std::size(dm) == std::size(mass));
0182
0183 real_type const total_integral
0184 = std::accumulate(zone_dens.begin(),
0185 zone_dens.end(),
0186 real_type{0},
0187 [](real_type sum, ZoneDensity const& zone) {
0188 return sum + zone.integral;
0189 });
0190
0191 for (auto i : range(components.size()))
0192 {
0193 for (auto ptype : range(ZoneComponent::NucleonArray::size()))
0194 {
0195 components[i].density[ptype]
0196 = static_cast<real_type>(num_nucleons[ptype])
0197 * zone_dens[i].integral
0198 / (total_integral * components[i].volume);
0199 components[i].fermi_mom[ptype]
0200 = options_.fermi_scale
0201 * std::cbrt(components[i].density[ptype]);
0202 components[i].potential[ptype]
0203 = ipow<2>(components[i].fermi_mom[ptype]) / (2 * mass[ptype])
0204 + dm[ptype];
0205 }
0206 }
0207 return components;
0208 }
0209
0210
0211
0212
0213
0214 auto NuclearZoneBuilder::calc_zones_light(AtomicMassNumber a) const
0215 -> VecZoneDensity
0216 {
0217 CELER_EXPECT(a <= AtomicMassNumber{4});
0218 ZoneDensity result;
0219 result.radius = options_.radius_small
0220 * (a == AtomicMassNumber{4} ? options_.radius_alpha : 1);
0221 result.integral = 1;
0222 return {result};
0223 }
0224
0225
0226
0227
0228
0229 auto NuclearZoneBuilder::calc_zones_small(AtomicMassNumber a) const
0230 -> VecZoneDensity
0231 {
0232 real_type const nuclear_radius = this->calc_nuclear_radius(a);
0233 real_type gauss_radius = std::sqrt(
0234 ipow<2>(nuclear_radius) * (1 - 1 / static_cast<real_type>(a.get()))
0235 + real_type{6.4});
0236
0237 Integrator integrate_gauss{
0238 [](real_type r) { return ipow<2>(r) * std::exp(-ipow<2>(r)); }};
0239
0240
0241 real_type const y[] = {0.597223, 1.09726, 2.14597};
0242 VecZoneDensity result(std::size(y));
0243
0244 real_type ymin = 0;
0245 for (auto i : range(result.size()))
0246 {
0247 result[i].radius = gauss_radius * y[i];
0248 result[i].integral = ipow<3>(gauss_radius)
0249 * integrate_gauss(ymin, y[i]);
0250 ymin = y[i];
0251 }
0252
0253 CELER_ENSURE(result.size() == 3);
0254 return result;
0255 }
0256
0257
0258
0259
0260
0261
0262
0263
0264
0265
0266
0267
0268
0269
0270 auto NuclearZoneBuilder::calc_zones_heavy(AtomicMassNumber a) const
0271 -> VecZoneDensity
0272 {
0273 real_type const nuclear_radius = this->calc_nuclear_radius(a);
0274 real_type const skin_ratio = nuclear_radius / skin_depth_;
0275 real_type const skin_decay = std::exp(-skin_ratio);
0276 Integrator integrate_ws{[ws_shift = 2 * skin_ratio](real_type r) {
0277 return r * (r + ws_shift) / (1 + std::exp(r));
0278 }};
0279
0280 Span<real_type const> alpha;
0281 if (a < AtomicMassNumber{100})
0282 {
0283 static real_type const alpha_i[] = {0.7, 0.3, 0.01};
0284 alpha = make_span(alpha_i);
0285 }
0286 else
0287 {
0288 static real_type const alpha_h[] = {0.9, 0.6, 0.4, 0.2, 0.1, 0.05};
0289 alpha = make_span(alpha_h);
0290 }
0291
0292 VecZoneDensity result(alpha.size());
0293 real_type ymin = -skin_ratio;
0294 for (auto i : range(result.size()))
0295 {
0296 real_type y = std::log((1 + skin_decay) / alpha[i] - 1);
0297 result[i].radius = nuclear_radius + skin_depth_ * y;
0298
0299 result[i].integral
0300 = ipow<3>(skin_depth_)
0301 * (integrate_ws(ymin, y)
0302 + ipow<2>(skin_ratio)
0303 * std::log((1 + std::exp(-ymin)) / (1 + std::exp(-y))));
0304 ymin = y;
0305 }
0306
0307 return result;
0308 }
0309
0310
0311
0312
0313
0314
0315
0316
0317
0318
0319
0320
0321 real_type NuclearZoneBuilder::calc_nuclear_radius(AtomicMassNumber a) const
0322 {
0323 CELER_EXPECT(a > AtomicMassNumber{4});
0324
0325
0326 real_type cbrt_a = std::cbrt(static_cast<real_type>(a.get()));
0327 real_type par_a = (options_.use_two_params ? 1.16 : 1.2);
0328 real_type par_b = (options_.use_two_params ? -1.3456 : 0);
0329 return options_.radius_scale * (par_a * cbrt_a + par_b / cbrt_a);
0330 }
0331
0332
0333 }
0334 }