8#include <dg/file/json_utilities.h>
76 std::string e = gs.get(
"equilibrium",
"solovev" ).asString();
79 equi = str2equilibrium.at( e);
80 }
catch ( std::out_of_range& err)
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);
95 double R0 = gs.get(
"R_0", 10.0).asDouble();
100 double I0 = gs.get(
"I_0", 20.0).asDouble();
101 double R0 = gs.get(
"R_0", 10.0).asDouble();
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();
133inline void transform_psi( TokamakMagneticField mag,
double& psi0,
double& alpha0,
double& sign0)
135 double RO=mag.
R0(), ZO=0.;
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));
145inline void createModifiedField(
147 dg::file::WrappedJsonValue jsmod,
149 CylindricalFunctorsLvl2& mod_psip,
150 CylindricalFunctor& wall, CylindricalFunctor& transition)
152 std::string m = jsmod.get(
"type",
"heaviside" ).asString();
153 mod = modifier::heaviside;
156 mod = str2modifier.at( m);
157 }
catch ( std::out_of_range& err)
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);
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)
168 if ( desc == description::standardX or desc == description::doubleX)
175 if ( desc == description::doubleX)
185 wall = mod::DampingRegion( mod::nowhere, mag.
psip(), 0, 1, -1);
186 transition = mod::MagneticTransition( mod::nowhere, mag.
psip(),
191 case modifier::heaviside:
193 double psi0 = jsmod.get(
"boundary", 1.1 ).asDouble();
194 double alpha = jsmod.get(
"alpha", 0.2 ).asDouble();
196 if( desc == description::standardX || desc == description::standardO ||
197 desc == description::doubleX)
198 detail::transform_psi( mag, psi0, alpha, sign);
200 sign = jsmod.get(
"sign", -1. ).asDouble();
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);
207 case modifier::sol_pfr:
209 double psi0 = jsmod[
"boundary"].get( 0, 1.1 ).asDouble();
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();
215 case description::standardX:
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);
227 mod::everywhere, mag.
psip(), psi0, alpha0, -sign0);
229 mod::everywhere, mag.
psip(), psi0, alpha0, sign0);
231 mod::HeavisideZ(ZX1, -1), mag.
psip(), psi1, alpha1, -sign1);
233 mod::HeavisideZ(ZX1, -1), mag.
psip(), psi1, alpha1, sign1);
234 wall = mod::SetUnion( wall0, wall1);
235 transition = mod::SetUnion( transition0, transition1);
238 case description::doubleX:
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);
251 mod::everywhere, mag.
psip(), psi0, alpha0, -sign0);
253 mod::HeavisideZ(ZX1, -1), mag.
psip(), psi1, alpha1, -sign1);
255 mod::HeavisideZ(ZX2, +1), mag.
psip(), psi1, alpha1, -sign1);
257 mod::everywhere, mag.
psip(), psi0, alpha0, sign0);
259 mod::HeavisideZ(ZX1, -1), mag.
psip(), psi1, alpha1, sign1);
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);
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);
276 mod::everywhere, mag.
psip(), psi0, alpha0, sign0);
278 mod::everywhere, mag.
psip(), psi0, alpha0, sign0);
280 mod::everywhere, mag.
psip(), psi1, alpha1, sign1);
282 mod::everywhere, mag.
psip(), psi1, alpha1, sign1);
283 wall = mod::SetUnion( wall0, wall1);
284 transition = mod::SetUnion( transition0, transition1);
290 case modifier::sol_pfr_2X:
292 if( desc != description::doubleX)
293 throw Error( Message(_ping_) <<
"Description must be doubleX");
295 std::vector<double> psi(num), alpha(num);
296 for(
unsigned u=0; u<num; u++)
298 psi[u] = jsmod[
"boundary"].get( u, 1.0) .asDouble();
299 alpha[u] = jsmod[
"alpha"].get( u, 0.1) .asDouble();
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]);
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),
308 mod::RightSideOf( {RX2,ZX2},{RO,ZO},{RX1,ZX1}),
309 mod::Above( {RX2, ZX2}, {RO, ZO},
false) };
311 CylindricalFunctorsLvl2 mod_psips[num];
312 mod_psips[0] = mod::createPsip( mods[0], mag.
get_psip(), psi[0],
314 for(
unsigned u=1; u<num; u++)
315 mod_psips[u] = mod::createPsip( mods[u], mod_psips[u-1], psi[u],
317 mod_psip = mod_psips[num-1];
320 for(
unsigned u=0; u<4; u++)
322 walls[u] = mod::DampingRegion( mods[u], mag.
psip(), psi[u],
324 transitions[u] = mod::MagneticTransition( mods[u], mag.
psip(),
325 psi[u], alpha[u], sign[u]);
327 transition = mod::SetUnion( transitions[0], transitions[1]);
328 wall = mod::SetUnion( walls[0], walls[1]);
329 for(
unsigned u=2; u<4; u++)
331 transition = mod::SetUnion( transition, transitions[u]);
332 wall = mod::SetUnion( wall, walls[u]);
423 dg::file::WrappedJsonValue gs, dg::file::WrappedJsonValue jsmod,
431 detail::createModifiedField( mag, jsmod, mod, mod_psip, wall, transition);
456 dg::file::WrappedJsonValue jsmod)
461 detail::createModifiedField( mag, jsmod, mod, mod_psip, wall, transition);
466 dg::file::WrappedJsonValue jsmod)
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++)
549 if( wall( R0+i*gR1d.
h(), Z0) == 0)
551 if( wall( R0+i*gR1d.
h(), Z1) == 0)
553 if( wall( R0, Z0 + i*gZ1d.
h()) == 0)
555 if( wall( R1, Z0 + i*gZ1d.
h()) == 0)
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();
564 double alpha = jsmod.get(
"alpha", 0.015625 ).asDouble();
566 mag), sheath_walls, (-boundary-1e-3)*2.0*
M_PI,
569 mag), sheath_walls, (+boundary+1e-3)*2.0*
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
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