Garfield++ 5.0
A toolkit for the detailed simulation of particle detectors based on ionisation measurement in gases and semiconductors
Loading...
Searching...
No Matches
Garfield::ComponentAnsys123 Class Reference

Component for importing and interpolating three-dimensional ANSYS field maps. More...

#include <ComponentAnsys123.hh>

+ Inheritance diagram for Garfield::ComponentAnsys123:

Public Member Functions

 ComponentAnsys123 ()
 Constructor.
 
 ~ComponentAnsys123 ()
 Destructor.
 
bool Initialise (const std::string &elist="ELIST.lis", const std::string &nlist="NLIST.lis", const std::string &mplist="MPLIST.lis", const std::string &prnsol="PRNSOL.lis", const std::string &unit="cm")
 
bool SetWeightingPotential (const std::string &prnsol, const std::string &label)
 Import weighting potentials.
 
bool SetWeightingField (const std::string &prnsol, const std::string &label)
 
- Public Member Functions inherited from Garfield::ComponentFieldMap
 ComponentFieldMap ()=delete
 Default constructor.
 
 ComponentFieldMap (const std::string &name)
 Constructor.
 
virtual ~ComponentFieldMap ()
 Destructor.
 
bool Check ()
 Check element aspect ratio.
 
void PrintRange ()
 Show x, y, z, V and angular ranges.
 
void PrintMaterials ()
 List all currently defined materials.
 
void DriftMedium (const size_t imat)
 Flag a field map material as a drift medium.
 
void NotDriftMedium (const size_t imat)
 Flag a field map materials as a non-drift medium.
 
size_t GetNumberOfMaterials () const
 Return the number of materials in the field map.
 
double GetPermittivity (const size_t imat) const
 Return the relative permittivity of a field map material.
 
double GetConductivity (const size_t imat) const
 Return the conductivity of a field map material.
 
void SetMedium (const size_t imat, Medium *medium)
 Associate a field map material with a Medium object.
 
MediumGetMedium (const size_t imat) const
 Return the Medium associated to a field map material.
 
void SetGas (Medium *medium)
 
virtual size_t GetNumberOfElements () const
 Return the number of mesh elements.
 
bool GetElement (const size_t i, double &vol, double &dmin, double &dmax) const
 Return the volume and aspect ratio of a mesh element.
 
virtual bool GetElement (const size_t i, size_t &mat, bool &drift, std::vector< size_t > &nodes) const
 Return the material and node indices of a mesh element.
 
virtual size_t GetNumberOfNodes () const
 
virtual bool GetNode (const size_t i, double &x, double &y, double &z) const
 
double GetPotential (const size_t i) const
 
void EnableCheckMapIndices (const bool on=true)
 
void EnableDeleteBackgroundElements (const bool on=true)
 Option to eliminate mesh elements in conductors (default: on).
 
void EnableTetrahedralTreeForElementSearch (const bool on=true)
 
void EnableConvergenceWarnings (const bool on=true)
 
MediumGetMedium (const double x, const double y, const double z) override
 Get the medium at a given location (x, y, z).
 
void ElectricField (const double x, const double y, const double z, double &ex, double &ey, double &ez, Medium *&m, int &status) override
 
void ElectricField (const double x, const double y, const double z, double &ex, double &ey, double &ez, double &v, Medium *&m, int &status) override
 Calculate the drift field [V/cm] and potential [V] at (x, y, z).
 
void WeightingField (const double x, const double y, const double z, double &wx, double &wy, double &wz, const std::string &label) override
 
double WeightingPotential (const double x, const double y, const double z, const std::string &label) override
 
double DelayedWeightingPotential (double x, double y, double z, const double t, const std::string &label) override
 
bool IsInBoundingBox (const double x, const double y, const double z) const
 
bool GetBoundingBox (double &xmin, double &ymin, double &zmin, double &xmax, double &ymax, double &zmax) override
 Get the bounding box coordinates.
 
bool GetElementaryCell (double &xmin, double &ymin, double &zmin, double &xmax, double &ymax, double &zmax) override
 Get the coordinates of the elementary cell.
 
bool GetVoltageRange (double &vmin, double &vmax) override
 Calculate the voltage range [V].
 
void CopyWeightingPotential (const std::string &label, const std::string &labelSource, const double x, const double y, const double z, const double alpha, const double beta, const double gamma)
 
std::array< double, 3 > ElectricField (const double x, const double y, const double z)
 Calculate the drift field [V/cm] at (x, y, z).
 
- Public Member Functions inherited from Garfield::Component
 Component ()=delete
 Default constructor.
 
 Component (const std::string &name)
 Constructor.
 
virtual ~Component ()
 Destructor.
 
virtual void SetGeometry (Geometry *geo)
 Define the geometry.
 
virtual void Clear ()
 Reset.
 
std::array< double, 3 > ElectricField (const double x, const double y, const double z)
 Calculate the drift field [V/cm] at (x, y, z).
 
virtual double ElectricPotential (const double x, const double y, const double z)
 Calculate the (drift) electrostatic potential [V] at (x, y, z).
 
virtual void DelayedWeightingField (const double x, const double y, const double z, const double t, double &wx, double &wy, double &wz, const std::string &label)
 
virtual void MagneticField (const double x, const double y, const double z, double &bx, double &by, double &bz, int &status)
 
void SetMagneticField (const double bx, const double by, const double bz)
 Set a constant magnetic field.
 
virtual bool IsReady ()
 Ready for use?
 
double IntegrateFluxCircle (const double xc, const double yc, const double r, const unsigned int nI=50)
 
double IntegrateFluxSphere (const double xc, const double yc, const double zc, const double r, const unsigned int nI=20)
 
double IntegrateFluxParallelogram (const double x0, const double y0, const double z0, const double dx1, const double dy1, const double dz1, const double dx2, const double dy2, const double dz2, const unsigned int nU=20, const unsigned int nV=20)
 
double IntegrateWeightingFluxParallelogram (const std::string &label, const double x0, const double y0, const double z0, const double dx1, const double dy1, const double dz1, const double dx2, const double dy2, const double dz2, const unsigned int nU=20, const unsigned int nV=20)
 
double IntegrateFluxLine (const double x0, const double y0, const double z0, const double x1, const double y1, const double z1, const double xp, const double yp, const double zp, const unsigned int nI, const int isign=0)
 
virtual bool CrossedWire (const double x0, const double y0, const double z0, const double x1, const double y1, const double z1, double &xc, double &yc, double &zc, const bool centre, double &rc)
 
virtual bool InTrapRadius (const double q0, const double x0, const double y0, const double z0, double &xw, double &yw, double &rw)
 
virtual bool CrossedPlane (const double x0, const double y0, const double z0, const double x1, const double y1, const double z1, double &xc, double &yc, double &zc)
 
void EnablePeriodicityX (const bool on=true)
 Enable simple periodicity in the $x$ direction.
 
void EnablePeriodicityY (const bool on=true)
 Enable simple periodicity in the $y$ direction.
 
void EnablePeriodicityZ (const bool on=true)
 Enable simple periodicity in the $z$ direction.
 
void IsPeriodic (bool &perx, bool &pery, bool &perz)
 Return periodicity flags.
 
void EnableMirrorPeriodicityX (const bool on=true)
 Enable mirror periodicity in the $x$ direction.
 
void EnableMirrorPeriodicityY (const bool on=true)
 Enable mirror periodicity in the $y$ direction.
 
void EnableMirrorPeriodicityZ (const bool on=true)
 Enable mirror periodicity in the $y$ direction.
 
void IsMirrorPeriodic (bool &perx, bool &pery, bool &perz)
 Return mirror periodicity flags.
 
void EnableAxialPeriodicityX (const bool on=true)
 Enable axial periodicity in the $x$ direction.
 
void EnableAxialPeriodicityY (const bool on=true)
 Enable axial periodicity in the $y$ direction.
 
void EnableAxialPeriodicityZ (const bool on=true)
 Enable axial periodicity in the $z$ direction.
 
void IsAxiallyPeriodic (bool &perx, bool &pery, bool &perz)
 Return axial periodicity flags.
 
void EnableRotationSymmetryX (const bool on=true)
 Enable rotation symmetry around the $x$ axis.
 
void EnableRotationSymmetryY (const bool on=true)
 Enable rotation symmetry around the $y$ axis.
 
void EnableRotationSymmetryZ (const bool on=true)
 Enable rotation symmetry around the $z$ axis.
 
void IsRotationSymmetric (bool &rotx, bool &roty, bool &rotz)
 Return rotation symmetry flags.
 
void EnableDebugging ()
 Switch on debugging messages.
 
void DisableDebugging ()
 Switch off debugging messages.
 
virtual bool HasMagneticField () const
 Does the component have a non-zero magnetic field?
 
virtual bool HasTownsendMap () const
 Does the component have maps of the Townsend coefficient?
 
virtual bool HasAttachmentMap () const
 Does the component have attachment maps?
 
virtual bool HasVelocityMap () const
 Does the component have velocity maps?
 
virtual bool ElectronAttachment (const double, const double, const double, double &eta)
 Get the electron attachment coefficient.
 
virtual bool HoleAttachment (const double, const double, const double, double &eta)
 Get the hole attachment coefficient.
 
virtual bool ElectronTownsend (const double, const double, const double, double &alpha)
 Get the electron Townsend coefficient.
 
virtual bool HoleTownsend (const double, const double, const double, double &alpha)
 Get the hole Townsend coefficient.
 
virtual bool ElectronVelocity (const double, const double, const double, double &vx, double &vy, double &vz)
 Get the electron drift velocity.
 
virtual bool HoleVelocity (const double, const double, const double, double &vx, double &vy, double &vz)
 Get the hole drift velocity.
 
virtual bool GetElectronLifetime (const double, const double, const double, double &etau)
 
virtual bool GetHoleLifetime (const double, const double, const double, double &htau)
 
virtual double StepSizeHint ()
 

Additional Inherited Members

- Protected Types inherited from Garfield::ComponentFieldMap
enum class  ElementType { Unknown = 0 , Serendipity = 5 , CurvedTetrahedron = 13 }
 
- Protected Member Functions inherited from Garfield::ComponentFieldMap
void Reset () override
 Reset the component.
 
void Prepare ()
 
virtual void SetRange ()
 
void UpdatePeriodicity () override
 Verify periodicities.
 
void UpdatePeriodicity2d ()
 
void UpdatePeriodicityCommon ()
 
bool SetDefaultDriftMedium ()
 Find lowest epsilon, check for eps = 0, set default drift media flags.
 
int Field (const double x, const double y, const double z, double &fx, double &fy, double &fz, int &iel, const std::vector< double > &potentials) const
 Compute the electric/weighting field.
 
double Potential (const double x, const double y, const double z, const std::vector< double > &potentials) const
 Compute the electrostatic/weighting potential.
 
int FindElement5 (const double x, const double y, double &t1, double &t2, double &t3, double &t4, double jac[4][4], double &det) const
 Find the element for a point in curved quadratic quadrilaterals.
 
int FindElement13 (const double x, const double y, const double z, double &t1, double &t2, double &t3, double &t4, double jac[4][4], double &det) const
 Find the element for a point in curved quadratic tetrahedra.
 
int FindElementCube (const double x, const double y, const double z, double &t1, double &t2, double &t3, TMatrixD *&jac, std::vector< TMatrixD * > &dN) const
 Find the element for a point in a cube.
 
void MapCoordinates (double &xpos, double &ypos, double &zpos, bool &xmirrored, bool &ymirrored, bool &zmirrored, double &rcoordinate, double &rotation) const
 Move (xpos, ypos, zpos) to field map coordinates.
 
void UnmapFields (double &ex, double &ey, double &ez, const double xpos, const double ypos, const double zpos, const bool xmirrored, const bool ymirrored, const bool zmirrored, const double rcoordinate, const double rotation) const
 Move (ex, ey, ez) to global coordinates.
 
virtual double GetElementVolume (const size_t i) const
 
virtual void GetAspectRatio (const size_t i, double &dmin, double &dmax) const
 
void PrintWarning (const std::string &header)
 
void PrintNotReady (const std::string &header) const
 
void PrintCouldNotOpen (const std::string &header, const std::string &filename) const
 
void PrintElement (const std::string &header, const double x, const double y, const double z, const double t1, const double t2, const double t3, const double t4, const size_t i, const std::vector< double > &potential) const
 
void TimeInterpolation (const double t, double &f0, double &f1, int &i0, int &i1)
 Interpolation of potential between two time slices.
 
- Static Protected Member Functions inherited from Garfield::ComponentFieldMap
static double ScalingFactor (std::string unit)
 
static double Potential3 (const std::array< double, 6 > &v, const std::array< double, 3 > &t)
 Interpolate the potential in a triangle.
 
static void Field3 (const std::array< double, 6 > &v, const std::array< double, 3 > &t, double jac[4][4], const double det, double &ex, double &ey)
 Interpolate the field in a triangle.
 
static double Potential5 (const std::array< double, 8 > &v, const std::array< double, 2 > &t)
 Interpolate the potential in a curved quadrilateral.
 
static void Field5 (const std::array< double, 8 > &v, const std::array< double, 2 > &t, double jac[4][4], const double det, double &ex, double &ey)
 Interpolate the field in a curved quadrilateral.
 
static double Potential13 (const std::array< double, 10 > &v, const std::array< double, 4 > &t)
 Interpolate the potential in a curved quadratic tetrahedron.
 
static void Field13 (const std::array< double, 10 > &v, const std::array< double, 4 > &t, double jac[4][4], const double det, double &ex, double &ey, double &ez)
 Interpolate the field in a curved quadratic tetrahedron.
 
static int ReadInteger (char *token, int def, bool &error)
 
static double ReadDouble (char *token, double def, bool &error)
 
- Protected Attributes inherited from Garfield::ComponentFieldMap
bool m_is3d = true
 
ElementType m_elementType = ElementType::CurvedTetrahedron
 
std::vector< Elementm_elements
 
std::vector< int > m_elementIndices
 
std::vector< bool > m_degenerate
 
std::vector< std::array< double, 3 > > m_bbMin
 
std::vector< std::array< double, 3 > > m_bbMax
 
std::vector< std::array< std::array< double, 3 >, 4 > > m_w12
 
std::vector< Nodem_nodes
 
std::vector< double > m_pot
 
std::map< std::string, std::vector< double > > m_wpot
 
std::map< std::string, std::vector< std::vector< double > > > m_dwpot
 
std::vector< Materialm_materials
 
std::map< std::string, WeightingFieldCopym_wfieldCopies
 
std::vector< double > m_wdtimes
 
bool m_hasBoundingBox = false
 
std::array< double, 3 > m_minBoundingBox = {{0., 0., 0.}}
 
std::array< double, 3 > m_maxBoundingBox = {{0., 0., 0.}}
 
std::array< double, 3 > m_mapmin = {{0., 0., 0.}}
 
std::array< double, 3 > m_mapmax = {{0., 0., 0.}}
 
std::array< double, 3 > m_mapamin = {{0., 0., 0.}}
 
std::array< double, 3 > m_mapamax = {{0., 0., 0.}}
 
std::array< double, 3 > m_mapna = {{0., 0., 0.}}
 
std::array< double, 3 > m_cells = {{0., 0., 0.}}
 
double m_mapvmin = 0.
 
double m_mapvmax = 0.
 
std::array< bool, 3 > m_setang
 
bool m_deleteBackground = true
 
bool m_warning = false
 
unsigned int m_nWarnings = 0
 
bool m_printConvergenceWarnings = true
 
- Protected Attributes inherited from Garfield::Component
std::string m_className = "Component"
 Class name.
 
Geometrym_geometry = nullptr
 Pointer to the geometry.
 
std::array< double, 3 > m_b0 = {{0., 0., 0.}}
 Constant magnetic field.
 
bool m_ready = false
 Ready for use?
 
bool m_debug = false
 Switch on/off debugging messages.
 
std::array< bool, 3 > m_periodic = {{false, false, false}}
 Simple periodicity in x, y, z.
 
std::array< bool, 3 > m_mirrorPeriodic = {{false, false, false}}
 Mirror periodicity in x, y, z.
 
std::array< bool, 3 > m_axiallyPeriodic = {{false, false, false}}
 Axial periodicity in x, y, z.
 
std::array< bool, 3 > m_rotationSymmetric = {{false, false, false}}
 Rotation symmetry around x-axis, y-axis, z-axis.
 

Detailed Description

Component for importing and interpolating three-dimensional ANSYS field maps.

Definition at line 10 of file ComponentAnsys123.hh.

Constructor & Destructor Documentation

◆ ComponentAnsys123()

Garfield::ComponentAnsys123::ComponentAnsys123 ( )

Constructor.

Definition at line 10 of file ComponentAnsys123.cc.

10: ComponentFieldMap("Ansys123") {}
ComponentFieldMap()=delete
Default constructor.

◆ ~ComponentAnsys123()

Garfield::ComponentAnsys123::~ComponentAnsys123 ( )
inline

Destructor.

Definition at line 15 of file ComponentAnsys123.hh.

15{}

Member Function Documentation

◆ Initialise()

bool Garfield::ComponentAnsys123::Initialise ( const std::string & elist = "ELIST.lis",
const std::string & nlist = "NLIST.lis",
const std::string & mplist = "MPLIST.lis",
const std::string & prnsol = "PRNSOL.lis",
const std::string & unit = "cm" )

Import a field map.

Parameters
elistname of the file containing the list of elements
nlistname of the file containing the list of nodes
mplistname of the file containing the list of materials
prnsolname of the file containing the nodal solutions
unitlength unit

Definition at line 12 of file ComponentAnsys123.cc.

16 {
17 Reset();
18 // Keep track of the success.
19 bool ok = true;
20
21 // Buffer for reading
22 constexpr int size = 100;
23 char line[size];
24
25 // Open the material list.
26 std::ifstream fmplist(mplist);
27 if (!fmplist) {
28 PrintCouldNotOpen("Initialise", mplist);
29 return false;
30 }
31
32 // Read the material list.
33 long il = 0;
34 unsigned int icurrmat = 0;
35 bool readerror = false;
36 while (fmplist.getline(line, size, '\n')) {
37 il++;
38 // Skip page feed.
39 if (strcmp(line, "1") == 0) {
40 for (size_t k = 0; k < 5; ++k) fmplist.getline(line, size, '\n');
41 il += 5;
42 continue;
43 }
44 // Split the line in tokens.
45 char* token = strtok(line, " ");
46 // Skip blank lines and headers.
47 if (!token || strcmp(token, " ") == 0 || strcmp(token, "\n") == 0 ||
48 strcmp(token, "TEMPERATURE") == 0 || strcmp(token, "PROPERTY=") == 0 ||
49 int(token[0]) == 10 || int(token[0]) == 13) {
50 continue;
51 }
52 // Read number of materials and initialise the list.
53 if (strcmp(token, "LIST") == 0) {
54 token = strtok(nullptr, " ");
55 token = strtok(nullptr, " ");
56 token = strtok(nullptr, " ");
57 token = strtok(nullptr, " ");
58 const int nMaterials = ReadInteger(token, -1, readerror);
59 if (readerror || nMaterials < 0) {
60 std::cerr << m_className << "::Initialise:\n"
61 << " Error reading file " << mplist << " (line " << il
62 << ").\n";
63 fmplist.close();
64 return false;
65 }
66 m_materials.resize(nMaterials);
67 for (auto& material : m_materials) {
68 material.ohm = -1;
69 material.eps = -1;
70 material.medium = nullptr;
71 }
72 if (m_debug) {
73 std::cout << m_className << "::Initialise: " << nMaterials
74 << " materials.\n";
75 }
76 } else if (strcmp(token, "MATERIAL") == 0) {
77 // Version 12 format: read material number
78 token = strtok(nullptr, " ");
79 token = strtok(nullptr, " ");
80 const int imat = ReadInteger(token, -1, readerror);
81 if (readerror || imat < 0) {
82 std::cerr << m_className << "::Initialise:\n"
83 << " Error reading file " << mplist << " (line " << il
84 << ").\n";
85 fmplist.close();
86 return false;
87 }
88 icurrmat = imat;
89 } else if (strcmp(token, "TEMP") == 0) {
90 // Version 12 format: read property tag and value
91 token = strtok(nullptr, " ");
92 int itype = 0;
93 if (strncmp(token, "PERX", 4) == 0) {
94 itype = 1;
95 } else if (strncmp(token, "RSVX", 4) == 0) {
96 itype = 2;
97 } else {
98 std::cerr << m_className << "::Initialise:\n"
99 << " Unknown material property flag " << token << "\n"
100 << " in material properties file " << mplist << " (line "
101 << il << ").\n";
102 ok = false;
103 }
104 fmplist.getline(line, size, '\n');
105 il++;
106 token = nullptr;
107 token = strtok(line, " ");
108 if (icurrmat < 1 || icurrmat > m_materials.size()) {
109 std::cerr << m_className << "::Initialise:\n"
110 << " Found out-of-range current material index "
111 << icurrmat << "\n"
112 << " in material properties file " << mplist << ".\n";
113 ok = false;
114 readerror = false;
115 } else if (itype == 1) {
116 m_materials[icurrmat - 1].eps = ReadDouble(token, -1, readerror);
117 } else if (itype == 2) {
118 m_materials[icurrmat - 1].ohm = ReadDouble(token, -1, readerror);
119 }
120 if (readerror) {
121 std::cerr << m_className << "::Initialise:\n"
122 << " Error reading file " << mplist << " (line " << il
123 << ").\n";
124 fmplist.close();
125 return false;
126 }
127 } else if (strcmp(token, "PROPERTY") == 0) {
128 // Version 11 format
129 token = strtok(nullptr, " ");
130 token = strtok(nullptr, " ");
131 int itype = 0;
132 if (strcmp(token, "PERX") == 0) {
133 itype = 1;
134 } else if (strcmp(token, "RSVX") == 0) {
135 itype = 2;
136 } else {
137 std::cerr << m_className << "::Initialise:\n"
138 << " Unknown material property flag " << token << "\n"
139 << " in material properties file " << mplist << " (line "
140 << il << ").\n";
141 ok = false;
142 }
143 token = strtok(nullptr, " ");
144 token = strtok(nullptr, " ");
145 int imat = ReadInteger(token, -1, readerror);
146 if (readerror) {
147 std::cerr << m_className << "::Initialise:\n"
148 << " Error reading file " << mplist << " (line " << il
149 << ").\n";
150 fmplist.close();
151 return false;
152 } else if (imat < 1 || imat > (int)m_materials.size()) {
153 std::cerr << m_className << "::Initialise:\n"
154 << " Found out-of-range material index " << imat << "\n"
155 << " in material properties file " << mplist << ".\n";
156 ok = false;
157 } else {
158 fmplist.getline(line, size, '\n');
159 il++;
160 fmplist.getline(line, size, '\n');
161 il++;
162 token = nullptr;
163 token = strtok(line, " ");
164 token = strtok(nullptr, " ");
165 if (itype == 1) {
166 m_materials[imat - 1].eps = ReadDouble(token, -1, readerror);
167 } else if (itype == 2) {
168 m_materials[imat - 1].ohm = ReadDouble(token, -1, readerror);
169 }
170 if (readerror) {
171 std::cerr << m_className << "::Initialise:\n"
172 << " Error reading file " << mplist << " (line " << il
173 << ").\n";
174 fmplist.close();
175 return false;
176 }
177 }
178 }
179 }
180 // Close the file
181 fmplist.close();
182
183 // Find lowest epsilon, check for eps = 0, set default drift medium.
184 if (!SetDefaultDriftMedium()) ok = false;
185
186 // Tell how many lines read
187 std::cout << m_className << "::Initialise:\n"
188 << " Read properties of " << m_materials.size()
189 << " materials from file " << mplist << ".\n";
190 if (m_debug) PrintMaterials();
191
192 // Open the element list
193 std::ifstream felist(elist);
194 if (!felist) {
195 PrintCouldNotOpen("Initialise", elist);
196 return false;
197 }
198
199 // Read the element list.
200 int nbackground = 0;
201 il = 0;
202 int highestnode = 0;
203 while (felist.getline(line, size, '\n')) {
204 il++;
205 // Skip page feed.
206 if (strcmp(line, "1") == 0) {
207 for (size_t k = 0; k < 5; ++k) felist.getline(line, size, '\n');
208 il += 5;
209 continue;
210 }
211 // Skip page feed if Ansys > v15.x
212 if (strstr(line, "***") != nullptr) {
213 for (size_t k = 0; k < 3; ++k) felist.getline(line, size, '\n');
214 il += 3;
215 continue;
216 }
217 // Split the line in tokens.
218 char* token = strtok(line, " ");
219 // Skip blank lines and headers.
220 if (!token || strcmp(token, " ") == 0 || strcmp(token, "\n") == 0 ||
221 int(token[0]) == 10 || int(token[0]) == 13 ||
222 strcmp(token, "LIST") == 0 || strcmp(token, "ELEM") == 0) {
223 continue;
224 }
225 // Read the element.
226 int ielem = ReadInteger(token, -1, readerror);
227 token = strtok(nullptr, " ");
228 int imat = ReadInteger(token, -1, readerror);
229 token = strtok(nullptr, " ");
230 token = strtok(nullptr, " ");
231 token = strtok(nullptr, " ");
232 token = strtok(nullptr, " ");
233 std::vector<int> inode;
234 for (size_t k = 0; k < 8; ++k) {
235 token = strtok(nullptr, " ");
236 const int in = ReadInteger(token, -1, readerror);
237 if (!readerror) inode.push_back(in);
238 }
239 if (!felist.getline(line, size, '\n')) {
240 std::cerr << m_className << "::Initialise:\n"
241 << " Error reading element " << ielem << ".\n";
242 ok = false;
243 break;
244 }
245 ++il;
246 token = strtok(line, " ");
247 const int in8 = ReadInteger(token, -1, readerror);
248 if (!readerror) inode.push_back(in8);
249 token = strtok(nullptr, " ");
250 const int in9 = ReadInteger(token, -1, readerror);
251 if (!readerror) inode.push_back(in9);
252
253 if (inode.size() != 10) {
254 std::cerr << m_className << "::Initialise:\n"
255 << " Error reading file " << elist << " (line "
256 << il << ").\n"
257 << " Read " << inode.size() << " node indices for element "
258 << ielem << " (expected 10).\n";
259 ok = false;
260 break;
261 }
262 // Check synchronisation.
263 if (ielem - 1 != (int)m_elements.size() + nbackground) {
264 std::cerr << m_className << "::Initialise:\n"
265 << " Synchronisation lost on file " << elist << " (line "
266 << il << ").\n"
267 << " Element: " << ielem << " (expected "
268 << m_elements.size() << ").\n";
269 ok = false;
270 break;
271 }
272
273 // Check the material number and ensure that epsilon is non-negative
274 if (imat < 1 || imat > (int)m_materials.size()) {
275 std::cerr << m_className << "::Initialise:\n"
276 << " Out-of-range material number on file " << elist
277 << " (line " << il << ").\n"
278 << " Element: " << ielem << ", material: " << imat << ".\n";
279 ok = false;
280 break;
281 }
282 if (m_materials[imat - 1].eps < 0) {
283 std::cerr << m_className << "::Initialise:\n"
284 << " Element " << ielem << " in " << elist << "\n"
285 << " uses material " << imat << " which does not have\n"
286 << " a positive permittivity in " << mplist << ".\n";
287 ok = false;
288 break;
289 }
290
291 // Check the node numbers.
292 bool degenerate = false;
293 for (size_t k = 0; k < 10; ++k) {
294 if (inode[k] < 1) {
295 std::cerr << m_className << "::Initialise:\n"
296 << " Found a node number < 1 in " << elist << " (line "
297 << il << ").\n"
298 << " Element: " << ielem << ", material: " << imat << ".\n";
299 ok = false;
300 }
301 if (inode[k] > highestnode) highestnode = inode[k];
302 for (size_t kk = k + 1; kk < 10; ++kk) {
303 if (inode[k] == inode[kk]) degenerate = true;
304 }
305 }
306 if (!ok) break;
307
308 // Skip elements in conductors.
309 if (m_deleteBackground && m_materials[imat - 1].ohm == 0) {
310 nbackground++;
311 continue;
312 }
313
314 // These elements must not be degenerate.
315 if (degenerate) {
316 std::cerr << m_className << "::Initialise:\n"
317 << " Element " << ielem << " of file " << elist
318 << " is degenerate,\n"
319 << " no such elements allowed in this type of map.\n";
320 ok = false;
321 break;
322 }
323 Element element;
324 // Store the material reference.
325 element.matmap = imat - 1;
326 // Store the node references.
327 element.emap[0] = inode[0] - 1;
328 element.emap[1] = inode[1] - 1;
329 element.emap[2] = inode[2] - 1;
330 element.emap[3] = inode[3] - 1;
331 element.emap[4] = inode[4] - 1;
332 element.emap[7] = inode[5] - 1;
333 element.emap[5] = inode[6] - 1;
334 element.emap[6] = inode[7] - 1;
335 element.emap[8] = inode[8] - 1;
336 element.emap[9] = inode[9] - 1;
337 m_elements.push_back(std::move(element));
338 }
339 // Close the file
340 felist.close();
341 if (!ok) return false;
342
343 if (m_elements.empty()) {
344 std::cerr << m_className << "::Initialise:\n"
345 << " Found no valid elements in file " << elist << ".\n";
346 return false;
347 }
348 m_degenerate.assign(m_elements.size(), false);
349
350 // Tell how many lines read.
351 std::cout << " Read " << m_elements.size() << " elements from file "
352 << elist << ".\n"
353 << " Highest node number: " << highestnode << "\n"
354 << " Background elements skipped: " << nbackground << "\n";
355 // Check the value of the unit
356 double funit = ScalingFactor(unit);
357 if (funit <= 0.) {
358 std::cerr << m_className << "::Initialise:\n"
359 << " Unknown length unit " << unit << ". Will use cm.\n";
360 funit = 1.0;
361 }
362 if (m_debug) {
363 std::cout << m_className << ":Initialise: Unit scaling factor = "
364 << funit << ".\n";
365 }
366
367 // Open the node list
368 std::ifstream fnlist(nlist);
369 if (!fnlist) {
370 PrintCouldNotOpen("Initialise", nlist);
371 return false;
372 }
373
374 // Read the node list.
375 il = 0;
376 while (fnlist.getline(line, size, '\n')) {
377 il++;
378 // Skip page feed
379 if (strcmp(line, "1") == 0) {
380 for (size_t k = 0; k < 5; ++k) fnlist.getline(line, size, '\n');
381 il += 5;
382 continue;
383 }
384 // Skip page feed if Ansys > v15.x
385 if (strstr(line, "***") != nullptr) {
386 for (size_t k = 0; k < 3; ++k) fnlist.getline(line, size, '\n');
387 il += 3;
388 continue;
389 }
390 // Split the line in tokens-
391 char* token = strtok(line, " ");
392 // Skip blank lines and headers.
393 if (!token || strcmp(token, " ") == 0 || strcmp(token, "\n") == 0 ||
394 int(token[0]) == 10 || int(token[0]) == 13 ||
395 strcmp(token, "LIST") == 0 || strcmp(token, "NODE") == 0 ||
396 strcmp(token, "SORT") == 0) {
397 continue;
398 }
399 // Read the node.
400 int inode = ReadInteger(token, -1, readerror);
401 token = strtok(nullptr, " ");
402 double xnode = ReadDouble(token, -1, readerror);
403 token = strtok(nullptr, " ");
404 double ynode = ReadDouble(token, -1, readerror);
405 token = strtok(nullptr, " ");
406 double znode = ReadDouble(token, -1, readerror);
407 // Check syntax.
408 if (readerror) {
409 std::cerr << m_className << "::Initialise:\n"
410 << " Error reading file " << nlist << " (line " << il
411 << ").\n";
412 fnlist.close();
413 return false;
414 }
415 // Check synchronisation.
416 if (inode - 1 != (int)m_nodes.size()) {
417 std::cerr << m_className << "::Initialise:\n"
418 << " Synchronisation lost on file " << nlist << " (line "
419 << il << ").\n"
420 << " Node: " << inode << " (expected " << m_nodes.size()
421 << "), (x,y,z) = (" << xnode << ", " << ynode << ", " << znode
422 << ")\n";
423 ok = false;
424 break;
425 }
426 // Store the point coordinates
427 Node node;
428 node.x = xnode * funit;
429 node.y = ynode * funit;
430 node.z = znode * funit;
431 m_nodes.push_back(std::move(node));
432 }
433 // Close the file.
434 fnlist.close();
435 if (!ok) return false;
436
437 // Tell how many lines read.
438 std::cout << " Read " << m_nodes.size() << " nodes from file "
439 << nlist << ".\n";
440 // Check number of nodes
441 if ((int)m_nodes.size() != highestnode) {
442 std::cerr << m_className << "::Initialise:\n"
443 << " Number of nodes read (" << m_nodes.size() << ") on " << nlist
444 << "\n"
445 << " does not match element list (" << highestnode << ").\n";
446 return false;
447 }
448
449 if (!LoadPotentials(prnsol, m_pot)) return false;
450
451 m_ready = true;
452 Prepare();
453 return true;
454}
void PrintMaterials()
List all currently defined materials.
bool SetDefaultDriftMedium()
Find lowest epsilon, check for eps = 0, set default drift media flags.
static double ReadDouble(char *token, double def, bool &error)
static double ScalingFactor(std::string unit)
static int ReadInteger(char *token, int def, bool &error)
std::vector< Material > m_materials
void PrintCouldNotOpen(const std::string &header, const std::string &filename) const
std::vector< Element > m_elements
void Reset() override
Reset the component.
bool m_debug
Switch on/off debugging messages.
Definition Component.hh:371
std::string m_className
Class name.
Definition Component.hh:359
bool m_ready
Ready for use?
Definition Component.hh:368

◆ SetWeightingField()

bool Garfield::ComponentAnsys123::SetWeightingField ( const std::string & prnsol,
const std::string & label )

Definition at line 544 of file ComponentAnsys123.cc.

545 {
546 if (!m_ready) {
547 PrintNotReady("SetWeightingField");
548 std::cerr << " Weighting field cannot be added.\n";
549 return false;
550 }
551
552 std::cout << m_className << "::SetWeightingField:\n"
553 << " Loading field map for electrode " << label << ".\n";
554 // Check if a weighting field with the same label already exists.
555 if (m_wpot.count(label) > 0) {
556 std::cout << " Replacing existing weighting field.\n";
557 m_wpot[label].clear();
558 }
559
560 std::vector<double> pot(m_nodes.size(), 0.);
561 if (!LoadPotentials(prnsol, pot)) return false;
562 m_wpot[label] = std::move(pot);
563 return true;
564}
std::map< std::string, std::vector< double > > m_wpot
void PrintNotReady(const std::string &header) const

Referenced by SetWeightingPotential().

◆ SetWeightingPotential()

bool Garfield::ComponentAnsys123::SetWeightingPotential ( const std::string & prnsol,
const std::string & label )
inline

Import weighting potentials.

Definition at line 31 of file ComponentAnsys123.hh.

32 {
33 return SetWeightingField(prnsol, label);
34 }
bool SetWeightingField(const std::string &prnsol, const std::string &label)

The documentation for this class was generated from the following files: