OPAL (Object Oriented Parallel Accelerator Library)  2021.1.99
OPAL
ElementBase.cpp
Go to the documentation of this file.
1 //
2 // Class ElementBase
3 // The very base class for beam line representation objects. A beam line
4 // is modelled as a composite structure having a single root object
5 // (the top level beam line), which contains both ``single'' leaf-type
6 // elements (Components), as well as sub-lines (composites).
7 //
8 // Interface for basic beam line object.
9 // This class defines the abstract interface for all objects which can be
10 // contained in a beam line. ElementBase forms the base class for two distinct
11 // but related hierarchies of objects:
12 // [OL]
13 // [LI]
14 // A set of concrete accelerator element classes, which compose the standard
15 // accelerator component library (SACL).
16 // [LI]
17 // [/OL]
18 // Instances of the concrete classes for single elements are by default
19 // sharable. Instances of beam lines and integrators are by
20 // default non-sharable, but they may be made sharable by a call to
21 // [b]makeSharable()[/b].
22 // [p]
23 // An ElementBase object can return two lengths, which may be different:
24 // [OL]
25 // [LI]
26 // The arc length along the geometry.
27 // [LI]
28 // The design length, often measured along a straight line.
29 // [/OL]
30 // Class ElementBase contains a map of name versus value for user-defined
31 // attributes (see file AbsBeamline/AttributeSet.hh). The map is primarily
32 // intended for processes that require algorithm-specific data in the
33 // accelerator model.
34 // [P]
35 // The class ElementBase has as base class the abstract class RCObject.
36 // Virtual derivation is used to make multiple inheritance possible for
37 // derived concrete classes. ElementBase implements three copy modes:
38 // [OL]
39 // [LI]
40 // Copy by reference: Call RCObject::addReference() and use [b]this[/b].
41 // [LI]
42 // Copy structure: use ElementBase::copyStructure().
43 // During copying of the structure, all sharable items are re-used, while
44 // all non-sharable ones are cloned.
45 // [LI]
46 // Copy by cloning: use ElementBase::clone().
47 // This returns a full deep copy.
48 // [/OL]
49 //
50 // Copyright (c) 200x - 2021, Paul Scherrer Institut, Villigen PSI, Switzerland
51 // All rights reserved
52 //
53 // This file is part of OPAL.
54 //
55 // OPAL is free software: you can redistribute it and/or modify
56 // it under the terms of the GNU General Public License as published by
57 // the Free Software Foundation, either version 3 of the License, or
58 // (at your option) any later version.
59 //
60 // You should have received a copy of the GNU General Public License
61 // along with OPAL. If not, see <https://www.gnu.org/licenses/>.
62 //
64 
65 #include "Channels/Channel.h"
67 #include "Solvers/WakeFunction.h"
69 
70 #include <string>
71 
73  ElementBase("")
74 {}
75 
76 
78  RCObject(),
79  shareFlag(true),
80  csTrafoGlobal2Local_m(right.csTrafoGlobal2Local_m),
81  misalignment_m(right.misalignment_m),
82  aperture_m(right.aperture_m),
83  elementEdge_m(right.elementEdge_m),
84  rotationZAxis_m(right.rotationZAxis_m),
85  elementID(right.elementID),
86  userAttribs(right.userAttribs),
87  wake_m(right.wake_m),
88  bgeometry_m(right.bgeometry_m),
89  parmatint_m(right.parmatint_m),
90  positionIsFixed(right.positionIsFixed),
91  elementPosition_m(right.elementPosition_m),
92  elemedgeSet_m(right.elemedgeSet_m),
93  outputfn_m(right.outputfn_m)
94 {
95 
96  if (parmatint_m) {
98  }
99  if (bgeometry_m) {
100  bgeometry_m->updateElement(this);
101  }
102 }
103 
104 
105 ElementBase::ElementBase(const std::string &name):
106  RCObject(),
107  shareFlag(true),
108  csTrafoGlobal2Local_m(),
109  misalignment_m(),
110  elementEdge_m(0),
111  rotationZAxis_m(0.0),
112  elementID(name),
113  userAttribs(),
114  wake_m(NULL),
115  bgeometry_m(NULL),
116  parmatint_m(NULL),
117  positionIsFixed(false),
118  elementPosition_m(0.0),
119  elemedgeSet_m(false),
120  outputfn_m("")
121 {}
122 
123 
125 
126 {}
127 
128 
129 const std::string &ElementBase::getName() const {
130  return elementID;
131 }
132 
133 
134 void ElementBase::setName(const std::string &name) {
135  elementID = name;
136 }
137 
138 
139 void ElementBase::setOutputFN(const std::string fn) {
140  outputfn_m = fn;
141 }
142 
143 
144 std::string ElementBase::getOutputFN() const {
145  if (outputfn_m.empty()) {
146  return getName();
147  } else {
148  return outputfn_m.substr(0, outputfn_m.rfind("."));
149  }
150 }
151 
152 
153 double ElementBase::getAttribute(const std::string &aKey) const {
154  const ConstChannel *aChannel = getConstChannel(aKey);
155 
156  if (aChannel != NULL) {
157  double val = *aChannel;
158  delete aChannel;
159  return val;
160  } else {
161  return 0.0;
162  }
163 }
164 
165 
166 bool ElementBase::hasAttribute(const std::string &aKey) const {
167  const ConstChannel *aChannel = getConstChannel(aKey);
168 
169  if (aChannel != NULL) {
170  delete aChannel;
171  return true;
172  } else {
173  return false;
174  }
175 }
176 
177 
178 void ElementBase::removeAttribute(const std::string &aKey) {
180 }
181 
182 
183 void ElementBase::setAttribute(const std::string &aKey, double val) {
184  Channel *aChannel = getChannel(aKey, true);
185 
186  if (aChannel != NULL && aChannel->isSettable()) {
187  *aChannel = val;
188  delete aChannel;
189  } else
190  std::cout << "Channel NULL or not Settable" << std::endl;
191 }
192 
193 
194 Channel *ElementBase::getChannel(const std::string &aKey, bool create) {
195  return userAttribs.getChannel(aKey, create);
196 }
197 
198 
199 const ConstChannel *ElementBase::getConstChannel(const std::string &aKey) const {
200  // Use const_cast to allow calling the non-const method GetChannel().
201  // The const return value of this method will nevertheless inhibit set().
202  return const_cast<ElementBase *>(this)->getChannel(aKey);
203 }
204 
205 
207  switch (type) {
208  case BEAMLINE:
209  return "Beamline";
210  case CCOLLIMATOR:
211  return "CCollimator";
212  case CORRECTOR:
213  return "Corrector";
214  case CYCLOTRON:
215  return "Cyclotron";
216  case DEGRADER:
217  return "Degrader";
218  case DRIFT:
219  return "Drift";
220  case MARKER:
221  return "Marker";
222  case MONITOR:
223  return "Monitor";
224  case MULTIPOLE:
225  return "Multipole";
226  case OFFSET:
227  return "Offset";
228  case PROBE:
229  return "Probe";
230  case RBEND:
231  return "RBend";
232  case RFCAVITY:
233  return "RFCavity";
234  case RING:
235  return "Ring";
236  case SBEND3D:
237  return "SBend3D";
238  case SBEND:
239  return "SBend";
240  case SEPTUM:
241  return "Septum";
242  case SOLENOID:
243  return "Solenoid";
244  case STRIPPER:
245  return "Stripper";
246  case TRAVELINGWAVE:
247  return "TravelingWave";
248 #ifdef ENABLE_OPAL_FEL
249  case UNDULATOR:
250  return "Undulator";
251 #endif
252  case VACUUM:
253  return "Vacuum";
254  case VARIABLERFCAVITY:
255  return "VariableRFCavity";
256  case ANY:
257  default:
258  return "'unknown' type";
259  }
260 }
261 
263  if (isSharable()) {
264  return this;
265  } else {
266  return clone();
267  }
268 }
269 
270 
272  shareFlag = true;
273 }
274 
275 
277  for (AttributeSet::const_iterator i = set.begin(); i != set.end(); ++i) {
278  setAttribute(i->first, i->second);
279  }
280 
281  return true;
282 }
283 
285  wake_m = wk;//->clone(getName() + std::string("_wake")); }
286 }
287 
289  bgeometry_m = geo;//->clone(getName() + std::string("_wake")); }
290 }
291 
293  parmatint_m = parmatint;
294 }
295 
297  if (actionRange_m.size() > 0 && actionRange_m.front().second < s) {
298  actionRange_m.pop();
299  if (actionRange_m.size() > 0) {
300  elementEdge_m = actionRange_m.front().first;
301  }
302  }
303 }
304 
306 {
307  const double &xLimit = aperture_m.second[0];
308  const double &yLimit = aperture_m.second[1];
309  double factor = 1.0;
310  if (aperture_m.first == CONIC_RECTANGULAR ||
311  aperture_m.first == CONIC_ELLIPTICAL) {
312  Vector_t rRelativeToBegin = getEdgeToBegin().transformTo(r);
313  double fractionLength = rRelativeToBegin(2) / getElementLength();
314  factor = fractionLength * aperture_m.second[2];
315  }
316 
317  switch(aperture_m.first) {
318  case RECTANGULAR:
319  return (std::abs(r[0]) < xLimit && std::abs(r[1]) < yLimit);
320  case ELLIPTICAL:
321  return (std::pow(r[0] / xLimit, 2) + std::pow(r[1] / yLimit, 2) < 1.0);
322  case CONIC_RECTANGULAR:
323  return (std::abs(r[0]) < factor * xLimit && std::abs(r[1]) < factor * yLimit);
324  case CONIC_ELLIPTICAL:
325  return (std::pow(r[0] / (factor * xLimit), 2) + std::pow(r[1] / (factor * yLimit), 2) < 1.0);
326  default:
327  return false;
328  }
329 }
330 
332  const Vector_t & point = points.front();
333  BoundingBox result{point, point};
334  for (const Vector_t & point: points) {
335  BoundingBox tmp{point, point};
336  result.getCombinedBoundingBox(tmp);
337  }
338 
339  return result;
340 }
341 
342 bool ElementBase::BoundingBox::isInside(const Vector_t & position) const {
343  Vector_t relativePosition = position - lowerLeftCorner;
344  Vector_t diagonal = upperRightCorner - lowerLeftCorner;
345 
346  for (unsigned int d = 0; d < 3; ++ d) {
347  if (relativePosition[d] < 0.0 ||
348  relativePosition[d] > diagonal[d]) {
349  return false;
350  }
351  }
352 
353  return true;
354 }
355 
356 boost::optional<Vector_t>
358  const Vector_t & direction) const {
359  Vector_t relativePosition = lowerLeftCorner - position;
360  Vector_t diagonal = upperRightCorner - lowerLeftCorner;
361  Vector_t normalizedDirection = direction / euclidean_norm(direction);
362 
363  for (int i : {-1, 1}) {
364  for (unsigned int d = 0; d < 3; ++ d) {
365  double projectedDirection = normalizedDirection[d];
366  if (std::abs(projectedDirection) < 1e-10) {
367  continue;
368  }
369 
370  double distanceNearestPoint = relativePosition[d];
371  double tau = distanceNearestPoint / projectedDirection;
372  if (tau < 0) {
373  continue;
374  }
375  Vector_t delta = tau * normalizedDirection;
376  Vector_t relativeIntersectionPoint = i * (relativePosition - delta);
377 
378  if (relativeIntersectionPoint[(d + 1) % 3] < 0.0 ||
379  relativeIntersectionPoint[(d + 1) % 3] > diagonal[(d + 1) % 3] ||
380  relativeIntersectionPoint[(d + 2) % 3] < 0.0 ||
381  relativeIntersectionPoint[(d + 2) % 3] > diagonal[(d + 2) % 3]) {
382  continue;
383  }
384 
385  return position + delta;
386  }
387  relativePosition = upperRightCorner - position;
388  }
389 
390  return boost::none;
391 }
392 
396 
397  const double &x = aperture_m.second[0];
398  const double &y = aperture_m.second[1];
399  const double &f = aperture_m.second[2];
400 
401  std::vector<Vector_t> corners(8);
402  for (int i = -1; i < 2; i += 2) {
403  for (int j = -1; j < 2; j += 2) {
404  unsigned int idx = (i + 1)/2 + (j + 1);
405  corners[idx] = toBegin.transformFrom(Vector_t(i * x, j * y, 0.0));
406  corners[idx + 4] = toEnd.transformFrom(Vector_t(i * f * x, j * f * y, 0.0));
407  }
408  }
409 
410  BoundingBox bb;
411  bb.lowerLeftCorner = bb.upperRightCorner = corners[0];
412 
413  for (unsigned int i = 1; i < 8u; ++ i) {
414  for (unsigned int d = 0; d < 3u; ++ d) {
415  if (bb.lowerLeftCorner(d) > corners[i](d)) {
416  bb.lowerLeftCorner(d) = corners[i](d);
417  }
418  if (bb.upperRightCorner(d) < corners[i](d)) {
419  bb.upperRightCorner(d) = corners[i](d);
420  }
421  }
422  }
423 
424  return bb;
425 }
426 
427 void ElementBase::BoundingBox::print(std::ostream & out) const {
428  const Vector_t & ll = lowerLeftCorner;
429  const Vector_t & ur = upperRightCorner;
430  Vector_t diagonal = ur - ll;
431  Vector_t dX(diagonal(0), 0, 0), dY(0, diagonal(1), 0), dZ(0, 0, diagonal(2));
432 
433  std::vector<Vector_t> corners{ll, ll + dX, ll + dX + dY, ll + dY, ur, ur - dX, ur - dX - dY, ur - dY};
434  std::vector<std::vector<unsigned int>> paths{{0, 1, 2, 3}, {0, 1, 7, 6}, {1, 2, 4, 7}, {2, 3, 5, 4}, {3, 0, 6, 5}, {4, 5, 6, 7}};
435 
436  out << std::setprecision(8);
437  for (const std::vector<unsigned int>& path: paths) {
438  for (unsigned int i : {0, 1, 2, 3, 0}) {
439  const Vector_t & corner = corners[path[i]];
440  out << std::setw(16) << corner(0)
441  << std::setw(16) << corner(1)
442  << std::setw(16) << corner(2)
443  << std::endl;
444  }
445  out << std::endl;
446  }
447 }
Tps< T > pow(const Tps< T > &x, int y)
Integer power.
Definition: TpsMath.h:76
T euclidean_norm(const Vector< T > &)
Euclidean norm.
Definition: Vector.h:243
PETE_TUTree< FnAbs, typename T::PETE_Expr_t > abs(const PETE_Expr< T > &l)
Inform & endl(Inform &inf)
Definition: Inform.cpp:42
const std::string name
constexpr double e
The value of.
Definition: Physics.h:39
boost::function< boost::tuple< double, bool >arguments_t)> type
Definition: function.hpp:21
Map of std::string versus double value.
Definition: AttributeSet.h:41
void removeAttribute(const std::string &aKey)
Remove an existing attribute.
const_iterator begin() const
Iterator accessing first member.
Definition: AttributeSet.h:105
NameMap::const_iterator const_iterator
An iterator for a map of name versus value.
Definition: AttributeSet.h:49
Channel * getChannel(const std::string &aKey, bool create=false)
Construct a read/write channel.
const_iterator end() const
Iterator marking the end of the list.
Definition: AttributeSet.h:108
virtual void setBoundaryGeometry(BoundaryGeometry *geo)
virtual ~ElementBase()
virtual Channel * getChannel(const std::string &aKey, bool create=false)
Construct a read/write channel.
virtual void setName(const std::string &name)
Set element name.
virtual const std::string & getName() const
Get element name.
virtual void removeAttribute(const std::string &aKey)
Remove an existing attribute.
virtual double getElementLength() const
Get design length.
Definition: ElementBase.h:432
bool update(const AttributeSet &)
Update element.
std::string outputfn_m
Definition: ElementBase.h:420
ParticleMatterInteractionHandler * parmatint_m
Definition: ElementBase.h:411
virtual const ConstChannel * getConstChannel(const std::string &aKey) const
Construct a read-only channel.
virtual void setAttribute(const std::string &aKey, double val)
Set value of an attribute.
virtual BoundingBox getBoundingBoxInLabCoords() const
bool isInsideTransverse(const Vector_t &r) const
std::string getOutputFN() const
Get output filename.
void setOutputFN(std::string fn)
Set output filename.
std::pair< ApertureType, std::vector< double > > aperture_m
Definition: ElementBase.h:390
virtual void setParticleMatterInteraction(ParticleMatterInteractionHandler *spys)
virtual void makeSharable()
Set sharable flag.
virtual bool hasAttribute(const std::string &aKey) const
Test for existence of an attribute.
WakeFunction * wake_m
Definition: ElementBase.h:407
std::queue< std::pair< double, double > > actionRange_m
Definition: ElementBase.h:418
virtual CoordinateSystemTrafo getEdgeToBegin() const
Definition: ElementBase.h:520
std::string getTypeString() const
Definition: ElementBase.h:596
AttributeSet userAttribs
Definition: ElementBase.h:405
virtual ElementBase * clone() const =0
Return clone.
virtual ElementBase * copyStructure()
Make a structural copy.
BoundaryGeometry * bgeometry_m
Definition: ElementBase.h:409
bool isSharable() const
Test if the element can be shared.
Definition: ElementBase.h:480
bool shareFlag
Definition: ElementBase.h:385
void setCurrentSCoordinate(double s)
std::string elementID
Definition: ElementBase.h:402
virtual double getAttribute(const std::string &aKey) const
Get attribute value.
virtual CoordinateSystemTrafo getEdgeToEnd() const
Definition: ElementBase.h:528
virtual void setWake(WakeFunction *wf)
attach a wake field to the element
double elementEdge_m
Definition: ElementBase.h:392
CoordinateSystemTrafo csTrafoGlobal2Local_m
Definition: ElementBase.h:387
bool isInside(const Vector_t &) const
static BoundingBox getBoundingBox(const std::vector< Vector_t > &points)
void print(std::ostream &) const
boost::optional< Vector_t > getPointOfIntersection(const Vector_t &position, const Vector_t &direction) const
Vector_t transformFrom(const Vector_t &r) const
Vector_t transformTo(const Vector_t &r) const
Abstract interface for read/write access to variable.
Definition: Channel.h:32
virtual bool isSettable() const
Test if settable.
Definition: Channel.cpp:36
Abstract interface for read-only access to variable.
Definition: ConstChannel.h:29
Abstract base class for reference counted objects.
Definition: RCObject.h:40
void updateElement(ElementBase *element)
Vektor< double, 3 > Vector_t
Definition: Vektor.h:6