OPAL (Object Oriented Parallel Accelerator Library)  2021.1.99
OPAL
FM1DMagnetoStatic.cpp
Go to the documentation of this file.
2 #include "Fields/Fieldmap.hpp"
3 #include "Physics/Physics.h"
5 #include "Utilities/Util.h"
6 
7 #include "gsl/gsl_fft_real.h"
8 
9 #include <fstream>
10 #include <ios>
11 
13  : Fieldmap(aFilename) {
14 
16 
17  std::ifstream fieldFile(Filename_m.c_str());
18  if(fieldFile.good()) {
19 
20  bool parsingPassed = readFileHeader(fieldFile);
21  parsingPassed = checkFileData(fieldFile, parsingPassed);
22  fieldFile.close();
23 
24  if(!parsingPassed) {
26  zEnd_m = zBegin_m - 1.0e-3;
27  } else
29 
31  / (numberOfGridPoints_m - 1);
32  } else {
34  zBegin_m = 0.0;
35  zEnd_m = -1.0e-3;
36  }
37 }
38 
40  freeMap();
41 }
42 
44 
45  if(fourierCoefs_m.empty()) {
46 
47  std::ifstream fieldFile(Filename_m.c_str());
48  stripFileHeader(fieldFile);
49 
50  double *fieldData = new double[2 * numberOfGridPoints_m - 1];
51  double maxBz = readFileData(fieldFile, fieldData);
52  fieldFile.close();
53  computeFourierCoefficients(maxBz, fieldData);
54  delete [] fieldData;
55 
56  INFOMSG(level3 << typeset_msg("read in fieldmap '" + Filename_m + "'", "info")
57  << endl);
58  }
59 }
60 
62 
63  if(!fourierCoefs_m.empty()) {
64  fourierCoefs_m.clear();
65 
66  INFOMSG(level3 << typeset_msg("freed fieldmap '" + Filename_m + "'", "info")
67  << endl);
68  }
69 }
70 
72  Vector_t &B) const {
73 
74  std::vector<double> fieldComponents;
75  computeFieldOnAxis(R(2) - zBegin_m, fieldComponents);
76  computeFieldOffAxis(R, E, B, fieldComponents);
77 
78  return false;
79 }
80 
82  Vector_t &/*E*/,
83  Vector_t &B,
84  const DiffDirection &/*dir*/) const {
85 
86  double kz = Physics::two_pi * (R(2) - zBegin_m) / length_m + Physics::pi;
87  double bZPrime = 0.0;
88 
89  int coefIndex = 1;
90  for(int n = 1; n < accuracy_m; n++) {
91 
92  bZPrime += n * Physics::two_pi / length_m
93  * (-fourierCoefs_m.at(coefIndex) * sin(kz * n)
94  - fourierCoefs_m.at(coefIndex + 1) * cos(kz * n));
95  coefIndex += 2;
96 
97  }
98 
99  B(2) += bZPrime;
100 
101  return false;
102 }
103 
104 void FM1DMagnetoStatic::getFieldDimensions(double &zBegin, double &zEnd) const {
105  zBegin = zBegin_m;
106  zEnd = zEnd_m;
107 }
108 void FM1DMagnetoStatic::getFieldDimensions(double &/*xIni*/, double &/*xFinal*/,
109  double &/*yIni*/, double &/*yFinal*/,
110  double &/*zIni*/, double &/*zFinal*/) const {
111 }
112 
114 { }
115 
117  (*msg) << Filename_m
118  << " (1D magnetostatic); zini= "
119  << zBegin_m << " m; zfinal= "
120  << zEnd_m << " m;" << endl;
121 }
122 
124  return 0.0;
125 }
126 
127 void FM1DMagnetoStatic::setFrequency(double /*freq*/)
128 { }
129 
130 bool FM1DMagnetoStatic::checkFileData(std::ifstream &fieldFile,
131  bool parsingPassed) {
132 
133  double tempDouble;
134  for(int dataIndex = 0; dataIndex < numberOfGridPoints_m; ++ dataIndex)
135  parsingPassed = parsingPassed
136  && interpretLine<double>(fieldFile, tempDouble);
137 
138  return parsingPassed && interpreteEOF(fieldFile);
139 
140 }
141 
143  Vector_t &/*E*/,
144  Vector_t &B,
145  std::vector<double> fieldComponents) const {
146 
147  double radiusSq = pow(R(0), 2.0) + pow(R(1), 2.0);
148  double transverseBFactor = -fieldComponents.at(1) / 2.0
149  + radiusSq * fieldComponents.at(3) / 16.0;
150 
151  B(0) += R(0) * transverseBFactor;
152  B(1) += R(1) * transverseBFactor;
153  B(2) += fieldComponents.at(0) - fieldComponents.at(2) * radiusSq / 4.0;
154 
155 }
156 
158  std::vector<double> &fieldComponents) const {
159 
160  double kz = Physics::two_pi * z / length_m + Physics::pi;
161  fieldComponents.push_back(fourierCoefs_m.at(0));
162  fieldComponents.push_back(0.0);
163  fieldComponents.push_back(0.0);
164  fieldComponents.push_back(0.0);
165 
166  int coefIndex = 1;
167  for(int n = 1; n < accuracy_m; n++) {
168 
169  double kn = n * Physics::two_pi / length_m;
170  double coskzn = cos(kz * n);
171  double sinkzn = sin(kz * n);
172 
173  fieldComponents.at(0) += fourierCoefs_m.at(coefIndex) * coskzn
174  - fourierCoefs_m.at(coefIndex + 1) * sinkzn;
175 
176  fieldComponents.at(1) += kn * (-fourierCoefs_m.at(coefIndex) * sinkzn
177  - fourierCoefs_m.at(coefIndex + 1) * coskzn);
178 
179  double derivCoeff = pow(kn, 2.0);
180  fieldComponents.at(2) += derivCoeff * (-fourierCoefs_m.at(coefIndex) * coskzn
181  + fourierCoefs_m.at(coefIndex + 1) * sinkzn);
182  derivCoeff *= kn;
183  fieldComponents.at(3) += derivCoeff * (fourierCoefs_m.at(coefIndex) * sinkzn
184  + fourierCoefs_m.at(coefIndex + 1) * coskzn);
185 
186  coefIndex += 2;
187  }
188 }
189 
191  double fieldData[]) {
192  const unsigned int totalSize = 2 * numberOfGridPoints_m - 1;
193  gsl_fft_real_wavetable *waveTable = gsl_fft_real_wavetable_alloc(totalSize);
194  gsl_fft_real_workspace *workSpace = gsl_fft_real_workspace_alloc(totalSize);
195 
196  gsl_fft_real_transform(fieldData, 1, totalSize, waveTable, workSpace);
197 
198  /*
199  * Normalize the Fourier coefficients such that the max field
200  * value is 1 V/m.
201  */
202 
203  fourierCoefs_m.push_back(fieldData[0] / (totalSize * maxBz));
204  for(int coefIndex = 1; coefIndex < 2 * accuracy_m - 1; coefIndex++)
205  fourierCoefs_m.push_back(2.0 * fieldData[coefIndex] / (totalSize * maxBz));
206 
207  gsl_fft_real_workspace_free(workSpace);
208  gsl_fft_real_wavetable_free(waveTable);
209 
210 }
211 
213 
214  // Convert to m.
215  rBegin_m /= 100.0;
216  rEnd_m /= 100.0;
217  zBegin_m /= 100.0;
218  zEnd_m /= 100.0;
219 }
220 
221 double FM1DMagnetoStatic::readFileData(std::ifstream &fieldFile,
222  double fieldData[]) {
223 
224  double maxBz = 0.0;
225  for(int dataIndex = 0; dataIndex < numberOfGridPoints_m; dataIndex++) {
226  interpretLine<double>(fieldFile, fieldData[numberOfGridPoints_m
227  - 1 + dataIndex]);
228  if(std::abs(fieldData[numberOfGridPoints_m + dataIndex]) > maxBz)
229  maxBz = std::abs(fieldData[numberOfGridPoints_m + dataIndex]);
230 
231  /*
232  * Mirror the field map about minimum z value to ensure that
233  * it is periodic.
234  */
235  if(dataIndex != 0)
236  fieldData[numberOfGridPoints_m - 1 - dataIndex]
237  = fieldData[numberOfGridPoints_m + dataIndex];
238  }
239 
240  if (!normalize_m)
241  maxBz = 1.0;
242 
243  return maxBz;
244 }
245 
246 bool FM1DMagnetoStatic::readFileHeader(std::ifstream &fieldFile) {
247 
248  std::string tempString;
249  int tempInt;
250 
251  bool parsingPassed = true;
252  try {
253  parsingPassed = interpretLine<std::string, int>(fieldFile,
254  tempString,
255  accuracy_m);
256  } catch (GeneralClassicException &e) {
257  parsingPassed = interpretLine<std::string, int, std::string>(fieldFile,
258  tempString,
259  accuracy_m,
260  tempString);
261 
262  tempString = Util::toUpper(tempString);
263  if (tempString != "TRUE" &&
264  tempString != "FALSE")
265  throw GeneralClassicException("FM1DMagnetoStatic::readFileHeader",
266  "The third string on the first line of 1D field "
267  "maps has to be either TRUE or FALSE");
268 
269  normalize_m = (tempString == "TRUE");
270  }
271  parsingPassed = parsingPassed &&
272  interpretLine<double, double, int>(fieldFile, zBegin_m,
273  zEnd_m,
275  parsingPassed = parsingPassed &&
276  interpretLine<double, double, int>(fieldFile, rBegin_m,
277  rEnd_m, tempInt);
278 
280 
283 
284  return parsingPassed;
285 }
286 
287 void FM1DMagnetoStatic::stripFileHeader(std::ifstream &fieldFile) {
288 
289  std::string tempString;
290 
291  getLine(fieldFile, tempString);
292  getLine(fieldFile, tempString);
293  getLine(fieldFile, tempString);
294 }
Tps< T > cos(const Tps< T > &x)
Cosine.
Definition: TpsMath.h:129
Tps< T > pow(const Tps< T > &x, int y)
Integer power.
Definition: TpsMath.h:76
Tps< T > sin(const Tps< T > &x)
Sine.
Definition: TpsMath.h:111
@ T1DMagnetoStatic
Definition: Fieldmap.h:20
DiffDirection
Definition: Fieldmap.h:54
PETE_TUTree< FnAbs, typename T::PETE_Expr_t > abs(const PETE_Expr< T > &l)
Inform & endl(Inform &inf)
Definition: Inform.cpp:42
Inform & level3(Inform &inf)
Definition: Inform.cpp:47
#define INFOMSG(msg)
Definition: IpplInfo.h:348
constexpr double two_pi
The value of.
Definition: Physics.h:33
constexpr double e
The value of.
Definition: Physics.h:39
constexpr double pi
The value of.
Definition: Physics.h:30
std::string toUpper(const std::string &str)
Definition: Util.cpp:132
MapType Type
Definition: Fieldmap.h:114
bool interpreteEOF(std::ifstream &in)
Definition: Fieldmap.cpp:558
void disableFieldmapWarning()
Definition: Fieldmap.cpp:613
bool normalize_m
Definition: Fieldmap.h:120
static std::string typeset_msg(const std::string &msg, const std::string &title)
Definition: Fieldmap.cpp:652
std::string Filename_m
Definition: Fieldmap.h:117
void getLine(std::ifstream &in, std::string &buffer)
Definition: Fieldmap.h:121
void noFieldmapWarning()
Definition: Fieldmap.cpp:621
bool readFileHeader(std::ifstream &fieldFile)
virtual void readMap()
std::vector< double > fourierCoefs_m
Number of Fourier coefficients to use reconstructing field.
virtual void setFrequency(double freq)
int numberOfGridPoints_m
Field length.
int accuracy_m
Number of grid points in field input file.
void stripFileHeader(std::ifstream &fieldFile)
virtual bool getFieldDerivative(const Vector_t &R, Vector_t &E, Vector_t &B, const DiffDirection &dir) const
virtual double getFrequency() const
double length_m
Longitudinal end of field.
virtual void freeMap()
void computeFourierCoefficients(double maxEz, double fieldData[])
double rEnd_m
Minimum radius of field.
bool checkFileData(std::ifstream &fieldFile, bool parsingPassed)
virtual bool getFieldstrength(const Vector_t &R, Vector_t &E, Vector_t &B) const
double zBegin_m
Maximum radius of field.
virtual void getFieldDimensions(double &zBegin, double &zEnd) const
double readFileData(std::ifstream &fieldFile, double fieldData[])
void computeFieldOnAxis(double z, std::vector< double > &fieldComponents) const
double zEnd_m
Longitudinal start of field.
virtual void getInfo(Inform *)
FM1DMagnetoStatic(std::string aFilename)
void computeFieldOffAxis(const Vector_t &R, Vector_t &E, Vector_t &B, std::vector< double > fieldComponents) const
Definition: Inform.h:42