Coverage Report

Created: 2018-08-29 13:53

/src/openssl/crypto/siphash/siphash.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * Copyright 2017 The OpenSSL Project Authors. All Rights Reserved.
3
 *
4
 * Licensed under the OpenSSL license (the "License").  You may not use
5
 * this file except in compliance with the License.  You can obtain a copy
6
 * in the file LICENSE in the source distribution or at
7
 * https://www.openssl.org/source/license.html
8
 */
9
10
/* Based on https://131002.net/siphash C reference implementation */
11
/*
12
   SipHash reference C implementation
13
14
   Copyright (c) 2012-2016 Jean-Philippe Aumasson
15
   Copyright (c) 2012-2014 Daniel J. Bernstein
16
17
   To the extent possible under law, the author(s) have dedicated all copyright
18
   and related and neighboring rights to this software to the public domain
19
   worldwide. This software is distributed without any warranty.
20
21
   You should have received a copy of the CC0 Public Domain Dedication along
22
   with this software. If not, see
23
   <http://creativecommons.org/publicdomain/zero/1.0/>.
24
 */
25
26
#include <stdlib.h>
27
#include <string.h>
28
#include <openssl/crypto.h>
29
30
#include "internal/siphash.h"
31
#include "siphash_local.h"
32
33
/* default: SipHash-2-4 */
34
0
#define SIPHASH_C_ROUNDS 2
35
0
#define SIPHASH_D_ROUNDS 4
36
37
0
#define ROTL(x, b) (uint64_t)(((x) << (b)) | ((x) >> (64 - (b))))
38
39
#define U32TO8_LE(p, v)                                                        \
40
0
    (p)[0] = (uint8_t)((v));                                                   \
41
0
    (p)[1] = (uint8_t)((v) >> 8);                                              \
42
0
    (p)[2] = (uint8_t)((v) >> 16);                                             \
43
0
    (p)[3] = (uint8_t)((v) >> 24);
44
45
#define U64TO8_LE(p, v)                                                        \
46
0
    U32TO8_LE((p), (uint32_t)((v)));                                           \
47
0
    U32TO8_LE((p) + 4, (uint32_t)((v) >> 32));
48
49
#define U8TO64_LE(p)                                                           \
50
0
    (((uint64_t)((p)[0])) | ((uint64_t)((p)[1]) << 8) |                        \
51
0
     ((uint64_t)((p)[2]) << 16) | ((uint64_t)((p)[3]) << 24) |                 \
52
0
     ((uint64_t)((p)[4]) << 32) | ((uint64_t)((p)[5]) << 40) |                 \
53
0
     ((uint64_t)((p)[6]) << 48) | ((uint64_t)((p)[7]) << 56))
54
55
#define SIPROUND                                                               \
56
0
    do {                                                                       \
57
0
        v0 += v1;                                                              \
58
0
        v1 = ROTL(v1, 13);                                                     \
59
0
        v1 ^= v0;                                                              \
60
0
        v0 = ROTL(v0, 32);                                                     \
61
0
        v2 += v3;                                                              \
62
0
        v3 = ROTL(v3, 16);                                                     \
63
0
        v3 ^= v2;                                                              \
64
0
        v0 += v3;                                                              \
65
0
        v3 = ROTL(v3, 21);                                                     \
66
0
        v3 ^= v0;                                                              \
67
0
        v2 += v1;                                                              \
68
0
        v1 = ROTL(v1, 17);                                                     \
69
0
        v1 ^= v2;                                                              \
70
0
        v2 = ROTL(v2, 32);                                                     \
71
0
    } while (0)
72
73
size_t SipHash_ctx_size(void)
74
0
{
75
0
    return sizeof(SIPHASH);
76
0
}
77
78
size_t SipHash_hash_size(SIPHASH *ctx)
79
0
{
80
0
    return ctx->hash_size;
81
0
}
82
83
/* hash_size = crounds = drounds = 0 means SipHash24 with 16-byte output */
84
int SipHash_Init(SIPHASH *ctx, const unsigned char *k, int hash_size, int crounds, int drounds)
85
0
{
86
0
    uint64_t k0 = U8TO64_LE(k);
87
0
    uint64_t k1 = U8TO64_LE(k + 8);
88
0
89
0
    if (hash_size == 0)
90
0
        hash_size = SIPHASH_MAX_DIGEST_SIZE;
91
0
    else if (hash_size != SIPHASH_MIN_DIGEST_SIZE &&
92
0
             hash_size != SIPHASH_MAX_DIGEST_SIZE)
93
0
        return 0;
94
0
95
0
    if (drounds == 0)
96
0
        drounds = SIPHASH_D_ROUNDS;
97
0
    if (crounds == 0)
98
0
        crounds = SIPHASH_C_ROUNDS;
99
0
100
0
    ctx->crounds = crounds;
101
0
    ctx->drounds = drounds;
102
0
    ctx->hash_size = hash_size;
103
0
104
0
    ctx->len = 0;
105
0
    ctx->total_inlen = 0;
106
0
107
0
    ctx->v0 = 0x736f6d6570736575ULL ^ k0;
108
0
    ctx->v1 = 0x646f72616e646f6dULL ^ k1;
109
0
    ctx->v2 = 0x6c7967656e657261ULL ^ k0;
110
0
    ctx->v3 = 0x7465646279746573ULL ^ k1;
111
0
112
0
    if (ctx->hash_size == SIPHASH_MAX_DIGEST_SIZE)
113
0
        ctx->v1 ^= 0xee;
114
0
115
0
    return 1;
116
0
}
117
118
void SipHash_Update(SIPHASH *ctx, const unsigned char *in, size_t inlen)
119
0
{
120
0
    uint64_t m;
121
0
    const uint8_t *end;
122
0
    int left;
123
0
    int i;
124
0
    uint64_t v0 = ctx->v0;
125
0
    uint64_t v1 = ctx->v1;
126
0
    uint64_t v2 = ctx->v2;
127
0
    uint64_t v3 = ctx->v3;
128
0
129
0
    ctx->total_inlen += inlen;
130
0
131
0
    if (ctx->len) {
132
0
        /* deal with leavings */
133
0
        size_t available = SIPHASH_BLOCK_SIZE - ctx->len;
134
0
135
0
        /* not enough to fill leavings */
136
0
        if (inlen < available) {
137
0
            memcpy(&ctx->leavings[ctx->len], in, inlen);
138
0
            ctx->len += inlen;
139
0
            return;
140
0
        }
141
0
142
0
        /* copy data into leavings and reduce input */
143
0
        memcpy(&ctx->leavings[ctx->len], in, available);
144
0
        inlen -= available;
145
0
        in += available;
146
0
147
0
        /* process leavings */
148
0
        m = U8TO64_LE(ctx->leavings);
149
0
        v3 ^= m;
150
0
        for (i = 0; i < ctx->crounds; ++i)
151
0
            SIPROUND;
152
0
        v0 ^= m;
153
0
    }
154
0
    left = inlen & (SIPHASH_BLOCK_SIZE-1); /* gets put into leavings */
155
0
    end = in + inlen - left;
156
0
157
0
    for (; in != end; in += 8) {
158
0
        m = U8TO64_LE(in);
159
0
        v3 ^= m;
160
0
        for (i = 0; i < ctx->crounds; ++i)
161
0
            SIPROUND;
162
0
        v0 ^= m;
163
0
    }
164
0
165
0
    /* save leavings and other ctx */
166
0
    if (left)
167
0
        memcpy(ctx->leavings, end, left);
168
0
    ctx->len = left;
169
0
170
0
    ctx->v0 = v0;
171
0
    ctx->v1 = v1;
172
0
    ctx->v2 = v2;
173
0
    ctx->v3 = v3;
174
0
}
175
176
int SipHash_Final(SIPHASH *ctx, unsigned char *out, size_t outlen)
177
0
{
178
0
    /* finalize hash */
179
0
    int i;
180
0
    uint64_t b = ctx->total_inlen << 56;
181
0
    uint64_t v0 = ctx->v0;
182
0
    uint64_t v1 = ctx->v1;
183
0
    uint64_t v2 = ctx->v2;
184
0
    uint64_t v3 = ctx->v3;
185
0
186
0
    if (outlen != (size_t)ctx->hash_size)
187
0
        return 0;
188
0
189
0
    switch (ctx->len) {
190
0
    case 7:
191
0
        b |= ((uint64_t)ctx->leavings[6]) << 48;
192
0
        /* fall thru */
193
0
    case 6:
194
0
        b |= ((uint64_t)ctx->leavings[5]) << 40;
195
0
        /* fall thru */
196
0
    case 5:
197
0
        b |= ((uint64_t)ctx->leavings[4]) << 32;
198
0
        /* fall thru */
199
0
    case 4:
200
0
        b |= ((uint64_t)ctx->leavings[3]) << 24;
201
0
        /* fall thru */
202
0
    case 3:
203
0
        b |= ((uint64_t)ctx->leavings[2]) << 16;
204
0
        /* fall thru */
205
0
    case 2:
206
0
        b |= ((uint64_t)ctx->leavings[1]) <<  8;
207
0
        /* fall thru */
208
0
    case 1:
209
0
        b |= ((uint64_t)ctx->leavings[0]);
210
0
    case 0:
211
0
        break;
212
0
    }
213
0
214
0
    v3 ^= b;
215
0
    for (i = 0; i < ctx->crounds; ++i)
216
0
        SIPROUND;
217
0
    v0 ^= b;
218
0
    if (ctx->hash_size == SIPHASH_MAX_DIGEST_SIZE)
219
0
        v2 ^= 0xee;
220
0
    else
221
0
        v2 ^= 0xff;
222
0
    for (i = 0; i < ctx->drounds; ++i)
223
0
        SIPROUND;
224
0
    b = v0 ^ v1 ^ v2  ^ v3;
225
0
    U64TO8_LE(out, b);
226
0
    if (ctx->hash_size == SIPHASH_MIN_DIGEST_SIZE)
227
0
        return 1;
228
0
    v1 ^= 0xdd;
229
0
    for (i = 0; i < ctx->drounds; ++i)
230
0
        SIPROUND;
231
0
    b = v0 ^ v1 ^ v2  ^ v3;
232
0
    U64TO8_LE(out + 8, b);
233
0
    return 1;
234
0
}