-
Notifications
You must be signed in to change notification settings - Fork 712
/
galois.h
176 lines (150 loc) · 6.28 KB
/
galois.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT license.
#pragma once
#include "seal/memorymanager.h"
#include "seal/modulus.h"
#include "seal/util/defines.h"
#include "seal/util/iterator.h"
#include "seal/util/pointer.h"
#include <cstddef>
#include <cstdint>
#include <stdexcept>
namespace seal
{
namespace util
{
class GaloisTool
{
public:
GaloisTool(int coeff_count_power, MemoryPoolHandle pool) : pool_(std::move(pool))
{
if (!pool_)
{
throw std::invalid_argument("pool is uninitialized");
}
initialize(coeff_count_power);
}
void apply_galois(
ConstCoeffIter operand, std::uint32_t galois_elt, const Modulus &modulus, CoeffIter result) const;
inline void apply_galois(
ConstRNSIter operand, std::size_t coeff_modulus_size, std::uint32_t galois_elt,
ConstModulusIter modulus, RNSIter result) const
{
#ifdef SEAL_DEBUG
if ((!operand && coeff_modulus_size > 0) || (operand.poly_modulus_degree() != coeff_count_))
{
throw std::invalid_argument("operand");
}
if ((!result && coeff_modulus_size > 0) || (result.poly_modulus_degree() != coeff_count_))
{
throw std::invalid_argument("result");
}
#endif
SEAL_ITERATE(iter(operand, modulus, result), coeff_modulus_size, [&](auto I) {
this->apply_galois(get<0>(I), galois_elt, get<1>(I), get<2>(I));
});
}
void apply_galois(
ConstPolyIter operand, std::size_t size, std::uint32_t galois_elt, ConstModulusIter modulus,
PolyIter result) const
{
#ifdef SEAL_DEBUG
if (!operand && size > 0)
{
throw std::invalid_argument("operand");
}
if (!result && size > 0)
{
throw std::invalid_argument("result");
}
if (operand.coeff_modulus_size() != result.coeff_modulus_size())
{
throw std::invalid_argument("incompatible iterators");
}
#endif
auto coeff_modulus_size = result.coeff_modulus_size();
SEAL_ITERATE(iter(operand, result), size, [&](auto I) {
this->apply_galois(get<0>(I), coeff_modulus_size, galois_elt, modulus, get<1>(I));
});
}
void apply_galois_ntt(ConstCoeffIter operand, std::uint32_t galois_elt, CoeffIter result) const;
void apply_galois_ntt(
ConstRNSIter operand, std::size_t coeff_modulus_size, std::uint32_t galois_elt, RNSIter result) const
{
#ifdef SEAL_DEBUG
if ((!operand && coeff_modulus_size > 0) || (operand.poly_modulus_degree() != coeff_count_))
{
throw std::invalid_argument("operand");
}
if ((!result && coeff_modulus_size > 0) || (result.poly_modulus_degree() != coeff_count_))
{
throw std::invalid_argument("result");
}
#endif
SEAL_ITERATE(iter(operand, result), coeff_modulus_size, [&](auto I) {
this->apply_galois_ntt(get<0>(I), galois_elt, get<1>(I));
});
}
void apply_galois_ntt(
ConstPolyIter operand, std::size_t size, std::uint32_t galois_elt, PolyIter result) const
{
#ifdef SEAL_DEBUG
if (!operand && size > 0)
{
throw std::invalid_argument("operand");
}
if (!result && size > 0)
{
throw std::invalid_argument("result");
}
if (operand.coeff_modulus_size() != result.coeff_modulus_size())
{
throw std::invalid_argument("incompatible iterators");
}
#endif
auto coeff_modulus_size = result.coeff_modulus_size();
SEAL_ITERATE(iter(operand, result), size, [&](auto I) {
this->apply_galois_ntt(get<0>(I), coeff_modulus_size, galois_elt, get<1>(I));
});
}
/**
Compute the Galois element corresponding to a given rotation step.
*/
SEAL_NODISCARD std::uint32_t get_elt_from_step(int step) const;
/**
Compute the Galois elements corresponding to a vector of given rotation steps.
*/
SEAL_NODISCARD std::vector<std::uint32_t> get_elts_from_steps(const std::vector<int> &steps) const;
/**
Compute a vector of all necessary galois_elts.
*/
SEAL_NODISCARD std::vector<std::uint32_t> get_elts_all() const noexcept;
/**
Compute the index in the range of 0 to (coeff_count_ - 1) of a given Galois element.
*/
SEAL_NODISCARD static inline std::size_t GetIndexFromElt(std::uint32_t galois_elt)
{
#ifdef SEAL_DEBUG
if (!(galois_elt & 1))
{
throw std::invalid_argument("galois_elt is not valid");
}
#endif
return util::safe_cast<std::size_t>((galois_elt - 1) >> 1);
}
private:
GaloisTool(const GaloisTool ©) = delete;
GaloisTool(GaloisTool &&source) = delete;
GaloisTool &operator=(const GaloisTool &assign) = delete;
GaloisTool &operator=(GaloisTool &&assign) = delete;
void initialize(int coeff_count_power);
void generate_table_ntt(std::uint32_t galois_elt, Pointer<std::uint32_t> &result) const;
MemoryPoolHandle pool_;
int coeff_count_power_ = 0;
std::size_t coeff_count_ = 0;
static constexpr std::uint32_t generator_ = 3;
mutable Pointer<Pointer<std::uint32_t>> permutation_tables_;
mutable util::ReaderWriterLocker permutation_tables_locker_;
};
} // namespace util
} // namespace seal