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"
15#include "Framework/Exception/Exception.h"
17static const double DIPOLE_OFFSET = 400.;
19using InterpolatedMagneticField3 = Acts::InterpolatedBFieldMap<
20 Acts::Grid<Acts::Vector3, Acts::Axis<Acts::AxisType::Equidistant>,
21 Acts::Axis<Acts::AxisType::Equidistant>,
22 Acts::Axis<Acts::AxisType::Equidistant>>>;
24using GenericTransformPos = std::function<Acts::Vector3(
const Acts::Vector3&)>;
25using GenericTransformBField =
26 std::function<Acts::Vector3(
const Acts::Vector3&,
const Acts::Vector3&)>;
36Acts::Vector3 defaultTransformPos(
const Acts::Vector3& pos_);
44Acts::Vector3 defaultTransformBField(
const Acts::Vector3& field,
45 const Acts::Vector3& );
47void testField(
const std::shared_ptr<Acts::MagneticFieldProvider> bfield,
48 const Acts::Vector3& eval_pos,
49 const Acts::MagneticFieldContext& bctx);
56size_t localToGlobalBinXyz(std::array<size_t, 3> bins,
57 std::array<size_t, 3> sizes);
59inline InterpolatedMagneticField3 rotateFieldMapXYZ(
60 const std::function<
size_t(std::array<size_t, 3> binsXYZ,
61 std::array<size_t, 3> nBinsXYZ)>&
63 std::vector<double> xPos, std::vector<double> yPos,
64 std::vector<double> zPos, std::vector<Acts::Vector3> bField,
65 double lengthUnit,
double BFieldUnit,
bool firstOctant,
66 GenericTransformPos transformPosition,
67 GenericTransformBField transformMagneticField) {
70 std::sort(xPos.begin(), xPos.end());
71 std::sort(yPos.begin(), yPos.end());
72 std::sort(zPos.begin(), zPos.end());
75 xPos.erase(std::unique(xPos.begin(), xPos.end()), xPos.end());
76 yPos.erase(std::unique(yPos.begin(), yPos.end()), yPos.end());
77 zPos.erase(std::unique(zPos.begin(), zPos.end()), zPos.end());
83 size_t n_bins_x = xPos.size();
84 size_t n_bins_y = yPos.size();
85 size_t n_bins_z = zPos.size();
88 auto min_max_x = std::minmax_element(xPos.begin(), xPos.end());
89 auto min_max_y = std::minmax_element(yPos.begin(), yPos.end());
90 auto min_max_z = std::minmax_element(zPos.begin(), zPos.end());
93 double x_min = *min_max_x.first;
94 double y_min = *min_max_y.first;
95 double z_min = *min_max_z.first;
97 double x_max = *min_max_x.second;
98 double y_max = *min_max_y.second;
99 double z_max = *min_max_z.second;
102 double step_z = std::fabs(z_max - z_min) / (n_bins_z - 1);
103 double step_y = std::fabs(y_max - y_min) / (n_bins_y - 1);
104 double step_x = std::fabs(x_max - x_min) / (n_bins_x - 1);
111 x_min = -*min_max_x.second;
112 y_min = -*min_max_y.second;
113 z_min = -*min_max_z.second;
114 n_bins_x = 2 * n_bins_x - 1;
115 n_bins_y = 2 * n_bins_y - 1;
116 n_bins_z = 2 * n_bins_z - 1;
118 Acts::Axis<Acts::AxisType::Equidistant> x_axis(x_min * lengthUnit,
119 x_max * lengthUnit, n_bins_x);
120 Acts::Axis<Acts::AxisType::Equidistant> y_axis(y_min * lengthUnit,
121 y_max * lengthUnit, n_bins_y);
122 Acts::Axis<Acts::AxisType::Equidistant> z_axis(z_min * lengthUnit,
123 z_max * lengthUnit, n_bins_z);
126 Acts::Grid<Acts::Vector3, Acts::Axis<Acts::AxisType::Equidistant>,
127 Acts::Axis<Acts::AxisType::Equidistant>,
128 Acts::Axis<Acts::AxisType::Equidistant>>;
130 std::make_tuple(std::move(x_axis), std::move(y_axis), std::move(z_axis)));
133 for (
size_t i = 1; i <= n_bins_x; ++i) {
134 for (
size_t j = 1; j <= n_bins_y; ++j) {
135 for (
size_t k = 1; k <= n_bins_z; ++k) {
136 Grid_t::index_t indices = {{i, j, k}};
137 std::array<size_t, 3> n_indices = {
138 {xPos.size(), yPos.size(), zPos.size()}};
143 size_t m = std::abs(
int(i) - (
int(xPos.size())));
144 size_t n = std::abs(
int(j) - (
int(yPos.size())));
145 size_t l = std::abs(
int(k) - (
int(zPos.size())));
146 Grid_t::index_t indices_first_octant = {{m, n, l}};
148 grid.atLocalBins(indices) =
149 bField.at(localToGlobalBin(indices_first_octant, n_indices)) *
156 grid.atLocalBins(indices) =
157 bField.at(localToGlobalBin({{i - 1, j - 1, k - 1}}, n_indices)) *
163 grid.setExteriorBins(Acts::Vector3::Zero());
199 return Acts::InterpolatedBFieldMap<Grid_t>(
200 {transformPosition, transformMagneticField, std::move(grid)});
208inline InterpolatedMagneticField3 makeMagneticFieldMapXyzFromText(
209 std::function<
size_t(std::array<size_t, 3> binsXYZ,
210 std::array<size_t, 3> nBinsXYZ)>
212 GenericTransformPos transformPosition,
213 GenericTransformBField transformMagneticField,
214 const std::string& fieldMapFile, Acts::ActsScalar lengthUnit,
215 Acts::ActsScalar BFieldUnit,
bool firstOctant,
bool rotateAxes) {
218 std::vector<double> x_pos;
219 std::vector<double> y_pos;
220 std::vector<double> z_pos;
222 std::vector<Acts::Vector3> b_field;
224 constexpr size_t k_default_size = 1 << 15;
226 x_pos.reserve(k_default_size);
227 y_pos.reserve(k_default_size);
228 z_pos.reserve(k_default_size);
229 b_field.reserve(k_default_size);
231 std::ifstream map_file(fieldMapFile.c_str(), std::ios::in);
232 if (!map_file.is_open()) {
233 EXCEPTION_RAISE(
"BadConf",
"BFieldXYZUtils: cannot open field map file '" +
237 double pos_x = 0., pos_y = 0., pos_z = 0.;
238 double bx = 0., by = 0., bz = 0.;
240 bool header_found =
false;
242 while (std::getline(map_file, line)) {
243 if (line.empty() || line[0] ==
'%' || line[0] ==
'#' || line[0] ==
' ' ||
244 line.find_first_not_of(
' ') == std::string::npos || !header_found) {
245 if (line.find(
"Header") != std::string::npos) header_found =
true;
248 std::istringstream tmp(line);
249 tmp >> pos_x >> pos_y >> pos_z >> bx >> by >> bz;
251 x_pos.push_back(pos_x);
252 y_pos.push_back(pos_y);
253 z_pos.push_back(pos_z);
254 b_field.push_back(Acts::Vector3(bx, by, bz));
259 EXCEPTION_RAISE(
"BadConf",
260 "BFieldXYZUtils: no 'Header' line found in field map "
264 if (b_field.empty()) {
265 EXCEPTION_RAISE(
"BadConf",
"BFieldXYZUtils: no field data read from '" +
269 x_pos.shrink_to_fit();
270 y_pos.shrink_to_fit();
271 z_pos.shrink_to_fit();
272 b_field.shrink_to_fit();
275 return rotateFieldMapXYZ(localToGlobalBin, x_pos, y_pos, z_pos, b_field,
276 lengthUnit, BFieldUnit, firstOctant,
277 transformPosition, transformMagneticField);
279 return Acts::fieldMapXYZ(localToGlobalBin, x_pos, y_pos, z_pos, b_field,
280 lengthUnit, BFieldUnit, firstOctant);
283inline InterpolatedMagneticField3 loadDefaultBField(
284 const std::string& fieldMapFile, GenericTransformPos transformPosition,
285 GenericTransformBField transformMagneticField) {
290 return makeMagneticFieldMapXyzFromText(
291 std::move(localToGlobalBinXyz), transformPosition, transformMagneticField,
293 1. * Acts::UnitConstants::mm,
294 1000. * Acts::UnitConstants::T,