Discontinuous Galerkin Library
#include "dg/algorithm.h"
Loading...
Searching...
No Matches
simpsons.h
Go to the documentation of this file.
1#pragma once
2
3#include <list>
5#include "blas1.h"
6
10namespace dg{
11
33template<class ContainerType>
35{
37 using container_type = ContainerType;
42 Simpsons( unsigned order = 3): m_counter(0), m_order(order), m_t0(0)
43 {
44 set_order(order);
45 }
47 void set_order( unsigned order){
48 m_order=order;
49 m_t.resize( order-1);
50 m_u.resize(order-1);
51 if( !(order == 2 || order == 3))
52 throw dg::Error(dg::Message()<<"Integration order must be either 2 or 3!");
53 }
55 unsigned get_order() const{return m_order;}
61 void init( value_type t0, const ContainerType& u0) {
62 m_integral = u0;
63 for( auto& u: m_u)
64 u = u0;
65 for( auto& t: m_t)
66 t = 0;
67 m_t.front() = t0;
68 flush();
69 m_t0 = t0;
70 }
73 void flush() {
74 m_counter = 0; //since the counter becomes zero we do not need to touch m_u and m_t since the next add triggers the Trapezoidal rule
75 dg::blas1::scal( m_integral, 0.);
76 m_t0 = m_t.front();
77 }
85 void add( value_type t_new, const ContainerType& u_new){
86 if( t_new < m_t.front())
87 throw dg::Error(dg::Message()<<"New time must be strictly larger than old time (or you forgot to call the init function)!");
88 auto pt0 = m_t.begin();
89 auto pt1 = std::next( pt0);
90 auto pu0 = m_u.begin();
91 auto pu1 = std::next( pu0);
92 value_type t0 = *pt1, t1 = *pt0, t2 = t_new;
93 if( m_counter % 2 == 0 || m_order == 2)
94 {
95 //Trapezoidal rule
96 dg::blas1::axpbypgz( 0.5*(t2 - t1), u_new,
97 0.5*(t2 - t1), *pu0 , 1., m_integral);
98 }
99 else
100 {
101 //Simpson's rule
102 value_type pre0 = (2.*t0-3.*t1+t2)*(t2-t0)/(6.*(t0-t1));
103 value_type pre1 = (t2-t0)*(t2-t0)*(t2-t0)/(6.*(t0-t1)*(t1-t2));
104 value_type pre2 = (t0-3.*t1+2.*t2)*(t0-t2)/(6.*(t1-t2));
105
106 dg::blas1::axpby( pre2, u_new, 1., m_integral);
108 pre1-0.5*(t1-t0), *pu0, //subtract last Trapezoidal step
109 pre0-0.5*(t1-t0), *pu1,
110 1., m_integral);
111 }
112 //splice does not copy or move anything, only the internal pointers of the list nodes are re-pointed
113 m_t.splice( pt0, m_t, pt1, m_t.end());//permute elements
114 m_u.splice( pu0, m_u, pu1, m_u.end());
115 m_t.front() = t_new; //and now remove zeroth element
116 dg::blas1::copy( u_new, m_u.front());
117 m_counter++;
118 }
119
123 const ContainerType& get_integral() const{
124 return m_integral;
125 }
131 std::array<value_type,2> get_boundaries() const{
132 std::array<value_type,2> times{ m_t0, m_t.front()};
133 return times;
134 }
135 private:
136 unsigned m_counter, m_order;
137 ContainerType m_integral;
138 //we use a list here to avoid explicitly calling the swap function
139 std::list<value_type> m_t;
140 std::list<ContainerType> m_u;
141 value_type m_t0;
142};
143
144}//namespace dg
class intended for the use in throw statements
Definition exceptions.h:83
small class holding a stringstream
Definition exceptions.h:29
Error classes or the dg library.
void copy(const ContainerTypeIn &source, ContainerTypeOut &target)
Definition blas1.h:243
void axpbypgz(value_type alpha, const ContainerType1 &x, value_type1 beta, const ContainerType2 &y, value_type2 gamma, ContainerType &z)
Definition blas1.h:337
void axpby(value_type alpha, const ContainerType1 &x, value_type1 beta, ContainerType &y)
Definition blas1.h:306
void scal(ContainerType &x, value_type alpha)
Definition blas1.h:263
typename TensorTraits< std::decay_t< Vector > >::value_type get_value_type
Definition tensor_traits.h:45
This is the namespace for all functions and classes defined and used by the discontinuous Galerkin li...
Time integration based on Simpson's rule.
Definition simpsons.h:35
unsigned get_order() const
Access current integration order.
Definition simpsons.h:55
Simpsons(unsigned order=3)
Set integration order without initializing values.
Definition simpsons.h:42
void init(value_type t0, const ContainerType &u0)
Initialize the left-side boundary of the integration.
Definition simpsons.h:61
void flush()
Reset the integral to zero and the last (t,u) pair in the add function as the new left-side.
Definition simpsons.h:73
void add(value_type t_new, const ContainerType &u_new)
Add a new (t,u) pair to the time integral.
Definition simpsons.h:85
std::array< value_type, 2 > get_boundaries() const
Access the left and right boundary in time.
Definition simpsons.h:131
ContainerType container_type
the type of the vector class in use
Definition simpsons.h:37
const ContainerType & get_integral() const
Access the current value of the time integral.
Definition simpsons.h:123
get_value_type< ContainerType > value_type
the value type of the time variable (float or double)
Definition simpsons.h:36
void set_order(unsigned order)
Set integration order without initializing values.
Definition simpsons.h:47