powerful.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  */
12 #ifndef HELIB_POWERFUL_H
13 #define HELIB_POWERFUL_H
14 
18 #include <helib/NumbTh.h>
19 #include <helib/clonedPtr.h>
20 #include <helib/bluestein.h>
21 #include <helib/hypercube.h>
22 #include <helib/DoubleCRT.h>
23 #include <helib/Context.h>
24 
25 namespace helib {
26 
30 {
31 public:
32  long m; // product of mi's
33  long phim; // phi(m) = product of phi(mi)'s
34  NTL::Vec<long> mvec; // mvec[i] = mi
35  NTL::Vec<long> phivec; // phivec[i] = phi(mi)
36  NTL::Vec<long> divvec; // divvec[i] = m/mi
37  NTL::Vec<long> invvec; // invvec[i] = (m/mi)^{-1} mod mi
38 
39  CubeSignature longSig; // hypercube of dimensions mi
40  CubeSignature shortSig; // hypercube of dimensions phi(mi)
41 
42  NTL::Vec<long> polyToCubeMap; // index translation tables
43  NTL::Vec<long> cubeToPolyMap;
44  NTL::Vec<long> shortToLongMap;
45 
46  NTL::Vec<NTL::ZZX> cycVec; // cycvec[i] = Phi_mi(X)
47  NTL::ZZX phimX;
48 
49  PowerfulTranslationIndexes(const NTL::Vec<long>& mv);
50 };
51 
84 {
85  const PowerfulTranslationIndexes* indexes;
86  NTL::zz_pContext zzpContext;
87  NTL::Vec<NTL::zz_pXModulus> cycVec_p; // cycvec[i] = Phi_mi(X)
88  NTL::zz_pXModulus phimX_p;
89 
90 public:
91  PowerfulConversion() : indexes(nullptr) {}
92 
94  indexes(nullptr)
95  {
96  initPConv(ind);
97  }
98 
100  {
101  if (indexes != nullptr)
102  return; // cannot re-initialize a non-nullptr object
103  indexes = &ind;
104 
105  cycVec_p.SetLength(ind.cycVec.length());
106  zzpContext.save(); // store the current modulus
107  for (long i = 0; i < ind.cycVec.length(); i++) {
108  cycVec_p[i] =
109  NTL::conv<NTL::zz_pX>(ind.cycVec[i]); // convert to zz_pXModulus
110  }
111  phimX_p = NTL::conv<NTL::zz_pX>(ind.phimX); // convert to zz_pXModulus
112  }
113 
114  void restoreModulus() const { zzpContext.restore(); }
115  const CubeSignature& getLongSig() const { return indexes->longSig; }
116  const CubeSignature& getShortSig() const { return indexes->shortSig; }
117 
120  long powerfulToPoly(NTL::zz_pX& poly,
121  const HyperCube<NTL::zz_p>& powerful) const;
122  long polyToPowerful(HyperCube<NTL::zz_p>& powerful,
123  const NTL::zz_pX& poly) const;
124 };
125 
131 {
132  const Context& context; // points to the context for the DoubleCRT's
133 
134  PowerfulTranslationIndexes indexes; // modulus-independent tables
135 
136  // a vector of PowerfulConversion tables, one for each modulus
137  NTL::Vec<PowerfulConversion> pConvVec;
138 
139  // product_bits[i] is the number of bits in the product of primes [0..i)
140  NTL::Vec<long> product_bits;
141 
142  // number of excess bits needed to ensure correct conversion
143  long to_pwfl_excess_bits;
144  long to_poly_excess_bits;
145 
146  bool triv;
147 
148 public:
149  PowerfulDCRT(const Context& _context, const NTL::Vec<long>& mvec);
150 
152  {
153  return indexes;
154  }
155  const PowerfulConversion& getPConv(long i) const { return pConvVec.at(i); }
156 
157  // coefficients are reduced to the interval [-Q/2,Q/2], where
158  // Q = product of primes in dcrt.getIndexSet();
159  void dcrtToPowerful(NTL::Vec<NTL::ZZ>& powerful, const DoubleCRT& dcrt) const;
160 
161  void ZZXtoPowerful(NTL::Vec<NTL::ZZ>& powerful, const NTL::ZZX& poly) const;
162  void powerfulToZZX(NTL::ZZX& poly, const NTL::Vec<NTL::ZZ>& powerful) const;
163 };
164 
165 /********************************************************************/
166 /**************** UNUSED CODE - COMMENTED OUT ******************/
167 /********************************************************************/
168 
169 #if 0
170 long computeProd(const Vec< Pair<long, long> >& vec);
172 
174 inline long computePhi(const Pair<long, long>& x)
175 {
176  long p = x.a;
177  long e = x.b;
178  return power_long(p, e - 1) * (p-1);
179 }
180 
183 void computePhiVec(Vec<long>& phiVec,
184  const Vec< Pair<long, long> >& factors);
185 
188 void computeDivVec(Vec<long>& divVec, long m,
189  const Vec<long>& powVec);
190 
193 void computeInvVec(Vec<long>& invVec,
194  const Vec<long>& divVec, const Vec<long>& powVec);
195 
196 
202 void computeShortToLongMap(Vec<long>& shortToLongMap,
203  const CubeSignature& shortSig,
204  const CubeSignature& longSig);
205 
208 void computeLongToShortMap(Vec<long>& longToShortMap,
209  long m,
210  const Vec<long>& shortToLongMap);
211 
219 void recursiveReduce(const CubeSlice<zz_p>& s,
220  const Vec<zz_pXModulus>& cycVec,
221  long d,
222  zz_pX& tmp1,
223  zz_pX& tmp2);
224 
231 void convertPolyToPowerful(HyperCube<zz_p>& cube,
232  HyperCube<zz_p>& tmpCube,
233  const zz_pX& poly,
234  const Vec<zz_pXModulus>& cycVec,
235  const Vec<long>& polyToCubeMap,
236  const Vec<long>& shortToLongMap);
237 
239 void convertPowerfulToPoly(zz_pX& poly,
240  const HyperCube<zz_p>& cube,
241  long m,
242  const Vec<long>& shortToLongMap,
243  const Vec<long>& cubeToPolyMap,
244  const zz_pXModulus& phimX);
245 #endif
246 /********************************************************************/
247 #if 0
248 inline long computePow(const Pair<long, long>& x)
250 {
251  long p = x.a;
252  long e = x.b;
253  return power_long(p, e);
254 }
255 
257 void computePowVec(Vec<long>& powVec,
258  const Vec< Pair<long, long> >& factors);
259 
262 void mapIndexToPowerful(Vec<long>& pow, long j, const Vec<long>& phiVec);
263 
265 void computeCycVec(Vec<zz_pXModulus>& cycVec, const Vec<long>& powVec);
266 
277 void computePowerToCubeMap(Vec<long>& polyToCubeMap,
278  Vec<long>& cubeToPolyMap,
279  long m,
280  const Vec<long>& powVec,
281  const Vec<long>& invVec,
282  const CubeSignature& longSig);
283 
287 void computeMultiEvalPoints(Vec< Vec<zz_p> >& multiEvalPoints,
288  const zz_p& base,
289  long m,
290  const Vec<long>& powVec,
291  const Vec<long>& phiVec);
292 
294 void computeLinearEvalPoints(Vec<zz_p>& linearEvalPoints,
295  const zz_p& base,
296  long m, long phim);
297 
303 void computeCompressedIndex(Vec< Vec<long> >& compressedIndex,
304  const Vec<long>& powVec);
305 
310 void computePowToCompressedIndexMap(Vec<long>& powToCompressedIndexMap,
311  long m,
312  const Vec<long>& powVec,
313  const Vec< Vec<long> >& compressedIndex,
314  const CubeSignature& shortSig);
315 
316 void recursiveEval(const CubeSlice<zz_p>& s,
317  const Vec< Vec<zz_p> >& multiEvalPoints,
318  long d,
319  zz_pX& tmp1,
320  Vec<zz_p>& tmp2);
321 
322 inline void eval(HyperCube<zz_p>& cube,
323  const Vec< Vec<zz_p> >& multiEvalPoints)
324 {
325  zz_pX tmp1;
326  Vec<zz_p> tmp2;
327 
328  recursiveEval(CubeSlice<zz_p>(cube), multiEvalPoints, 0, tmp1, tmp2);
329 }
330 
331 void mapPowerfulToPoly(ZZX& poly,
332  const Vec<long>& pow,
333  const Vec<long>& divVec,
334  long m,
335  const ZZX& phimX);
336 #endif
337 /********************************************************************/
338 #if 0
339 class FFTHelper {
351 
352 private:
353  long m;
354  zz_p m_inv;
355  zz_p root, iroot;
356  zz_pXModulus phimx;
357  Vec<bool> coprime;
358  long phim;
359 
360  mutable zz_pX powers, ipowers;
361  mutable Vec<mulmod_precon_t> powers_aux, ipowers_aux;
362  mutable fftRep Rb, iRb;
363  mutable fftrep_aux Rb_aux, iRb_aux;
364  mutable fftRep Ra;
365  mutable zz_pX tmp;
366 
367 public:
368  FFTHelper(long _m, zz_p x);
369 
370  void FFT(const zz_pX& f, Vec<zz_p>& v) const;
371  // compute v = { f(x^i) }_{i in Z_m^*}
372  void iFFT(zz_pX& f, const Vec<zz_p>& v, bool normalize = true) const;
373  // computes inverse transform. If !normalize, result is scaled by m
374 
375  const zz_p& get_m_inv() const { return m_inv; }
376  // useful for aggregate normalization
377 
378 };
379 
380 
383 void computeMultiEvalPoints(Vec< copied_ptr<FFTHelper> >& multiEvalPoints,
384  const zz_p& base,
385  long m,
386  const Vec<long>& powVec,
387  const Vec<long>& phiVec);
388 
389 void recursiveEval(const CubeSlice<zz_p>& s,
390  const Vec< copied_ptr<FFTHelper> >& multiEvalPoints,
391  long d,
392  zz_pX& tmp1,
393  Vec<zz_p>& tmp2);
394 
395 inline void eval(HyperCube<zz_p>& cube,
396  const Vec< copied_ptr<FFTHelper> >& multiEvalPoints)
397 {
398  zz_pX tmp1;
399  Vec<zz_p> tmp2;
400 
401  recursiveEval(CubeSlice<zz_p>(cube), multiEvalPoints, 0, tmp1, tmp2);
402 }
403 
404 void recursiveInterp(const CubeSlice<zz_p>& s,
405  const Vec< copied_ptr<FFTHelper> >& multiEvalPoints,
406  long d,
407  zz_pX& tmp1,
408  Vec<zz_p>& tmp2);
409 
410 void interp(HyperCube<zz_p>& cube,
411  const Vec< copied_ptr<FFTHelper> >& multiEvalPoints);
412 #endif
413 
414 } // namespace helib
415 
416 #endif // ifndef HELIB_POWERFUL_H
NTL::Vec< long > divvec
Definition: powerful.h:36
NTL::Vec< long > mvec
Definition: powerful.h:34
long polyToPowerful(HyperCube< NTL::zz_p > &powerful, const NTL::zz_pX &poly) const
Definition: powerful.cpp:199
long phim
Definition: powerful.h:33
void restoreModulus() const
Definition: powerful.h:114
PowerfulConversion(const PowerfulTranslationIndexes &ind)
Definition: powerful.h:93
const PowerfulConversion & getPConv(long i) const
Definition: powerful.h:155
NTL::Vec< long > cubeToPolyMap
Definition: powerful.h:43
Implementing polynomials (elements in the ring R_Q) in double-CRT form.
Definition: DoubleCRT.h:76
const PowerfulTranslationIndexes & getIndexTranslation() const
Definition: powerful.h:151
PowerfulDCRT(const Context &_context, const NTL::Vec< long > &mvec)
Definition: powerful.cpp:246
CubeSignature longSig
Definition: powerful.h:39
void normalize(zzX &f)
Definition: zzX.cpp:65
void computeInvVec(NTL::Vec< long > &invVec, const NTL::Vec< long > &divVec, const NTL::Vec< long > &powVec)
Definition: powerful.cpp:35
const CubeSignature & getShortSig() const
Definition: powerful.h:116
long computeProd(const NTL::Vec< long > &vec)
returns \prod_d vec[d]
Definition: NumbTh.cpp:92
void computeDivVec(NTL::Vec< long > &divVec, long m, const NTL::Vec< long > &powVec)
Definition: powerful.cpp:22
CubeSignature shortSig
Definition: powerful.h:40
NTL::Vec< long > invvec
Definition: powerful.h:37
NTL::Vec< long > phivec
Definition: powerful.h:35
NTL::Vec< long > polyToCubeMap
Definition: powerful.h:42
NTL::Vec< long > shortToLongMap
Definition: powerful.h:44
long m
Definition: powerful.h:32
void initPConv(const PowerfulTranslationIndexes &ind)
Definition: powerful.h:99
void powerfulToZZX(NTL::ZZX &poly, const NTL::Vec< NTL::ZZ > &powerful) const
Definition: powerful.cpp:354
PowerfulTranslationIndexes(const NTL::Vec< long > &mv)
Definition: powerful.cpp:152
long powerfulToPoly(NTL::zz_pX &poly, const HyperCube< NTL::zz_p > &powerful) const
Definition: powerful.cpp:223
Definition: apiAttributes.h:21
const CubeSignature & getLongSig() const
Definition: powerful.h:115
Holds a vector of dimensions for a hypercube and some additional data.
Definition: hypercube.h:28
PowerfulConversion()
Definition: powerful.h:91
void dcrtToPowerful(NTL::Vec< NTL::ZZ > &powerful, const DoubleCRT &dcrt) const
Definition: powerful.cpp:393
A multi-dimensional cube.
Definition: hypercube.h:221
Maintaining the parameters.
Definition: Context.h:121
Conversion between powerful representation in R_m/(q) and zz_pX.
Definition: powerful.h:84
void ZZXtoPowerful(NTL::Vec< NTL::ZZ > &powerful, const NTL::ZZX &poly) const
Definition: powerful.cpp:309
NTL::ZZX phimX
Definition: powerful.h:47
Holds index tables for translation between powerful and zz_pX.
Definition: powerful.h:30
NTL::Vec< NTL::ZZX > cycVec
Definition: powerful.h:46
Conversion between powerful representation, DoubleCRT, and ZZX.
Definition: powerful.h:131