forked from ph4r05/Whitebox-crypto-AES
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBGEAttack.h
315 lines (258 loc) · 9.98 KB
/
BGEAttack.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
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
/*
* BGEAttack.h
*
* Created on: Apr 7, 2013
* Author: ph4r05
*/
#ifndef BGEATTACK_H_
#define BGEATTACK_H_
#include "base.h"
#include "NTLUtils.h"
#include "MixingBijections.h"
#include "LinearAffineEq.h"
#include "GenericAES.h"
#include "WBAES.h"
#include "WBAESGenerator.h"
#include <time.h>
NTL_CLIENT
using wbacr::laeqv::affineEquiv_t;
namespace wbacr {
namespace attack {
// compares columns of two state arrays for equality
#define ISEQ_W128B_COL(a,cola,b,colb) ( \
(a).B[(cola) + 4*0] == (b).B[(colb) + 4*0] && \
(a).B[(cola) + 4*1] == (b).B[(colb) + 4*1] && \
(a).B[(cola) + 4*2] == (b).B[(colb) + 4*2] && \
(a).B[(cola) + 4*3] == (b).B[(colb) + 4*3] \
)
// initializes column of state array from long long. Requires semicolon (do-while trick)
#define INIT_W128B_COL(a, cola, lng) do { \
(a).B[(cola) + 4*0] = (lng) & 0xFF; \
(a).B[(cola) + 4*1] = ((lng) >> 8) & 0xFF; \
(a).B[(cola) + 4*2] = ((lng) >> 16) & 0xFF; \
(a).B[(cola) + 4*3] = ((lng) >> 24) & 0xFF; \
} while(0)
// Function map hash -> function idx
#ifdef FORCE_DETERMINISM
// std map is sorted, so it always leads to same solutions using this map
typedef std::map<std::string, int>fctionMap_t;
#else
// unordered version is faster - implemented by hash table, but brings also some
// nondeterminism when iterating over it
typedef boost::unordered_map<std::string, int>fctionMap_t;
#endif
typedef std::pair<std::string, int> fctionMap_elem_t;
// Psi mapping, function idx -> [\Beta]
typedef BYTE psiFction_t[256];
// R set in algorithm for PSI recovery
#ifdef FORCE_DETERMINISM
typedef std::map<BYTE, BYTE> Rmap_t;
#else
typedef boost::unordered_map<BYTE, BYTE> Rmap_t;
#endif
typedef std::pair<BYTE, BYTE> Rmap_elem_t;
// Hashes GF256 function with MD5
inline std::string hashFunction(GF256_func_t f){
std::ostringstream out;
for(int i=0; i<GF256; i++){
out << "["<<CHEX(i)<<"]="<<CHEX(f[i])<<",";
}
return hashString(out.str());
}
// composes two functions and return hash of resulting function. Computes (f \ocirc g)(x)
std::string composeFunction(GF256_func_t f, GF256_func_t g);
// Element of Sset - set of functions;
// S = {Q \op \oplus_\beta \op Q^{-1}}_{\beta \in GF(2^8)}
//
// or just ordinary function of Rbox, x2=0, x3=0
typedef struct fction_t_ {
// Rbox function
// f(x, c1, 0, 0)
BYTE c1;
GF256_func_t f;
GF256_func_t finv;
std::string hash;
fction_t_(void) {
c1=0;
for(int i=0; i<GF256; i++){
f[i] = i;
finv[i] = i;
}
initHash();
}
void initHash(void){
hash=hashFunction(f);
}
} fction_t;
// S set of functions.
// Note: all functions in this set should form vector space, thus
// by composing any two of them one should obtain another from set
typedef struct Sset_t_ {
// Each function in array below is constructed as
// f_c1(f^{-1}_00(x))
// = f(f^{-1}(x,0,0,0), c1, 0, 0)
fction_t_ fctions[GF256];
fctionMap_t fmap;
fction_t f_00;
// PSI mapping (isomorphism S -> GF(2)^8)
psiFction_t psi;
} Sset_t;
// set S per each round (each for one state array element)
typedef struct SsetPerRound_t_{
//
// +------- Column (entering to MixCol operation)
// | +---- Row (y0, y1, y2, y3)
// | |
Sset_t S[4][4];
} SsetPerRound_t;
// Q~
typedef struct Qaffine_t_{
// +--------------- 9 Rounds with usable output encoding
// | +-------- For each state array element (y0, y1, y2, y3, for each column)
// | |
fction_t Q[9][AES_BYTES];
BYTE qj[9][AES_BYTES]; // affine constant determined in later phase of attack
mat_GF2 Aj[9][AES_BYTES]; // linear part of affine transformation determined in later phase of attack
}Qaffine_t;
// set for beta coefficients from 3.3 section
typedef boost::unordered_set<BYTE> Bset;
// characteristic polynomial multimap for resolving to beta
typedef boost::unordered_multimap<int, BYTE> charPolynomialMultimap_t;
typedef std::pair<int, BYTE> charPolynomialMultimap_elem_t;
// square matrix 8x8 (maximum) of GF2X elements
typedef struct mat_GF2X_t_ {
NTL::GF2X x[8][8];
int n;
}mat_GF2X_t;
// Affine equivalence
typedef std::map<BYTE, BYTE> lmap_t;
typedef std::pair<BYTE, BYTE> lmap_elem_t;
typedef struct affineEquiv_t_ {
mat_GF2 L;
mat_GF2 Linv;
lmap_t Lm;
lmap_t Lminv;
BYTE c;
} affineEquiv_t;
// vector of base vectors - used for solving singular system of homogeouns equations
typedef std::vector<NTL::vec_GF2> baseVectors_t;
// Structure for 1 affine relation in proposition 3
typedef struct prop3affineRelation_t_{
BYTE delta; // delta_i according to the paper, defining affine mapping
BYTE c; // c_i according to the paper, defining affine mapping
BYTE affineConst; // affine constant of final affine mapping determined
mat_GF2 L; // linear part of the mapping
mat_GF2 Linv; // inversion of linear part of the mapping
GF256_func_t affineMap; // resulting affine mapping P~_i
bool valid; // should be this record taken seriously?
BYTE alfa_0; // MixColumn coefficient alfa_{0,i}
} prop3affineRelation_t;
// Return structure for proposition 3
typedef struct prop3struct_t_{
prop3affineRelation_t P[4];
BYTE gamma;
BYTE c4;
} prop3struct_t;
class BGEAttack {
public:
BGEAttack();
virtual ~BGEAttack();
// AES to attack on
WBAES * wbaes;
// External coding of the AES above, just to test conversion to affine transformation [OPTIONAL]
ExtEncoding * coding;
//
// Demonstration of an attack - generates new WB AES instance and calls attack()
int run(BYTE * key = nullptr, keySize keyLen = KEY_SIZE_16);
//
// Main entry point
// Performs the whole BGE attack on this->wbaes
int attack(void);
//
// round function using AES this->wbaes instance
void Rbox(W128b& state, bool encrypt=true, int r=1, bool noShift=false, int colMask2compute=15);
//
// Recovers PSI isomorphism from (S,o) -> (GF(2^8), +) according to the paper
void recoverPsi(Sset_t & set);
//
// Derives set B from the BGE attack from MixColumn matrix coefficients
int deriveBset(Bset & bset, GenericAES & aes, bool permissive=true);
// Computes characteristic polynomial of matrix with coefficients from GF2.
// Uses recursive coMatrices algorithm.
GF2X characteristicPolynomial(mat_GF2 & m);
// Proposition 1 solver - finding affine equivalences between (yi, yj)
// yi is computed with Rbox() function using current wbaes.
//
// @param r round to compute on
// @param col column in given round
// @param syi 1..4, which row to use for yi
// @param syj 1..4, which row to use for yj
// @param sx which x 1..4 should be used to compute yi, yj
int proposition1(affineEquiv_t & ret, int r, int col, int syi, int syj, int sx);
// Proposition 2 solver - Finding A~0 for given matrix L and Beta.
// L = A0 * beta * A0^{-1}
// A~0 = A0 * gamma for some unique gamma \in GF(2^8)
//
// Function returns result in out variable.
// If a system of equations obtained is non-singular, out will contain only
// one vector - solution of the system, but can be trivial (null).
// If the only one found solution is trivial, return value = -1
// If found solution is invalid, return value = -2 (something wrong, maybe bug)
// Otherwise return value = 0
//
// If the system is singular, the out contains the set of orthogonal base vectors
// that span subspace that solves the system. Return value = 1.
int proposition2(mat_GF2 & inp, baseVectors_t & out, mat_GF2 beta);
// Proposition 3 according to the paper.
// Finding affine relations with embedded round key
//
// @param aes AES to use for Sbox inverse
// @param r computing prop3 for given round
// @param col column of state array to compute on
// @param row y0...y3 to compute on
// @param Atild result from proposition 2, A~ relation
// @param vectorIdx if Atild is whole vector space, which vector to use
// Returns:
// -1 in case Atild is empty
// -2 in case A~Inv cannot be constructed from Atild
// Flags determining whether P0,P1,P2,P3 were computed successfully (bit 1 if yes)
int proposition3(prop3struct_t * out, GenericAES & aes, int r, int col, int row, baseVectors_t & Atild, long int vectorIdx=0);
// Modified version of proposition 3 - with exactly given A~_row{-1} matrix which to use
// This is used when we have determined Q_row affine transformation completely and Aj linear part of Qj
// to find also cj, constant part of Qj.
int proposition3(prop3struct_t * out, GenericAES & aes, int r, int col, int row, const mat_GF2 AtildMatInv);
// Helper method to recover Qj if we already know A0 for Q0
//
// @param r computing prop3 for given round
// @param col column of state array to compute on
// @param row recovers A_row, should be \in {1,2,3}
// @param A0 linear part A0 of Q0 derived from y0
// @param alfa00 MixColumn coefficient alfa_{0,0} - should be 2 in general
// @param alfa_row_0 MixColumn coefficient alfa_{row,0}
// @param Aj computed matrix - linear part of affine mapping Qj
// @param qj computed constant - constant part of affine mapping Qj
//
int recoverQj(GenericAES & aes, int r, int col, int row, const mat_GF2 A0, BYTE alfa00, BYTE alfa_row_0, mat_GF2 & Aj, BYTE * qj);
// Recovers encryption key from two consecutive round keys
// For now it works only for AES-128
int recoverCipherKey(GenericAES & aes, BYTE roundKeys[2][16], vec_GF2E& encKey);
// Tries to invert encryption cipher to decryption
int invertCipherTest();
// just identity on 16 elements - used when shift rows operation ignored
const static int shiftIdentity[16];
const static int shiftT2[16];
const vec_GF2E &getEncKey() const;
void setEncKey(const vec_GF2E &encKey);
bool isDoCout() const;
void setDoCout(bool doCout);
protected:
// used internally for comptuting characteristic polynomial
GF2X characteristicPolynomial(mat_GF2X_t & m);
// Recovered enc key
vec_GF2E encKey;
// Dumping debugging info to cout
bool doCout = true;
};
} /* namespace attack */
} /* namespace wbacr */
#endif /* BGEATTACK_H_ */