Extension: Geometries
#include "dg/geometries/geometries.h"
Loading...
Searching...
No Matches
sheath.h
Go to the documentation of this file.
1#pragma once
2
3#include <cmath>
4#include <array>
5
6#include "dg/algorithm.h"
7#include "fieldaligned.h"
8#include "modified.h"
9#include "magnetic_field.h"
10#include "fluxfunctions.h"
11
12namespace dg{
13namespace geo{
14
15
31struct WallFieldlineDistance : public aCylindricalFunctor<WallFieldlineDistance>
32{
46 const dg::aRealTopology2d<double>& domain,
47 double maxPhi, double eps, std::string type,
48 std::function<bool(double,double)> predicate = mod::everywhere):
49 m_pred( predicate),
50 m_domain( domain), m_cyl_field(vec),
51 m_deltaPhi( maxPhi), m_eps( eps), m_type(type)
52 {
53 if( m_type != "phi" && m_type != "s")
54 throw std::runtime_error( "Distance type "+m_type+" not recognized!\n");
55 }
66 double do_compute( double R, double Z) const
67 {
68 std::array<double,3> coords{ R, Z, 0}, coordsP(coords);
69 // determine sign
70 m_cyl_field( 0., coords, coordsP);
71 double sign = coordsP[2] > 0 ? +1. : -1.;
72 double phi1 = sign*m_deltaPhi; // we integrate negative ...
73 if( m_pred(R,Z)) // only integrate if necessary
74 {
75 try{
77 "Dormand-Prince-7-4-5", coords);
79 m_cyl_field, dg::pid_control, dg::fast_l2norm, m_eps,
80 1e-10); // using 1e-10 instead of eps may cost 10% of performance but is what we also use in Fieldaligned
81 odeint.integrate_in_domain( 0., coords, phi1, coordsP, 0.,
82 m_domain, m_eps);
83 //integration
84 }catch (std::exception& e)
85 {
86 // if not possible the distance is large
87 //std::cerr << e.what();
88 phi1 = sign*m_deltaPhi;
89 coordsP[2] = 1e6*phi1;
90 }
91 }
92 else
93 coordsP[2] = 1e6*phi1;
94 if( m_type == "phi")
95 return sign*phi1;
96 return coordsP[2];
97 }
98
99 private:
100 std::function<bool(double,double)> m_pred;
101 const dg::Grid2d m_domain;
102 dg::geo::detail::DSFieldCylindrical3 m_cyl_field;
103 double m_deltaPhi, m_eps;
104 std::string m_type;
105};
106
127struct WallFieldlineCoordinate : public aCylindricalFunctor<WallFieldlineCoordinate>
128{
133 const dg::aRealTopology2d<double>& domain,
134 double maxPhi, double eps, std::string type,
135 std::function<bool(double,double)> predicate = mod::everywhere
136 ) :
137 m_pred(predicate),
138 m_domain( domain), m_cyl_field(vec),
139 m_deltaPhi( maxPhi), m_eps( eps), m_type(type)
140 {
141 if( m_type != "phi" && m_type != "s")
142 throw std::runtime_error( "Distance type "+m_type+" not recognized!\n");
143 }
144 double do_compute( double R, double Z) const
145 {
146 double phiP = m_deltaPhi, phiM = -m_deltaPhi;
147 std::array<double,3> coords{ R, Z, 0}, coordsP(coords), coordsM(coords);
148 // determine sign
149 m_cyl_field( 0., coords, coordsP);
150 double sign = coordsP[2] > 0 ? +1. : -1.;
151 if( m_pred(R,Z)) // only integrate if necessary
152 {
153 try{
155 dg::Adaptive<dg::ERKStep<std::array<double,3>>>(
156 "Dormand-Prince-7-4-5", coords), m_cyl_field,
158 1e-10); // using 1e-10 instead of eps may cost 10% of performance but is what we also use in Fieldaligned
159 odeint.integrate_in_domain( 0., coords, phiP, coordsP, 0.,
160 m_domain, m_eps);
161 odeint.integrate_in_domain( 0., coords, phiM, coordsM, 0.,
162 m_domain, m_eps);
163 }catch (std::exception& e)
164 {
165 // if not possible the distance is large
166 phiP = m_deltaPhi;
167 coordsP[2] = 1e6*phiP;
168 phiM = -m_deltaPhi;
169 coordsM[2] = 1e6*phiM;
170 }
171 }
172 else
173 {
174 coordsP[2] = 1e6*phiP;
175 coordsM[2] = 1e6*phiM;
176 }
177 if( m_type == "phi")
178 return sign*(-phiP-phiM)/(phiP-phiM);
179 double sP = coordsP[2], sM = coordsM[2];
180 double value = sign*(-sP-sM)/(sP-sM);
181 if( (phiM <= -m_deltaPhi) and (phiP >= m_deltaPhi))
182 return 0.; //return exactly zero if sheath not reached
183 if( (phiM <= -m_deltaPhi))
184 return value*sign > 0 ? value : 0.; // avoid false negatives
185 if( (phiP >= m_deltaPhi))
186 return value*sign < 0 ? value : 0.; // avoid false positives
187 return value;
188 }
189
190 private:
191 std::function<bool(double,double)> m_pred;
192 const dg::Grid2d m_domain;
193 dg::geo::detail::DSFieldCylindrical3 m_cyl_field;
194 double m_deltaPhi, m_eps;
195 std::string m_type;
196};
197
198}//namespace geo
199}//namespace dg
static auto pid_control
static auto fast_l2norm
void integrate_in_domain(value_type t0, const ContainerType &u0, value_type &t1, ContainerType &u1, value_type dt, Domain &&domain, value_type eps_root)
Definition fluxfunctions.h:416
Normalized coordinate relative to wall along fieldline in phi or s coordinate.
Definition sheath.h:128
WallFieldlineCoordinate(const dg::geo::CylindricalVectorLvl0 &vec, const dg::aRealTopology2d< double > &domain, double maxPhi, double eps, std::string type, std::function< bool(double, double)> predicate=mod::everywhere)
Construct with vector field, domain.
Definition sheath.h:131
double do_compute(double R, double Z) const
Definition sheath.h:144
Distance to wall along fieldline in phi or s coordinate
Definition sheath.h:32
double do_compute(double R, double Z) const
Integrate fieldline until wall is reached.
Definition sheath.h:66
WallFieldlineDistance(const dg::geo::CylindricalVectorLvl0 &vec, const dg::aRealTopology2d< double > &domain, double maxPhi, double eps, std::string type, std::function< bool(double, double)> predicate=mod::everywhere)
Construct with vector field, domain.
Definition sheath.h:44
Represent functions written in cylindrical coordinates that are independent of the angle phi serving ...
Definition fluxfunctions.h:66