Extension: Geometries
#include "dg/geometries/geometries.h"
Loading...
Searching...
No Matches
curvilinear.h
Go to the documentation of this file.
1#pragma once
2
3#include "dg/algorithm.h"
4#include "generator.h"
5
6namespace dg
7{
8namespace geo
9{
10
62
65{
66 unsigned n;
67 unsigned N;
69};
70
71
73template<class real_type>
75namespace detail
76{
77template<class host_vector>
78dg::SparseTensor<host_vector> square( const dg::SparseTensor<host_vector >& jac, const host_vector& R, bool orthogonal)
79{
80 std::vector<host_vector> values( 5, R);
81 {
82 dg::blas1::scal( values[0], 0); //0
83 dg::blas1::pointwiseDot( 1., jac.value(0,0), jac.value(0,0), 1., jac.value(0,1), jac.value(0,1), 0., values[1]); //xx
84 dg::blas1::pointwiseDot( 1., jac.value(1,0), jac.value(1,0), 1., jac.value(1,1), jac.value(1,1), 0., values[2]); //yy
85 dg::blas1::pointwiseDot( values[3], values[3], values[3]);
86 dg::blas1::pointwiseDivide( 1., values[3], values[3]); //pp == 1/R^2
87
88 dg::blas1::pointwiseDot( 1., jac.value(0,0), jac.value(1,0), 1., jac.value(0,1), jac.value(1,1), 0., values[4]); //xy
89 }
90 SparseTensor<host_vector> metric(values[0]); //unit tensor
91 metric.values().pop_back(); //remove the one
92 metric.idx(0,0) = 1; metric.values().push_back( values[1]);
93 metric.idx(1,1) = 2; metric.values().push_back( values[2]);
94 metric.idx(2,2) = 3; metric.values().push_back( values[3]);
95 if( !orthogonal)
96 {
97 metric.idx(1,0) = metric.idx(0,1) = 4;
98 metric.values().push_back( values[4]);
99 }
100 return metric;
101}
102}//namespace detail
104
105//when we make a 3d grid with eta and phi swapped the metric structure and the transformation changes
106//In practise it can only be orthogonal due to the projection tensor in the elliptic operator
107
111template<class real_type>
113{
120 Topology1d tx, Topology1d ty) :
121 dg::aRealGeometry2d<real_type>( {0.,0.},
122 {generator.width(), generator.height()},
123 {tx.n,ty.n},{tx.N,ty.N}, { tx.b, ty.b}),
124 m_handle(generator)
125 {
126 construct( tx.n, tx.N, ty.n, ty.N);
127 }
128
134
136 const aRealGenerator2d<real_type>& generator() const{return *m_handle;}
137 virtual RealCurvilinearGrid2d* clone()const override final{return new RealCurvilinearGrid2d(*this);}
138 private:
139 virtual void do_set(std::array<unsigned,2> new_n, std::array<unsigned,2> new_N) override final
140 {
142 construct( new_n[0], new_N[0], new_n[1], new_N[1]);
143 }
144 virtual void do_set(std::array<dg::bc,2> new_bc) override final
145 {
147 }
148 virtual void do_set_pq(std::array<real_type,2> new_x0, std::array<real_type,2> new_x1) override final
149 {
150 throw dg::Error(dg::Message(_ping_)<<"This grid cannot change boundaries\n");
151 }
152 void construct( unsigned nx, unsigned Nx, unsigned ny, unsigned Ny);
153 virtual SparseTensor<thrust::host_vector<real_type> > do_compute_jacobian( ) const override final{
154 return m_jac;
155 }
156 virtual SparseTensor<thrust::host_vector<real_type>> do_compute_metric( ) const override final{
157 return m_metric;
158 }
159 virtual std::vector<thrust::host_vector<real_type>> do_compute_map()const override final{return m_map;}
161 std::vector<thrust::host_vector<real_type>> m_map;
163};
164
165
171template<class real_type>
173{
176
180
184 dg::aRealProductGeometry3d<real_type>(
185 {0.,0., gz.x0()},{ generator.width(),
186 generator.height(),gz.x1()}, {tx.n,ty.n, gz.n()},
187 {tx.N, ty.N, gz.N()},{ tx.b, ty.b, gz.bcx()}), m_handle(generator)
188 {
189 m_map.resize(3);
190 constructPerp( this->nx(), this->Nx(), this->ny(), this->Ny());
191 constructParallel(this->nz(), this->Nz());
192 }
193
194
196 const aRealGenerator2d<real_type> & generator() const{return *m_handle;}
197 virtual RealCurvilinearProductGrid3d* clone()const override final{return new RealCurvilinearProductGrid3d(*this);}
198 private:
199 virtual RealCurvilinearGrid2d<real_type>* do_perp_grid() const override final;
200 virtual void do_set( std::array<unsigned,3> new_n, std::array<unsigned,3> new_N) override final
201 {
202 auto old_n = this->get_n(), old_N = this->get_N();
204 if( !( new_n[0] == old_n[0] && new_N[0] == old_N[0] &&
205 new_n[1] == old_n[1] && new_N[1] == old_N[1] ) )
206 constructPerp( new_n[0], new_N[0], new_n[1],new_N[1]);
207 constructParallel(new_n[2],new_N[2]);
208 }
209 virtual void do_set(std::array<dg::bc,3> new_bc) override final
210 {
212 }
213 virtual void do_set_pq(std::array<real_type,3> new_x0, std::array<real_type,3> new_x1) override final
214 {
215 throw dg::Error(dg::Message(_ping_)<<"This grid cannot change boundaries\n");
216 }
217 //construct phi and lift rest to 3d
218 void constructParallel(unsigned nz,unsigned Nz)
219 {
220 m_map[2]=dg::evaluate(dg::cooZ3d, *this);
221 unsigned size = this->size();
222 unsigned size2d = this->nx()*this->ny()*this->Nx()*this->Ny();
223 //resize for 3d values
224 for( unsigned r=0; r<6;r++)
225 m_jac.values()[r].resize(size);
226 m_map[0].resize(size);
227 m_map[1].resize(size);
228 //lift to 3D grid
229 for( unsigned k=1; k<nz*Nz; k++)
230 for( unsigned i=0; i<size2d; i++)
231 {
232 for(unsigned r=0; r<6; r++)
233 m_jac.values()[r][k*size2d+i] = m_jac.values()[r][(k-1)*size2d+i];
234 m_map[0][k*size2d+i] = m_map[0][(k-1)*size2d+i];
235 m_map[1][k*size2d+i] = m_map[1][(k-1)*size2d+i];
236 }
237 }
238 //construct 2d plane
239 void constructPerp( unsigned nx, unsigned Nx, unsigned ny, unsigned Ny)
240 {
241 dg::Grid1d gX1d( this->x0(), this->x1(), nx, Nx);
242 dg::Grid1d gY1d( this->y0(), this->y1(), ny, Ny);
243 thrust::host_vector<real_type> x_vec = dg::evaluate( dg::cooX1d, gX1d);
244 thrust::host_vector<real_type> y_vec = dg::evaluate( dg::cooX1d, gY1d);
245 m_jac = SparseTensor< thrust::host_vector<real_type>>( x_vec);//unit tensor
246 m_jac.values().resize( 6);
247 m_handle->generate( x_vec, y_vec, m_map[0], m_map[1], m_jac.values()[2], m_jac.values()[3], m_jac.values()[4], m_jac.values()[5]);
248 m_jac.idx(0,0) = 2, m_jac.idx(0,1) = 3, m_jac.idx(1,0)=4, m_jac.idx(1,1) = 5;
249 }
250 virtual SparseTensor<thrust::host_vector<real_type> > do_compute_jacobian( ) const override final{
251 return m_jac;
252 }
253 virtual SparseTensor<thrust::host_vector<real_type> > do_compute_metric( ) const override final
254 {
255 return detail::square( m_jac, m_map[0], m_handle->isOrthogonal());
256 }
257 virtual std::vector<thrust::host_vector<real_type> > do_compute_map()const override final{return m_map;}
258 std::vector<thrust::host_vector<real_type> > m_map;
259 SparseTensor<thrust::host_vector<real_type> > m_jac;
261};
262
265#ifndef MPI_VERSION
270#endif
271
274template<class real_type>
276 dg::aRealGeometry2d<real_type>( std::array{g.gx(), g.gy()} ), m_handle(g.generator())
277{
278 g.set( {this->nx(), this->ny(), 1}, {this->Nx(), this->Ny(),1}); //shouldn't trigger 2d grid generator
279 m_map=g.map();
280 m_jac=g.jacobian();
281 m_metric=g.metric();
282 // we rely on the fact that the 3d grid uses square to compute its metric
283 // so the (2,2) entry is value 3 that we need to set to 1 (for the
284 // create::volume function to work properly)
285 dg::blas1::copy( 1., m_metric.values()[3]);
286 m_map.pop_back();
287}
288template<class real_type>
289void RealCurvilinearGrid2d<real_type>::construct( unsigned nx, unsigned Nx, unsigned ny, unsigned Ny)
290{
291 RealCurvilinearProductGrid3d<real_type> g( *m_handle, {nx,Nx,this->bcx()}, {ny,Ny,this->bcy()}, {0., 2.*M_PI, 1,1});
292 *this = RealCurvilinearGrid2d<real_type>(g);
293}
294template<class real_type>
295typename RealCurvilinearProductGrid3d<real_type>::perpendicular_grid* RealCurvilinearProductGrid3d<real_type>::do_perp_grid() const { return new typename RealCurvilinearProductGrid3d<real_type>::perpendicular_grid(*this);}
297
298}//namespace geo
299}//namespace dg
#define M_PI
DG_DEVICE double cooZ3d(double x, double y, double z)
DG_DEVICE double cooX1d(double x)
void copy(const ContainerTypeIn &source, ContainerTypeOut &target)
void pointwiseDot(value_type alpha, const ContainerType1 &x1, const ContainerType2 &x2, value_type1 beta, ContainerType &y)
void scal(ContainerType &x, value_type alpha)
void pointwiseDivide(value_type alpha, const ContainerType1 &x1, const ContainerType2 &x2, value_type1 beta, ContainerType &y)
auto evaluate(Functor &&f, const Topology &g)
@ square
closed flux surfaces centered around an O-point and bordered by a square with four X-points in the co...
CurvilinearProductGrid3d CurvilinearProductGrid3d
Definition curvilinear.h:268
CurvilinearGrid2d CurvilinearGrid2d
Definition curvilinear.h:267
int idx(unsigned i, unsigned j) const
std::vector< container > & values()
const container & value(size_t i, size_t j) const
SparseTensor< thrust::host_vector< real_type > > jacobian() const
SparseTensor< thrust::host_vector< real_type > > metric() const
std::vector< thrust::host_vector< real_type > > map() const
std::array< unsigned, Nd > get_n() const
unsigned n(unsigned u=0) const
std::array< unsigned, Nd > get_N() const
std::enable_if_t<(Md==1), void > set(unsigned new_n, unsigned new_Nx)
virtual void do_set(std::array< unsigned, Nd > new_n, std::array< unsigned, Nd > new_N)=0
RealGrid< real_type, 1 > gz() const
A two-dimensional grid based on curvilinear coordinates.
Definition curvilinear.h:113
RealCurvilinearGrid2d(const aRealGenerator2d< real_type > &generator, unsigned n, unsigned Nx, unsigned Ny, dg::bc bcx=dg::DIR, bc bcy=dg::PER)
Construct a 2D grid.
Definition curvilinear.h:116
const aRealGenerator2d< real_type > & generator() const
read access to the generator
Definition curvilinear.h:136
RealCurvilinearGrid2d(const aRealGenerator2d< real_type > &generator, Topology1d tx, Topology1d ty)
Construct the computational space as the product of two 1d grids.
Definition curvilinear.h:119
virtual RealCurvilinearGrid2d * clone() const override final
Definition curvilinear.h:137
RealCurvilinearGrid2d(RealCurvilinearProductGrid3d< real_type > g)
Explicitly convert 3d product grid to the perpendicular grid.
A 2x1 curvilinear product space grid.
Definition curvilinear.h:173
const aRealGenerator2d< real_type > & generator() const
read access to the generator
Definition curvilinear.h:196
RealCurvilinearProductGrid3d(const aRealGenerator2d< real_type > &generator, unsigned n, unsigned Nx, unsigned Ny, unsigned Nz, bc bcx=dg::DIR, bc bcy=dg::PER, bc bcz=dg::PER)
Construct a 3D grid.
Definition curvilinear.h:178
virtual RealCurvilinearProductGrid3d * clone() const override final
Definition curvilinear.h:197
RealCurvilinearGrid2d< real_type > perpendicular_grid
Definition curvilinear.h:174
RealCurvilinearProductGrid3d(const aRealGenerator2d< real_type > &generator, Topology1d tx, Topology1d ty, RealGrid1d< real_type > gz)
Construct the computational space as the product of three 1d grids.
Definition curvilinear.h:182
Helper class for construction.
Definition curvilinear.h:65
unsigned n
Definition curvilinear.h:66
bc b
Definition curvilinear.h:68
unsigned N
Definition curvilinear.h:67
The abstract generator base class.
Definition generator.h:20