/src/libressl/crypto/chacha/chacha.c
Line | Count | Source (jump to first uncovered line) |
1 | | /* $OpenBSD: chacha.c,v 1.8 2019/01/22 00:59:21 dlg Exp $ */ |
2 | | /* |
3 | | * Copyright (c) 2014 Joel Sing <jsing@openbsd.org> |
4 | | * |
5 | | * Permission to use, copy, modify, and distribute this software for any |
6 | | * purpose with or without fee is hereby granted, provided that the above |
7 | | * copyright notice and this permission notice appear in all copies. |
8 | | * |
9 | | * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES |
10 | | * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF |
11 | | * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR |
12 | | * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES |
13 | | * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN |
14 | | * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF |
15 | | * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. |
16 | | */ |
17 | | |
18 | | #include <stdint.h> |
19 | | |
20 | | #include <openssl/chacha.h> |
21 | | |
22 | | #include "chacha-merged.c" |
23 | | |
24 | | void |
25 | | ChaCha_set_key(ChaCha_ctx *ctx, const unsigned char *key, uint32_t keybits) |
26 | 0 | { |
27 | 0 | chacha_keysetup((chacha_ctx *)ctx, key, keybits); |
28 | 0 | ctx->unused = 0; |
29 | 0 | } |
30 | | |
31 | | void |
32 | | ChaCha_set_iv(ChaCha_ctx *ctx, const unsigned char *iv, |
33 | | const unsigned char *counter) |
34 | 0 | { |
35 | 0 | chacha_ivsetup((chacha_ctx *)ctx, iv, counter); |
36 | 0 | ctx->unused = 0; |
37 | 0 | } |
38 | | |
39 | | void |
40 | | ChaCha(ChaCha_ctx *ctx, unsigned char *out, const unsigned char *in, size_t len) |
41 | 0 | { |
42 | 0 | unsigned char *k; |
43 | 0 | int i, l; |
44 | | |
45 | | /* Consume remaining keystream, if any exists. */ |
46 | 0 | if (ctx->unused > 0) { |
47 | 0 | k = ctx->ks + 64 - ctx->unused; |
48 | 0 | l = (len > ctx->unused) ? ctx->unused : len; |
49 | 0 | for (i = 0; i < l; i++) |
50 | 0 | *(out++) = *(in++) ^ *(k++); |
51 | 0 | ctx->unused -= l; |
52 | 0 | len -= l; |
53 | 0 | } |
54 | |
|
55 | 0 | chacha_encrypt_bytes((chacha_ctx *)ctx, in, out, (uint32_t)len); |
56 | 0 | } |
57 | | |
58 | | void |
59 | | CRYPTO_chacha_20(unsigned char *out, const unsigned char *in, size_t len, |
60 | | const unsigned char key[32], const unsigned char iv[8], uint64_t counter) |
61 | 466 | { |
62 | 466 | struct chacha_ctx ctx; |
63 | | |
64 | | /* |
65 | | * chacha_ivsetup expects the counter to be in u8. Rather than |
66 | | * converting size_t to u8 and then back again, pass a counter of |
67 | | * NULL and manually assign it afterwards. |
68 | | */ |
69 | 466 | chacha_keysetup(&ctx, key, 256); |
70 | 466 | chacha_ivsetup(&ctx, iv, NULL); |
71 | 466 | if (counter != 0) { |
72 | 466 | ctx.input[12] = (uint32_t)counter; |
73 | 466 | ctx.input[13] = (uint32_t)(counter >> 32); |
74 | 466 | } |
75 | | |
76 | 466 | chacha_encrypt_bytes(&ctx, in, out, (uint32_t)len); |
77 | 466 | } |
78 | | |
79 | | void |
80 | | CRYPTO_xchacha_20(unsigned char *out, const unsigned char *in, size_t len, |
81 | | const unsigned char key[32], const unsigned char iv[24]) |
82 | 0 | { |
83 | 0 | uint8_t subkey[32]; |
84 | |
|
85 | 0 | CRYPTO_hchacha_20(subkey, key, iv); |
86 | 0 | CRYPTO_chacha_20(out, in, len, subkey, iv + 16, 0); |
87 | 0 | } |