OPAL (Object Oriented Parallel Accelerator Library)  2021.1.99
OPAL
SOffset.h
Go to the documentation of this file.
1 // -*- C++ -*-
2 /***************************************************************************
3  *
4  * The IPPL Framework
5  *
6  *
7  * Visit http://people.web.psi.ch/adelmann/ for more details
8  *
9  ***************************************************************************/
10 
11 #ifndef SOFFSET_H
12 #define SOFFSET_H
13 
14 // include files
15 #include "Index/NDIndex.h"
16 #include "Utility/PAssert.h"
17 #include "Message/Message.h"
18 
19 #include <iostream>
20 
21 // forward declarations
22 template <unsigned Dim> class SOffset;
23 
24 template <unsigned Dim>
26 template <unsigned Dim>
27 SOffset<Dim> operator+(const SOffset<Dim>&, const int *);
28 template <unsigned Dim>
29 SOffset<Dim> operator+(const int *, const SOffset<Dim>&);
30 template <unsigned Dim>
32 template <unsigned Dim>
34 
35 template <unsigned Dim>
37 template <unsigned Dim>
38 SOffset<Dim> operator-(const SOffset<Dim>&, const int *);
39 template <unsigned Dim>
40 SOffset<Dim> operator-(const int *, const SOffset<Dim>&);
41 template <unsigned Dim>
43 template <unsigned Dim>
45 
46 template <unsigned Dim>
48 template <unsigned Dim>
49 SOffset<Dim> operator*(const SOffset<Dim>&, const int *);
50 template <unsigned Dim>
51 SOffset<Dim> operator*(const int *, const SOffset<Dim>&);
52 template <unsigned Dim>
54 template <unsigned Dim>
56 
57 template <unsigned Dim>
59 template <unsigned Dim>
60 SOffset<Dim> operator/(const SOffset<Dim>&, const int *);
61 template <unsigned Dim>
62 SOffset<Dim> operator/(const int *, const SOffset<Dim>&);
63 template <unsigned Dim>
65 
66 template <unsigned Dim>
68 
69 template <unsigned Dim>
70 std::ostream& operator<<(std::ostream&, const SOffset<Dim>&);
71 
72 /***********************************************************************
73  *
74  * SOffset is a simple class which stores an array of N integers
75  * for use by the SIndex class. It represents an integer offset
76  * for the numbers in the SIndex set of indices. Otherwise, it acts
77  * just like a set of N integers. The class is templated on the number
78  * of elements N.
79  *
80  ***********************************************************************/
81 
82 template <unsigned Dim>
83 class SOffset {
84 
85 public:
86  typedef int * iterator;
87  typedef const int * const_iterator;
88 
89 public:
90  // constructors: note the default constructor initializes data to 0
91  SOffset() { for (unsigned int d=0; d < Dim; v[d++] = 0); }
92  SOffset(int v0) { for (unsigned int d=0; d < Dim; v[d++] = v0); }
93  SOffset(int v0, int v1); // only works for a 2D SOffset
94  SOffset(int v0, int v1, int v2); // only works for a 3D SOffset
95  SOffset(int v0, int v1, int v2, int v3); // only works for a 4D SOffset
96  SOffset(int v0, int v1, int v2, int v3, int v4); // only for 5D SOffset
97  SOffset(int v0, int v1, int v2, int v3, int v4, int v5); // only for 6D
98  SOffset(const int *vv); // Takes array of offsets, for any dimensionality
100 
101  // copy and index operators
103  SOffset<Dim>& operator=(const int *);
104  int& operator[](unsigned int d) { return v[d]; }
105  const int& operator[](unsigned int d) const { return v[d]; }
106 
107  // comparison operators
108  bool operator==(const SOffset<Dim>&) const;
109  bool operator<(const SOffset<Dim>&) const;
110  bool operator!=(const SOffset<Dim>& a) const { return !(*this == a); }
111  bool operator<=(const SOffset<Dim>& a) const { return !(*this > a); }
112  bool operator>=(const SOffset<Dim>& a) const { return !(*this < a); }
113  bool operator>(const SOffset<Dim>& a) const {
114  return !(*this < a || *this == a);
115  }
116 
117  // arithmetic operators
119  SOffset<Dim>& operator+=(const int *);
121  SOffset<Dim>& operator-=(const int *);
123  SOffset<Dim>& operator*=(const int *);
125  SOffset<Dim>& operator/=(const int *);
126 
127  // put or get to/from a Message
130 
131  // iterators for the offset, and information about the data
132  iterator begin() { return v; }
133  iterator end() { return (v + Dim); }
134  const_iterator begin() const { return v; }
135  const_iterator end() const { return (v + Dim); }
136  unsigned int size() const { return Dim; }
137 
138  // is this point within an NDIndex object (includes the edges) ?
139  bool inside(const NDIndex<Dim>&) const;
140 
141 private:
142  int v[Dim];
143 };
144 
145 
146 /***********************************************************************
147  *
148  * Inlined constructors and SOffset functions.
149  *
150  ***********************************************************************/
151 
152 template<unsigned int Dim>
153 inline
154 SOffset<Dim>::SOffset(int v0, int v1) {
155  CTAssert(Dim==2);
156  v[0] = v0;
157  v[1] = v1;
158 }
159 
160 template<unsigned int Dim>
161 inline
162 SOffset<Dim>::SOffset(int v0, int v1, int v2) {
163  CTAssert(Dim==3);
164  v[0] = v0;
165  v[1] = v1;
166  v[2] = v2;
167 }
168 
169 template<unsigned int Dim>
170 inline
171 SOffset<Dim>::SOffset(int v0, int v1, int v2, int v3) {
172  CTAssert(Dim==4);
173  v[0] = v0;
174  v[1] = v1;
175  v[2] = v2;
176  v[3] = v3;
177 }
178 
179 template<unsigned int Dim>
180 inline
181 SOffset<Dim>::SOffset(int v0, int v1, int v2, int v3, int v4) {
182  CTAssert(Dim==5);
183  v[0] = v0;
184  v[1] = v1;
185  v[2] = v2;
186  v[3] = v3;
187  v[4] = v4;
188 }
189 
190 template<unsigned int Dim>
191 inline
192 SOffset<Dim>::SOffset(int v0, int v1, int v2, int v3, int v4, int v5) {
193  CTAssert(Dim==6);
194  v[0] = v0;
195  v[1] = v1;
196  v[2] = v2;
197  v[3] = v3;
198  v[4] = v4;
199  v[5] = v5;
200 }
201 
202 template<unsigned int Dim>
203 inline bool
205  for (unsigned int i=0; i < Dim; ++i)
206  if (v[i] != a.v[i])
207  return false;
208  return true;
209 }
210 
211 template<unsigned int Dim>
212 inline bool
213 SOffset<Dim>::operator<(const SOffset<Dim>& a) const {
214  for (unsigned int i=0; i < Dim; ++i) {
215  if (v[i] < a.v[i])
216  return true;
217  else if (v[i] > a.v[i])
218  return false;
219  }
220 
221  // if we're here, they're exactly equal
222  return false;
223 }
224 
225 template<unsigned int Dim>
226 inline Message&
228  ::putMessage(m, v, v + Dim);
229  return m;
230 }
231 
232 template<unsigned int Dim>
233 inline Message&
235  ::getMessage_iter(m, v);
236  return m;
237 }
238 
239 template<unsigned int Dim>
240 inline std::ostream&
241 operator<<(std::ostream& o, const SOffset<Dim>& a) {
242  o << "[";
243  for (unsigned int i=0; i < Dim; ++i)
244  o << a[i] << (i < (Dim-1) ? "," : "");
245  o << "]";
246  return o;
247 }
248 
249 // is this point within an NDIndex object (includes the edges) ?
250 // NOTE: this intersects with the whole region of the NDIndex, basically
251 // assuming unit stride.
252 template<unsigned int Dim>
253 inline bool
255  for (unsigned int d=0; d < Dim; ++d) {
256  if (v[d] < ndi[d].first() || v[d] > ndi[d].last())
257  return false;
258  }
259  return true;
260 }
261 
262 
263 /***********************************************************************
264  *
265  * Template classes and functions to unroll the loops into subloops of
266  * a maximum of 3 elements.
267  * This is taken from the Utility/Vec.h class with modifications.
268  *
269  ***********************************************************************/
270 
271 // a tag class used to indicate how many elements and when to split a loop
272 template<unsigned int Dim, int Flag>
274 };
275 
276 // simple functors for performing operations on SOffset elements
278  void operator()(int& a, int b) { a = b; }
279 };
280 
282  void operator()(int& a, int b) { a += b; }
283 };
284 
286  void operator()(int& a, int b) { a -= b; }
287 };
288 
290  void operator()(int& a, int b) { a *= b; }
291 };
292 
294  void operator()(int& a, int b) { a /= b; }
295 };
296 
297 
298 // Operate on a vector of dim 4 or greater. Split it in half and do each half.
299 // Do it this way so that the depth of inlining is O(log(Dim)).
300 template<class T1, class T2, class Op, unsigned int L>
301 inline void
303  divide_soffset_op(p1, p2, op, DivideSOffsetCopyTag<L/2, (L>=8 ? 4 : L/2)>());
304  divide_soffset_op(p1+(L/2), p2+(L/2), op,
305  DivideSOffsetCopyTag<L-L/2, (L>=8?4:L-L/2)>());
306 }
307 
308 
309 // Modify a vector of length 0, 1, 2 or 3.
310 template<class T1, class T2, class Op>
311 inline void
313  op(p1[0], p2[0]);
314  op(p1[1], p2[1]);
315  op(p1[2], p2[2]);
316 }
317 
318 template<class T1, class T2, class Op>
319 inline void
321  op(p1[0], p2[0]);
322  op(p1[1], p2[1]);
323 }
324 
325 template<class T1, class T2, class Op>
326 inline void
328  op(*p1, *p2);
329 }
330 
331 template<class T1, class T2, class Op>
332 inline void
334 }
335 
336 //
337 // The SOffset operators just call divide_soffset_op to do the operation
338 //
339 
340 template<unsigned int L>
341 inline
343 {
345  DivideSOffsetCopyTag<L,( L>=4 ? 4 : L)>() );
346 }
347 
348 template<unsigned int L>
349 inline
350 SOffset<L>::SOffset(const int *SO)
351 {
353  DivideSOffsetCopyTag<L,( L>=4 ? 4 : L)>() );
354 }
355 
356 template<unsigned int L>
357 inline SOffset<L>&
359 {
360  if ( this != &SO )
362  DivideSOffsetCopyTag<L,( L>=4 ? 4 : L)>() );
363  return *this;
364 }
365 
366 template<unsigned int L>
367 inline SOffset<L>&
368 SOffset<L>::operator=(const int *SO)
369 {
371  DivideSOffsetCopyTag<L,( L>=4 ? 4 : L)>() );
372  return *this;
373 }
374 
375 template<unsigned int L>
376 inline SOffset<L>&
378 {
380  DivideSOffsetCopyTag<L,( L>=4 ? 4 : L)>() );
381  return *this;
382 }
383 
384 template<unsigned int L>
385 inline SOffset<L>&
387 {
389  DivideSOffsetCopyTag<L,( L>=4 ? 4 : L)>() );
390  return *this;
391 }
392 
393 template<unsigned int L>
394 inline SOffset<L>&
396 {
398  DivideSOffsetCopyTag<L,( L>=4 ? 4 : L)>() );
399  return *this;
400 }
401 
402 template<unsigned int L>
403 inline SOffset<L>&
405 {
407  DivideSOffsetCopyTag<L,( L>=4 ? 4 : L)>() );
408  return *this;
409 }
410 
411 template<unsigned int L>
412 inline SOffset<L>&
414 {
416  DivideSOffsetCopyTag<L,( L>=4 ? 4 : L)>() );
417  return *this;
418 }
419 
420 template<unsigned int L>
421 inline SOffset<L>&
423 {
425  DivideSOffsetCopyTag<L,( L>=4 ? 4 : L)>() );
426  return *this;
427 }
428 
429 template<unsigned int L>
430 inline SOffset<L>&
432 {
434  DivideSOffsetCopyTag<L,( L>=4 ? 4 : L)>() );
435  return *this;
436 }
437 
438 template<unsigned int L>
439 inline SOffset<L>&
441 {
443  DivideSOffsetCopyTag<L,( L>=4 ? 4 : L)>() );
444  return *this;
445 }
446 
447 
448 //
449 // binary operators
450 // SOffset +-*/ SOffset --> SOffset
451 // SOffset +-*/ int * --> SOffset
452 // SOffset +-*/ NDIndex --> NDIndex
453 //
454 
455 template<unsigned int L>
456 inline SOffset<L>
457 operator+(const SOffset<L>& a, const SOffset<L>& b)
458 {
459  SOffset<L> retval(a);
460  return (retval += b);
461 }
462 
463 template<unsigned int L>
464 inline SOffset<L>
465 operator+(const SOffset<L>& a, const int *b)
466 {
467  SOffset<L> retval(a);
468  return (retval += b);
469 }
470 
471 template<unsigned int L>
472 inline SOffset<L>
473 operator+(const int *b, const SOffset<L>& a)
474 {
475  SOffset<L> retval(a);
476  return (retval += b);
477 }
478 
479 template<unsigned int L>
480 inline NDIndex<L>
481 operator+(const SOffset<L>& a, const NDIndex<L>& b)
482 {
483  NDIndex<L> retval;
484  for (unsigned int d=0; d < L; ++d)
485  retval[d] = b[d] + a[d];
486  return retval;
487 }
488 
489 template<unsigned int L>
490 inline NDIndex<L>
491 operator+(const NDIndex<L>& b, const SOffset<L>& a)
492 {
493  NDIndex<L> retval;
494  for (unsigned int d=0; d < L; ++d)
495  retval[d] = b[d] + a[d];
496  return retval;
497 }
498 
499 template<unsigned int L>
500 inline SOffset<L>
501 operator-(const SOffset<L>& a, const SOffset<L>& b)
502 {
503  SOffset<L> retval(a);
504  return (retval -= b);
505 }
506 
507 template<unsigned int L>
508 inline SOffset<L>
509 operator-(const SOffset<L>& a, const int *b)
510 {
511  SOffset<L> retval(a);
512  return (retval -= b);
513 }
514 
515 template<unsigned int L>
516 inline SOffset<L>
517 operator-(const int *b, const SOffset<L>& a)
518 {
519  SOffset<L> retval(a);
520  return (retval -= b);
521 }
522 
523 template<unsigned int L>
524 inline NDIndex<L>
525 operator-(const SOffset<L>& a, const NDIndex<L>& b)
526 {
527  NDIndex<L> retval;
528  for (int d=0; d < L; ++d)
529  retval[d] = b[d] - a[d];
530  return retval;
531 }
532 
533 template<unsigned int L>
534 inline NDIndex<L>
535 operator-(const NDIndex<L>& b, const SOffset<L>& a)
536 {
537  NDIndex<L> retval;
538  for (int d=0; d < L; ++d)
539  retval[d] = b[d] - a[d];
540  return retval;
541 }
542 
543 template<unsigned int L>
544 inline SOffset<L>
545 operator*(const SOffset<L>& a, const SOffset<L>& b)
546 {
547  SOffset<L> retval(a);
548  return (retval *= b);
549 }
550 
551 template<unsigned int L>
552 inline SOffset<L>
553 operator*(const SOffset<L>& a, const int *b)
554 {
555  SOffset<L> retval(a);
556  return (retval *= b);
557 }
558 
559 template<unsigned int L>
560 inline SOffset<L>
561 operator*(const int *b, const SOffset<L>& a)
562 {
563  SOffset<L> retval(a);
564  return (retval *= b);
565 }
566 
567 template<unsigned int L>
568 inline NDIndex<L>
569 operator*(const SOffset<L>& a, const NDIndex<L>& b)
570 {
571  NDIndex<L> retval;
572  for (unsigned int d=0; d < L; ++d)
573  retval[d] = b[d] * a[d];
574  return retval;
575 }
576 
577 template<unsigned int L>
578 inline NDIndex<L>
579 operator*(const NDIndex<L>& b, const SOffset<L>& a)
580 {
581  NDIndex<L> retval;
582  for (unsigned int d=0; d < L; ++d)
583  retval[d] = b[d] * a[d];
584  return retval;
585 }
586 
587 template<unsigned int L>
588 inline SOffset<L>
589 operator/(const SOffset<L>& a, const SOffset<L>& b)
590 {
591  SOffset<L> retval(a);
592  return (retval /= b);
593 }
594 
595 template<unsigned int L>
596 inline SOffset<L>
597 operator/(const SOffset<L>& a, const int *b)
598 {
599  SOffset<L> retval(a);
600  return (retval /= b);
601 }
602 
603 template<unsigned int L>
604 inline SOffset<L>
605 operator/(const int *b, const SOffset<L>& a)
606 {
607  SOffset<L> retval(a);
608  return (retval /= b);
609 }
610 
611 template<unsigned int L>
612 inline NDIndex<L>
613 operator/(const NDIndex<L>& b, const SOffset<L>& a)
614 {
615  NDIndex<L> retval;
616  for (int d=0; d < L; ++d)
617  retval[d] = b[d] / a[d];
618  return retval;
619 }
620 
621 
622 //
623 // unary operators
624 //
625 
626 template<unsigned int L>
627 inline SOffset<L>&
629 {
630  SOffset<L> retval;
631  return (retval -= a);
632 }
633 
634 #endif // SOFFSET_H
635 
636 /***************************************************************************
637  * $RCSfile: SOffset.h,v $ $Author: adelmann $
638  * $Revision: 1.1.1.1 $ $Date: 2003/01/23 07:40:27 $
639  * IPPL_VERSION_ID: $Id: SOffset.h,v 1.1.1.1 2003/01/23 07:40:27 adelmann Exp $
640  ***************************************************************************/
const unsigned Dim
SOffset< Dim > operator/(const SOffset< Dim > &, const SOffset< Dim > &)
std::ostream & operator<<(std::ostream &, const SOffset< Dim > &)
Definition: SOffset.h:241
SOffset< Dim > operator*(const SOffset< Dim > &, const SOffset< Dim > &)
SOffset< Dim > operator+(const SOffset< Dim > &, const SOffset< Dim > &)
void divide_soffset_op(T1 *p1, T2 *p2, Op op, DivideSOffsetCopyTag< L, 4 >)
Definition: SOffset.h:302
SOffset< Dim > operator-(const SOffset< Dim > &, const SOffset< Dim > &)
void putMessage(Message &m, const T &t)
Definition: Message.h:549
void getMessage_iter(Message &m, OutputIterator o)
Definition: Message.h:595
std::complex< double > a
#define CTAssert(c)
Definition: PAssert.h:35
int & operator[](unsigned int d)
Definition: SOffset.h:104
bool inside(const NDIndex< Dim > &) const
Definition: SOffset.h:254
SOffset(const SOffset< Dim > &)
const int & operator[](unsigned int d) const
Definition: SOffset.h:105
bool operator>(const SOffset< Dim > &a) const
Definition: SOffset.h:113
SOffset< Dim > & operator=(const SOffset< Dim > &)
SOffset< Dim > & operator*=(const SOffset< Dim > &)
iterator begin()
Definition: SOffset.h:132
const int * const_iterator
Definition: SOffset.h:87
SOffset(int v0)
Definition: SOffset.h:92
int * iterator
Definition: SOffset.h:86
Message & getMessage(Message &)
Definition: SOffset.h:234
const_iterator begin() const
Definition: SOffset.h:134
bool operator!=(const SOffset< Dim > &a) const
Definition: SOffset.h:110
SOffset< Dim > & operator+=(const SOffset< Dim > &)
bool operator<(const SOffset< Dim > &) const
Definition: SOffset.h:213
SOffset()
Definition: SOffset.h:91
unsigned int size() const
Definition: SOffset.h:136
int v[Dim]
Definition: SOffset.h:142
Message & putMessage(Message &)
Definition: SOffset.h:227
const_iterator end() const
Definition: SOffset.h:135
bool operator>=(const SOffset< Dim > &a) const
Definition: SOffset.h:112
bool operator<=(const SOffset< Dim > &a) const
Definition: SOffset.h:111
SOffset< Dim > & operator/=(const SOffset< Dim > &)
SOffset< Dim > & operator-=(const SOffset< Dim > &)
iterator end()
Definition: SOffset.h:133
bool operator==(const SOffset< Dim > &) const
Definition: SOffset.h:204
void operator()(int &a, int b)
Definition: SOffset.h:278
void operator()(int &a, int b)
Definition: SOffset.h:282
void operator()(int &a, int b)
Definition: SOffset.h:286
void operator()(int &a, int b)
Definition: SOffset.h:290
void operator()(int &a, int b)
Definition: SOffset.h:294