Loading [MathJax]/extensions/tex2jax.js

DUNE MultiDomainGrid (2.8)

subdomainset.hh
1#ifndef DUNE_MULTIDOMAINGRID_SUBDOMAINSET_HH
2#define DUNE_MULTIDOMAINGRID_SUBDOMAINSET_HH
3
4#include <algorithm>
5#include <cassert>
6#include <cstddef>
7#include <cstdint>
8#include <limits>
9#include <strings.h>
10#include <dune/common/typetraits.hh>
11#include <dune/common/iteratorfacades.hh>
12#include <dune/common/exceptions.hh>
13
14#include <dune/grid/multidomaingrid/utility.hh>
15
16namespace Dune {
17
18namespace mdgrid {
19
20// forward declarations
21template<typename SubDomainIndex, std::size_t capacity>
22class IntegralTypeSubDomainSet;
23
24template<typename SubDomainIndex, std::size_t capacity>
25bool setContains(const IntegralTypeSubDomainSet<SubDomainIndex,capacity>& a,
26 const IntegralTypeSubDomainSet<SubDomainIndex,capacity>& b);
27
28template<typename SubDomainIndex, std::size_t capacity>
29void setAdd(IntegralTypeSubDomainSet<SubDomainIndex,capacity>& a,
30 const IntegralTypeSubDomainSet<SubDomainIndex,capacity>& b);
31
33
34// \internal
35namespace sds_detail {
36
38 template<typename T>
39 struct Candidate;
40
42 template<>
43 struct Candidate<uint8_t> {
44 typedef uint8_t type;
45 typedef Candidate<uint16_t> next_candidate;
46 };
47
49 template<>
50 struct Candidate<uint16_t> {
51 typedef uint16_t type;
52 typedef Candidate<uint32_t> next_candidate;
53 };
54
56 template<>
57 struct Candidate<uint32_t> {
58 typedef uint32_t type;
59 typedef Candidate<uint64_t> next_candidate;
60 };
61
63 template<>
64 struct Candidate<uint64_t> {
65 typedef uint64_t type;
66 typedef void next_candidate;
67 };
68
70 template<std::size_t capacity, typename candidate>
71 struct SetStorageTester {
72
73 static_assert(std::numeric_limits<typename candidate::type>::is_specialized,"numeric_limits<> lacks specialization");
74
75 typedef typename std::conditional<capacity <= static_cast<std::size_t>(std::numeric_limits<typename candidate::type>::digits),
76 typename candidate::type,
77 typename SetStorageTester<capacity,
78 typename candidate::next_candidate
79 >::type
80 >::type type;
81
82 static_assert((!std::is_same<type,void>::value),"unsupported maximum number of subdomains");
83
84 };
85
89 template<std::size_t capacity>
90 struct SetStorageTester<capacity,void> {
91
92 typedef void type;
93
94 };
95
97 template<std::size_t capacity>
98 struct SetStorageChooser {
99 typedef typename SetStorageTester<capacity,Candidate<uint8_t> >::type type;
100 };
101
103 template<typename T>
104 class Log2 {
105
106 private:
107 inline static std::size_t doCalculation(T value);
108
109 public:
110
111 inline static std::size_t calculate(T value) {
112 assert(value != 0 && (value & (value-1)) == 0); // make sure value is a power of 2
113 return doCalculation(value);
114 }
115 };
116
118 template<>
119 inline std::size_t Log2<uint8_t>::doCalculation(uint8_t value) {
120 return __builtin_ffs(value)-1;
121 }
122
124 template<>
125 inline std::size_t Log2<uint16_t>::doCalculation(uint16_t value) {
126 return __builtin_ffs(value)-1;
127 }
128
130 template<>
131 inline std::size_t Log2<uint32_t>::doCalculation(uint32_t value) {
132 return __builtin_ffsl(value)-1;
133 }
134
136 template<>
137 inline std::size_t Log2<uint64_t>::doCalculation(uint64_t value) {
138 return __builtin_ffsll(value)-1;
139 }
140
142 template<typename SubDomainIndex, typename SetStorage>
143 class Iterator : public ForwardIteratorFacade<Iterator<SubDomainIndex,SetStorage>,
144 SubDomainIndex,
145 SubDomainIndex,
146 std::ptrdiff_t> {
147
148 template<typename,std::size_t>
149 friend class ::Dune::mdgrid::IntegralTypeSubDomainSet;
150
151 public:
152
153 typedef Iterator<SubDomainIndex,SetStorage> ThisType;
154 static const SetStorage base = 1;
155
156 SubDomainIndex dereference() const {
157 assert(_state != 0);
158 return _value;
159 }
160
161 bool equals(const ThisType& rhs) const {
162 return _state == rhs._state;
163 }
164
165 void increment() {
166 _state &= ~(base << _value);
167 if (_state != 0) {
168 findNextValue();
169 }
170 }
171
172 private:
173
174 void findNextValue() {
175 SetStorage lowestBit = _state & ((~_state) + 1);
176 _value = Log2<SetStorage>::calculate(lowestBit);
177 }
178
179 explicit Iterator(SetStorage state) :
180 _state(state),
181 _value(0)
182 {
183 if (_state != 0)
184 findNextValue();
185 }
186
187 // optimized constructor for end iterator
188 explicit Iterator() :
189 _state(0),
190 _value(0)
191 {}
192
193 SetStorage _state;
194 SubDomainIndex _value;
195
196 };
197
198}
199
201
202
203template<typename SubDomainIndexT, std::size_t capacity>
204class IntegralTypeSubDomainSet {
205
206 friend bool setContains<>(const IntegralTypeSubDomainSet<SubDomainIndexT,capacity>& a,
207 const IntegralTypeSubDomainSet<SubDomainIndexT,capacity>& b);
208
209 friend void setAdd<>(IntegralTypeSubDomainSet<SubDomainIndexT,capacity>& a,
210 const IntegralTypeSubDomainSet<SubDomainIndexT,capacity>& b);
211
212 typedef typename sds_detail::SetStorageChooser<capacity>::type SetStorage;
213 static const SetStorage base = 1;
214
215public:
216 static const std::size_t maxSize = capacity;
217 typedef SubDomainIndexT SubDomainIndex;
218 typedef sds_detail::Iterator<SubDomainIndex,SetStorage> Iterator;
219 typedef IntegralTypeSubDomainSet<SubDomainIndex,capacity> This;
220
221 struct DataHandle
222 {
223 typedef SetStorage DataType;
224
225 static bool fixedSize(int dim, int codim)
226 {
227 return true;
228 }
229
230 static std::size_t size(const IntegralTypeSubDomainSet& sds)
231 {
232 return 1;
233 }
234
235 template<typename MessageBufferImp>
236 static void gather(MessageBufferImp& buf, const IntegralTypeSubDomainSet& sds)
237 {
238 buf.write(sds._set);
239 }
240
241 template<typename MessageBufferImp>
242 static void scatter(MessageBufferImp& buf, IntegralTypeSubDomainSet& sds, std::size_t n)
243 {
244 IntegralTypeSubDomainSet h;
245 buf.read(h._set);
246 sds.addAll(h);
247 }
248
249 };
250
251 enum SetState {emptySet,simpleSet,multipleSet};
252
253 Iterator begin() const {
254 return Iterator(_set);
255 }
256
257 Iterator end() const {
258 return Iterator();
259 }
260
261 bool contains(SubDomainIndex domain) const {
262 assert(domain < maxSize);
263 return (base << domain) & _set;
264 }
265
266 template<typename Set>
267 bool containsAll(const Set& set) const {
268 return setContains(*this,set);
269 }
270
271 void difference(const IntegralTypeSubDomainSet& minuend, const IntegralTypeSubDomainSet& subtrahend)
272 {
273 _set = minuend._set & ~subtrahend._set;
274 }
275
276 bool simple() const {
277 return (!empty()) && ((_set & (_set - 1)) == 0);
278 }
279
280 bool empty() const {
281 return _set == 0;
282 }
283
284 SetState state() const {
285 return (_set == 0 ? emptySet : (_set & (_set - 1)) == 0 ? simpleSet : multipleSet);
286 }
287
288 std::size_t size() const {
289 std::size_t c = 0;
290 for (SetStorage t = _set; t; ++c) {
291 t &= t - 1;
292 }
293 return c;
294 }
295
296 void clear() {
297 _set = 0;
298 }
299
300 void add(SubDomainIndex domain) {
301 assert(domain < maxSize);
302 _set |= base << domain;
303 }
304
305 void remove(SubDomainIndex domain) {
306 assert(domain < maxSize);
307 _set &= ~(base << domain);
308 }
309
310 void set(SubDomainIndex domain) {
311 assert(domain < maxSize);
312 _set = base << domain;
313 }
314
315 template<typename Set>
316 void addAll(const Set& rhs) {
317 setAdd(*this,rhs);
318 }
319
320 int domainOffset(SubDomainIndex domain) const {
321 assert(domain >= 0 && domain < maxSize);
322 return domain;
323 }
324
325 IntegralTypeSubDomainSet() :
326 _set(0)
327 {}
328
329 bool operator==(const IntegralTypeSubDomainSet& r) const {
330 return _set == r._set;
331 }
332
333 bool operator!=(const IntegralTypeSubDomainSet& r) const {
334 return !operator==(r);
335 }
336
337private:
338 SetStorage _set;
339
340};
341
342
343template<typename A, typename B>
344inline bool setContains(const A& a, const B& b) {
345 return std::all_of(b.begin(),b.end(),[&a](decltype(*(b.begin())) i) { return a.contains(i); });
346}
347
348template<typename A, typename B>
349inline void setAdd(A& a, const B& b) {
350 std::for_each(b.begin(),b.end(),[&a](decltype(*(b.begin())) i) { a.add(i); });
351}
352
353
354template<typename SubDomainIndex, std::size_t capacity>
355inline bool setContains(const IntegralTypeSubDomainSet<SubDomainIndex,capacity>& a,
356 const IntegralTypeSubDomainSet<SubDomainIndex,capacity>& b) {
357 return (a._set & b._set) == b._set;
358}
359
360template<typename SubDomainIndex, std::size_t capacity>
361inline void setAdd(IntegralTypeSubDomainSet<SubDomainIndex,capacity>& a,
362 const IntegralTypeSubDomainSet<SubDomainIndex,capacity>& b) {
363 a._set |= b._set;
364}
365
366} // namespace mdgrid
367
368} // namespace Dune
369
370#endif // DUNE_MULTIDOMAINGRID_SUBDOMAINSET_HH
Creative Commons License   |  Legal Statements / Impressum  |  Hosted by TU Dresden & Uni Heidelberg  |  generated with Hugo v0.111.3 (Apr 3, 22:46, 2025)