Coverage Report

Created: 2026-02-09 07:07

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/php-src/ext/hash/hash_gost.c
Line
Count
Source
1
/*
2
  +----------------------------------------------------------------------+
3
  | Copyright (c) The PHP Group                                          |
4
  +----------------------------------------------------------------------+
5
  | This source file is subject to version 3.01 of the PHP license,      |
6
  | that is bundled with this package in the file LICENSE, and is        |
7
  | available through the world-wide-web at the following url:           |
8
  | https://www.php.net/license/3_01.txt                                 |
9
  | If you did not receive a copy of the PHP license and are unable to   |
10
  | obtain it through the world-wide-web, please send a note to          |
11
  | license@php.net so we can mail you a copy immediately.               |
12
  +----------------------------------------------------------------------+
13
  | Authors: Michael Wallner <mike@php.net>                              |
14
  |          Sara Golemon <pollita@php.net>                              |
15
  +----------------------------------------------------------------------+
16
*/
17
18
#include "php_hash.h"
19
#include "php_hash_gost.h"
20
#include "php_hash_gost_tables.h"
21
22
/* {{{ Gost()
23
 * derived from gost_compress() by Markku-Juhani Saarinen <mjos@ssh.fi>
24
 */
25
26
#define round(tables, k1, k2) \
27
1.33M
  t = (k1) + r; \
28
1.33M
  l ^= tables[0][t & 0xff] ^ tables[1][(t >> 8) & 0xff] ^ \
29
1.33M
    tables[2][(t >> 16) & 0xff] ^ tables[3][t >> 24]; \
30
1.33M
  t = (k2) + l; \
31
1.33M
  r ^= tables[0][t & 0xff] ^ tables[1][(t >> 8) & 0xff] ^ \
32
1.33M
    tables[2][(t >> 16) & 0xff] ^ tables[3][t >> 24];
33
34
#define R(tables, key, h, i, t, l, r) \
35
83.1k
  r = h[i]; \
36
83.1k
  l = h[i + 1]; \
37
83.1k
  round(tables, key[0], key[1]) \
38
83.1k
  round(tables, key[2], key[3]) \
39
83.1k
  round(tables, key[4], key[5]) \
40
83.1k
  round(tables, key[6], key[7]) \
41
83.1k
  round(tables, key[0], key[1]) \
42
83.1k
  round(tables, key[2], key[3]) \
43
83.1k
  round(tables, key[4], key[5]) \
44
83.1k
  round(tables, key[6], key[7]) \
45
83.1k
  round(tables, key[0], key[1]) \
46
83.1k
  round(tables, key[2], key[3]) \
47
83.1k
  round(tables, key[4], key[5]) \
48
83.1k
  round(tables, key[6], key[7]) \
49
83.1k
  round(tables, key[7], key[6]) \
50
83.1k
  round(tables, key[5], key[4]) \
51
83.1k
  round(tables, key[3], key[2]) \
52
83.1k
  round(tables, key[1], key[0]) \
53
83.1k
  t = r; \
54
83.1k
  r = l; \
55
83.1k
  l = t; \
56
57
#define X(w, u, v) \
58
83.1k
  w[0] = u[0] ^ v[0]; \
59
83.1k
  w[1] = u[1] ^ v[1]; \
60
83.1k
  w[2] = u[2] ^ v[2]; \
61
83.1k
  w[3] = u[3] ^ v[3]; \
62
83.1k
  w[4] = u[4] ^ v[4]; \
63
83.1k
  w[5] = u[5] ^ v[5]; \
64
83.1k
  w[6] = u[6] ^ v[6]; \
65
83.1k
  w[7] = u[7] ^ v[7];
66
67
#define P(key, w) \
68
83.1k
  key[0] = (w[0]  & 0x000000ff) | ((w[2] & 0x000000ff) << 8) | \
69
83.1k
    ((w[4] & 0x000000ff) << 16) | ((w[6] & 0x000000ff) << 24); \
70
83.1k
  key[1] = ((w[0] & 0x0000ff00) >> 8)  | (w[2]  & 0x0000ff00) | \
71
83.1k
    ((w[4] & 0x0000ff00) << 8) | ((w[6] & 0x0000ff00) << 16); \
72
83.1k
  key[2] = ((w[0] & 0x00ff0000) >> 16) | ((w[2] & 0x00ff0000) >> 8) | \
73
83.1k
    (w[4] & 0x00ff0000) | ((w[6] & 0x00ff0000) << 8); \
74
83.1k
  key[3] = ((w[0] & 0xff000000) >> 24) | ((w[2] & 0xff000000) >> 16) | \
75
83.1k
    ((w[4] & 0xff000000) >> 8) | (w[6] & 0xff000000); \
76
83.1k
  key[4] = (w[1] & 0x000000ff) | ((w[3] & 0x000000ff) << 8) | \
77
83.1k
    ((w[5] & 0x000000ff) << 16) | ((w[7] & 0x000000ff) << 24); \
78
83.1k
  key[5] = ((w[1] & 0x0000ff00) >> 8) | (w[3]  & 0x0000ff00) | \
79
83.1k
    ((w[5] & 0x0000ff00) << 8) | ((w[7] & 0x0000ff00) << 16); \
80
83.1k
  key[6] = ((w[1] & 0x00ff0000) >> 16) | ((w[3] & 0x00ff0000) >> 8) | \
81
83.1k
    (w[5] & 0x00ff0000) | ((w[7] & 0x00ff0000) << 8); \
82
83.1k
  key[7] = ((w[1] & 0xff000000) >> 24) | ((w[3] & 0xff000000) >> 16) | \
83
83.1k
    ((w[5] & 0xff000000) >> 8) | (w[7] & 0xff000000);
84
85
#define A(x, l, r) \
86
62.3k
  l = x[0] ^ x[2]; \
87
62.3k
  r = x[1] ^ x[3]; \
88
62.3k
  x[0] = x[2]; \
89
62.3k
  x[1] = x[3]; \
90
62.3k
  x[2] = x[4]; \
91
62.3k
  x[3] = x[5]; \
92
62.3k
  x[4] = x[6]; \
93
62.3k
  x[5] = x[7]; \
94
62.3k
  x[6] = l; \
95
62.3k
  x[7] = r;
96
97
#define AA(x, l, r) \
98
62.3k
  l = x[0]; \
99
62.3k
  r = x[2]; \
100
62.3k
  x[0] = x[4]; \
101
62.3k
  x[2] = x[6]; \
102
62.3k
  x[4] = l ^ r; \
103
62.3k
  x[6] = x[0] ^ r; \
104
62.3k
  l = x[1]; \
105
62.3k
  r = x[3]; \
106
62.3k
  x[1] = x[5]; \
107
62.3k
  x[3] = x[7]; \
108
62.3k
  x[5] = l ^ r; \
109
62.3k
  x[7] = x[1] ^ r;
110
111
#define C(x) \
112
20.7k
  x[0] ^= 0xff00ff00; \
113
20.7k
  x[1] ^= 0xff00ff00; \
114
20.7k
  x[2] ^= 0x00ff00ff; \
115
20.7k
  x[3] ^= 0x00ff00ff; \
116
20.7k
  x[4] ^= 0x00ffff00; \
117
20.7k
  x[5] ^= 0xff0000ff; \
118
20.7k
  x[6] ^= 0x000000ff; \
119
20.7k
  x[7] ^= 0xff00ffff;
120
121
#define S(s, l, r) \
122
83.1k
    s[i] = r; \
123
83.1k
    s[i + 1] = l;
124
125
#define SHIFT12(u, m, s) \
126
20.7k
  u[0] = m[0] ^ s[6]; \
127
20.7k
  u[1] = m[1] ^ s[7]; \
128
20.7k
  u[2] = m[2] ^ (s[0] << 16) ^ (s[0] >> 16) ^ (s[0] & 0xffff) ^ \
129
20.7k
    (s[1] & 0xffff) ^ (s[1] >> 16) ^ (s[2] << 16) ^ s[6] ^ (s[6] << 16) ^ \
130
20.7k
    (s[7] & 0xffff0000) ^ (s[7] >> 16); \
131
20.7k
  u[3] = m[3] ^ (s[0] & 0xffff) ^ (s[0] << 16) ^ (s[1] & 0xffff) ^ \
132
20.7k
    (s[1] << 16) ^ (s[1] >> 16) ^ (s[2] << 16) ^ (s[2] >> 16) ^ \
133
20.7k
    (s[3] << 16) ^ s[6] ^ (s[6] << 16) ^ (s[6] >> 16) ^ (s[7] & 0xffff) ^ \
134
20.7k
    (s[7] << 16) ^ (s[7] >> 16); \
135
20.7k
  u[4] = m[4] ^ \
136
20.7k
    (s[0] & 0xffff0000) ^ (s[0] << 16) ^ (s[0] >> 16) ^ \
137
20.7k
    (s[1] & 0xffff0000) ^ (s[1] >> 16) ^ (s[2] << 16) ^ (s[2] >> 16) ^ \
138
20.7k
    (s[3] << 16) ^ (s[3] >> 16) ^ (s[4] << 16) ^ (s[6] << 16) ^ \
139
20.7k
    (s[6] >> 16) ^(s[7] & 0xffff) ^ (s[7] << 16) ^ (s[7] >> 16); \
140
20.7k
  u[5] = m[5] ^ (s[0] << 16) ^ (s[0] >> 16) ^ (s[0] & 0xffff0000) ^ \
141
20.7k
    (s[1] & 0xffff) ^ s[2] ^ (s[2] >> 16) ^ (s[3] << 16) ^ (s[3] >> 16) ^ \
142
20.7k
    (s[4] << 16) ^ (s[4] >> 16) ^ (s[5] << 16) ^  (s[6] << 16) ^ \
143
20.7k
    (s[6] >> 16) ^ (s[7] & 0xffff0000) ^ (s[7] << 16) ^ (s[7] >> 16); \
144
20.7k
  u[6] = m[6] ^ s[0] ^ (s[1] >> 16) ^ (s[2] << 16) ^ s[3] ^ (s[3] >> 16) ^ \
145
20.7k
    (s[4] << 16) ^ (s[4] >> 16) ^ (s[5] << 16) ^ (s[5] >> 16) ^ s[6] ^ \
146
20.7k
    (s[6] << 16) ^ (s[6] >> 16) ^ (s[7] << 16); \
147
20.7k
  u[7] = m[7] ^ (s[0] & 0xffff0000) ^ (s[0] << 16) ^ (s[1] & 0xffff) ^ \
148
20.7k
    (s[1] << 16) ^ (s[2] >> 16) ^ (s[3] << 16) ^ s[4] ^ (s[4] >> 16) ^ \
149
20.7k
    (s[5] << 16) ^ (s[5] >> 16) ^ (s[6] >> 16) ^ (s[7] & 0xffff) ^ \
150
20.7k
    (s[7] << 16) ^ (s[7] >> 16);
151
152
#define SHIFT16(h, v, u) \
153
20.7k
  v[0] = h[0] ^ (u[1] << 16) ^ (u[0] >> 16); \
154
20.7k
  v[1] = h[1] ^ (u[2] << 16) ^ (u[1] >> 16); \
155
20.7k
  v[2] = h[2] ^ (u[3] << 16) ^ (u[2] >> 16); \
156
20.7k
  v[3] = h[3] ^ (u[4] << 16) ^ (u[3] >> 16); \
157
20.7k
  v[4] = h[4] ^ (u[5] << 16) ^ (u[4] >> 16); \
158
20.7k
  v[5] = h[5] ^ (u[6] << 16) ^ (u[5] >> 16); \
159
20.7k
  v[6] = h[6] ^ (u[7] << 16) ^ (u[6] >> 16); \
160
20.7k
  v[7] = h[7] ^ (u[0] & 0xffff0000) ^ (u[0] << 16) ^ (u[7] >> 16) ^ \
161
20.7k
    (u[1] & 0xffff0000) ^ (u[1] << 16) ^ (u[6] << 16) ^ (u[7] & 0xffff0000);
162
163
#define SHIFT61(h, v) \
164
20.7k
  h[0] = (v[0] & 0xffff0000) ^ (v[0] << 16) ^ (v[0] >> 16) ^ (v[1] >> 16) ^ \
165
20.7k
    (v[1] & 0xffff0000) ^ (v[2] << 16) ^ (v[3] >> 16) ^ (v[4] << 16) ^ \
166
20.7k
    (v[5] >> 16) ^ v[5] ^ (v[6] >> 16) ^ (v[7] << 16) ^ (v[7] >> 16) ^ \
167
20.7k
    (v[7] & 0xffff); \
168
20.7k
  h[1] = (v[0] << 16) ^ (v[0] >> 16) ^ (v[0] & 0xffff0000) ^ (v[1] & 0xffff) ^ \
169
20.7k
  v[2] ^ (v[2] >> 16) ^ (v[3] << 16) ^ (v[4] >> 16) ^ (v[5] << 16) ^ \
170
20.7k
    (v[6] << 16) ^ v[6] ^ (v[7] & 0xffff0000) ^ (v[7] >> 16); \
171
20.7k
  h[2] = (v[0] & 0xffff) ^ (v[0] << 16) ^ (v[1] << 16) ^ (v[1] >> 16) ^ \
172
20.7k
    (v[1] & 0xffff0000) ^ (v[2] << 16) ^ (v[3] >> 16) ^ v[3] ^ (v[4] << 16) ^ \
173
20.7k
    (v[5] >> 16) ^ v[6] ^ (v[6] >> 16) ^ (v[7] & 0xffff) ^ (v[7] << 16) ^ \
174
20.7k
    (v[7] >> 16); \
175
20.7k
  h[3] = (v[0] << 16) ^ (v[0] >> 16) ^ (v[0] & 0xffff0000) ^ \
176
20.7k
    (v[1] & 0xffff0000) ^ (v[1] >> 16) ^ (v[2] << 16) ^ (v[2] >> 16) ^ v[2] ^ \
177
20.7k
    (v[3] << 16) ^ (v[4] >> 16) ^ v[4] ^ (v[5] << 16) ^ (v[6] << 16) ^ \
178
20.7k
    (v[7] & 0xffff) ^ (v[7] >> 16); \
179
20.7k
  h[4] = (v[0] >> 16) ^ (v[1] << 16) ^ v[1] ^ (v[2] >> 16) ^ v[2] ^ \
180
20.7k
    (v[3] << 16) ^ (v[3] >> 16) ^ v[3] ^ (v[4] << 16) ^ (v[5] >> 16) ^ \
181
20.7k
  v[5] ^ (v[6] << 16) ^ (v[6] >> 16) ^ (v[7] << 16); \
182
20.7k
  h[5] = (v[0] << 16) ^ (v[0] & 0xffff0000) ^ (v[1] << 16) ^ (v[1] >> 16) ^ \
183
20.7k
    (v[1] & 0xffff0000) ^ (v[2] << 16) ^ v[2] ^ (v[3] >> 16) ^ v[3] ^ \
184
20.7k
    (v[4] << 16) ^ (v[4] >> 16) ^ v[4] ^ (v[5] << 16) ^ (v[6] << 16) ^ \
185
20.7k
    (v[6] >> 16) ^ v[6] ^ (v[7] << 16) ^ (v[7] >> 16) ^ (v[7] & 0xffff0000); \
186
20.7k
  h[6] = v[0] ^ v[2] ^ (v[2] >> 16) ^ v[3] ^ (v[3] << 16) ^ v[4] ^ \
187
20.7k
    (v[4] >> 16) ^ (v[5] << 16) ^ (v[5] >> 16) ^ v[5] ^ (v[6] << 16) ^ \
188
20.7k
    (v[6] >> 16) ^ v[6] ^ (v[7] << 16) ^ v[7]; \
189
20.7k
  h[7] = v[0] ^ (v[0] >> 16) ^ (v[1] << 16) ^ (v[1] >> 16) ^ (v[2] << 16) ^ \
190
20.7k
    (v[3] >> 16) ^ v[3] ^ (v[4] << 16) ^ v[4] ^ (v[5] >> 16) ^ v[5] ^ \
191
20.7k
    (v[6] << 16) ^ (v[6] >> 16) ^ (v[7] << 16) ^ v[7];
192
193
#define PASS(tables) \
194
83.1k
  X(w, u, v); \
195
83.1k
  P(key, w); \
196
83.1k
  R((tables), key, h, i, t, l, r); \
197
83.1k
  S(s, l, r); \
198
83.1k
  if (i != 6) { \
199
62.3k
    A(u, l, r); \
200
62.3k
    if (i == 2) { \
201
20.7k
      C(u); \
202
20.7k
    } \
203
62.3k
    AA(v, l, r); \
204
62.3k
  }
205
206
static inline void Gost(PHP_GOST_CTX *context, uint32_t data[8])
207
20.7k
{
208
20.7k
  int i;
209
20.7k
  uint32_t l, r, t, key[8], u[8], v[8], w[8], s[8], *h = context->state, *m = data;
210
211
20.7k
  memcpy(u, context->state, sizeof(u));
212
20.7k
  memcpy(v, data, sizeof(v));
213
214
103k
  for (i = 0; i < 8; i += 2) {
215
83.1k
    PASS(*context->tables);
216
83.1k
  }
217
20.7k
  SHIFT12(u, m, s);
218
20.7k
  SHIFT16(h, v, u);
219
20.7k
  SHIFT61(h, v);
220
20.7k
}
221
/* }}} */
222
223
static inline void GostTransform(PHP_GOST_CTX *context, const unsigned char input[32])
224
20.5k
{
225
20.5k
  int i, j;
226
20.5k
  uint32_t data[8], temp = 0;
227
228
185k
  for (i = 0, j = 0; i < 8; ++i, j += 4) {
229
164k
    data[i] = ((uint32_t) input[j]) | (((uint32_t) input[j + 1]) << 8) |
230
164k
          (((uint32_t) input[j + 2]) << 16) | (((uint32_t) input[j + 3]) << 24);
231
164k
    context->state[i + 8] += data[i] + temp;
232
164k
    temp = context->state[i + 8] < data[i] ? 1 : (context->state[i + 8] == data[i] ? temp : 0);
233
164k
  }
234
235
20.5k
  Gost(context, data);
236
20.5k
}
237
238
PHP_HASH_API void PHP_GOSTInit(PHP_GOST_CTX *context, ZEND_ATTRIBUTE_UNUSED HashTable *args)
239
121
{
240
121
  memset(context, 0, sizeof(*context));
241
121
  context->tables = &tables_test;
242
121
}
243
244
PHP_HASH_API void PHP_GOSTInitCrypto(PHP_GOST_CTX *context, ZEND_ATTRIBUTE_UNUSED HashTable *args)
245
1
{
246
1
  PHP_GOSTInit(context, NULL);
247
1
  context->tables = &tables_crypto;
248
1
}
249
250
static const uint32_t MAX32 = 0xffffffffLU;
251
252
PHP_HASH_API void PHP_GOSTUpdate(PHP_GOST_CTX *context, const unsigned char *input, size_t len)
253
114
{
254
114
  if ((MAX32 - context->count[0]) < (len * 8)) {
255
42
    context->count[1]++;
256
42
    context->count[0] = MAX32 - context->count[0];
257
42
    context->count[0] = (len * 8) - context->count[0];
258
72
  } else {
259
72
    context->count[0] += len * 8;
260
72
  }
261
262
114
  if (context->length + len < 32) {
263
5
    memcpy(&context->buffer[context->length], input, len);
264
5
    context->length += (unsigned char)len;
265
109
  } else {
266
109
    size_t i = 0, r = (context->length + len) % 32;
267
268
109
    if (context->length) {
269
91
      i = 32 - context->length;
270
91
      memcpy(&context->buffer[context->length], input, i);
271
91
      GostTransform(context, context->buffer);
272
91
    }
273
274
20.4k
    for (; i + 32 <= len; i += 32) {
275
20.3k
      GostTransform(context, input + i);
276
20.3k
    }
277
278
109
    memcpy(context->buffer, input + i, r);
279
109
    ZEND_SECURE_ZERO(&context->buffer[r], 32 - r);
280
109
    context->length = (unsigned char)r;
281
109
  }
282
114
}
283
284
PHP_HASH_API void PHP_GOSTFinal(unsigned char digest[32], PHP_GOST_CTX *context)
285
114
{
286
114
  uint32_t i, j, l[8] = {0};
287
288
114
  if (context->length) {
289
103
    GostTransform(context, context->buffer);
290
103
  }
291
292
114
  memcpy(l, context->count, sizeof(context->count));
293
114
  Gost(context, l);
294
114
  memcpy(l, &context->state[8], sizeof(l));
295
114
  Gost(context, l);
296
297
1.02k
  for (i = 0, j = 0; j < 32; i++, j += 4) {
298
912
    digest[j] = (unsigned char) (context->state[i] & 0xff);
299
912
    digest[j + 1] = (unsigned char) ((context->state[i] >> 8) & 0xff);
300
912
    digest[j + 2] = (unsigned char) ((context->state[i] >> 16) & 0xff);
301
912
    digest[j + 3] = (unsigned char) ((context->state[i] >> 24) & 0xff);
302
912
  }
303
304
114
  ZEND_SECURE_ZERO(context, sizeof(*context));
305
114
}
306
307
static hash_spec_result php_gost_unserialize(php_hashcontext_object *hash, zend_long magic, const zval *zv)
308
121
{
309
121
  PHP_GOST_CTX *ctx = (PHP_GOST_CTX *) hash->context;
310
121
  hash_spec_result r = HASH_SPEC_FAILURE;
311
121
  if (magic == PHP_HASH_SERIALIZE_MAGIC_SPEC
312
120
    && (r = php_hash_unserialize_spec(hash, zv, PHP_GOST_SPEC)) == HASH_SPEC_SUCCESS
313
114
    && ctx->length < sizeof(ctx->buffer)) {
314
114
    return HASH_SPEC_SUCCESS;
315
114
  }
316
317
7
    return r != HASH_SPEC_SUCCESS ? r : CONTEXT_VALIDATION_FAILURE;
318
121
}
319
320
const php_hash_ops php_hash_gost_ops = {
321
  "gost",
322
  (php_hash_init_func_t) PHP_GOSTInit,
323
  (php_hash_update_func_t) PHP_GOSTUpdate,
324
  (php_hash_final_func_t) PHP_GOSTFinal,
325
  php_hash_copy,
326
  php_hash_serialize,
327
  php_gost_unserialize,
328
  PHP_GOST_SPEC,
329
  32,
330
  32,
331
  sizeof(PHP_GOST_CTX),
332
  1
333
};
334
335
const php_hash_ops php_hash_gost_crypto_ops = {
336
  "gost-crypto",
337
  (php_hash_init_func_t) PHP_GOSTInitCrypto,
338
  (php_hash_update_func_t) PHP_GOSTUpdate,
339
  (php_hash_final_func_t) PHP_GOSTFinal,
340
  php_hash_copy,
341
  php_hash_serialize,
342
  php_gost_unserialize,
343
  PHP_GOST_SPEC,
344
  32,
345
  32,
346
  sizeof(PHP_GOST_CTX),
347
  1
348
};