Discontinuous Galerkin Library
#include "dg/algorithm.h"
Loading...
Searching...
No Matches
stencil.h
Go to the documentation of this file.
1#pragma once
2
3#include "xspacelib.h"
4#ifdef MPI_VERSION
5#include "mpi_projection.h" // for make_mpi_matrix function
6#endif // MPI_VERSION
7
11namespace dg
12{
13namespace create
14{
16namespace detail
17{
18template<class real_type>
19 void set_boundary(
20 thrust::host_vector<real_type>& values,
21 thrust::host_vector<int>& column_indices,
22 dg::bc bcx,
23 int num_cols)
24{
25 // Fix this leads to duplicate values (do not sort or remove them though)
26 for( unsigned k=0; k<values.size(); k++)
27 {
28 if( column_indices[k] < 0 )
29 {
30 if( bcx == dg::NEU || bcx == dg::NEU_DIR)
31 column_indices[k] = -(column_indices[k]+1);
32 else if( bcx == dg::DIR || bcx == dg::DIR_NEU)
33 {
34 column_indices[k] = -(column_indices[k]+1);
35 values[k] *= -1;
36 }
37 else if( bcx == dg::PER)
38 column_indices[k] += num_cols;
39 }
40 else if( column_indices[k] >= num_cols)
41 {
42 if( bcx == dg::NEU || bcx == dg::DIR_NEU)
43 column_indices[k] = 2*num_cols-1-column_indices[k];
44 else if( bcx == dg::DIR || bcx == dg::NEU_DIR)
45 {
46 column_indices[k] = 2*num_cols-1-column_indices[k];
47 values[k] *= -1;
48 }
49 else if( bcx == dg::PER)
50 column_indices[k] -= num_cols;
51 }
52 }
53}
54
55template<class real_type>
57 unsigned stencil_size,
58 const RealGrid1d<real_type>& local,
59 const RealGrid1d<real_type>& global,
60 dg::bc bcx)
61{
62 thrust::host_vector<real_type> values;
63 thrust::host_vector<int> row_offsets;
64 thrust::host_vector<int> column_indices;
65
66 unsigned num_rows = local.size();
67 unsigned num_cols = global.size();
68 unsigned radius = stencil_size/2;
69 int L0 = round((local.x0() - global.x0())/global.h())*global.n();
70
71 row_offsets.push_back(0);
72 for( unsigned k=0; k<num_rows; k++)
73 {
74 row_offsets.push_back(stencil_size + row_offsets[k]);
75 for( unsigned l=0; l<stencil_size; l++)
76 {
77 column_indices.push_back( L0 + (int)(k + l) - (int)radius);
78 values.push_back( 1.0);
79 }
80 }
81 set_boundary( values, column_indices, bcx, num_cols);
82
83 // DO NOT SORT and KEEP DUPLICATES
84 return {num_rows, num_cols, row_offsets, column_indices, values};
85}
86
87// Rethink this approach if we ever need to make it work with MPI again
88template<class real_type>
90 const RealGrid1d<real_type>& local,
91 const RealGrid1d<real_type>& global,
92 dg::bc bcx)
93{
94 thrust::host_vector<real_type> values;
95 thrust::host_vector<int> row_offsets;
96 thrust::host_vector<int> column_indices;
97
98 unsigned num_rows = local.size();
99 unsigned num_cols = global.size();
100 int L0 = round((local.x0() - global.x0())/global.h())*global.n();
101 // We need the first two lines of forward
103 // and the second column of backward
105 if( global.n() == 1)
106 throw dg::Error( dg::Message(_ping_) << "Limiter stencil not possible for n==1!");
107
108
109 row_offsets.push_back( 0);
110 for( unsigned k=0; k<local.N(); k++)
111 {
112 row_offsets.push_back(row_offsets.back() + 3*local.n());
113 for( unsigned j=1; j<local.n(); j++)
114 row_offsets.push_back(row_offsets.back());
115 // Order is important
116 for( unsigned j=0; j<local.n(); j++)
117 {
118 column_indices.push_back( L0 + (int)((k-1)*global.n() + j) );
119 values.push_back( forward(0, j ));
120 }
121 for( unsigned j=0; j<local.n(); j++)
122 {
123 column_indices.push_back( L0 + (int)(k*global.n() + j ));
124 values.push_back( forward(1, j ));
125 }
126 for( unsigned j=0; j<local.n(); j++)
127 {
128 column_indices.push_back( L0 + (int)((k+1)*global.n() + j));
129 values.push_back( backward(j, 1) );
130 }
131 }
132 assert( row_offsets.size() == num_rows+1);
133 set_boundary( values, column_indices, bcx, num_cols);
134
135 // DO NOT SORT and KEEP DUPLICATES
136 return {num_rows, num_cols, row_offsets, column_indices, values};
137}
138
139template<class real_type>
140dg::SparseMatrix< int, real_type, thrust::host_vector> identity_matrix( const RealGrid1d<real_type>& local, const RealGrid1d<real_type>& global)
141{
142 int L0 = round((local.x0() - global.x0())/global.h())*global.n();
143 thrust::host_vector<real_type> values( local.size());
144 thrust::host_vector<int> row_offsets( local.size()+1);
145 thrust::host_vector<int> column_indices( local.size());
146 row_offsets[0] = 0;
147 for( unsigned i=0; i<local.size(); i++)
148 {
149 row_offsets[i+1] = 1 + row_offsets[i];
150 column_indices[i] = L0 + i;
151 values[i] = 1.;
152 }
153 return {local.size(), global.size(), row_offsets, column_indices, values};
154}
155
156} //namespace detail
158
161
176template<class real_type>
178 unsigned window_size,
179 const RealGrid1d<real_type>& g,
180 dg::bc bcx)
181{
182 return detail::window_stencil( window_size, g, g, bcx);
183}
184
199template<class real_type>
201 const RealGrid1d<real_type>& g,
202 dg::bc bound)
203{
204 return detail::limiter_stencil( g, g, bound);
205}
206
207
208
228template<class real_type>
230 std::array<int,2> window_size,
232 dg::bc bcx, dg::bc bcy)
233{
234 auto mx = detail::window_stencil(window_size[0], g.gx(), g.gx(), bcx);
235 auto my = detail::window_stencil(window_size[1], g.gy(), g.gy(), bcy);
236 return dg::tensorproduct( my, mx);
237}
238
241template<class real_type>
243 enum coo3d direction,
245 dg::bc bound)
246{
247 if( direction == dg::coo3d::x)
248 {
249 auto mx = detail::limiter_stencil(g.gx(), g.gx(), bound);
250 auto einsy = detail::identity_matrix( g.gy(), g.gy());
251 return dg::tensorproduct( einsy, mx);
252 }
253 auto my = detail::limiter_stencil(g.gy(), g.gy(), bound);
254 auto einsx = detail::identity_matrix( g.gx(), g.gx());
255 return dg::tensorproduct( my, einsx);
256}
257
260template<class real_type>
262 enum coo3d direction,
264 dg::bc bound)
265{
266 if( direction == dg::coo3d::x)
267 {
268 auto mx = detail::limiter_stencil(g.gx(), g.gx(), bound);
269 auto einsy = detail::identity_matrix( g.gy(), g.gy());
270 auto einsz = detail::identity_matrix( g.gz(), g.gz());
271 auto temp = dg::tensorproduct( einsy, mx);
272 return dg::tensorproduct( einsz, temp);
273 }
274 if( direction == dg::coo3d::y)
275 {
276 auto einsx = detail::identity_matrix( g.gx(), g.gx());
277 auto my = detail::limiter_stencil(g.gy(), g.gy(), bound);
278 auto einsz = detail::identity_matrix( g.gz(), g.gz());
279 return dg::tensorproduct( einsz, dg::tensorproduct( my, einsx));
280 }
281 auto mz = detail::limiter_stencil(g.gz(), g.gz(), bound);
282 auto einsy = detail::identity_matrix( g.gy(), g.gy());
283 auto einsx = detail::identity_matrix( g.gx(), g.gx());
284 return dg::tensorproduct( mz, dg::tensorproduct( einsy, einsx));
285}
286
306template<class real_type>
308 std::array<int,2> window_size,
310 dg::bc bcx, dg::bc bcy)
311{
312 auto mx = detail::window_stencil(window_size[0], g.gx(), g.gx(), bcx);
313 auto my = detail::window_stencil(window_size[1], g.gy(), g.gy(), bcy);
314 auto mz = detail::identity_matrix( g.gz(), g.gz());
315 return dg::tensorproduct( mz, dg::tensorproduct( my, mx));
316}
317
318#ifdef MPI_VERSION
320template<class real_type>
322 std::array<int,2> window_size,
324 dg::bc bcx, dg::bc bcy)
325{
326 auto mx = detail::window_stencil(window_size[0], g.local().gx(), g.global().gx(), bcx);
327 auto my = detail::window_stencil(window_size[1], g.local().gy(), g.global().gy(), bcy);
328 auto local = dg::tensorproduct( my, mx);
329 return dg::make_mpi_matrix( local, g);
330}
331
333template<class real_type>
335 std::array<int,2> window_size,
337 dg::bc bcx, dg::bc bcy)
338{
339 auto mx = detail::window_stencil(window_size[0], g.local().gx(), g.global().gx(), bcx);
340 auto my = detail::window_stencil(window_size[1], g.local().gy(), g.global().gy(), bcy);
341 auto mz = detail::identity_matrix( g.local().gz(), g.global().gz());
342 auto out = dg::tensorproduct( mz, dg::tensorproduct( my, mx));
343 return dg::make_mpi_matrix( out, g);
344}
345
346#endif // MPI_VERSION
347
349} // namespace create
350} // namespace dg
class intended for the use in throw statements
Definition exceptions.h:83
small class holding a stringstream
Definition exceptions.h:29
A square nxn matrix.
Definition operator.h:31
#define _ping_
Definition exceptions.h:12
coo3d
3d coordinates
Definition enums.h:179
bc
Switch between boundary conditions.
Definition enums.h:15
direction
Direction of a discrete derivative.
Definition enums.h:97
@ y
y direction
@ x
x direction
@ NEU_DIR
Neumann on left, Dirichlet on right boundary.
Definition enums.h:19
@ PER
periodic boundaries
Definition enums.h:16
@ NEU
Neumann on both boundaries.
Definition enums.h:20
@ DIR
homogeneous dirichlet boundaries
Definition enums.h:17
@ DIR_NEU
Dirichlet on left, Neumann on right boundary.
Definition enums.h:18
@ backward
backward derivative (cell to the left and current cell)
Definition enums.h:99
@ forward
forward derivative (cell to the right and current cell)
Definition enums.h:98
SquareMatrix< T > tensorproduct(const SquareMatrix< T > &op1, const SquareMatrix< T > &op2)
Form the tensor product between two operators.
Definition operator_tensor.h:25
dg::MIHMatrix_t< real_type > make_mpi_matrix(const dg::IHMatrix_t< real_type > &global_cols, const ConversionPolicy &col_policy)
Convert a (row-distributed) matrix with local row and global column indices to a row distributed MPI ...
Definition mpi_projection.h:50
dg::IHMatrix_t< real_type > limiter_stencil(const RealGrid1d< real_type > &g, dg::bc bound)
A stencil for the dg Slope limiter.
Definition stencil.h:200
dg::IHMatrix_t< real_type > window_stencil(unsigned window_size, const RealGrid1d< real_type > &g, dg::bc bcx)
A 1d centered window stencil.
Definition stencil.h:177
Useful MPI typedefs and overloads of interpolation and projection.
This is the namespace for all functions and classes defined and used by the discontinuous Galerkin li...
static std::vector< real_type > backward(unsigned n)
Return backward DLT trafo matrix.
Definition dlt.h:139
static std::vector< real_type > forward(unsigned n)
Return forward DLT trafo matrix.
Definition dlt.h:195
Distributed memory matrix class, asynchronous communication.
Definition mpi_matrix.h:395
The simplest implementation of aRealTopology.
Definition grid.h:710
A CSR formatted sparse matrix.
Definition sparsematrix.h:96
An abstract base class for MPI distributed Nd-dimensional dG grids.
Definition mpi_grid.h:91
const RealGrid< real_type, Nd > & global() const
The global grid as a shared memory grid.
Definition mpi_grid.h:279
const RealGrid< real_type, Nd > & local() const
The local grid as a shared memory grid.
Definition mpi_grid.h:274
An abstract base class for Nd-dimensional dG grids.
Definition grid.h:95
RealGrid< real_type, 1 > gy() const
Equivalent to grid(1)
Definition grid.h:360
RealGrid< real_type, 1 > gx() const
Equivalent to grid(0)
Definition grid.h:354
RealGrid< real_type, 1 > gz() const
Equivalent to grid(2)
Definition grid.h:366
utility functions