Extension: Geometries
#include "dg/geometries/geometries.h"
Loading...
Searching...
No Matches
make_field.h
Go to the documentation of this file.
1#include "magnetic_field.h"
2#include "solovev.h"
3#include "guenter.h"
4#include "polynomial.h"
5#include "toroidal.h"
6#include "sheath.h"
7#include "modified.h"
8#include <dg/file/json_utilities.h>
9
14namespace dg{
15namespace geo{
16
74inline TokamakMagneticField createMagneticField( dg::file::WrappedJsonValue gs)
75{
76 std::string e = gs.get( "equilibrium", "solovev" ).asString();
78 try{
79 equi = str2equilibrium.at( e);
80 }catch ( std::out_of_range& err)
81 {
82 std::string message = "ERROR: Key \"" + e
83 + "\" not valid in field:\n\t"
84 + gs.access_string() + "\"equilibrium\" \n";
85 throw std::out_of_range(message);
86 }
87 switch( equi){
89 {
91 return createPolynomialField( gp);
92 }
94 {
95 double R0 = gs.get( "R_0", 10.0).asDouble();
96 return createToroidalField( R0);
97 }
99 {
100 double I0 = gs.get( "I_0", 20.0).asDouble();
101 double R0 = gs.get( "R_0", 10.0).asDouble();
102 return createGuenterField( R0, I0);
103 }
105 {
106 double I0 = gs.get( "I_0", 20.0).asDouble();
107 double R0 = gs.get( "R_0", 10.0).asDouble();
108 double a = gs.get( "a", 1.0).asDouble();
109 double b = gs.get( "b", 1.0).asDouble();
110 return createCircularField( R0, I0, a, b);
111 }
112#ifdef BOOST_VERSION
114 {
115 solovev::Parameters gp( gs);
116 return createTaylorField( gp);
117 }
118#endif
119 default:
120 {
121 solovev::Parameters gp( gs);
122 return createSolovevField( gp);
123 }
124 }
125}
126
127
128//TODO The place where these functions are somewhat tested is geometry_diag.cpp Formalise?
129
131namespace detail{
132
133inline void transform_psi( TokamakMagneticField mag, double& psi0, double& alpha0, double& sign0)
134{
135 double RO=mag.R0(), ZO=0.;
136 dg::geo::findOpoint( mag.get_psip(), RO, ZO);
137 double psipO = mag.psip()( RO, ZO);
138 double wall_psi0p = (1.-psi0*psi0)*psipO;
139 double wall_alpha0p = -(2.*psi0+alpha0)*alpha0*psipO;
140 psi0 = wall_psi0p + sign0*wall_alpha0p/2.;
141 alpha0 = fabs( wall_alpha0p/2.);
142 sign0 = sign0*((psipO>0)-(psipO<0));
143}
144
145inline void createModifiedField(
147 dg::file::WrappedJsonValue jsmod,
148 modifier& mod,
149 CylindricalFunctorsLvl2& mod_psip,
150 CylindricalFunctor& wall, CylindricalFunctor& transition)
151{
152 std::string m = jsmod.get( "type", "heaviside" ).asString();
153 mod = modifier::heaviside;
154 description desc = mag.params().getDescription();
155 try{
156 mod = str2modifier.at( m);
157 }catch ( std::out_of_range& err)
158 {
159 std::string message = "ERROR: Key \"" + m
160 + "\" not valid in field:\n\t"
161 + jsmod.access_string() + "\"type\" \n";
162 throw std::out_of_range(message);
163 }
164 double RX1 = 0., ZX1 = 0., RX2 = 0., ZX2 = 0.;
165 double RO=mag.R0(), ZO=0.;
166 if( desc != description::none and desc != description::centeredX)
167 dg::geo::findOpoint( mag.get_psip(), RO, ZO);
168 if ( desc == description::standardX or desc == description::doubleX)
169 {
170 // Find first X-point
171 RX1 = mag.R0()-1.1*mag.params().triangularity()*mag.params().a();
172 ZX1 = -1.1*mag.params().elongation()*mag.params().a();
173 dg::geo::findXpoint( mag.get_psip(), RX1, ZX1);
174 }
175 if ( desc == description::doubleX)
176 {
177 // Find second X-point
178 RX2 = mag.R0()-1.1*mag.params().triangularity()*mag.params().a();
179 ZX2 = +1.1*mag.params().elongation()*mag.params().a();
180 dg::geo::findXpoint( mag.get_psip(), RX2, ZX2);
181 }
182 switch (mod) {
183 default: //none
184 {
185 wall = mod::DampingRegion( mod::nowhere, mag.psip(), 0, 1, -1);
186 transition = mod::MagneticTransition( mod::nowhere, mag.psip(),
187 0, 1, -1);
188 mod_psip = mag.get_psip();
189 break;
190 }
191 case modifier::heaviside:
192 {
193 double psi0 = jsmod.get( "boundary", 1.1 ).asDouble();
194 double alpha = jsmod.get( "alpha", 0.2 ).asDouble();
195 double sign = +1;
196 if( desc == description::standardX || desc == description::standardO ||
197 desc == description::doubleX)
198 detail::transform_psi( mag, psi0, alpha, sign);
199 else
200 sign = jsmod.get( "sign", -1. ).asDouble();
201
202 mod_psip = mod::createPsip( mod::everywhere, mag.get_psip(), psi0, alpha, sign);
203 wall = mod::DampingRegion( mod::everywhere, mag.psip(), psi0, alpha, -sign);
204 transition = mod::MagneticTransition( mod::everywhere, mag.psip(), psi0, alpha, sign);
205 break;
206 }
207 case modifier::sol_pfr:
208 {
209 double psi0 = jsmod["boundary"].get( 0, 1.1 ).asDouble(); //given in rho_p
210 double alpha0 = jsmod["alpha"].get(0, 0.2 ).asDouble();
211 double psi1 = jsmod["boundary"].get(1, 0.97 ).asDouble();
212 double alpha1 = jsmod[ "alpha"].get(1, 0.2 ).asDouble();
213 switch( desc){
214 // TODO Maybe this can be unified with modifier::sol_pfr_2X
215 case description::standardX:
216 {
217 double sign0 = +1., sign1 = -1.;
218 detail::transform_psi( mag, psi0, alpha0, sign0);
219 detail::transform_psi( mag, psi1, alpha1, sign1);
220 CylindricalFunctorsLvl2 mod0_psip;
221 mod0_psip = mod::createPsip(
222 mod::everywhere, mag.get_psip(), psi0, alpha0, sign0);
223 mod_psip = mod::createPsip(
224 mod::HeavisideZ( ZX1, -1), mod0_psip, psi1, alpha1, sign1);
225
226 CylindricalFunctor wall0 = mod::DampingRegion(
227 mod::everywhere, mag.psip(), psi0, alpha0, -sign0);
228 CylindricalFunctor transition0 = mod::MagneticTransition(
229 mod::everywhere, mag.psip(), psi0, alpha0, sign0);
230 CylindricalFunctor wall1 = mod::DampingRegion(
231 mod::HeavisideZ(ZX1, -1), mag.psip(), psi1, alpha1, -sign1);
232 CylindricalFunctor transition1 = mod::MagneticTransition(
233 mod::HeavisideZ(ZX1, -1), mag.psip(), psi1, alpha1, sign1);
234 wall = mod::SetUnion( wall0, wall1);
235 transition = mod::SetUnion( transition0, transition1);
236 break;
237 }
238 case description::doubleX:
239 {
240 double sign0 = +1., sign1 = -1.;
241 detail::transform_psi( mag, psi0, alpha0, sign0);
242 detail::transform_psi( mag, psi1, alpha1, sign1);
243 CylindricalFunctorsLvl2 mod0_psip, mod1_psip;
244 mod0_psip = mod::createPsip(
245 mod::everywhere, mag.get_psip(), psi0, alpha0, sign0);
246 mod1_psip = mod::createPsip(
247 mod::HeavisideZ( ZX1, -1), mod0_psip, psi1, alpha1, sign1);
248 mod_psip = mod::createPsip(
249 mod::HeavisideZ( ZX2, +1), mod1_psip, psi1, alpha1, sign1);
250 CylindricalFunctor wall0 = mod::DampingRegion(
251 mod::everywhere, mag.psip(), psi0, alpha0, -sign0);
252 CylindricalFunctor wall1 = mod::DampingRegion(
253 mod::HeavisideZ(ZX1, -1), mag.psip(), psi1, alpha1, -sign1);
254 CylindricalFunctor wall2 = mod::DampingRegion(
255 mod::HeavisideZ(ZX2, +1), mag.psip(), psi1, alpha1, -sign1);
256 CylindricalFunctor transition0 = mod::MagneticTransition(
257 mod::everywhere, mag.psip(), psi0, alpha0, sign0);
258 CylindricalFunctor transition1 = mod::MagneticTransition(
259 mod::HeavisideZ(ZX1, -1), mag.psip(), psi1, alpha1, sign1);
260 CylindricalFunctor transition2 = mod::MagneticTransition(
261 mod::HeavisideZ(ZX2, +1), mag.psip(), psi1, alpha1, sign1);
262 transition = mod::SetUnion( mod::SetUnion( transition0, transition1), transition2);
263 wall = mod::SetUnion( mod::SetUnion( wall0, wall1), wall2);
264 break;
265 }
266 default:
267 {
268 double sign0 = jsmod[ "sign"].get( 0, -1. ).asDouble();
269 double sign1 = jsmod[ "sign"].get( 1, +1. ).asDouble();
270 CylindricalFunctorsLvl2 mod0_psip;
271 mod0_psip = mod::createPsip(
272 mod::everywhere, mag.get_psip(), psi0, alpha0, sign0);
273 mod_psip = mod::createPsip(
274 mod::everywhere, mod0_psip, psi1, alpha1, sign1);
275 CylindricalFunctor wall0 = mod::DampingRegion(
276 mod::everywhere, mag.psip(), psi0, alpha0, sign0);
277 CylindricalFunctor transition0 = mod::MagneticTransition(
278 mod::everywhere, mag.psip(), psi0, alpha0, sign0);
279 CylindricalFunctor wall1 = mod::DampingRegion(
280 mod::everywhere, mag.psip(), psi1, alpha1, sign1);
281 CylindricalFunctor transition1 = mod::MagneticTransition(
282 mod::everywhere, mag.psip(), psi1, alpha1, sign1);
283 wall = mod::SetUnion( wall0, wall1);
284 transition = mod::SetUnion( transition0, transition1);
285 break;
286 }
287 }
288 break;
289 }
290 case modifier::sol_pfr_2X:
291 {
292 if( desc != description::doubleX)
293 throw Error( Message(_ping_) << "Description must be doubleX");
294 unsigned num = 4;
295 std::vector<double> psi(num), alpha(num);
296 for( unsigned u=0; u<num; u++)
297 {
298 psi[u] = jsmod["boundary"].get( u, 1.0) .asDouble();
299 alpha[u] = jsmod["alpha"].get( u, 0.1) .asDouble();
300 }
301 std::vector<double> sign = {+1,-1.,+1.,-1};
302 for( unsigned u=0; u<num; u++)
303 detail::transform_psi( mag, psi[u], alpha[u], sign[u]);
304
305 std::vector<std::function< bool( double,double)>> mods = {
306 mod::RightSideOf( {RX1,ZX1},{RO,ZO},{RX2,ZX2}),
307 mod::Above( {RX1, ZX1}, {RO,ZO}, false), // = Below
308 mod::RightSideOf( {RX2,ZX2},{RO,ZO},{RX1,ZX1}),
309 mod::Above( {RX2, ZX2}, {RO, ZO}, false) };
310
311 CylindricalFunctorsLvl2 mod_psips[num];
312 mod_psips[0] = mod::createPsip( mods[0], mag.get_psip(), psi[0],
313 alpha[0], sign[0]);
314 for( unsigned u=1; u<num; u++)
315 mod_psips[u] = mod::createPsip( mods[u], mod_psips[u-1], psi[u],
316 alpha[u], sign[u]);
317 mod_psip = mod_psips[num-1];
318
319 CylindricalFunctor walls[num], transitions[num];
320 for( unsigned u=0; u<4; u++)
321 {
322 walls[u] = mod::DampingRegion( mods[u], mag.psip(), psi[u],
323 alpha[u], -sign[u]);
324 transitions[u] = mod::MagneticTransition( mods[u], mag.psip(),
325 psi[u], alpha[u], sign[u]);
326 }
327 transition = mod::SetUnion( transitions[0], transitions[1]);
328 wall = mod::SetUnion( walls[0], walls[1]);
329 for( unsigned u=2; u<4; u++)
330 {
331 transition = mod::SetUnion( transition, transitions[u]);
332 wall = mod::SetUnion( wall, walls[u]);
333 }
334 break;
335 }
336 }
337}
338}//namespace detail
340
343
423 dg::file::WrappedJsonValue gs, dg::file::WrappedJsonValue jsmod,
424 CylindricalFunctor& wall, CylindricalFunctor& transition)
425{
428 const MagneticFieldParameters& inp = mag.params();
429
430 modifier mod;
431 detail::createModifiedField( mag, jsmod, mod, mod_psip, wall, transition);
432 description desc = inp.getDescription();
433 equilibrium equi = inp.getEquilibrium();
434 MagneticFieldParameters mod_params{ inp.a(), inp.elongation(),
435 inp.triangularity(), equi, mod, desc};
436 switch( equi)
437 {
439 {
440 solovev::Parameters gp( gs);
441 return TokamakMagneticField( gp.R_0,mod_psip,
442 solovev::createIpol( gp, mod_psip), mod_params);
443 }
444 default:
445 {
446 return TokamakMagneticField( mag.R0(), mod_psip,
447 mag.get_ipol(), mod_params);
448 }
449 }
450}
451
452
456 dg::file::WrappedJsonValue jsmod)
457{
458 CylindricalFunctor wall, transition;
460 modifier mod;
461 detail::createModifiedField( mag, jsmod, mod, mod_psip, wall, transition);
462 return wall;
463}
465inline CylindricalFunctor createWallRegion( dg::file::WrappedJsonValue gs,
466 dg::file::WrappedJsonValue jsmod)
467{
468 return createWallRegion( createMagneticField(gs), jsmod);
469}
470
471
538 dg::file::WrappedJsonValue jsmod, TokamakMagneticField mag,
539 CylindricalFunctor wall, dg::Grid2d& sheath_walls,
540 CylindricalFunctor& sheath)
541{
542 double R0 = sheath_walls.x0(), R1 = sheath_walls.x1();
543 double Z0 = sheath_walls.y0(), Z1 = sheath_walls.y1();
544 Grid1d gR1d ( R0, R1, 1, 100);
545 Grid1d gZ1d ( Z0, Z1, 1, 100);
546 std::array<bool,2> sheathR = {false,false}, sheathZ = {false,false};
547 for ( unsigned i=0; i<100; i++)
548 {
549 if( wall( R0+i*gR1d.h(), Z0) == 0)
550 sheathR[0] = true;
551 if( wall( R0+i*gR1d.h(), Z1) == 0)
552 sheathR[1] = true;
553 if( wall( R0, Z0 + i*gZ1d.h()) == 0)
554 sheathZ[0] = true;
555 if( wall( R1, Z0 + i*gZ1d.h()) == 0)
556 sheathZ[1] = true;
557 }
558 if( false == sheathR[0]) Z0 = -1e10;
559 if( false == sheathR[1]) Z1 = 1e10;
560 if( false == sheathZ[0]) R0 = -1e10;
561 if( false == sheathZ[1]) R1 = 1e10;
562 sheath_walls = dg::Grid2d( R0, R1, Z0, Z1, 1,1,1);
563 double boundary = jsmod.get( "boundary", 0.0625 ).asDouble(); // 1/16
564 double alpha = jsmod.get( "alpha", 0.015625 ).asDouble(); // 1/64
566 mag), sheath_walls, (-boundary-1e-3)*2.0*M_PI,
567 1e-6, "phi", mod::SOLRegion(mag,wall));
569 mag), sheath_walls, (+boundary+1e-3)*2.0*M_PI,
570 1e-6, "phi", mod::SOLRegion(mag,wall));
571 dg::PolynomialHeaviside polyM( -boundary*2.*M_PI + alpha*M_PI, alpha*M_PI, +1);
572 dg::PolynomialHeaviside polyP( boundary*2.*M_PI - alpha*M_PI, alpha*M_PI, -1);
573 auto sheathM = dg::compose( polyM, distM); //positive (because distance)
574 auto sheathP = dg::compose( polyP, distP);
575 sheath = mod::SetUnion( sheathM, sheathP);
576 sheath = mod::SetIntersection( mod::SetNot( wall), sheath);
577}
579
580} //namespace geo
581}//namespace dg
#define M_PI
dg::geo::TokamakMagneticField createCircularField(double R0, double I0, double a=1, double b=1)
Definition toroidal.h:136
auto compose(UnaryOp f, Functor g)
RealCylindricalFunctor< double > CylindricalFunctor
Most of the times we use double.
Definition fluxfunctions.h:50
TokamakMagneticField createMagneticField(dg::file::WrappedJsonValue gs)
Create a Magnetic field based on the given parameters.
Definition make_field.h:74
dg::RealGrid< double, 2 > Grid2d
dg::geo::TokamakMagneticField createGuenterField(double R_0, double I_0)
Create a Guenter Magnetic field.
Definition guenter.h:158
modifier
How flux-function is modified.
Definition magnetic_field.h:36
equilibrium
How flux-function is computed. Decides how to construct magnetic field.
Definition magnetic_field.h:26
description
How flux function looks like. Decider on whether and what flux aligned grid to construct.
Definition magnetic_field.h:50
CylindricalVectorLvl1 createBHat(const TokamakMagneticField &mag)
Contravariant components of the magnetic unit vector field and its Divergence and derivative in cylin...
Definition magnetic_field.h:934
@ solovev
dg::geo::solovev::Psip
@ taylor
dg::geo::taylor::Psip
@ polynomial
dg::geo::polynomial::Psip
@ guenter
dg::geo::guenter::Psip
@ toroidal
dg::geo::createToroidalField
@ circular
dg::geo::circular::Psip
void findXpoint(const CylindricalFunctorsLvl2 &psi, double &RC, double &ZC)
This function finds X-points of psi.
Definition fluxfunctions.h:354
int findOpoint(const CylindricalFunctorsLvl2 &psi, double &RC, double &ZC)
This function finds O-points of psi.
Definition fluxfunctions.h:336
dg::geo::TokamakMagneticField createPolynomialField(dg::geo::polynomial::Parameters gp)
Create a Polynomial Magnetic field.
Definition polynomial.h:175
dg::geo::CylindricalFunctorsLvl1 createIpol(const Parameters &gp, const CylindricalFunctorsLvl1 &psip)
Definition solovev.h:354
dg::geo::TokamakMagneticField createSolovevField(dg::geo::solovev::Parameters gp)
Create a Solovev Magnetic field.
Definition solovev.h:375
dg::geo::TokamakMagneticField createTaylorField(dg::geo::solovev::Parameters gp)
Create a Taylor Magnetic field.
Definition taylor.h:314
dg::geo::TokamakMagneticField createToroidalField(double R0)
Create a Toroidal Magnetic field.
Definition toroidal.h:118
void createSheathRegion(dg::file::WrappedJsonValue jsmod, TokamakMagneticField mag, CylindricalFunctor wall, dg::Grid2d &sheath_walls, CylindricalFunctor &sheath)
Create the sheath region where fieldlines intersect the boundary.
Definition make_field.h:537
CylindricalFunctor createWallRegion(dg::geo::TokamakMagneticField mag, dg::file::WrappedJsonValue jsmod)
A convenience function call for dg::geo::createModifiedField that ignores the transition parameter an...
Definition make_field.h:455
TokamakMagneticField createModifiedField(dg::file::WrappedJsonValue gs, dg::file::WrappedJsonValue jsmod, CylindricalFunctor &wall, CylindricalFunctor &transition)
Modify Magnetic Field and create wall above or below certain Psi values according to given parameters...
Definition make_field.h:422
real_type h(unsigned u=0) const
real_type x0() const
real_type y0() const
real_type y1() const
real_type x1() const
This struct bundles a function and its first and second derivatives.
Definition fluxfunctions.h:223
Meta-data about the magnetic field in particular the flux function.
Definition magnetic_field.h:94
double a() const
The minor radius.
Definition magnetic_field.h:125
description getDescription() const
how the flux function looks
Definition magnetic_field.h:143
double elongation() const
Definition magnetic_field.h:131
double triangularity() const
Definition magnetic_field.h:137
equilibrium getEquilibrium() const
the way the flux function is computed
Definition magnetic_field.h:139
A tokamak field as given by R0, Psi and Ipol plus Meta-data like shape and equilibrium.
Definition magnetic_field.h:165
const CylindricalFunctorsLvl2 & get_psip() const
Definition magnetic_field.h:200
const CylindricalFunctorsLvl1 & get_ipol() const
Definition magnetic_field.h:201
const MagneticFieldParameters & params() const
Access Meta-data of the field.
Definition magnetic_field.h:207
double R0() const
Definition magnetic_field.h:180
const CylindricalFunctor & psip() const
, where R, Z are cylindrical coordinates
Definition magnetic_field.h:182
Distance to wall along fieldline in phi or s coordinate
Definition sheath.h:32
The default predicate for sheath integration.
Definition modified.h:425
Definition modified.h:497
Definition modified.h:517
Definition modified.h:477
Constructs and display geometric parameters for the polynomial fields.
Definition polynomial_parameters.h:43
Constructs and display geometric parameters for the solovev and taylor fields.
Definition solovev_parameters.h:44
double R_0
major tokamak radius
Definition solovev_parameters.h:46