OPAL (Object Oriented Parallel Accelerator Library)  2021.1.99
OPAL
FM1DDynamic.cpp
Go to the documentation of this file.
1 #include "Fields/FM1DDynamic.h"
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 <iostream>
11 
12 FM1DDynamic::FM1DDynamic(std::string aFilename):
13  Fieldmap(aFilename) {
14 
15  Type = T1DDynamic;
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 
33  } else {
35  zBegin_m = 0.0;
36  zEnd_m = -1e-3;
37  }
38 }
39 
41  freeMap();
42 }
43 
45 
46  if(fourierCoefs_m.empty()) {
47 
48  std::ifstream fieldFile(Filename_m.c_str());
49  stripFileHeader(fieldFile);
50 
51  double *fieldData = new double[2 * numberOfGridPoints_m - 1];
52  double maxEz = readFileData(fieldFile, fieldData);
53  fieldFile.close();
54  computeFourierCoefficients(maxEz, fieldData);
55  delete [] fieldData;
56 
57  INFOMSG(typeset_msg("Read in field map '" + Filename_m + "'", "info") << endl);
58  }
59 }
60 
62 
63  if(!fourierCoefs_m.empty()) {
64  fourierCoefs_m.clear();
65 
66  INFOMSG(typeset_msg("Freed field map '" + 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 eZPrime = 0.0;
88 
89  int coefIndex = 1;
90  for(int n = 1; n < accuracy_m; ++ n) {
91 
92  eZPrime += (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  E(2) += eZPrime;
100 
101  return false;
102 }
103 
104 void FM1DDynamic::getFieldDimensions(double &zBegin, double &zEnd) const {
105  zBegin = zBegin_m;
106  zEnd = zEnd_m;
107 }
108 
109 void FM1DDynamic::getFieldDimensions(double &/*xIni*/, double &/*xFinal*/,
110  double &/*yIni*/, double &/*yFinal*/,
111  double &/*zIni*/, double &/*zFinal*/) const {
112 }
113 
115 { }
116 
118  (*msg) << Filename_m
119  << " (1D dynamic); zini= "
120  << zBegin_m << " m; zfinal= "
121  << zEnd_m << " m;" << endl;
122 }
123 
125  return frequency_m;
126 }
127 
128 void FM1DDynamic::setFrequency(double frequency) {
129  frequency_m = frequency;
130 }
131 
132 void FM1DDynamic::getOnaxisEz(std::vector<std::pair<double, double> > &eZ) {
133 
134  eZ.resize(numberOfGridPoints_m);
135  std::ifstream fieldFile(Filename_m.c_str());
136  stripFileHeader(fieldFile);
137  double maxEz = readFileData(fieldFile, eZ);
138  fieldFile.close();
139  scaleField(maxEz, eZ);
140 
141 }
142 
143 bool FM1DDynamic::checkFileData(std::ifstream &fieldFile, bool parsingPassed) {
144 
145  double tempDouble;
146  for(int dataIndex = 0; dataIndex < numberOfGridPoints_m; ++ dataIndex)
147  parsingPassed = parsingPassed
148  && interpretLine<double>(fieldFile, tempDouble);
149 
150  return parsingPassed && interpreteEOF(fieldFile);
151 
152 }
153 
155  Vector_t &E,
156  Vector_t &B,
157  std::vector<double> fieldComponents) const {
158 
159  double radiusSq = pow(R(0), 2.0) + pow(R(1), 2.0);
160  double transverseEFactor = (fieldComponents.at(1)
161  * (0.5 - radiusSq * twoPiOverLambdaSq_m / 16.0)
162  - radiusSq * fieldComponents.at(3) / 16.0);
163  double transverseBFactor = ((fieldComponents.at(0)
164  * (0.5 - radiusSq * twoPiOverLambdaSq_m / 16.0)
165  - radiusSq * fieldComponents.at(2) / 16.0)
167 
168  E(0) += - R(0) * transverseEFactor;
169  E(1) += - R(1) * transverseEFactor;
170  E(2) += (fieldComponents.at(0)
171  * (1.0 - radiusSq * twoPiOverLambdaSq_m / 4.0)
172  - radiusSq * fieldComponents.at(2) / 4.0);
173 
174  B(0) += - R(1) * transverseBFactor;
175  B(1) += R(0) * transverseBFactor;
176 
177 }
178 
180  std::vector<double> &fieldComponents) const {
181 
182  double kz = Physics::two_pi * z / length_m + Physics::pi;
183  fieldComponents.push_back(fourierCoefs_m.at(0));
184  fieldComponents.push_back(0.0);
185  fieldComponents.push_back(0.0);
186  fieldComponents.push_back(0.0);
187 
188  int coefIndex = 1;
189  for(int n = 1; n < accuracy_m; ++ n) {
190 
191  double kn = n * Physics::two_pi / length_m;
192  double coskzn = cos(kz * n);
193  double sinkzn = sin(kz * n);
194 
195  fieldComponents.at(0) += (fourierCoefs_m.at(coefIndex) * coskzn
196  - fourierCoefs_m.at(coefIndex + 1) * sinkzn);
197 
198  fieldComponents.at(1) += kn * (-fourierCoefs_m.at(coefIndex) * sinkzn
199  - fourierCoefs_m.at(coefIndex + 1) * coskzn);
200 
201  double derivCoeff = pow(kn, 2.0);
202  fieldComponents.at(2) += derivCoeff * (-fourierCoefs_m.at(coefIndex) * coskzn
203  + fourierCoefs_m.at(coefIndex + 1) * sinkzn);
204  derivCoeff *= kn;
205  fieldComponents.at(3) += derivCoeff * (fourierCoefs_m.at(coefIndex) * sinkzn
206  + fourierCoefs_m.at(coefIndex + 1) * coskzn);
207 
208  coefIndex += 2;
209  }
210 }
211 
212 void FM1DDynamic::computeFourierCoefficients(double maxEz, double fieldData[]) {
213  const unsigned int totalSize = 2 * numberOfGridPoints_m - 1;
214  gsl_fft_real_wavetable *waveTable = gsl_fft_real_wavetable_alloc(totalSize);
215  gsl_fft_real_workspace *workSpace = gsl_fft_real_workspace_alloc(totalSize);
216  gsl_fft_real_transform(fieldData, 1, totalSize, waveTable, workSpace);
217 
218  /*
219  * Normalize the Fourier coefficients such that the max field value
220  * is 1 V/m.
221  */
222  maxEz /= 1.0e6;
223 
224  fourierCoefs_m.push_back(fieldData[0] / (totalSize * maxEz));
225  for(int coefIndex = 1; coefIndex < 2 * accuracy_m - 1; ++ coefIndex)
226  fourierCoefs_m.push_back(2.0 * fieldData[coefIndex] / (totalSize * maxEz));
227 
228  gsl_fft_real_workspace_free(workSpace);
229  gsl_fft_real_wavetable_free(waveTable);
230 
231 }
232 
234 
235  // Convert to angular frequency in Hz.
236  frequency_m *= Physics::two_pi * 1.0e6;
237 
238  // Convert to m.
239  rBegin_m /= 100.0;
240  rEnd_m /= 100.0;
241  zBegin_m /= 100.0;
242  zEnd_m /= 100.0;
243 
245 }
246 
247 double FM1DDynamic::readFileData(std::ifstream &fieldFile, double fieldData[]) {
248 
249  double maxEz = 0.0;
250  for(int dataIndex = 0; dataIndex < numberOfGridPoints_m; ++ dataIndex) {
251  interpretLine<double>(fieldFile, fieldData[numberOfGridPoints_m
252  - 1 + dataIndex]);
253  if(std::abs(fieldData[numberOfGridPoints_m + dataIndex]) > maxEz)
254  maxEz = std::abs(fieldData[numberOfGridPoints_m + dataIndex]);
255 
256  /*
257  * Mirror the field map about minimum z value to ensure that it
258  * is periodic.
259  */
260  if(dataIndex != 0)
261  fieldData[numberOfGridPoints_m - 1 - dataIndex]
262  = fieldData[numberOfGridPoints_m + dataIndex];
263  }
264 
265  if (!normalize_m)
266  maxEz = 1.0;
267 
268  return maxEz;
269 }
270 
271 double FM1DDynamic::readFileData(std::ifstream &fieldFile,
272  std::vector<std::pair<double, double>> &eZ) {
273 
274  double maxEz = 0.0;
275  double deltaZ = (zEnd_m - zBegin_m) / (numberOfGridPoints_m - 1);
276  for(int dataIndex = 0; dataIndex < numberOfGridPoints_m; ++ dataIndex) {
277  eZ.at(dataIndex).first = deltaZ * dataIndex;
278  interpretLine<double>(fieldFile, eZ.at(dataIndex).second);
279  if(std::abs(eZ.at(dataIndex).second) > maxEz)
280  maxEz = std::abs(eZ.at(dataIndex).second);
281  }
282 
283  if (!normalize_m)
284  maxEz = 1.0;
285 
286  return maxEz;
287 }
288 
289 bool FM1DDynamic::readFileHeader(std::ifstream &fieldFile) {
290 
291  std::string tempString;
292  int tempInt;
293 
294  bool parsingPassed = true;
295  try {
296  parsingPassed = interpretLine<std::string, int>(fieldFile,
297  tempString,
298  accuracy_m);
299  } catch (GeneralClassicException &e) {
300  parsingPassed = interpretLine<std::string, int, std::string>(fieldFile,
301  tempString,
302  accuracy_m,
303  tempString);
304 
305  tempString = Util::toUpper(tempString);
306  if (tempString != "TRUE" &&
307  tempString != "FALSE")
308  throw GeneralClassicException("FM1DDynamic::FM1DDynamic",
309  "The third string on the first line of 1D field "
310  "maps has to be either TRUE or FALSE");
311 
312  normalize_m = (tempString == "TRUE");
313  }
314 
315  parsingPassed = parsingPassed &&
316  interpretLine<double, double, int>(fieldFile,
317  zBegin_m,
318  zEnd_m,
320  parsingPassed = parsingPassed &&
321  interpretLine<double>(fieldFile, frequency_m);
322  parsingPassed = parsingPassed &&
323  interpretLine<double, double, int>(fieldFile,
324  rBegin_m,
325  rEnd_m,
326  tempInt);
327 
329 
332 
333  return parsingPassed;
334 }
335 
336 void FM1DDynamic::scaleField(double maxEz, std::vector<std::pair<double, double>> &eZ) {
337  for(int dataIndex = 0; dataIndex < numberOfGridPoints_m; ++ dataIndex)
338  eZ.at(dataIndex).second /= maxEz;
339 }
340 
341 void FM1DDynamic::stripFileHeader(std::ifstream &fieldFile) {
342 
343  std::string tempString;
344 
345  getLine(fieldFile, tempString);
346  getLine(fieldFile, tempString);
347  getLine(fieldFile, tempString);
348  getLine(fieldFile, tempString);
349 }
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
@ T1DDynamic
Definition: Fieldmap.h:16
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
#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 c
The velocity of light in m/s.
Definition: Physics.h:51
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
double zBegin_m
Maximum radius of field.
Definition: FM1DDynamic.h:49
virtual void swap()
void computeFieldOnAxis(double z, std::vector< double > &fieldComponents) const
int numberOfGridPoints_m
Field length.
Definition: FM1DDynamic.h:52
void stripFileHeader(std::ifstream &fieldFile)
FM1DDynamic(std::string aFilename)
Definition: FM1DDynamic.cpp:12
double twoPiOverLambdaSq_m
Field angular frequency (Hz).
Definition: FM1DDynamic.h:45
double readFileData(std::ifstream &fieldFile, double fieldData[])
std::vector< double > fourierCoefs_m
Number of Fourier coefficients to use reconstructing field.
Definition: FM1DDynamic.h:55
void scaleField(double maxEz, std::vector< std::pair< double, double >> &eZ)
double rEnd_m
Minimum radius of field.
Definition: FM1DDynamic.h:48
double length_m
Longitudinal end of field.
Definition: FM1DDynamic.h:51
void convertHeaderData()
bool checkFileData(std::ifstream &fieldFile, bool parsingPassed)
virtual double getFrequency() const
double zEnd_m
Longitudinal start of field.
Definition: FM1DDynamic.h:50
virtual void freeMap()
Definition: FM1DDynamic.cpp:61
virtual bool getFieldstrength(const Vector_t &R, Vector_t &E, Vector_t &B) const
Definition: FM1DDynamic.cpp:71
virtual void readMap()
Definition: FM1DDynamic.cpp:44
virtual void getFieldDimensions(double &zBegin, double &zEnd) const
int accuracy_m
Number of grid points in field input file.
Definition: FM1DDynamic.h:54
void computeFourierCoefficients(double maxEz, double fieldData[])
double frequency_m
Definition: FM1DDynamic.h:44
double rBegin_m
2 Pi divided by the field RF wavelength squared.
Definition: FM1DDynamic.h:47
virtual void getInfo(Inform *)
void computeFieldOffAxis(const Vector_t &R, Vector_t &E, Vector_t &B, std::vector< double > fieldComponents) const
virtual bool getFieldDerivative(const Vector_t &R, Vector_t &E, Vector_t &B, const DiffDirection &dir) const
Definition: FM1DDynamic.cpp:81
bool readFileHeader(std::ifstream &fieldFile)
virtual void setFrequency(double freq)
virtual void getOnaxisEz(std::vector< std::pair< double, double >> &eZ)
Definition: Inform.h:42