hypercube.h
1 /* Copyright (C) 2012-2020 IBM Corp.
2  * This program is Licensed under the Apache License, Version 2.0
3  * (the "License"); you may not use this file except in compliance
4  * with the License. You may obtain a copy of the License at
5  * http://www.apache.org/licenses/LICENSE-2.0
6  * Unless required by applicable law or agreed to in writing, software
7  * distributed under the License is distributed on an "AS IS" BASIS,
8  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
9  * See the License for the specific language governing permissions and
10  * limitations under the License. See accompanying LICENSE file.
11  */
16 #ifndef HELIB_HYPERCUBE_H
17 #define HELIB_HYPERCUBE_H
18 
19 #include <helib/NumbTh.h>
20 
21 namespace helib {
22 
23 class PAlgebra; // forward declaration
24 
28 {
29 private:
30  NTL::Vec<long> dims; // dims[i] is the size along the i'th dimension
31  NTL::Vec<long> prods; // prods[i] = \prod_{j=i}^{n-1} dims[i]
32 
33 public:
34  CubeSignature() {} // a nullptr signature
35 
36  void initSignature(const long _dims[], long _ndims)
37  {
38 
39  // can only initialize a nullptr signature
40  assertEq(dims.length(),
41  0l,
42  "Can only initialize an un-initialized signature");
43 
44  assertTrue<InvalidArgument>(_ndims >= 0l, "Dimension count is negative");
45 
46  dims.SetLength(_ndims);
47  prods.SetLength(_ndims + 1);
48  prods[_ndims] = 1;
49  for (long i = _ndims - 1; i >= 0; i--) {
50  assertTrue<InvalidArgument>(
51  _dims[i] > 0,
52  "Bad (non-positive) cube signature dimension");
53  dims[i] = _dims[i];
54  prods[i] = prods[i + 1] * _dims[i];
55  }
56  }
57  // VecType is either std::vector<intType> or NTL:Vec<intType>
58  template <typename VecType>
59  void initSignature(const VecType& _dims)
60  {
61  initSignature(_dims.data(), lsize(_dims));
62  }
63 
64  CubeSignature(const long _dims[], long _ndims)
65  {
66  initSignature(_dims, _ndims);
67  }
68  CubeSignature(const NTL::Vec<long>& _dims) { initSignature(_dims); }
69  CubeSignature(const std::vector<long>& _dims) { initSignature(_dims); }
70 
71  /* TODO: We could #include <initializer_list> and then do e.g.,
72  * CubeSignature s {1,2,3};
73  CubeSignature(std::initializer_list<long> _dims)
74  {
75  dims.SetLength(_dims.size());
76  long i;
77  std::initializer_list<long>::iterator it;
78  for (i=0, it=_dims.begin(); it!=_dims.end(); ++i, ++it)
79  dims[i] = *it;
80 
81  [...] // continue as above, initialize prods
82  }
83  */
84 
86  long getNumDims() const { return dims.length(); }
87 
89  long getSize() const { return ((getNumDims() > 0) ? prods[0] : 1); }
90 
92  long getDim(long d) const { return dims.at(d); }
93 
95  long getProd(long d) const { return prods.at(d); }
96 
98  long getProd(long from, long to) const
99  {
100  return prods.at(from) / prods.at(to);
101  }
102 
104  long getCoord(long i, long d) const
105  {
106  assertInRange(i,
107  0l,
108  getSize(),
109  "Coordinate does not exist (index i out of range)");
110 
111  return (i % prods.at(d)) / prods.at(d + 1);
112  }
113 
115  long addCoord(long i, long d, long offset) const
116  {
117  assertInRange(i,
118  0l,
119  getSize(),
120  "Coordinate does not exist (index i out of range)");
121 
122  offset = offset % dims.at(d);
123  if (offset < 0)
124  offset += dims.at(d);
125 
126  long i_d = getCoord(i, d);
127  long i_d1 = (i_d + offset) % dims.at(d);
128 
129  long i1 = i + (i_d1 - i_d) * prods.at(d + 1);
130 
131  return i1;
132  }
133 
137  template <typename VecType>
138  bool incrementCoords(VecType& v) const
139  {
140  for (long i = getNumDims() - 1; i >= 0; --i) {
141  if (i >= lsize(v))
142  continue; // sanity check
143 
144  // increment current index, set all the ones after it to zero
145  if (long(v[i]) < getDim(i) - 1) {
146  v[i]++;
147  for (long j = i + 1; j < lsize(v); j++)
148  v[j] = 0;
149  return true; // succeeded in incrementing the vector
150  }
151  // if buffer[i] >= getDim(i)-1, move to previous index i
152  }
153  return false; // cannot increment the vector anymore
154  }
155 
158  template <typename VecType>
159  void getAllCoords(VecType& v, long i) const
160  {
161  assertInRange(i,
162  0l,
163  getSize(),
164  "Coordinate does not exist (index i out of range)");
165 
166  resize(v, getNumDims()); // resize(*), lsize(*) defined in NumbTh.h
167  for (long j = getNumDims() - 1; j >= 0; --j) {
168  v[j] = i % getDim(j);
169  i = (i - v[j]) / getDim(j);
170  }
171  }
172 
175  template <typename VecType>
176  long assembleCoords(VecType& v) const
177  {
178  assertEq(lsize(v),
179  getNumDims(),
180  "Vector size is different to the number of dimensions");
181 
182  long idx = 0;
183  for (long i = 0; i < getNumDims(); i++) {
184  idx += v[i] * prods[i + 1];
185  }
186  return idx;
187  }
188 
190  long numSlices(long d = 1) const { return getProd(0, d); }
191 
193  long sliceSize(long d = 1) const { return getProd(d); }
194 
196  long numCols() const { return getProd(1); }
197 
200  std::pair<long, long> breakIndexByDim(long idx, long dim) const;
201 
203  long assembleIndexByDim(std::pair<long, long> idx, long dim) const;
204 
205  friend std::ostream& operator<<(std::ostream& s, const CubeSignature& sig);
206 };
207 
208 inline std::ostream& operator<<(std::ostream& s, const CubeSignature& sig)
209 {
210  return s << sig.dims;
211 }
212 
219 template <typename T>
221 {
222 private:
223  const CubeSignature& sig;
224  NTL::Vec<T> data;
225 
226  HyperCube(); // disable default constructor
227 
228 public:
229  // Default copy-constructor
230  HyperCube(const HyperCube& other) = default;
231 
233  HyperCube(const CubeSignature& _sig) : sig(_sig)
234  {
235  data.FixLength(sig.getSize());
236  }
237 
238  // use default copy constructor
239 
242  {
243  assertEq(&this->sig,
244  &other.sig,
245  "Cannot assign HyperCubes with different signatures");
246  data = other.data;
247  return *this;
248  }
249 
251  bool operator==(const HyperCube<T>& other) const
252  {
253  assertEq(&this->sig,
254  &other.sig,
255  "Cannot compare HyperCubes with different signatures");
256  return data == other.data;
257  }
258 
259  bool operator!=(const HyperCube<T>& other) const { return !(*this == other); }
260 
262  const CubeSignature& getSig() const { return sig; }
263 
267  NTL::Vec<T>& getData() { return data; }
268 
270  const NTL::Vec<T>& getData() const { return data; }
271 
273  long getSize() const { return sig.getSize(); }
274 
276  long getNumDims() const { return sig.getNumDims(); }
277 
279  long getDim(long d) const { return sig.getDim(d); }
280 
282  long getProd(long d) const { return sig.getProd(d); }
283 
285  long getProd(long from, long to) const { return sig.getProd(from, to); }
286 
288  long getCoord(long i, long d) const { return sig.getCoord(i, d); }
289 
291  long addCoord(long i, long d, long offset) const
292  {
293  return sig.addCoord(i, d, offset);
294  }
295 
297  long numSlices(long d = 1) const { return getProd(0, d); }
298 
300  long sliceSize(long d = 1) const { return getProd(d); }
301 
303  long numCols() const { return getProd(1); }
304 
306  T& at(long i) { return data.at(i); }
307 
309  T& operator[](long i) { return data[i]; }
310 
312  const T& at(long i) const { return data.at(i); }
313 
315  const T& operator[](long i) const { return data[i]; }
316 
318  void rotate1D(long i, long k);
319 
321  void shift1D(long i, long k);
322 };
323 
335 
336 template <typename T>
338 {
339 private:
340  const NTL::Vec<T>* data;
341  const CubeSignature* sig;
342  long dimOffset; // # of "missing dimensions" is this slice vs. the full cube
343  long sizeOffset; // index in the cube of the first element in this slice
344 
345  ConstCubeSlice(); // disabled
346 
347 public:
349  explicit ConstCubeSlice(const HyperCube<T>& _cube)
350  {
351  data = &_cube.getData();
352  sig = &_cube.getSig();
353  dimOffset = 0;
354  sizeOffset = 0;
355  }
356 
357  ConstCubeSlice(const NTL::Vec<T>& _data, const CubeSignature& _sig)
358  {
359  assertEq<InvalidArgument>(_data.length(),
360  _sig.getSize(),
361  "Data and signature sizes are different");
362  data = &_data;
363  sig = &_sig;
364  dimOffset = 0;
365  sizeOffset = 0;
366  }
367 
370  ConstCubeSlice(const ConstCubeSlice& bigger, long i, long _dimOffset = 1);
371  ConstCubeSlice(const HyperCube<T>& _cube, long i, long _dimOffset = 1);
372 
373  // use default copy constructor and assignment operators,
374  // which means shallow copy
375 
376  // the following mimic the corresponding methods
377  // in the HyperCube class, restricted to the slice
378 
380  const CubeSignature& getSig() const { return *sig; }
381 
383  long getSize() const { return sig->getProd(dimOffset); }
384 
386  long getNumDims() const { return sig->getNumDims() - dimOffset; }
387 
389  long getDim(long d) const { return sig->getDim(d + dimOffset); }
390 
392  long getProd(long d) const { return sig->getProd(d + dimOffset); }
393 
395  long getProd(long from, long to) const
396  {
397  return sig->getProd(from + dimOffset, to + dimOffset);
398  }
399 
401  long getCoord(long i, long d) const
402  {
403  assertInRange(i,
404  0l,
405  getSize(),
406  "Coordinate does not exist (index i out of range)");
407  return sig->getCoord(i + sizeOffset, d + dimOffset);
408  }
409 
411  long addCoord(long i, long d, long offset) const
412  {
413  assertInRange(i,
414  0l,
415  getSize(),
416  "Coordinate does not exist (index i out of range)");
417  return sig->addCoord(i + sizeOffset, d + dimOffset, offset);
418  }
419 
421  long numSlices(long d = 1) const { return getProd(0, d); }
422 
424  long sliceSize(long d = 1) const { return getProd(d); }
425 
427  long numCols() const { return getProd(1); }
428 
430  const T& at(long i) const
431  {
432  assertInRange(i,
433  0l,
434  getSize(),
435  "Coordinate does not exist (index i out of range)");
436  return (*data)[i + sizeOffset];
437  }
438 
440  const T& operator[](long i) const { return (*data)[i + sizeOffset]; }
441 };
442 
445 template <typename T>
446 class CubeSlice : public ConstCubeSlice<T>
447 {
448 private:
449  CubeSlice(); // disabled
450 public:
451  // initialize the slice to the full cube
452  explicit CubeSlice(HyperCube<T>& _cube) : ConstCubeSlice<T>(_cube) {}
453 
454  CubeSlice(NTL::Vec<T>& _data, const CubeSignature& _sig) :
455  ConstCubeSlice<T>(_data, _sig)
456  {}
457 
458  // initialize the slice to point to the i-th subcube
459  // of the cube pointed to by bigger
460  CubeSlice(const CubeSlice<T>& bigger, long i, long _dimOffset = 1) :
461  ConstCubeSlice<T>(bigger, i, _dimOffset)
462  {}
463 
464  CubeSlice(HyperCube<T>& _cube, long i, long _dimOffset = 1) :
465  ConstCubeSlice<T>(_cube, i, _dimOffset)
466  {}
467 
468  // deep copy of a slice: copies other into this
469  void copy(const ConstCubeSlice<T>& other) const;
470 
471  // reference to element at position i, with bounds check
472  T& at(long i) const { return const_cast<T&>(this->ConstCubeSlice<T>::at(i)); }
473 
474  // reference to element at position i, without bounds check
475  T& operator[](long i) const
476  {
477  return const_cast<T&>(this->ConstCubeSlice<T>::operator[](i));
478  }
479 };
480 
486 template <typename T>
487 void getHyperColumn(NTL::Vec<T>& v, const ConstCubeSlice<T>& s, long pos);
488 
491 template <typename T>
492 void setHyperColumn(const NTL::Vec<T>& v, const CubeSlice<T>& s, long pos);
493 
496 template <typename T>
497 void setHyperColumn(const NTL::Vec<T>& v,
498  const CubeSlice<T>& s,
499  long pos,
500  const T& val);
501 
502 template <typename T>
503 void print3D(const HyperCube<T>& c);
504 
505 } // namespace helib
506 
507 #endif // ifndef HELIB_HYPERCUBE_H
CubeSignature(const long _dims[], long _ndims)
Definition: hypercube.h:64
long sliceSize(long d=1) const
size of one slice
Definition: hypercube.h:300
long lsize(const std::vector< T > &v)
Size of STL vector as a long (rather than unsigned long)
Definition: NumbTh.h:702
long addCoord(long i, long d, long offset) const
add offset to coordinate in dimension d of index i
Definition: hypercube.h:291
T & at(long i)
reference to element at position i, with bounds check
Definition: hypercube.h:306
long getCoord(long i, long d) const
get coordinate in dimension d of index i
Definition: hypercube.h:104
long getCoord(long i, long d) const
get coordinate in dimension d of index i
Definition: hypercube.h:288
long assembleCoords(VecType &v) const
Definition: hypercube.h:176
T & operator[](long i)
reference to element at position i, without bounds check
Definition: hypercube.h:309
long getProd(long from, long to) const
product of sizes of dimensions from, from+1, ..., to-1
Definition: hypercube.h:395
long numCols() const
number of columns
Definition: hypercube.h:303
long getSize() const
total size
Definition: hypercube.h:383
long getDim(long d) const
size of dimension d
Definition: hypercube.h:389
HyperCube(const HyperCube &other)=default
long sliceSize(long d=1) const
size of one slice
Definition: hypercube.h:193
long getNumDims() const
number of dimensions
Definition: hypercube.h:276
CubeSignature()
Definition: hypercube.h:34
const T & at(long i) const
read-only reference to element at position i, with bounds check
Definition: hypercube.h:312
long getCoord(long i, long d) const
get coordinate in dimension d of index i
Definition: hypercube.h:401
void assertEq(const T &a, const T &b, const std::string &message)
Definition: assertions.h:108
long addCoord(long i, long d, long offset) const
add offset to coordinate in dimension d of index i
Definition: hypercube.h:115
bool incrementCoords(VecType &v) const
Definition: hypercube.h:138
long numSlices(long d=1) const
number of slices
Definition: hypercube.h:297
void copy(const ConstCubeSlice< T > &other) const
Definition: hypercube.cpp:152
const T & operator[](long i) const
read-only reference to element at position i, without bounds check
Definition: hypercube.h:440
long sliceSize(long d=1) const
size of one slice
Definition: hypercube.h:424
void assertInRange(const T &elem, const T &min, const T &max, const std::string &message, bool right_inclusive=false)
Definition: assertions.h:183
CubeSignature(const std::vector< long > &_dims)
Definition: hypercube.h:69
const T & operator[](long i) const
read-only reference to element at position i, without bounds check
Definition: hypercube.h:315
void shift1D(long i, long k)
Shift k positions along the i'th dimension with zero fill.
Definition: hypercube.cpp:67
long getNumDims() const
number of dimensions
Definition: hypercube.h:86
T & operator[](long i) const
Definition: hypercube.h:475
long getProd(long from, long to) const
product of sizes of dimensions from, from+1, ..., to-1
Definition: hypercube.h:285
long getDim(long d) const
size of dimension d
Definition: hypercube.h:92
long assembleIndexByDim(std::pair< long, long > idx, long dim) const
The inverse of breakIndexByDim.
Definition: hypercube.cpp:29
void getAllCoords(VecType &v, long i) const
Definition: hypercube.h:159
HyperCube(const CubeSignature &_sig)
initialize a HyperCube with a CubeSignature
Definition: hypercube.h:233
long getProd(long d) const
product of sizes of dimensions d, d+1, ...
Definition: hypercube.h:95
long numSlices(long d=1) const
number of slices
Definition: hypercube.h:421
void rotate1D(long i, long k)
rotate k positions along the i'th dimension
Definition: hypercube.cpp:42
long getProd(long d) const
product of sizes of dimensions d, d+1, ...
Definition: hypercube.h:392
CubeSlice(NTL::Vec< T > &_data, const CubeSignature &_sig)
Definition: hypercube.h:454
A constant lower-dimension slice of a hypercube.
Definition: hypercube.h:338
long getProd(long d) const
product of sizes of dimensions d, d+1, ...
Definition: hypercube.h:282
long numCols() const
number of columns
Definition: hypercube.h:427
bool operator==(const HyperCube< T > &other) const
equality testing: signatures must be the same
Definition: hypercube.h:251
void initSignature(const long _dims[], long _ndims)
Definition: hypercube.h:36
void resize(NTL::Vec< T > &v, long sz, const T &val)
Definition: NumbTh.h:744
long numSlices(long d=1) const
number of slices
Definition: hypercube.h:190
void print3D(const HyperCube< T > &c)
Definition: hypercube.cpp:237
HyperCube & operator=(const HyperCube< T > &other)
assignment: signatures must be the same
Definition: hypercube.h:241
Definition: apiAttributes.h:21
Holds a vector of dimensions for a hypercube and some additional data.
Definition: hypercube.h:28
CubeSignature(const NTL::Vec< long > &_dims)
Definition: hypercube.h:68
const CubeSignature & getSig() const
const ref to signature
Definition: hypercube.h:380
long getDim(long d) const
size of dimension d
Definition: hypercube.h:279
std::ostream & operator<<(std::ostream &s, const SKHandle &handle)
Definition: Ctxt.h:190
A multi-dimensional cube.
Definition: hypercube.h:221
const NTL::Vec< T > & getData() const
read-only ref to data vector
Definition: hypercube.h:270
long getSize() const
total size of cube
Definition: hypercube.h:89
void getHyperColumn(NTL::Vec< T > &v, const ConstCubeSlice< T > &s, long pos)
Definition: hypercube.cpp:174
const T & at(long i) const
read-only reference to element at position i, with bounds check
Definition: hypercube.h:430
long getSize() const
total size of cube
Definition: hypercube.h:273
ConstCubeSlice(const NTL::Vec< T > &_data, const CubeSignature &_sig)
Definition: hypercube.h:357
friend std::ostream & operator<<(std::ostream &s, const CubeSignature &sig)
Definition: hypercube.h:208
CubeSlice(HyperCube< T > &_cube)
Definition: hypercube.h:452
T & at(long i) const
Definition: hypercube.h:472
void initSignature(const VecType &_dims)
Definition: hypercube.h:59
long getProd(long from, long to) const
product of sizes of dimensions from, from+1, ..., to-1
Definition: hypercube.h:98
std::pair< long, long > breakIndexByDim(long idx, long dim) const
Definition: hypercube.cpp:20
void setHyperColumn(const NTL::Vec< T > &v, const CubeSlice< T > &s, long pos)
Definition: hypercube.cpp:193
A lower-dimension slice of a hypercube.
Definition: hypercube.h:447
CubeSlice(const CubeSlice< T > &bigger, long i, long _dimOffset=1)
Definition: hypercube.h:460
ConstCubeSlice(const HyperCube< T > &_cube)
initialize the slice to the full cube
Definition: hypercube.h:349
NTL::Vec< T > & getData()
Definition: hypercube.h:267
CubeSlice(HyperCube< T > &_cube, long i, long _dimOffset=1)
Definition: hypercube.h:464
long addCoord(long i, long d, long offset) const
add offset to coordinate in dimension d of index i
Definition: hypercube.h:411
bool operator!=(const HyperCube< T > &other) const
Definition: hypercube.h:259
long numCols() const
number of columns
Definition: hypercube.h:196
const CubeSignature & getSig() const
const ref to signature
Definition: hypercube.h:262
long getNumDims() const
number of dimensions
Definition: hypercube.h:386