7#include "Acts/Definitions/Algebra.hpp"
8#include "Acts/MagneticField/BFieldMapUtils.hpp"
9#include "Acts/MagneticField/InterpolatedBFieldMap.hpp"
10#include "Acts/MagneticField/MagneticFieldContext.hpp"
11#include "Acts/Utilities/AxisFwd.hpp"
12#include "Acts/Utilities/Grid.hpp"
13#include "Acts/Utilities/Interpolation.hpp"
14#include "Acts/Utilities/Result.hpp"
16static const double DIPOLE_OFFSET = 400.;
18using InterpolatedMagneticField3 = Acts::InterpolatedBFieldMap<
19 Acts::Grid<Acts::Vector3, Acts::Axis<Acts::AxisType::Equidistant>,
20 Acts::Axis<Acts::AxisType::Equidistant>,
21 Acts::Axis<Acts::AxisType::Equidistant>>>;
23using GenericTransformPos = std::function<Acts::Vector3(
const Acts::Vector3&)>;
24using GenericTransformBField =
25 std::function<Acts::Vector3(
const Acts::Vector3&,
const Acts::Vector3&)>;
35Acts::Vector3 defaultTransformPos(
const Acts::Vector3& pos_);
43Acts::Vector3 defaultTransformBField(
const Acts::Vector3& field,
44 const Acts::Vector3& );
46void testField(
const std::shared_ptr<Acts::MagneticFieldProvider> bfield,
47 const Acts::Vector3& eval_pos,
48 const Acts::MagneticFieldContext& bctx);
55size_t localToGlobalBinXyz(std::array<size_t, 3> bins,
56 std::array<size_t, 3> sizes);
58inline InterpolatedMagneticField3 rotateFieldMapXYZ(
59 const std::function<
size_t(std::array<size_t, 3> binsXYZ,
60 std::array<size_t, 3> nBinsXYZ)>&
62 std::vector<double> xPos, std::vector<double> yPos,
63 std::vector<double> zPos, std::vector<Acts::Vector3> bField,
64 double lengthUnit,
double BFieldUnit,
bool firstOctant,
65 GenericTransformPos transformPosition,
66 GenericTransformBField transformMagneticField) {
69 std::sort(xPos.begin(), xPos.end());
70 std::sort(yPos.begin(), yPos.end());
71 std::sort(zPos.begin(), zPos.end());
74 xPos.erase(std::unique(xPos.begin(), xPos.end()), xPos.end());
75 yPos.erase(std::unique(yPos.begin(), yPos.end()), yPos.end());
76 zPos.erase(std::unique(zPos.begin(), zPos.end()), zPos.end());
82 size_t n_bins_x = xPos.size();
83 size_t n_bins_y = yPos.size();
84 size_t n_bins_z = zPos.size();
87 auto min_max_x = std::minmax_element(xPos.begin(), xPos.end());
88 auto min_max_y = std::minmax_element(yPos.begin(), yPos.end());
89 auto min_max_z = std::minmax_element(zPos.begin(), zPos.end());
92 double x_min = *min_max_x.first;
93 double y_min = *min_max_y.first;
94 double z_min = *min_max_z.first;
96 double x_max = *min_max_x.second;
97 double y_max = *min_max_y.second;
98 double z_max = *min_max_z.second;
101 double step_z = std::fabs(z_max - z_min) / (n_bins_z - 1);
102 double step_y = std::fabs(y_max - y_min) / (n_bins_y - 1);
103 double step_x = std::fabs(x_max - x_min) / (n_bins_x - 1);
110 x_min = -*min_max_x.second;
111 y_min = -*min_max_y.second;
112 z_min = -*min_max_z.second;
113 n_bins_x = 2 * n_bins_x - 1;
114 n_bins_y = 2 * n_bins_y - 1;
115 n_bins_z = 2 * n_bins_z - 1;
117 Acts::Axis<Acts::AxisType::Equidistant> x_axis(x_min * lengthUnit,
118 x_max * lengthUnit, n_bins_x);
119 Acts::Axis<Acts::AxisType::Equidistant> y_axis(y_min * lengthUnit,
120 y_max * lengthUnit, n_bins_y);
121 Acts::Axis<Acts::AxisType::Equidistant> z_axis(z_min * lengthUnit,
122 z_max * lengthUnit, n_bins_z);
125 Acts::Grid<Acts::Vector3, Acts::Axis<Acts::AxisType::Equidistant>,
126 Acts::Axis<Acts::AxisType::Equidistant>,
127 Acts::Axis<Acts::AxisType::Equidistant>>;
129 std::make_tuple(std::move(x_axis), std::move(y_axis), std::move(z_axis)));
132 for (
size_t i = 1; i <= n_bins_x; ++i) {
133 for (
size_t j = 1; j <= n_bins_y; ++j) {
134 for (
size_t k = 1; k <= n_bins_z; ++k) {
135 Grid_t::index_t indices = {{i, j, k}};
136 std::array<size_t, 3> n_indices = {
137 {xPos.size(), yPos.size(), zPos.size()}};
142 size_t m = std::abs(
int(i) - (
int(xPos.size())));
143 size_t n = std::abs(
int(j) - (
int(yPos.size())));
144 size_t l = std::abs(
int(k) - (
int(zPos.size())));
145 Grid_t::index_t indices_first_octant = {{m, n, l}};
147 grid.atLocalBins(indices) =
148 bField.at(localToGlobalBin(indices_first_octant, n_indices)) *
155 grid.atLocalBins(indices) =
156 bField.at(localToGlobalBin({{i - 1, j - 1, k - 1}}, n_indices)) *
162 grid.setExteriorBins(Acts::Vector3::Zero());
198 return Acts::InterpolatedBFieldMap<Grid_t>(
199 {transformPosition, transformMagneticField, std::move(grid)});
207inline InterpolatedMagneticField3 makeMagneticFieldMapXyzFromText(
208 std::function<
size_t(std::array<size_t, 3> binsXYZ,
209 std::array<size_t, 3> nBinsXYZ)>
211 GenericTransformPos transformPosition,
212 GenericTransformBField transformMagneticField,
213 const std::string& fieldMapFile, Acts::ActsScalar lengthUnit,
214 Acts::ActsScalar BFieldUnit,
bool firstOctant,
bool rotateAxes) {
217 std::vector<double> x_pos;
218 std::vector<double> y_pos;
219 std::vector<double> z_pos;
221 std::vector<Acts::Vector3> b_field;
223 constexpr size_t k_default_size = 1 << 15;
225 x_pos.reserve(k_default_size);
226 y_pos.reserve(k_default_size);
227 z_pos.reserve(k_default_size);
228 b_field.reserve(k_default_size);
230 std::ifstream map_file(fieldMapFile.c_str(), std::ios::in);
232 double pos_x = 0., pos_y = 0., pos_z = 0.;
233 double bx = 0., by = 0., bz = 0.;
235 bool header_found =
false;
237 while (std::getline(map_file, line)) {
238 if (line.empty() || line[0] ==
'%' || line[0] ==
'#' || line[0] ==
' ' ||
239 line.find_first_not_of(
' ') == std::string::npos || !header_found) {
240 if (line.find(
"Header") != std::string::npos) header_found =
true;
243 std::istringstream tmp(line);
244 tmp >> pos_x >> pos_y >> pos_z >> bx >> by >> bz;
246 x_pos.push_back(pos_x);
247 y_pos.push_back(pos_y);
248 z_pos.push_back(pos_z);
249 b_field.push_back(Acts::Vector3(bx, by, bz));
254 std::cout <<
"MAP LOADING ERROR:: line containing the word 'Header' not "
259 x_pos.shrink_to_fit();
260 y_pos.shrink_to_fit();
261 z_pos.shrink_to_fit();
262 b_field.shrink_to_fit();
265 return rotateFieldMapXYZ(localToGlobalBin, x_pos, y_pos, z_pos, b_field,
266 lengthUnit, BFieldUnit, firstOctant,
267 transformPosition, transformMagneticField);
269 return Acts::fieldMapXYZ(localToGlobalBin, x_pos, y_pos, z_pos, b_field,
270 lengthUnit, BFieldUnit, firstOctant);
273inline InterpolatedMagneticField3 loadDefaultBField(
274 const std::string& fieldMapFile, GenericTransformPos transformPosition,
275 GenericTransformBField transformMagneticField) {
280 return makeMagneticFieldMapXyzFromText(
281 std::move(localToGlobalBinXyz), transformPosition, transformMagneticField,
283 1. * Acts::UnitConstants::mm,
284 1000. * Acts::UnitConstants::T,