// This file is dual-licensed. Choose whichever licence you want from
// the two licences listed below.
//
// The first licence is a regular 2-clause BSD licence. The second licence
// is the CC-0 from Creative Commons. It is intended to release Monocypher
// to the public domain. The BSD licence serves as a fallback option.
//
// SPDX-License-Identifier: BSD-2-Clause OR CC0-1.0
//
// ------------------------------------------------------------------------
//
// Copyright (c) 2020, Mike Pechkin and Loup Vaillant
// All rights reserved.
//
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
// 1. Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the
// distribution.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
//
// ------------------------------------------------------------------------
//
// Written in 2017-2020 by Mike Pechkin and Loup Vaillant
//
// To the extent possible under law, the author(s) have dedicated all copyright
// and related neighboring rights to this software to the public domain
// worldwide. This software is distributed without any warranty.
//
// You should have received a copy of the CC0 Public Domain Dedication along
// with this software. If not, see
//
#include "monocypher.h"
#include "monocypher-ed25519.h"
#include "utils.h"
#include "tis-ci-vectors.h"
#include
#include
#include
static void chacha20(vector_reader *reader)
{
vector key = next_input(reader);
vector nonce = next_input(reader);
vector plain = next_input(reader);
u64 ctr = load64_le(next_input(reader).buf);
vector out = next_output(reader);
u64 nb_blocks = plain.size / 64 + (plain.size % 64 != 0);
u64 new_ctr = crypto_chacha20_djb(out.buf, plain.buf, plain.size,
key.buf, nonce.buf, ctr);
if (new_ctr - ctr != nb_blocks) {
printf("FAILURE: Chacha20 returned counter not correct: ");
}
}
static void ietf_chacha20(vector_reader *reader)
{
vector key = next_input(reader);
vector nonce = next_input(reader);
vector plain = next_input(reader);
u64 ctr = load64_le(next_input(reader).buf);
vector out = next_output(reader);
u32 nb_blocks = (u32)(plain.size / 64 + (plain.size % 64 != 0));
u32 new_ctr = crypto_chacha20_ietf(out.buf, plain.buf, plain.size,
key.buf, nonce.buf, (u32)ctr);
if (new_ctr - ctr != nb_blocks) {
printf("FAILURE: IETF Chacha20 returned counter not correct: ");
}
}
static void hchacha20(vector_reader *reader)
{
vector key = next_input(reader);
vector nonce = next_input(reader);
vector out = next_output(reader);
crypto_chacha20_h(out.buf, key.buf, nonce.buf);
}
static void xchacha20(vector_reader *reader)
{
vector key = next_input(reader);
vector nonce = next_input(reader);
vector plain = next_input(reader);
u64 ctr = load64_le(next_input(reader).buf);
vector out = next_output(reader);
u64 nb_blocks = plain.size / 64 + (plain.size % 64 != 0);
u64 new_ctr = crypto_chacha20_x(out.buf, plain.buf, plain.size,
key.buf, nonce.buf, ctr);
if (new_ctr - ctr != nb_blocks) {
printf("FAILURE: XChacha20 returned counter not correct: ");
}
}
static void poly1305(vector_reader *reader)
{
vector key = next_input(reader);
vector msg = next_input(reader);
vector out = next_output(reader);
crypto_poly1305(out.buf, msg.buf, msg.size, key.buf);
}
static void aead_ietf(vector_reader *reader)
{
vector key = next_input(reader);
vector nonce = next_input(reader);
vector ad = next_input(reader);
vector text = next_input(reader);
vector out = next_output(reader);
crypto_aead_lock(out.buf + 16, out.buf, key.buf, nonce.buf,
ad.buf, ad.size, text.buf, text.size);
}
static void blake2b(vector_reader *reader)
{
vector msg = next_input(reader);
vector key = next_input(reader);
vector out = next_output(reader);
crypto_blake2b_keyed(out.buf, out.size,
key.buf, key.size,
msg.buf, msg.size);
}
static void sha512(vector_reader *reader)
{
vector in = next_input(reader);
vector out = next_output(reader);
crypto_sha512(out.buf, in.buf, in.size);
}
static void sha512_hmac(vector_reader *reader)
{
vector key = next_input(reader);
vector msg = next_input(reader);
vector out = next_output(reader);
crypto_sha512_hmac(out.buf, key.buf, key.size, msg.buf, msg.size);
}
static void sha512_hkdf(vector_reader *reader)
{
vector ikm = next_input(reader);
vector salt = next_input(reader);
vector info = next_input(reader);
vector okm = next_output(reader);
crypto_sha512_hkdf(okm .buf, okm .size,
ikm .buf, ikm .size,
salt.buf, salt.size,
info.buf, info.size);
}
static void argon2(vector_reader *reader)
{
crypto_argon2_config config;
config.algorithm = load32_le(next_input(reader).buf);
config.nb_blocks = load32_le(next_input(reader).buf);
config.nb_passes = load32_le(next_input(reader).buf);
config.nb_lanes = load32_le(next_input(reader).buf);
vector pass = next_input(reader);
vector salt = next_input(reader);
vector key = next_input(reader);
vector ad = next_input(reader);
vector out = next_output(reader);
void *work_area = alloc(config.nb_blocks * 1024);
crypto_argon2_inputs inputs;
inputs.pass = pass.buf;
inputs.salt = salt.buf;
inputs.pass_size = (u32)pass.size;
inputs.salt_size = (u32)salt.size;
crypto_argon2_extras extras;
extras.key = key.buf;
extras.ad = ad.buf;
extras.key_size = (u32)key.size;
extras.ad_size = (u32)ad.size;
crypto_argon2(out.buf, (u32)out.size, work_area, config, inputs, extras);
free(work_area);
}
static void x25519(vector_reader *reader)
{
vector scalar = next_input(reader);
vector point = next_input(reader);
vector out = next_output(reader);
crypto_x25519(out.buf, scalar.buf, point.buf);
}
static void edDSA(vector_reader *reader)
{
vector secret_k = next_input(reader);
vector public_k = next_input(reader);
vector msg = next_input(reader);
vector out = next_output(reader);
u8 fat_secret_key[64];
memcpy(fat_secret_key , secret_k.buf, 32);
memcpy(fat_secret_key + 32, public_k.buf, 32);
crypto_eddsa_sign(out.buf, fat_secret_key, msg.buf, msg.size);
}
static void edDSA_pk(vector_reader *reader)
{
vector in = next_input(reader);
vector out = next_output(reader);
u8 seed [32];
u8 secret_key[64];
u8 public_key[32];
memcpy(seed, in.buf, 32);
crypto_eddsa_key_pair(secret_key, public_key, seed);
memcpy(out.buf, public_key, 32);
u8 zeroes[32] = {0};
ASSERT_EQUAL(seed , zeroes , 32);
ASSERT_EQUAL(secret_key , in.buf , 32);
ASSERT_EQUAL(secret_key + 32, public_key, 32);
}
static void ed_25519(vector_reader *reader)
{
vector secret_k = next_input(reader);
vector public_k = next_input(reader);
vector msg = next_input(reader);
vector out = next_output(reader);
u8 fat_secret_key[64];
memcpy(fat_secret_key , secret_k.buf, 32);
memcpy(fat_secret_key + 32, public_k.buf, 32);
crypto_ed25519_sign(out.buf, fat_secret_key, msg.buf, msg.size);
}
static void ed_25519_check(vector_reader *reader)
{
vector public_k = next_input(reader);
vector msg = next_input(reader);
vector sig = next_input(reader);
vector out = next_output(reader);
out.buf[0] = (u8)crypto_ed25519_check(sig.buf, public_k.buf,
msg.buf, msg.size);
}
static void elligator_dir(vector_reader *reader)
{
vector in = next_input(reader);
vector out = next_output(reader);
crypto_elligator_map(out.buf, in.buf);
}
static void elligator_inv(vector_reader *reader)
{
vector point = next_input(reader);
u8 tweak = next_input(reader).buf[0];
u8 failure = next_input(reader).buf[0];
vector out = next_output(reader);
int check = crypto_elligator_rev(out.buf, point.buf, tweak);
ASSERT((u8)check == failure);
if (check) {
out.buf[0] = 0;
}
}
//@ ensures \result == 0;
static int p_wipe(void)
{
printf("\tcrypto_wipe\n");
u8 zeroes[50] = {0};
FOR (i, 0, 50) {
RANDOM_INPUT(buf, 50);
crypto_wipe(buf, i);
ASSERT_EQUAL(zeroes, buf, i);
}
return 0;
}
//@ ensures \result == 0;
static int p_eddsa_x25519(void)
{
RANDOM_INPUT(e_seed, 32);
u8 secret [64];
u8 e_public1[32]; crypto_eddsa_key_pair(secret, e_public1, e_seed);
u8 x_private[64]; crypto_blake2b(x_private, 64, secret, 32);
u8 x_public1[32]; crypto_eddsa_to_x25519 (x_public1, e_public1);
u8 x_public2[32]; crypto_x25519_public_key(x_public2, x_private);
ASSERT_EQUAL(x_public1, x_public2, 32);
u8 e_public2[32]; crypto_x25519_to_eddsa (e_public2, x_public1);
ASSERT((e_public2[31] & 0x80) == 0); // x coordinate always positive
e_public1[31] &= 0x7f; // y coordinate back to original
ASSERT_EQUAL(e_public1, e_public2, 32);
return 0;
}
//@ ensures \result == 0;
static int p_dirty(void)
{
int status = 0;
RANDOM_INPUT(sk1, 32); sk1[0] |= 1; // make sure it's dirty
u8 skc [32]; memcpy(skc, sk1, 32); skc[0] &= 248; // make sure it's clean
u8 pks [32]; crypto_x25519_dirty_small(pks , sk1);
u8 pksc[32]; crypto_x25519_dirty_small(pksc, skc);
u8 pkf [32]; crypto_x25519_dirty_fast (pkf , sk1);
u8 pkfc[32]; crypto_x25519_dirty_fast (pkfc, skc);
u8 pk1 [32]; crypto_x25519_public_key (pk1 , sk1);
// Both dirty functions behave the same
status |= memcmp(pks, pkf, 32);
// Dirty functions behave cleanly if we clear the 3 msb first
status |= memcmp(pksc, pk1, 32);
status |= memcmp(pkfc, pk1, 32);
printf("%s: x25519 dirty\n", status != 0 ? "FAILED" : "OK");
return status;
}
//@ ensures \result == 0;
static int p_x25519_inverse(void)
{
int status = 0;
RANDOM_INPUT(b, 32);
u8 base[32]; // random point (cofactor is cleared).
crypto_x25519_public_key(base, b);
// check round trip
RANDOM_INPUT(sk, 32);
u8 pk [32];
u8 blind[32];
crypto_x25519(pk, sk, base);
crypto_x25519_inverse(blind, sk, pk);
status |= memcmp(blind, base, 32);
// check cofactor clearing
// (Multiplying by a low order point yields zero
u8 low_order[32] = {
0x5f, 0x9c, 0x95, 0xbc, 0xa3, 0x50, 0x8c, 0x24,
0xb1, 0xd0, 0xb1, 0x55, 0x9c, 0x83, 0xef, 0x5b,
0x04, 0x44, 0x5c, 0xc4, 0x58, 0x1c, 0x8e, 0x86,
0xd8, 0x22, 0x4e, 0xdd, 0xd0, 0x9f, 0x11, 0x57,
};
u8 zero[32] = {0};
crypto_x25519_inverse(blind, sk, low_order);
status |= memcmp(blind, zero, 32);
printf("%s: x25519_inverse\n", status != 0 ? "FAILED" : "OK");
return status;
}
//@ ensures \result == 0;
static int p_verify(size_t size, int (*compare)(const u8*, const u8*))
{
int status = 0;
u8 a[64]; // size <= 64
u8 b[64]; // size <= 64
FOR (i, 0, 2) {
FOR (j, 0, 2) {
// Set every byte to the chosen value, then compare
FOR (k, 0, size) {
a[k] = (u8)i;
b[k] = (u8)j;
}
int cmp = compare(a, b);
status |= (i == j ? cmp : ~cmp);
// Set only two bytes to the chosen value, then compare
FOR (k, 0, size / 2) {
FOR (l, 0, size) {
a[l] = 0;
b[l] = 0;
}
a[k] = (u8)i; a[k + size/2 - 1] = (u8)i;
b[k] = (u8)j; b[k + size/2 - 1] = (u8)j;
cmp = compare(a, b);
status |= (i == j ? cmp : ~cmp);
}
}
}
printf("%s: crypto_verify%zu\n", status != 0 ? "FAILED" : "OK", size);
return status;
}
//@ ensures \result == 0;
static int p_verify16(void){ return p_verify(16, crypto_verify16); }
//@ ensures \result == 0;
static int p_verify32(void){ return p_verify(32, crypto_verify32); }
//@ ensures \result == 0;
static int p_verify64(void){ return p_verify(64, crypto_verify64); }
#define TEST(name) \
int v_##name(void) { \
return vector_test(name, #name, nb_##name##_vectors, name##_vectors); \
}
//@ ensures \result == 0;
TEST(chacha20)
//@ ensures \result == 0;
TEST(ietf_chacha20)
//@ ensures \result == 0;
TEST(hchacha20)
//@ ensures \result == 0;
TEST(xchacha20)
//@ ensures \result == 0;
TEST(poly1305)
//@ ensures \result == 0;
TEST(aead_ietf)
//@ ensures \result == 0;
TEST(blake2b)
//@ ensures \result == 0;
TEST(sha512)
//@ ensures \result == 0;
TEST(sha512_hmac)
//@ ensures \result == 0;
TEST(sha512_hkdf)
//@ ensures \result == 0;
TEST(argon2)
//@ ensures \result == 0;
TEST(x25519)
//@ ensures \result == 0;
TEST(edDSA)
//@ ensures \result == 0;
TEST(edDSA_pk)
//@ ensures \result == 0;
TEST(ed_25519)
//@ ensures \result == 0;
TEST(ed_25519_check)
//@ ensures \result == 0;
TEST(elligator_dir)
//@ ensures \result == 0;
TEST(elligator_inv)
//@ ensures \result == 0;
int main(void) {
ASSERT(v_chacha20 () == 0);
ASSERT(v_ietf_chacha20 () == 0);
ASSERT(v_hchacha20 () == 0);
ASSERT(v_xchacha20 () == 0);
ASSERT(v_poly1305 () == 0);
ASSERT(v_aead_ietf () == 0);
ASSERT(v_blake2b () == 0);
ASSERT(v_sha512 () == 0);
ASSERT(v_sha512_hmac () == 0);
ASSERT(v_sha512_hkdf () == 0);
ASSERT(v_argon2 () == 0);
ASSERT(v_x25519 () == 0);
ASSERT(v_edDSA () == 0);
ASSERT(v_edDSA_pk () == 0);
ASSERT(v_ed_25519 () == 0);
ASSERT(v_ed_25519_check() == 0);
ASSERT(v_elligator_dir () == 0);
ASSERT(v_elligator_inv () == 0);
ASSERT(p_wipe () == 0);
ASSERT(p_eddsa_x25519 () == 0);
ASSERT(p_dirty () == 0);
ASSERT(p_x25519_inverse() == 0);
ASSERT(p_verify16 () == 0);
ASSERT(p_verify32 () == 0);
ASSERT(p_verify64 () == 0);
return 0;
}