Discontinuous Galerkin Library
#include "dg/algorithm.h"
Loading...
Searching...
No Matches
evaluation.h
Go to the documentation of this file.
1#pragma once
2
3#include <cassert>
4#include <cmath>
5#include <thrust/host_vector.h>
6#include "dg/backend/config.h"
7#include "grid.h"
8#include "operator.h"
9
13namespace dg
14{
15
17template< class Functor, class Topology, size_t ...I>
18auto do_evaluate( Functor&& f, const Topology& g, std::index_sequence<I...>)
19{
20 return dg::kronecker( std::forward<Functor>(f), g.abscissas(I)...);
21}
22
26
73template< class Functor, class Topology>
74auto evaluate( Functor&& f, const Topology& g)
75{
76 // The evaluate function is the reason why our Topology needs to have fixed
77 // sized dimensions instead of dynamically sized dimensions
78 // even though if we really wanted we could maybe ask ndim = g.ndim()
79 // and then do use switch and manually implement until ndim < 10 say
80 // for now we keep fixed sized grids ...
81 //
82 // If we ever change the order of the fastest dimension we need to rethink
83 // NetCDF hyperslab construction
84 return do_evaluate( std::forward<Functor>(f), g, std::make_index_sequence<Topology::ndim()>());
85};
86
88//These overloads help the compiler in a situation where a free function has
89//several overloads of different dimensions e.g.
90//double zero( double);
91//double zero( double,double);
92//In such a case dg::evaluate( zero, grid); cannot determine the Functor type...
93template<class Topology, class value_type = typename Topology::value_type, class result_type = typename Topology::value_type, typename = std::enable_if_t<Topology::ndim() == 1 > >
94auto evaluate( result_type (*f)( value_type), const Topology& g)
95{
96 return do_evaluate( f, g, std::make_index_sequence<Topology::ndim()>());
97}
98template<class Topology, class value_type0 = typename Topology::value_type, class value_type1 = typename Topology::value_type, class result_type = typename Topology::value_type, typename = std::enable_if_t<Topology::ndim() == 2 > >
99auto evaluate( result_type (*f)( value_type0, value_type1), const Topology& g)
100{
101 return do_evaluate( f, g, std::make_index_sequence<Topology::ndim()>());
102}
103template<class Topology, class value_type0 = typename Topology::value_type, class value_type1 = typename Topology::value_type, class value_type2 = typename Topology::value_type, class result_type = typename Topology::value_type, typename = std::enable_if_t<Topology::ndim() == 3 > >
104auto evaluate( result_type (*f)( value_type0, value_type1, value_type2), const Topology& g)
105{
106 return do_evaluate( f, g, std::make_index_sequence<Topology::ndim()>());
107}
109
110
111
113
125template<class real_type>
126thrust::host_vector<real_type> integrate( const thrust::host_vector<real_type>& in, const RealGrid<real_type,1>& g, dg::direction dir = dg::forward)
127{
128 double h = g.hx();
129 unsigned n = g.nx();
130 thrust::host_vector<real_type> to_out(g.size(), 0.);
131 thrust::host_vector<real_type> to_in(in);
132 if( dir == dg::backward ) //reverse input vector
133 {
134 for( unsigned i=0; i<in.size(); i++)
135 to_in[i] = in[ in.size()-1-i];
136 }
137
138
141 dg::SquareMatrix<real_type> ninj = create::ninj<real_type>( n );
142 SquareMatrix<real_type> t = create::pipj_inv<real_type>(n);
143 t *= h/2.;
144 ninj = backward*t*ninj*forward;
145 real_type constant = 0.;
146
147 for( unsigned i=0; i<g.Nx(); i++)
148 {
149 for( unsigned k=0; k<n; k++)
150 {
151 for( unsigned l=0; l<n; l++)
152 to_out[ i*n + k] += ninj(k,l)*to_in[ i*n + l];
153 to_out[ i*n + k] += constant;
154 }
155 for( unsigned l=0; l<n; l++)
156 constant += h*forward(0,l)*to_in[i*n+l];
157 }
158 thrust::host_vector<real_type> out(to_out);
159 if( dir == dg::backward ) //reverse output
160 {
161 for( unsigned i=0; i<in.size(); i++)
162 out[i] = -to_out[ in.size()-1-i]; // minus from reversing!
163 }
164 return out;
165}
166
167
177template< class UnaryOp,class real_type>
178thrust::host_vector<real_type> integrate( UnaryOp f, const RealGrid<real_type,1>& g, dg::direction dir = dg::forward)
179{
180 thrust::host_vector<real_type> vector = evaluate( f, g);
181 return integrate<real_type>(vector, g, dir);
182}
184template<class real_type>
185thrust::host_vector<real_type> integrate( real_type (f)(real_type), const RealGrid<real_type,1>& g, dg::direction dir = dg::forward)
186{
187 thrust::host_vector<real_type> vector = evaluate( f, g);
188 return integrate<real_type>(vector, g, dir);
189};
191
193}//namespace dg
194
A square nxn matrix.
Definition operator.h:31
base topology classes
auto kronecker(Functor &&f, const ContainerType &x0, const ContainerTypes &... xs)
Memory allocating version of dg::blas1::kronecker
Definition blas1.h:857
direction
Direction of a discrete derivative.
Definition enums.h:97
@ 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
thrust::host_vector< real_type > integrate(const thrust::host_vector< real_type > &in, const RealGrid< real_type, 1 > &g, dg::direction dir=dg::forward)
Indefinite integral of a function on a grid.
Definition evaluation.h:126
auto evaluate(Functor &&f, const Topology &g)
Evaluate a function on grid coordinates
Definition evaluation.h:74
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
The simplest implementation of aRealTopology.
Definition grid.h:710
real_type hx() const
Equivalent to h(0)
Definition grid.h:314
unsigned size() const
The total number of points.
Definition grid.h:532
unsigned Nx() const
Equivalent to N(0)
Definition grid.h:334
unsigned nx() const
Equivalent to n(0)
Definition grid.h:324