Discontinuous Galerkin Library
#include "dg/algorithm.h"
Loading...
Searching...
No Matches
blas2.h
Go to the documentation of this file.
1#pragma once
2
6#include "backend/blas2_dispatch_scalar.h"
7#include "backend/blas2_dispatch_shared.h"
8#include "backend/blas2_sparseblockmat.h"
9#include "backend/blas2_selfmade.h"
10#include "backend/blas2_densematrix.h"
11#ifdef MPI_VERSION
12#include "backend/blas2_dispatch_mpi.h"
13#endif //MPI_VERSION
14#include "backend/blas2_dispatch_vector.h"
15
16
21namespace dg{
27namespace blas2{
30
32namespace detail{
33
34template< class ContainerType1, class MatrixType, class ContainerType2>
35inline std::vector<int64_t> doDot_superacc( int* status, const ContainerType1& x, const MatrixType& m, const ContainerType2& y)
36{
37 static_assert(
40 "The container types must have a vector data layout (AnyVector)!");
41 //check ContainerTypes: must be either scalar or same base category
42 using vector_type = find_if_t<dg::is_not_scalar, ContainerType1, ContainerType1, ContainerType2>;
43 using vector_category = get_tensor_category<vector_type>;
44 static_assert(
45 dg::is_scalar_or_same_base_category<ContainerType1, vector_category>::value &&
46 dg::is_scalar_or_same_base_category<ContainerType2, vector_category>::value,
47 "All container types must be either Scalar or have compatible Vector categories (AnyVector or Same base class)!");
48 return doDot_superacc( status, x, m, y, get_tensor_category<MatrixType>(), vector_category());
49}
50
51}//namespace detail
53
93template< class ContainerType1, class MatrixType, class ContainerType2>
94inline auto dot( const ContainerType1& x, const MatrixType& m, const ContainerType2& y)
95{
96 if constexpr (std::is_floating_point_v<get_value_type<ContainerType1>> &&
97 std::is_floating_point_v<get_value_type<MatrixType>> &&
98 std::is_floating_point_v<get_value_type<ContainerType2>>)
99 {
100 int status = 0;
101 std::vector<int64_t> acc = dg::blas2::detail::doDot_superacc( &status,
102 x,m,y);
103 if( status != 0)
104 throw dg::Error(dg::Message(_ping_)<<"dg::blas2::dot failed "
105 <<"since one of the inputs contains NaN or Inf");
106 return exblas::cpu::Round(acc.data());
107 }
108 else
109 {
110 return dg::blas1::vdot( dg::Product(), x, m, y);
111 }
112}
113
130template< class MatrixType, class ContainerType>
131inline get_value_type<MatrixType> dot( const MatrixType& m, const ContainerType& x)
132{
133 return dg::blas2::dot( x, m, x);
134}
136namespace detail{
137//resolve tags in two stages: first the matrix and then the container type
138template< class MatrixType, class ContainerType1, class ContainerType2>
139inline void doSymv( get_value_type<ContainerType1> alpha,
140 MatrixType&& M,
141 const ContainerType1& x,
143 ContainerType2& y,
145{
146 dg::blas1::pointwiseDot( alpha, std::forward<MatrixType>(M), x, beta, y);
147}
148template< class MatrixType, class ContainerType1, class ContainerType2>
149inline void doSymv( MatrixType&& M,
150 const ContainerType1& x,
151 ContainerType2& y,
152 AnyScalarTag)
153{
154 dg::blas1::pointwiseDot( std::forward<MatrixType>(M), x, y);
155}
156
157template< class MatrixType, class ContainerType1, class ContainerType2>
158inline void doSymv( get_value_type<ContainerType1> alpha,
159 MatrixType&& M,
160 const ContainerType1& x,
162 ContainerType2& y,
163 AnyMatrixTag)
164{
165 static_assert( std::is_same_v<get_execution_policy<ContainerType1>,
167 "Vector types must have same execution policy");
168 // We want to allow double matrix on complex vector ...
169 //static_assert( std::is_same_v<get_value_type<ContainerType1>,
170 // get_value_type<MatrixType>> &&
171 // std::is_same_v<get_value_type<ContainerType2>,
172 // get_value_type<MatrixType>>,
173 // "Vector and Matrix types must have same value type");
174 static_assert( std::is_same_v<get_tensor_category<ContainerType1>,
176 "Vector types must have same data layout");
177 dg::blas2::detail::doSymv( alpha, std::forward<MatrixType>(M), x, beta, y,
180}
181template< class MatrixType, class ContainerType1, class ContainerType2>
182inline void doSymv( MatrixType&& M,
183 const ContainerType1& x,
184 ContainerType2& y,
185 AnyMatrixTag)
186{
187 static_assert( std::is_same_v<get_execution_policy<ContainerType1>,
189 "Vector types must have same execution policy");
190 // We want to allow double matrix on complex vector ...
191 //static_assert( std::is_same_v<get_value_type<ContainerType1>,
192 // get_value_type<MatrixType>> &&
193 // std::is_same_v<get_value_type<ContainerType2>,
194 // get_value_type<MatrixType>>,
195 // "Vector and Matrix types must have same value type");
196 static_assert( std::is_same_v<get_tensor_category<ContainerType1>,
198 "Vector types must have same data layout");
199 dg::blas2::detail::doSymv( std::forward<MatrixType>(M), x, y,
202}
203template< class FunctorType, class MatrixType, class ContainerType1, class ContainerType2>
204inline void doStencil(
205 FunctorType f,
206 MatrixType&& M,
207 const ContainerType1& x,
208 ContainerType2& y,
209 AnyMatrixTag)
210{
211 static_assert( std::is_same_v<get_execution_policy<ContainerType1>,
213 "Vector types must have same execution policy");
214 // We want to allow double matrix on complex vector ...
215 //static_assert( std::is_same_v<get_value_type<ContainerType1>,
216 // get_value_type<MatrixType>> &&
217 // std::is_same_v<get_value_type<ContainerType2>,
218 // get_value_type<MatrixType>>,
219 // "Vector and Matrix types must have same value type");
220 static_assert( std::is_same_v<get_tensor_category<ContainerType1>,
222 "Vector types must have same data layout");
223 dg::blas2::detail::doStencil( f, std::forward<MatrixType>(M), x, y,
226}
227
228template< class MatrixType, class ContainerType1, class ContainerType2>
229inline void doSymv( get_value_type<ContainerType1> alpha,
230 MatrixType&& M,
231 const ContainerType1& x,
233 ContainerType2& y,
234 NotATensorTag)
235{
236 // if a compiler error message brings you here then you need to overload
237 // the parenthesis operator for your class
238 // void operator()( value_type alpha, const ContainerType1& x, value_type beta, ContainerType2& y)
239 M(alpha,x,beta,y);
240}
241template< class MatrixType, class ContainerType1, class ContainerType2>
242inline void doSymv( MatrixType&& M,
243 const ContainerType1& x,
244 ContainerType2& y,
245 NotATensorTag)
246{
247 // if a compiler error message brings you here then you need to overload
248 // the parenthesis operator for your class
249 // void operator()( const ContainerType1& x, ContainerType2& y)
250 M(x,y);
251}
252
253}//namespace detail
255
282template< class MatrixType, class ContainerType1, class ContainerType2>
284 MatrixType&& M,
285 const ContainerType1& x,
287 ContainerType2& y)
288{
289 if(alpha == (get_value_type<ContainerType1>)0) {
290 dg::blas1::scal( y, beta);
291 return;
292 }
293 dg::blas2::detail::doSymv( alpha, std::forward<MatrixType>(M), x, beta, y, get_tensor_category<MatrixType>());
294}
295
296
297
324template< class MatrixType, class ContainerType1, class ContainerType2>
325inline void symv( MatrixType&& M,
326 const ContainerType1& x,
327 ContainerType2& y)
328{
329 dg::blas2::detail::doSymv( std::forward<MatrixType>(M), x, y, get_tensor_category<MatrixType>());
330}
338template< class MatrixType, class ContainerType1, class ContainerType2>
340 MatrixType&& M,
341 const ContainerType1& x,
343 ContainerType2& y)
344{
345 dg::blas2::symv( alpha, std::forward<MatrixType>(M), x, beta, y);
346}
347
355template< class MatrixType, class ContainerType1, class ContainerType2>
356inline void gemv( MatrixType&& M,
357 const ContainerType1& x,
358 ContainerType2& y)
359{
360 dg::blas2::symv( std::forward<MatrixType>(M), x, y);
361}
362
412template< class Stencil, class ContainerType, class ...ContainerTypes>
413inline void parallel_for( Stencil f, unsigned N, ContainerType&& x, ContainerTypes&&... xs)
414{
415 // Is the assumption that results are automatically ready on return still true?
416 // Do we have to introduce barriers around this function?
417 static_assert( (dg::is_vector_v<ContainerType> &&
419 "All container types must have a vector data layout (AnyVector)!");
420 using vector_type = find_if_t<dg::is_not_scalar, ContainerType, ContainerType, ContainerTypes...>;
421 using tensor_category = get_tensor_category<vector_type>;
422 static_assert(( dg::is_scalar_or_same_base_category<ContainerType, tensor_category>::value &&
423 ... &&dg::is_scalar_or_same_base_category<ContainerTypes, tensor_category>::value),
424 "All container types must be either Scalar or have compatible Vector categories (AnyVector or Same base class)!");
425 dg::blas2::detail::doParallelFor(tensor_category(), f, N, std::forward<ContainerType>(x), std::forward<ContainerTypes>(xs)...);
426}
453template< class FunctorType, class MatrixType, class ContainerType1, class ContainerType2>
454inline void stencil(
455 FunctorType f,
456 MatrixType&& M,
457 const ContainerType1& x,
458 ContainerType2& y)
459{
460 dg::blas2::detail::doStencil( f, std::forward<MatrixType>(M), x, y, get_tensor_category<MatrixType>());
461}
472template<class MatrixType, class AnotherMatrixType>
473inline void transfer( const MatrixType& x, AnotherMatrixType& y)
474{
475 dg::blas2::detail::doTransfer( x,y,
478}
480
481} //namespace blas2
489template< class MatrixType, class ContainerType1, class ContainerType2>
491 MatrixType&& M,
492 const ContainerType1& x,
494 ContainerType2& y)
495{
496 dg::blas2::symv( alpha, std::forward<MatrixType>(M), x, beta, y);
497}
498
506template< class MatrixType, class ContainerType1, class ContainerType2>
507inline void apply( MatrixType&& M,
508 const ContainerType1& x,
509 ContainerType2& y)
510{
511 dg::blas2::symv( std::forward<MatrixType>(M), x, y);
512}
513} //namespace dg
class intended for the use in throw statements
Definition exceptions.h:83
small class holding a stringstream
Definition exceptions.h:29
#define _ping_
Definition exceptions.h:12
void pointwiseDot(value_type alpha, const ContainerType1 &x1, const ContainerType2 &x2, value_type1 beta, ContainerType &y)
Definition blas1.h:406
auto vdot(Functor f, const ContainerType &x, const ContainerTypes &...xs) -> std::invoke_result_t< Functor, dg::get_value_type< ContainerType >, dg::get_value_type< ContainerTypes >... >
Extended Precision transform reduce
Definition blas1.h:90
void scal(ContainerType &x, value_type alpha)
Definition blas1.h:263
void gemv(get_value_type< ContainerType1 > alpha, MatrixType &&M, const ContainerType1 &x, get_value_type< ContainerType1 > beta, ContainerType2 &y)
Alias for blas2::symv ;.
Definition blas2.h:339
auto dot(const ContainerType1 &x, const MatrixType &m, const ContainerType2 &y)
; Binary reproducible general dot product
Definition blas2.h:94
void parallel_for(Stencil f, unsigned N, ContainerType &&x, ContainerTypes &&... xs)
; Customizable and generic for loop
Definition blas2.h:413
void transfer(const MatrixType &x, AnotherMatrixType &y)
; Generic way to copy and/or convert a Matrix type to a different Matrix type
Definition blas2.h:473
void apply(get_value_type< ContainerType1 > alpha, MatrixType &&M, const ContainerType1 &x, get_value_type< ContainerType1 > beta, ContainerType2 &y)
Alias for dg::blas2::symv ;.
Definition blas2.h:490
void symv(MatrixType &&M, const ContainerType1 &x, ContainerType2 &y)
Definition blas2.h:325
void stencil(FunctorType f, MatrixType &&M, const ContainerType1 &x, ContainerType2 &y)
Definition blas2.h:454
@ y
y direction
@ x
x direction
typename TensorTraits< std::decay_t< Vector > >::tensor_category get_tensor_category
Definition tensor_traits.h:47
typename TensorTraits< std::decay_t< Vector > >::execution_policy get_execution_policy
Definition tensor_traits.h:49
constexpr bool is_vector_v
Utility typedef.
Definition predicate.h:75
typename TensorTraits< std::decay_t< Vector > >::value_type get_value_type
Definition tensor_traits.h:45
void doParallelFor(SharedVectorTag, Stencil f, unsigned N, ContainerType &&x, ContainerTypes &&... xs)
double Round(int64_t *accumulator)
This is the namespace for all functions and classes defined and used by the discontinuous Galerkin li...
Scalar Tag base class.
Definition scalar_categories.h:11
Definition subroutines.h:100