Coverage Report

Created: 2025-09-27 06:26

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/php-src/ext/hash/hash_whirlpool.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
20
/*
21
 * TODO: simplify Update and Final, those look ridiculously complex
22
 * Mike, 2005-11-23
23
 */
24
25
#include "php_hash_whirlpool.h"
26
#include "php_hash_whirlpool_tables.h"
27
28
600k
#define DIGESTBYTES 64
29
600k
#define DIGESTBITS  (8*DIGESTBYTES) /* 512 */
30
31
294
#define WBLOCKBYTES 64
32
#define WBLOCKBITS  (8*WBLOCKBYTES) /* 512 */
33
34
293
#define LENGTHBYTES 32
35
#define LENGTHBITS  (8*LENGTHBYTES) /* 256 */
36
37
static void WhirlpoolTransform(PHP_WHIRLPOOL_CTX *context)
38
9.46k
{
39
9.46k
    int i, r;
40
9.46k
    uint64_t K[8];        /* the round key */
41
9.46k
    uint64_t block[8];    /* mu(buffer) */
42
9.46k
    uint64_t state[8];    /* the cipher state */
43
9.46k
    uint64_t L[8];
44
9.46k
    unsigned char *buffer = context->buffer.data;
45
46
    /*
47
     * map the buffer to a block:
48
     */
49
85.2k
    for (i = 0; i < 8; i++, buffer += 8) {
50
75.7k
        block[i] =
51
75.7k
            (((uint64_t)buffer[0]        ) << 56) ^
52
75.7k
            (((uint64_t)buffer[1] & 0xffL) << 48) ^
53
75.7k
            (((uint64_t)buffer[2] & 0xffL) << 40) ^
54
75.7k
            (((uint64_t)buffer[3] & 0xffL) << 32) ^
55
75.7k
            (((uint64_t)buffer[4] & 0xffL) << 24) ^
56
75.7k
            (((uint64_t)buffer[5] & 0xffL) << 16) ^
57
75.7k
            (((uint64_t)buffer[6] & 0xffL) <<  8) ^
58
75.7k
            (((uint64_t)buffer[7] & 0xffL)      );
59
75.7k
    }
60
    /*
61
     * compute and apply K^0 to the cipher state:
62
     */
63
9.46k
    state[0] = block[0] ^ (K[0] = context->state[0]);
64
9.46k
    state[1] = block[1] ^ (K[1] = context->state[1]);
65
9.46k
    state[2] = block[2] ^ (K[2] = context->state[2]);
66
9.46k
    state[3] = block[3] ^ (K[3] = context->state[3]);
67
9.46k
    state[4] = block[4] ^ (K[4] = context->state[4]);
68
9.46k
    state[5] = block[5] ^ (K[5] = context->state[5]);
69
9.46k
    state[6] = block[6] ^ (K[6] = context->state[6]);
70
9.46k
    state[7] = block[7] ^ (K[7] = context->state[7]);
71
    /*
72
     * iterate over all rounds:
73
     */
74
104k
    for (r = 1; r <= R; r++) {
75
        /*
76
         * compute K^r from K^{r-1}:
77
         */
78
94.6k
        L[0] =
79
94.6k
            C0[(int)(K[0] >> 56)       ] ^
80
94.6k
            C1[(int)(K[7] >> 48) & 0xff] ^
81
94.6k
            C2[(int)(K[6] >> 40) & 0xff] ^
82
94.6k
            C3[(int)(K[5] >> 32) & 0xff] ^
83
94.6k
            C4[(int)(K[4] >> 24) & 0xff] ^
84
94.6k
            C5[(int)(K[3] >> 16) & 0xff] ^
85
94.6k
            C6[(int)(K[2] >>  8) & 0xff] ^
86
94.6k
            C7[(int)(K[1]      ) & 0xff] ^
87
94.6k
            rc[r];
88
94.6k
        L[1] =
89
94.6k
            C0[(int)(K[1] >> 56)       ] ^
90
94.6k
            C1[(int)(K[0] >> 48) & 0xff] ^
91
94.6k
            C2[(int)(K[7] >> 40) & 0xff] ^
92
94.6k
            C3[(int)(K[6] >> 32) & 0xff] ^
93
94.6k
            C4[(int)(K[5] >> 24) & 0xff] ^
94
94.6k
            C5[(int)(K[4] >> 16) & 0xff] ^
95
94.6k
            C6[(int)(K[3] >>  8) & 0xff] ^
96
94.6k
            C7[(int)(K[2]      ) & 0xff];
97
94.6k
        L[2] =
98
94.6k
            C0[(int)(K[2] >> 56)       ] ^
99
94.6k
            C1[(int)(K[1] >> 48) & 0xff] ^
100
94.6k
            C2[(int)(K[0] >> 40) & 0xff] ^
101
94.6k
            C3[(int)(K[7] >> 32) & 0xff] ^
102
94.6k
            C4[(int)(K[6] >> 24) & 0xff] ^
103
94.6k
            C5[(int)(K[5] >> 16) & 0xff] ^
104
94.6k
            C6[(int)(K[4] >>  8) & 0xff] ^
105
94.6k
            C7[(int)(K[3]      ) & 0xff];
106
94.6k
        L[3] =
107
94.6k
            C0[(int)(K[3] >> 56)       ] ^
108
94.6k
            C1[(int)(K[2] >> 48) & 0xff] ^
109
94.6k
            C2[(int)(K[1] >> 40) & 0xff] ^
110
94.6k
            C3[(int)(K[0] >> 32) & 0xff] ^
111
94.6k
            C4[(int)(K[7] >> 24) & 0xff] ^
112
94.6k
            C5[(int)(K[6] >> 16) & 0xff] ^
113
94.6k
            C6[(int)(K[5] >>  8) & 0xff] ^
114
94.6k
            C7[(int)(K[4]      ) & 0xff];
115
94.6k
        L[4] =
116
94.6k
            C0[(int)(K[4] >> 56)       ] ^
117
94.6k
            C1[(int)(K[3] >> 48) & 0xff] ^
118
94.6k
            C2[(int)(K[2] >> 40) & 0xff] ^
119
94.6k
            C3[(int)(K[1] >> 32) & 0xff] ^
120
94.6k
            C4[(int)(K[0] >> 24) & 0xff] ^
121
94.6k
            C5[(int)(K[7] >> 16) & 0xff] ^
122
94.6k
            C6[(int)(K[6] >>  8) & 0xff] ^
123
94.6k
            C7[(int)(K[5]      ) & 0xff];
124
94.6k
        L[5] =
125
94.6k
            C0[(int)(K[5] >> 56)       ] ^
126
94.6k
            C1[(int)(K[4] >> 48) & 0xff] ^
127
94.6k
            C2[(int)(K[3] >> 40) & 0xff] ^
128
94.6k
            C3[(int)(K[2] >> 32) & 0xff] ^
129
94.6k
            C4[(int)(K[1] >> 24) & 0xff] ^
130
94.6k
            C5[(int)(K[0] >> 16) & 0xff] ^
131
94.6k
            C6[(int)(K[7] >>  8) & 0xff] ^
132
94.6k
            C7[(int)(K[6]      ) & 0xff];
133
94.6k
        L[6] =
134
94.6k
            C0[(int)(K[6] >> 56)       ] ^
135
94.6k
            C1[(int)(K[5] >> 48) & 0xff] ^
136
94.6k
            C2[(int)(K[4] >> 40) & 0xff] ^
137
94.6k
            C3[(int)(K[3] >> 32) & 0xff] ^
138
94.6k
            C4[(int)(K[2] >> 24) & 0xff] ^
139
94.6k
            C5[(int)(K[1] >> 16) & 0xff] ^
140
94.6k
            C6[(int)(K[0] >>  8) & 0xff] ^
141
94.6k
            C7[(int)(K[7]      ) & 0xff];
142
94.6k
        L[7] =
143
94.6k
            C0[(int)(K[7] >> 56)       ] ^
144
94.6k
            C1[(int)(K[6] >> 48) & 0xff] ^
145
94.6k
            C2[(int)(K[5] >> 40) & 0xff] ^
146
94.6k
            C3[(int)(K[4] >> 32) & 0xff] ^
147
94.6k
            C4[(int)(K[3] >> 24) & 0xff] ^
148
94.6k
            C5[(int)(K[2] >> 16) & 0xff] ^
149
94.6k
            C6[(int)(K[1] >>  8) & 0xff] ^
150
94.6k
            C7[(int)(K[0]      ) & 0xff];
151
94.6k
        K[0] = L[0];
152
94.6k
        K[1] = L[1];
153
94.6k
        K[2] = L[2];
154
94.6k
        K[3] = L[3];
155
94.6k
        K[4] = L[4];
156
94.6k
        K[5] = L[5];
157
94.6k
        K[6] = L[6];
158
94.6k
        K[7] = L[7];
159
        /*
160
         * apply the r-th round transformation:
161
         */
162
94.6k
        L[0] =
163
94.6k
            C0[(int)(state[0] >> 56)       ] ^
164
94.6k
            C1[(int)(state[7] >> 48) & 0xff] ^
165
94.6k
            C2[(int)(state[6] >> 40) & 0xff] ^
166
94.6k
            C3[(int)(state[5] >> 32) & 0xff] ^
167
94.6k
            C4[(int)(state[4] >> 24) & 0xff] ^
168
94.6k
            C5[(int)(state[3] >> 16) & 0xff] ^
169
94.6k
            C6[(int)(state[2] >>  8) & 0xff] ^
170
94.6k
            C7[(int)(state[1]      ) & 0xff] ^
171
94.6k
            K[0];
172
94.6k
        L[1] =
173
94.6k
            C0[(int)(state[1] >> 56)       ] ^
174
94.6k
            C1[(int)(state[0] >> 48) & 0xff] ^
175
94.6k
            C2[(int)(state[7] >> 40) & 0xff] ^
176
94.6k
            C3[(int)(state[6] >> 32) & 0xff] ^
177
94.6k
            C4[(int)(state[5] >> 24) & 0xff] ^
178
94.6k
            C5[(int)(state[4] >> 16) & 0xff] ^
179
94.6k
            C6[(int)(state[3] >>  8) & 0xff] ^
180
94.6k
            C7[(int)(state[2]      ) & 0xff] ^
181
94.6k
            K[1];
182
94.6k
        L[2] =
183
94.6k
            C0[(int)(state[2] >> 56)       ] ^
184
94.6k
            C1[(int)(state[1] >> 48) & 0xff] ^
185
94.6k
            C2[(int)(state[0] >> 40) & 0xff] ^
186
94.6k
            C3[(int)(state[7] >> 32) & 0xff] ^
187
94.6k
            C4[(int)(state[6] >> 24) & 0xff] ^
188
94.6k
            C5[(int)(state[5] >> 16) & 0xff] ^
189
94.6k
            C6[(int)(state[4] >>  8) & 0xff] ^
190
94.6k
            C7[(int)(state[3]      ) & 0xff] ^
191
94.6k
            K[2];
192
94.6k
        L[3] =
193
94.6k
            C0[(int)(state[3] >> 56)       ] ^
194
94.6k
            C1[(int)(state[2] >> 48) & 0xff] ^
195
94.6k
            C2[(int)(state[1] >> 40) & 0xff] ^
196
94.6k
            C3[(int)(state[0] >> 32) & 0xff] ^
197
94.6k
            C4[(int)(state[7] >> 24) & 0xff] ^
198
94.6k
            C5[(int)(state[6] >> 16) & 0xff] ^
199
94.6k
            C6[(int)(state[5] >>  8) & 0xff] ^
200
94.6k
            C7[(int)(state[4]      ) & 0xff] ^
201
94.6k
            K[3];
202
94.6k
        L[4] =
203
94.6k
            C0[(int)(state[4] >> 56)       ] ^
204
94.6k
            C1[(int)(state[3] >> 48) & 0xff] ^
205
94.6k
            C2[(int)(state[2] >> 40) & 0xff] ^
206
94.6k
            C3[(int)(state[1] >> 32) & 0xff] ^
207
94.6k
            C4[(int)(state[0] >> 24) & 0xff] ^
208
94.6k
            C5[(int)(state[7] >> 16) & 0xff] ^
209
94.6k
            C6[(int)(state[6] >>  8) & 0xff] ^
210
94.6k
            C7[(int)(state[5]      ) & 0xff] ^
211
94.6k
            K[4];
212
94.6k
        L[5] =
213
94.6k
            C0[(int)(state[5] >> 56)       ] ^
214
94.6k
            C1[(int)(state[4] >> 48) & 0xff] ^
215
94.6k
            C2[(int)(state[3] >> 40) & 0xff] ^
216
94.6k
            C3[(int)(state[2] >> 32) & 0xff] ^
217
94.6k
            C4[(int)(state[1] >> 24) & 0xff] ^
218
94.6k
            C5[(int)(state[0] >> 16) & 0xff] ^
219
94.6k
            C6[(int)(state[7] >>  8) & 0xff] ^
220
94.6k
            C7[(int)(state[6]      ) & 0xff] ^
221
94.6k
            K[5];
222
94.6k
        L[6] =
223
94.6k
            C0[(int)(state[6] >> 56)       ] ^
224
94.6k
            C1[(int)(state[5] >> 48) & 0xff] ^
225
94.6k
            C2[(int)(state[4] >> 40) & 0xff] ^
226
94.6k
            C3[(int)(state[3] >> 32) & 0xff] ^
227
94.6k
            C4[(int)(state[2] >> 24) & 0xff] ^
228
94.6k
            C5[(int)(state[1] >> 16) & 0xff] ^
229
94.6k
            C6[(int)(state[0] >>  8) & 0xff] ^
230
94.6k
            C7[(int)(state[7]      ) & 0xff] ^
231
94.6k
            K[6];
232
94.6k
        L[7] =
233
94.6k
            C0[(int)(state[7] >> 56)       ] ^
234
94.6k
            C1[(int)(state[6] >> 48) & 0xff] ^
235
94.6k
            C2[(int)(state[5] >> 40) & 0xff] ^
236
94.6k
            C3[(int)(state[4] >> 32) & 0xff] ^
237
94.6k
            C4[(int)(state[3] >> 24) & 0xff] ^
238
94.6k
            C5[(int)(state[2] >> 16) & 0xff] ^
239
94.6k
            C6[(int)(state[1] >>  8) & 0xff] ^
240
94.6k
            C7[(int)(state[0]      ) & 0xff] ^
241
94.6k
            K[7];
242
94.6k
        state[0] = L[0];
243
94.6k
        state[1] = L[1];
244
94.6k
        state[2] = L[2];
245
94.6k
        state[3] = L[3];
246
94.6k
        state[4] = L[4];
247
94.6k
        state[5] = L[5];
248
94.6k
        state[6] = L[6];
249
94.6k
        state[7] = L[7];
250
94.6k
    }
251
    /*
252
     * apply the Miyaguchi-Preneel compression function:
253
     */
254
9.46k
    context->state[0] ^= state[0] ^ block[0];
255
9.46k
    context->state[1] ^= state[1] ^ block[1];
256
9.46k
    context->state[2] ^= state[2] ^ block[2];
257
9.46k
    context->state[3] ^= state[3] ^ block[3];
258
9.46k
    context->state[4] ^= state[4] ^ block[4];
259
9.46k
    context->state[5] ^= state[5] ^ block[5];
260
9.46k
    context->state[6] ^= state[6] ^ block[6];
261
9.46k
    context->state[7] ^= state[7] ^ block[7];
262
263
9.46k
  ZEND_SECURE_ZERO(state, sizeof(state));
264
9.46k
}
265
266
PHP_HASH_API void PHP_WHIRLPOOLInit(PHP_WHIRLPOOL_CTX *context, ZEND_ATTRIBUTE_UNUSED HashTable *args)
267
93
{
268
93
  memset(context, 0, sizeof(*context));
269
93
}
270
271
PHP_HASH_API void PHP_WHIRLPOOLUpdate(PHP_WHIRLPOOL_CTX *context, const unsigned char *input, size_t len)
272
49
{
273
49
    uint64_t sourceBits = len * 8;
274
49
    int sourcePos    = 0; /* index of leftmost source unsigned char containing data (1 to 8 bits). */
275
49
    int sourceGap    = (8 - ((int)sourceBits & 7)) & 7; /* space on source[sourcePos]. */
276
49
    int bufferRem    = context->buffer.bits & 7; /* occupied bits on buffer[bufferPos]. */
277
49
    const unsigned char *source = input;
278
49
    unsigned char *buffer       = context->buffer.data;
279
49
    unsigned char *bitLength    = context->bitlength;
280
49
    int bufferBits   = context->buffer.bits;
281
49
    int bufferPos    = context->buffer.pos;
282
49
    uint32_t b, carry;
283
49
    int i;
284
285
    /*
286
     * tally the length of the added data:
287
     */
288
49
    uint64_t value = sourceBits;
289
168
    for (i = 31, carry = 0; i >= 0 && (carry != 0 || value != L64(0)); i--) {
290
119
        carry += bitLength[i] + ((uint32_t)value & 0xff);
291
119
        bitLength[i] = (unsigned char)carry;
292
119
        carry >>= 8;
293
119
        value >>= 8;
294
119
    }
295
    /*
296
     * process data in chunks of 8 bits (a more efficient approach would be to take whole-word chunks):
297
     */
298
600k
    while (sourceBits > 8) {
299
        /* N.B. at least source[sourcePos] and source[sourcePos+1] contain data. */
300
        /*
301
         * take a byte from the source:
302
         */
303
600k
        b = ((source[sourcePos] << sourceGap) & 0xff) |
304
600k
            ((source[sourcePos + 1] & 0xff) >> (8 - sourceGap));
305
        /*
306
         * process this byte:
307
         */
308
600k
        buffer[bufferPos++] |= (unsigned char)(b >> bufferRem);
309
600k
        bufferBits += 8 - bufferRem; /* bufferBits = 8*bufferPos; */
310
600k
        if (bufferBits == DIGESTBITS) {
311
            /*
312
             * process data block:
313
             */
314
9.39k
            WhirlpoolTransform(context);
315
            /*
316
             * reset buffer:
317
             */
318
9.39k
            bufferBits = bufferPos = 0;
319
9.39k
        }
320
600k
        buffer[bufferPos] = (unsigned char) (b << (8 - bufferRem));
321
600k
        bufferBits += bufferRem;
322
        /*
323
         * proceed to remaining data:
324
         */
325
600k
        sourceBits -= 8;
326
600k
        sourcePos++;
327
600k
    }
328
    /* now 0 <= sourceBits <= 8;
329
     * furthermore, all data (if any is left) is in source[sourcePos].
330
     */
331
49
    if (sourceBits > 0) {
332
48
        b = (source[sourcePos] << sourceGap) & 0xff; /* bits are left-justified on b. */
333
        /*
334
         * process the remaining bits:
335
         */
336
48
        buffer[bufferPos] |= b >> bufferRem;
337
48
    } else {
338
1
        b = 0;
339
1
    }
340
49
    if (bufferRem + sourceBits < 8) {
341
        /*
342
         * all remaining data fits on buffer[bufferPos],
343
         * and there still remains some space.
344
         */
345
1
        bufferBits += (int) sourceBits;
346
48
    } else {
347
        /*
348
         * buffer[bufferPos] is full:
349
         */
350
48
        bufferPos++;
351
48
        bufferBits += 8 - bufferRem; /* bufferBits = 8*bufferPos; */
352
48
        sourceBits -= 8 - bufferRem;
353
        /* now 0 <= sourceBits < 8;
354
         * furthermore, all data (if any is left) is in source[sourcePos].
355
         */
356
48
        if (bufferBits == DIGESTBITS) {
357
            /*
358
             * process data block:
359
             */
360
1
            WhirlpoolTransform(context);
361
            /*
362
             * reset buffer:
363
             */
364
1
            bufferBits = bufferPos = 0;
365
1
        }
366
48
        buffer[bufferPos] = (unsigned char) (b << (8 - bufferRem));
367
48
        bufferBits += (int)sourceBits;
368
48
    }
369
49
    context->buffer.bits   = bufferBits;
370
49
    context->buffer.pos    = bufferPos;
371
49
}
372
373
PHP_HASH_API void PHP_WHIRLPOOLFinal(unsigned char digest[64], PHP_WHIRLPOOL_CTX *context)
374
49
{
375
49
    int i;
376
49
    unsigned char *buffer      = context->buffer.data;
377
49
    unsigned char *bitLength   = context->bitlength;
378
49
    int bufferBits  = context->buffer.bits;
379
49
    int bufferPos   = context->buffer.pos;
380
381
    /*
382
     * append a '1'-bit:
383
     */
384
49
    buffer[bufferPos] |= 0x80U >> (bufferBits & 7);
385
49
    bufferPos++; /* all remaining bits on the current unsigned char are set to zero. */
386
    /*
387
     * pad with zero bits to complete (N*WBLOCKBITS - LENGTHBITS) bits:
388
     */
389
49
    if (bufferPos > WBLOCKBYTES - LENGTHBYTES) {
390
26
        if (bufferPos < WBLOCKBYTES) {
391
24
            memset(&buffer[bufferPos], 0, WBLOCKBYTES - bufferPos);
392
24
        }
393
        /*
394
         * process data block:
395
         */
396
26
        WhirlpoolTransform(context);
397
        /*
398
         * reset buffer:
399
         */
400
26
        bufferPos = 0;
401
26
    }
402
49
    if (bufferPos < WBLOCKBYTES - LENGTHBYTES) {
403
48
        memset(&buffer[bufferPos], 0, (WBLOCKBYTES - LENGTHBYTES) - bufferPos);
404
48
    }
405
49
    bufferPos = WBLOCKBYTES - LENGTHBYTES;
406
    /*
407
     * append bit length of hashed data:
408
     */
409
49
    memcpy(&buffer[WBLOCKBYTES - LENGTHBYTES], bitLength, LENGTHBYTES);
410
    /*
411
     * process data block:
412
     */
413
49
    WhirlpoolTransform(context);
414
    /*
415
     * return the completed message digest:
416
     */
417
441
    for (i = 0; i < DIGESTBYTES/8; i++) {
418
392
        digest[0] = (unsigned char)(context->state[i] >> 56);
419
392
        digest[1] = (unsigned char)(context->state[i] >> 48);
420
392
        digest[2] = (unsigned char)(context->state[i] >> 40);
421
392
        digest[3] = (unsigned char)(context->state[i] >> 32);
422
392
        digest[4] = (unsigned char)(context->state[i] >> 24);
423
392
        digest[5] = (unsigned char)(context->state[i] >> 16);
424
392
        digest[6] = (unsigned char)(context->state[i] >>  8);
425
392
        digest[7] = (unsigned char)(context->state[i]      );
426
392
        digest += 8;
427
392
    }
428
429
49
    ZEND_SECURE_ZERO(context, sizeof(*context));
430
49
}
431
432
static hash_spec_result php_whirlpool_unserialize(php_hashcontext_object *hash, zend_long magic, const zval *zv)
433
93
{
434
93
    PHP_WHIRLPOOL_CTX *ctx = (PHP_WHIRLPOOL_CTX *) hash->context;
435
93
    hash_spec_result r = HASH_SPEC_FAILURE;
436
93
    if (magic == PHP_HASH_SERIALIZE_MAGIC_SPEC
437
92
        && (r = php_hash_unserialize_spec(hash, zv, PHP_WHIRLPOOL_SPEC)) == HASH_SPEC_SUCCESS
438
88
        && ctx->buffer.pos >= 0
439
86
        && ctx->buffer.pos < (int) sizeof(ctx->buffer.data)
440
71
        && ctx->buffer.bits >= ctx->buffer.pos * 8
441
70
        && ctx->buffer.bits < ctx->buffer.pos * 8 + 8) {
442
49
        return HASH_SPEC_SUCCESS;
443
49
    }
444
445
44
    return r != HASH_SPEC_SUCCESS ? r : CONTEXT_VALIDATION_FAILURE;
446
93
}
447
448
const php_hash_ops php_hash_whirlpool_ops = {
449
  "whirlpool",
450
  (php_hash_init_func_t) PHP_WHIRLPOOLInit,
451
  (php_hash_update_func_t) PHP_WHIRLPOOLUpdate,
452
  (php_hash_final_func_t) PHP_WHIRLPOOLFinal,
453
  php_hash_copy,
454
  php_hash_serialize,
455
  php_whirlpool_unserialize,
456
  PHP_WHIRLPOOL_SPEC,
457
  64,
458
  64,
459
  sizeof(PHP_WHIRLPOOL_CTX),
460
  1
461
};