Discontinuous Galerkin Library
#include "dg/algorithm.h"
Loading...
Searching...
No Matches
filter.h
Go to the documentation of this file.
1#pragma once
2#include "dg/functors.h"
4
9namespace dg
10{
11
12namespace create
13{
14
39template<class UnaryOp>
41{
42 using real_type = std::invoke_result_t<UnaryOp, unsigned>;
45 SquareMatrix<real_type> filter( n, 0);
46 for( unsigned i=0; i<n; i++)
47 filter(i,i) = op( i);
48 filter = backward*filter*forward;
49 return filter;
50}
51
52} //namespace create
53
55namespace detail{
56
57template<class real_type>
58DG_DEVICE void pix_sort( real_type& a, real_type& b)
59{
60 if( a > b) // swap
61 {
62 real_type tmp = a;
63 a = b;
64 b = tmp;
65 }
66}
67
68template<class real_type>
69DG_DEVICE real_type median3( real_type* p)
70{
71 pix_sort(p[0],p[1]) ; pix_sort(p[1],p[2]) ; pix_sort(p[0],p[1]) ;
72 return (p[1]) ;
73}
74
75template<class real_type>
76DG_DEVICE real_type median5( real_type* p)
77{
78 pix_sort(p[0],p[1]) ; pix_sort(p[3],p[4]) ; pix_sort(p[0],p[3]) ;
79 pix_sort(p[1],p[4]) ; pix_sort(p[1],p[2]) ; pix_sort(p[2],p[3]) ;
80 pix_sort(p[1],p[2]) ; return (p[2]) ;
81}
82
83template<class real_type>
84DG_DEVICE real_type median9( real_type* p)
85{
86 pix_sort(p[1], p[2]) ; pix_sort(p[4], p[5]) ; pix_sort(p[7], p[8]) ;
87 pix_sort(p[0], p[1]) ; pix_sort(p[3], p[4]) ; pix_sort(p[6], p[7]) ;
88 pix_sort(p[1], p[2]) ; pix_sort(p[4], p[5]) ; pix_sort(p[7], p[8]) ;
89 pix_sort(p[0], p[3]) ; pix_sort(p[5], p[8]) ; pix_sort(p[4], p[7]) ;
90 pix_sort(p[3], p[6]) ; pix_sort(p[1], p[4]) ; pix_sort(p[2], p[5]) ;
91 pix_sort(p[4], p[7]) ; pix_sort(p[4], p[2]) ; pix_sort(p[6], p[4]) ;
92 pix_sort(p[4], p[2]) ; return (p[4]) ;
93}
94
95template<class real_type, class Functor>
96DG_DEVICE real_type median( unsigned i, const int* row_offsets,
97 const int* column_indices, Functor f, const real_type* x )
98{
99 int n = row_offsets[i+1]-row_offsets[i];
100 if( n == 3)
101 {
102 real_type p[3];
103 int k = row_offsets[i];
104 for( int l = 0; l<3; l++)
105 p[l] = f(x[column_indices[k+l]]);
106 return detail::median3( p);
107 }
108 if ( n == 5)
109 {
110 real_type p[5];
111 int k = row_offsets[i];
112 for( int l = 0; l<5; l++)
113 p[l] = f(x[column_indices[k+l]]);
114 return detail::median5(p);
115 }
116 if( n == 9)
117 {
118 real_type p[9];
119 int k = row_offsets[i];
120 for( int l = 0; l<9; l++)
121 p[l] = f(x[column_indices[k+l]]);
122 return detail::median9( p);
123
124 }
125 int less, greater, equal;
126 real_type min, max, guess, maxltguess, mingtguess;
127
128 min = max = f(x[column_indices[row_offsets[i]]]) ;
129 for (int k=row_offsets[i]+1 ; k<row_offsets[i+1] ; k++) {
130 if (f(x[column_indices[k]])<min) min=f(x[column_indices[k]]);
131 if (f(x[column_indices[k]])>max) max=f(x[column_indices[k]]);
132 }
133
134 while (1) {
135 guess = (min+max)/2;
136 less = 0; greater = 0; equal = 0;
137 maxltguess = min ;
138 mingtguess = max ;
139 for (int k=row_offsets[i]; k<row_offsets[i+1]; k++) {
140 if (f(x[column_indices[k]])<guess) {
141 less++;
142 if (f(x[column_indices[k]])>maxltguess)
143 maxltguess = f(x[column_indices[k]]) ;
144 } else if (f(x[column_indices[k]])>guess) {
145 greater++;
146 if (f(x[column_indices[k]])<mingtguess)
147 mingtguess = f(x[column_indices[k]]) ;
148 } else equal++;
149 }
150 if (less <= (n+1)/2 && greater <= (n+1)/2) break ;
151 else if (less>greater) max = maxltguess ;
152 else min = mingtguess;
153 }
154 if (less >= (n+1)/2) return maxltguess;
155 else if (less+equal >= (n+1)/2) return guess;
156 else return mingtguess;
157}
158
159}//namespace detail
161
164
175{
176 template<class real_type>
178 void operator()( unsigned i, const int* row_offsets,
179 const int* column_indices, const real_type* values,
180 const real_type* x, real_type* y)
181 {
182 // http://ndevilla.free.fr/median/median/index.html
183 // ignore the values array ...
184 y[i] = detail::median( i, row_offsets, column_indices, []DG_DEVICE(double x){return x;}, x);
185 }
186};
187
188
207template<class real_type>
209{
210 CSRSWMFilter( real_type alpha) : m_alpha( alpha) {}
212 void operator()( unsigned i, const int* row_offsets,
213 const int* column_indices, const real_type* values,
214 const real_type* x, real_type* y)
215 {
216 real_type median = detail::median( i, row_offsets, column_indices,
217 []DG_DEVICE(double x){return x;}, x);
218 real_type amd = detail::median( i, row_offsets, column_indices,
219 [median]DG_DEVICE(double x){return fabs(x-median);}, x);
220
221 if( fabs( x[i] - median) > m_alpha*amd)
222 {
223 y[i] = median;
224 }
225 else
226 y[i] = x[i];
227 }
228 private:
229 real_type m_alpha ;
230};
231
237{
238 template<class real_type>
240 void operator()( unsigned i, const int* row_offsets,
241 const int* column_indices, const real_type* values,
242 const real_type* x, real_type* y)
243 {
244 y[i] = 0;
245 int n = row_offsets[i+1]-row_offsets[i];
246 for( int k=row_offsets[i]; k<row_offsets[i+1]; k++)
247 y[i] += x[column_indices[k]]/(real_type)n;
248 }
249};
255{
256 template<class real_type>
258 void operator()( unsigned i, const int* row_offsets,
259 const int* column_indices, const real_type* values,
260 const real_type* x, real_type* y)
261 {
262 y[i] = 0;
263 for( int k=row_offsets[i]; k<row_offsets[i+1]; k++)
264 y[i] += x[column_indices[k]]*values[k];
265 }
266};
267
287template<class real_type>
289{
290 CSRSlopeLimiter( real_type mod = (real_type)0) :
291 m_mod(mod) {}
293 void operator()( unsigned i, const int* row_offsets,
294 const int* column_indices, const real_type* values,
295 const real_type* x, real_type* y)
296 {
297 int k = row_offsets[i];
298 int n = (row_offsets[i+1] - row_offsets[i])/3;
299 if( n == 0) //only every n-th thread does something
300 return;
301 for( int u=0; u<n; u++)
302 y[column_indices[k+1*n+u]] = x[column_indices[k+1*n + u]]; // copy input
303 // Transform
304 real_type uM = 0, u0 = 0, uP = 0, u1 = 0;
305 for( int u=0; u<n; u++)
306 {
307 uM += x[column_indices[k+0*n + u]]*fabs(values[k+u]);
308 u0 += x[column_indices[k+1*n + u]]*fabs(values[k+u]);
309 u1 += x[column_indices[k+1*n + u]]*values[k+n+u];
310 uP += x[column_indices[k+2*n + u]]*fabs(values[k+u]);
311 }
312 if( values[k]<0) //DIR boundary condition
313 uM *= -1;
314 if( values[k+2*n]>0) //DIR boundary condition
315 uP *= -1;
316
317 dg::MinMod minmod;
318 if( fabs( u1) <= m_mod)
319 return;
320 real_type m = minmod( u1, uP - u0, u0 - uM);
321 if( m == u1)
322 return;
323 // Else transform back
324 for( int u=0; u<n; u++)
325 y[column_indices[k+1*n+u]] =
326 values[k+2*n]>0 ? u0 - m*values[k+2*n+u] : u0 + m*values[k+2*n+u];
327 }
328 private:
329 real_type m_mod;
330};
331
332
334}//namespace dg
A square nxn matrix.
Definition operator.h:31
A matrix type for fast interpolations/projections.
@ 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
@ y
y direction
@ x
x direction
dg::SquareMatrix< std::invoke_result_t< UnaryOp, unsigned > > modal_filter(UnaryOp op, unsigned n)
Create a modal filter block .
Definition filter.h:40
#define DG_DEVICE
Expands to __host__ __device__ if compiled with nvcc else is empty.
Definition dg_doc.h:378
This is the namespace for all functions and classes defined and used by the discontinuous Galerkin li...
Average filter that computes the average of all points in the stencil
Definition filter.h:237
DG_DEVICE void operator()(unsigned i, const int *row_offsets, const int *column_indices, const real_type *values, const real_type *x, real_type *y)
Definition filter.h:240
Compute (lower) Median of input numbers.
Definition filter.h:175
DG_DEVICE void operator()(unsigned i, const int *row_offsets, const int *column_indices, const real_type *values, const real_type *x, real_type *y)
Definition filter.h:178
Switching median filter.
Definition filter.h:209
DG_DEVICE void operator()(unsigned i, const int *row_offsets, const int *column_indices, const real_type *values, const real_type *x, real_type *y)
Definition filter.h:212
CSRSWMFilter(real_type alpha)
Definition filter.h:210
Generalized slope limiter for dG methods.
Definition filter.h:289
CSRSlopeLimiter(real_type mod=(real_type) 0)
Definition filter.h:290
DG_DEVICE void operator()(unsigned i, const int *row_offsets, const int *column_indices, const real_type *values, const real_type *x, real_type *y)
Definition filter.h:293
Test filter that computes the symv csr matrix-vector product if used.
Definition filter.h:255
DG_DEVICE void operator()(unsigned i, const int *row_offsets, const int *column_indices, const real_type *values, const real_type *x, real_type *y)
Definition filter.h:258
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
Definition functors.h:256