10ed42a6fSJason A. Donenfeld /* SPDX-License-Identifier: GPL-2.0 OR MIT */
20ed42a6fSJason A. Donenfeld /*
30ed42a6fSJason A. Donenfeld * Copyright (C) 2015-2019 Jason A. Donenfeld <Jason@zx2c4.com>. All Rights Reserved.
40ed42a6fSJason A. Donenfeld */
50ed42a6fSJason A. Donenfeld
60ed42a6fSJason A. Donenfeld #ifndef CURVE25519_H
70ed42a6fSJason A. Donenfeld #define CURVE25519_H
80ed42a6fSJason A. Donenfeld
90ed42a6fSJason A. Donenfeld #include <crypto/algapi.h> // For crypto_memneq.
100ed42a6fSJason A. Donenfeld #include <linux/types.h>
110ed42a6fSJason A. Donenfeld #include <linux/random.h>
120ed42a6fSJason A. Donenfeld
130ed42a6fSJason A. Donenfeld enum curve25519_lengths {
140ed42a6fSJason A. Donenfeld CURVE25519_KEY_SIZE = 32
150ed42a6fSJason A. Donenfeld };
160ed42a6fSJason A. Donenfeld
170ed42a6fSJason A. Donenfeld extern const u8 curve25519_null_point[];
180ed42a6fSJason A. Donenfeld extern const u8 curve25519_base_point[];
190ed42a6fSJason A. Donenfeld
200ed42a6fSJason A. Donenfeld void curve25519_generic(u8 out[CURVE25519_KEY_SIZE],
210ed42a6fSJason A. Donenfeld const u8 scalar[CURVE25519_KEY_SIZE],
220ed42a6fSJason A. Donenfeld const u8 point[CURVE25519_KEY_SIZE]);
230ed42a6fSJason A. Donenfeld
240ed42a6fSJason A. Donenfeld void curve25519_arch(u8 out[CURVE25519_KEY_SIZE],
250ed42a6fSJason A. Donenfeld const u8 scalar[CURVE25519_KEY_SIZE],
260ed42a6fSJason A. Donenfeld const u8 point[CURVE25519_KEY_SIZE]);
270ed42a6fSJason A. Donenfeld
280ed42a6fSJason A. Donenfeld void curve25519_base_arch(u8 pub[CURVE25519_KEY_SIZE],
290ed42a6fSJason A. Donenfeld const u8 secret[CURVE25519_KEY_SIZE]);
300ed42a6fSJason A. Donenfeld
31*1201581cSHerbert Xu bool curve25519_selftest(void);
32*1201581cSHerbert Xu
330ed42a6fSJason A. Donenfeld static inline
curve25519(u8 mypublic[CURVE25519_KEY_SIZE],const u8 secret[CURVE25519_KEY_SIZE],const u8 basepoint[CURVE25519_KEY_SIZE])340ed42a6fSJason A. Donenfeld bool __must_check curve25519(u8 mypublic[CURVE25519_KEY_SIZE],
350ed42a6fSJason A. Donenfeld const u8 secret[CURVE25519_KEY_SIZE],
360ed42a6fSJason A. Donenfeld const u8 basepoint[CURVE25519_KEY_SIZE])
370ed42a6fSJason A. Donenfeld {
383f523e12SJason A. Donenfeld if (IS_ENABLED(CONFIG_CRYPTO_ARCH_HAVE_LIB_CURVE25519))
390ed42a6fSJason A. Donenfeld curve25519_arch(mypublic, secret, basepoint);
400ed42a6fSJason A. Donenfeld else
410ed42a6fSJason A. Donenfeld curve25519_generic(mypublic, secret, basepoint);
420ed42a6fSJason A. Donenfeld return crypto_memneq(mypublic, curve25519_null_point,
430ed42a6fSJason A. Donenfeld CURVE25519_KEY_SIZE);
440ed42a6fSJason A. Donenfeld }
450ed42a6fSJason A. Donenfeld
460ed42a6fSJason A. Donenfeld static inline bool
curve25519_generate_public(u8 pub[CURVE25519_KEY_SIZE],const u8 secret[CURVE25519_KEY_SIZE])470ed42a6fSJason A. Donenfeld __must_check curve25519_generate_public(u8 pub[CURVE25519_KEY_SIZE],
480ed42a6fSJason A. Donenfeld const u8 secret[CURVE25519_KEY_SIZE])
490ed42a6fSJason A. Donenfeld {
500ed42a6fSJason A. Donenfeld if (unlikely(!crypto_memneq(secret, curve25519_null_point,
510ed42a6fSJason A. Donenfeld CURVE25519_KEY_SIZE)))
520ed42a6fSJason A. Donenfeld return false;
530ed42a6fSJason A. Donenfeld
543f523e12SJason A. Donenfeld if (IS_ENABLED(CONFIG_CRYPTO_ARCH_HAVE_LIB_CURVE25519))
550ed42a6fSJason A. Donenfeld curve25519_base_arch(pub, secret);
560ed42a6fSJason A. Donenfeld else
570ed42a6fSJason A. Donenfeld curve25519_generic(pub, secret, curve25519_base_point);
580ed42a6fSJason A. Donenfeld return crypto_memneq(pub, curve25519_null_point, CURVE25519_KEY_SIZE);
590ed42a6fSJason A. Donenfeld }
600ed42a6fSJason A. Donenfeld
curve25519_clamp_secret(u8 secret[CURVE25519_KEY_SIZE])610ed42a6fSJason A. Donenfeld static inline void curve25519_clamp_secret(u8 secret[CURVE25519_KEY_SIZE])
620ed42a6fSJason A. Donenfeld {
630ed42a6fSJason A. Donenfeld secret[0] &= 248;
640ed42a6fSJason A. Donenfeld secret[31] = (secret[31] & 127) | 64;
650ed42a6fSJason A. Donenfeld }
660ed42a6fSJason A. Donenfeld
curve25519_generate_secret(u8 secret[CURVE25519_KEY_SIZE])670ed42a6fSJason A. Donenfeld static inline void curve25519_generate_secret(u8 secret[CURVE25519_KEY_SIZE])
680ed42a6fSJason A. Donenfeld {
690ed42a6fSJason A. Donenfeld get_random_bytes_wait(secret, CURVE25519_KEY_SIZE);
700ed42a6fSJason A. Donenfeld curve25519_clamp_secret(secret);
710ed42a6fSJason A. Donenfeld }
720ed42a6fSJason A. Donenfeld
730ed42a6fSJason A. Donenfeld #endif /* CURVE25519_H */
74