3#include <thrust/sort.h>
4#include <thrust/gather.h>
5#include <thrust/scatter.h>
6#include <thrust/sequence.h>
19template<
class Keys,
class Values>
25 for(
unsigned u=0; u<keys.size(); u++)
26 out[keys[u]] = vals[u];
31template<
class IntegerVec>
32IntegerVec combine_gather(
const IntegerVec& g1,
const IntegerVec& g2)
34 IntegerVec gather = g1;
35 thrust::gather( g1.begin(), g1.end(), g2.begin(), gather.begin());
50template<
class IntegerVec>
51IntegerVec invert_permutation(
const IntegerVec& p)
55 thrust::sequence( seq.begin(), seq.end());
56 thrust::scatter( seq.begin(), seq.end(), p.begin(), sort_map.begin());
87 thrust::host_vector<int> gather1;
88 thrust::host_vector<int> gather2;
89 thrust::host_vector<T> unique;
90 thrust::host_vector<int> howmany;
93template<
class host_vector >
94Unique<typename host_vector::value_type> find_unique_stable_sort(
95 const host_vector& unsorted_elements)
97 using T =
typename host_vector::value_type;
100 auto ids = unsorted_elements;
101 uni.gather1.resize( ids.size());
102 thrust::sequence( uni.gather1.begin(), uni.gather1.end());
103 thrust::stable_sort_by_key( ids.begin(), ids.end(),
104 uni.gather1.begin(), std::less());
106 thrust::host_vector<T> unique_ids( ids.size());
107 thrust::host_vector<int> ones( ids.size(), 1), howmany(ones);
108 auto new_end = thrust::reduce_by_key( ids.begin(), ids.end(),
109 ones.begin(), unique_ids.begin(), howmany.begin(), std::equal_to() );
110 uni.unique = thrust::host_vector<T>( unique_ids.begin(),
112 uni.howmany = thrust::host_vector<int>( howmany.begin(), new_end.second);
113 for(
unsigned i=0; i<uni.howmany.size(); i++)
114 uni.gather2.insert( uni.gather2.end(), uni.howmany[i], i);
117 uni.gather1 = invert_permutation( uni.gather1);
121template<
class host_vector >
122Unique<typename host_vector::value_type> find_unique_order_preserving(
123 const host_vector& unsorted_elements)
125 using T =
typename host_vector::value_type;
128 std::vector<std::vector<int>> sort;
129 for(
unsigned u=0; u<unsorted_elements.size(); u++)
131 auto it =thrust::find( uni.unique.begin(),
132 uni.unique.end(), unsorted_elements[u]);
133 if( it == uni.unique.end())
135 uni.unique.push_back( unsorted_elements[u]);
136 sort.push_back( std::vector<int>(1,u));
140 size_t idx = std::distance( uni.unique.begin(), it);
141 sort[idx].push_back( u);
145 for(
unsigned i=0; i<uni.unique.size(); i++)
147 uni.howmany.push_back( sort[i].size());
148 uni.gather2.insert( uni.gather2.end(), uni.howmany[i], i);
149 for(
int k=0; k<uni.howmany[i]; k++)
150 uni.gather1.push_back(sort[i][k]);
153 uni.gather1 = invert_permutation( uni.gather1);
163inline bool operator==(
const dg::detail::MsgChunk& a,
const dg::detail::MsgChunk& b)
165 return a.idx == b.idx and a.size == b.size;
168inline thrust::host_vector<MsgChunk>
169find_contiguous_chunks(
170 const thrust::host_vector<int>& in)
172 thrust::host_vector<MsgChunk> range;
173 int previous = std::numeric_limits<int>::max();
174 for(
unsigned u=0; u<in.size(); u++)
177 if( idx -1 == previous)
178 range.back().size ++;
180 range.push_back({idx,1});
187inline std::map<int,int> make_size_map(
const thrust::host_vector<int>& sizes)
189 std::map<int,int> out;
190 for(
unsigned u=0; u<sizes.size(); u++)
203auto flatten (
const T& t);
205inline thrust::host_vector<int> flatten (
const MsgChunk& t)
207 thrust::host_vector<int> flat(2);
208 flat[0] = t.idx, flat[1] = t.size;
215 return thrust::host_vector<T>(1, t);
217template<
class ContainerType>
221 decltype( flatten(t[0])) flat;
222 for(
unsigned u=0; u<t.size(); u++)
224 auto value = flatten(t[u]);
225 flat.insert(flat.end(), value.begin(), value.end());
231auto flatten (
const T& t)
234 return flatten( t, tensor_tag());
238template<
class Message>
239auto flatten_values(
const std::map<int,Message>& idx_map
242 using vector_type =
decltype( flatten( idx_map.at(0) ));
243 std::map<int,vector_type> flat;
244 for(
auto& idx : idx_map)
246 flat[idx.first] = flatten(idx.second);
252template<
class vector>
254 const std::map<int,vector>& idx_map
258 for(
auto& idx : idx_map)
259 flat.insert(flat.end(), idx.second.begin(), idx.second.end());
266inline void make_target(
267 const thrust::host_vector<int>& src, MsgChunk& target)
269 assert( src.size() == 2);
270 target = {src[0], src[1]};
272inline void make_target(
273 const thrust::host_vector<int>& src, thrust::host_vector<MsgChunk>& target)
275 assert( src.size() % 2 == 0);
277 for(
unsigned u=0; u<src.size()/2; u++)
278 target.push_back( { src[2*u], src[2*u+1]});
280template<
class T,
size_t N>
282 const thrust::host_vector<T>& src, std::array<T,N>& target)
284 assert( src.size() == N);
285 thrust::copy( src.begin(), src.end(), target.begin());
287template<
class T,
size_t N>
289 const thrust::host_vector<T>& src, thrust::host_vector<std::array<T,N>>& target)
291 assert( src.size() % N == 0);
293 for(
unsigned u=0; u<src.size()/N; u++)
296 thrust::copy( src.begin() + N*u, src.begin() + N*(u+1), t.begin());
297 target.push_back( t);
300template<
class Target,
class Source>
301void make_target(
const thrust::host_vector<Source>& src, Target& t, AnyScalarTag)
303 assert( src.size() == 1);
306template<
class Target,
class Source>
307void make_target(
const thrust::host_vector<Source>& src, Target& t, AnyVectorTag)
311template<
class Target,
class Source>
312void make_target(
const thrust::host_vector<Source>& src, Target& t)
315 make_target(src, t, tensor_tag() );
318template<
class Target,
class T>
319std::map<int, Target> make_map_t(
320 const thrust::host_vector<T>& flat_map,
321 const std::map<int,int>& size_map
325 std::map<int, Target> map;
327 for(
auto& size : size_map)
329 if( size.second != 0)
331 thrust::host_vector<T> partial(
332 flat_map.begin()+start, flat_map.begin() + start + size.second);
333 start += size.second;
334 make_target( partial, map[size.first]);
346std::map<int,int> get_size_map(
const std::map<
int, thrust::host_vector<T>>& idx_map)
348 std::map<int,int> out;
349 for(
auto& idx : idx_map)
350 out[idx.first] = idx.second.size();
369template<
class ArrayVec = thrust::host_vector<std::array<
int,2>>,
class IntVec = thrust::host_vector<
int>>
370std::map<int, IntVec> gIdx2unique_idx(
371 const ArrayVec& gIdx,
374 auto uni = detail::find_unique_order_preserving( gIdx);
376 IntVec pids(uni.unique.size()), lIdx(pids);
377 for(
int i=0; i<(int)pids.size(); i++)
379 pids[i] = uni.unique[i][0];
380 lIdx[i] = uni.unique[i][1];
382 auto uni_pids = detail::find_unique_stable_sort( pids);
383 bufferIdx = detail::combine_gather( uni.gather1,
384 detail::combine_gather( uni.gather2, uni_pids.gather1));
386 auto sorted_unique_gIdx = lIdx;
387 thrust::scatter( lIdx.begin(), lIdx.end(),
388 uni_pids.gather1.begin(), sorted_unique_gIdx.begin());
390 std::map<int,int> pids_howmany = detail::make_map( uni_pids.unique, uni_pids.howmany);
391 return detail::make_map_t<IntVec>( sorted_unique_gIdx, pids_howmany);
394template<
class ConversionPolicy,
class IntVec = thrust::host_vector<
int>>
395thrust::host_vector<std::array<int,2>> gIdx2gIdx(
const IntVec& gIdx,
const ConversionPolicy& p)
397 thrust::host_vector<std::array<int,2>> arrayIdx( gIdx.size());
398 for(
unsigned i=0; i<gIdx.size(); i++)
399 assert(p.global2localIdx(gIdx[i],
400 arrayIdx[i][1], arrayIdx[i][0]) );
425template<
class ConversionPolicy,
class IntVec = thrust::host_vector<
int>>
426std::map<int, IntVec> gIdx2unique_idx(
427 const IntVec& globalIndexMap,
429 const ConversionPolicy& p)
432 return gIdx2unique_idx( gIdx2gIdx(globalIndexMap), bufferIdx);
typename TensorTraits< std::decay_t< Vector > >::tensor_category get_tensor_category
Definition tensor_traits.h:47
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...
Scalar Tag base class.
Definition scalar_categories.h:11
Vector Tag base class, indicates the basic Vector/container concept.
Definition vector_categories.h:28