|
- /* Copyright (c) 2014 Cryptography Research, Inc.
- * Released under the MIT License. See LICENSE.txt for license information.
- */
-
- /**
- * @file goldilocks.h
- * @author Mike Hamburg
- * @brief Goldilocks high-level functions.
- */
- #ifndef __GOLDILOCKS_H__
- #define __GOLDILOCKS_H__ 1
-
- #include <stdint.h>
-
- #ifndef GOLDI_IMPLEMENT_PRECOMPUTED_KEYS
- /** If nonzero, implement precomputation for verify and ECDH. */
- #define GOLDI_IMPLEMENT_PRECOMPUTED_KEYS 1
- #endif
-
- /** The size of the Goldilocks field, in bits. */
- #define GOLDI_FIELD_BITS 448
-
- /** The size of the Goldilocks scalars, in bits. */
- #define GOLDI_SCALAR_BITS 446
-
- /** The same size, in bytes. */
- #define GOLDI_FIELD_BYTES (GOLDI_FIELD_BITS/8)
-
- /** The size of a Goldilocks public key, in bytes. */
- #define GOLDI_PUBLIC_KEY_BYTES GOLDI_FIELD_BYTES
-
- /** The extra bytes in a Goldilocks private key for the symmetric key. */
- #define GOLDI_SYMKEY_BYTES 32
-
- /** The size of a shared secret. */
- #define GOLDI_SHARED_SECRET_BYTES 64
-
- /** The size of a Goldilocks private key, in bytes. */
- #define GOLDI_PRIVATE_KEY_BYTES (2*GOLDI_FIELD_BYTES + GOLDI_SYMKEY_BYTES)
-
- /** The size of a Goldilocks private key, in bytes. */
- #define GOLDI_SIGNATURE_BYTES (2*GOLDI_FIELD_BYTES)
-
- /**
- * @brief Serialized form of a Goldilocks public key.
- *
- * @warning This isn't even my final form!
- */
- struct goldilocks_public_key_t {
- uint8_t opaque[GOLDI_PUBLIC_KEY_BYTES]; /**< Serialized data. */
- };
-
- /**
- * @brief Serialized form of a Goldilocks private key.
- *
- * Contains 56 bytes of actual private key, 56 bytes of
- * public key, and 32 bytes of symmetric key for randomization.
- *
- * @warning This isn't even my final form!
- */
- struct goldilocks_private_key_t {
- uint8_t opaque[GOLDI_PRIVATE_KEY_BYTES]; /**< Serialized data. */
- };
-
- #ifdef __cplusplus
- extern "C" {
- #endif
-
- /** @brief No error. */
- static const int GOLDI_EOK = 0;
-
- /** @brief Error: your key or other state is corrupt. */
- static const int GOLDI_ECORRUPT = 44801;
-
- /** @brief Error: other party's key is corrupt. */
- static const int GOLDI_EINVAL = 44802;
-
- /** @brief Error: not enough entropy. */
- static const int GOLDI_ENODICE = 44804;
-
- /** @brief Error: you need to initialize the library first. */
- static const int GOLDI_EUNINIT = 44805;
-
- /** @brief Error: called init() but we are already initialized. */
- static const int GOLDI_EALREADYINIT = 44805;
-
- /**
- * @brief Initialize Goldilocks' precomputed tables and
- * random number generator. This function must be called before
- * any of the other Goldilocks routines (except
- * goldilocks_shared_secret in the current version) and should be
- * called only once per process.
- *
- * There is currently no way to tear down this state. It is possible
- * that a future version of this library will not require this function.
- *
- * @retval GOLDI_EOK Success.
- * @retval GOLDI_EALREADYINIT Already initialized.
- * @retval GOLDI_ECORRUPT Memory is corrupted, or another thread is already init'ing.
- * @retval Nonzero An error occurred.
- */
- int
- goldilocks_init ()
- __attribute__((warn_unused_result,visibility ("default")));
-
-
- /**
- * @brief Generate a new random keypair.
- * @param [out] privkey The generated private key.
- * @param [out] pubkey The generated public key.
- *
- * @warning This isn't even my final form!
- *
- * @retval GOLDI_EOK Success.
- * @retval GOLDI_ENODICE Insufficient entropy.
- * @retval GOLDI_EUNINIT You must call goldilocks_init() first.
- */
- int
- goldilocks_keygen (
- struct goldilocks_private_key_t *privkey,
- struct goldilocks_public_key_t *pubkey
- ) __attribute__((warn_unused_result,nonnull(1,2),visibility ("default")));
-
- /**
- * @brief Derive a key from its compressed form.
- * @param [out] privkey The derived private key.
- * @param [in] proto The compressed or proto-key, which must be 32 random bytes.
- *
- * @warning This isn't even my final form!
- *
- * @retval GOLDI_EOK Success.
- * @retval GOLDI_EUNINIT You must call goldilocks_init() first.
- */
- int
- goldilocks_derive_private_key (
- struct goldilocks_private_key_t *privkey,
- const unsigned char proto[GOLDI_SYMKEY_BYTES]
- ) __attribute__((nonnull(1,2),visibility ("default")));
-
- /**
- * @brief Compress a private key (by copying out the proto-key)
- * @param [out] proto The proto-key.
- * @param [in] privkey The private key.
- *
- * @warning This isn't even my final form!
- * @todo test.
- *
- * @retval GOLDI_EOK Success.
- * @retval GOLDI_EUNINIT You must call goldilocks_init() first.
- */
- void
- goldilocks_underive_private_key (
- unsigned char proto[GOLDI_SYMKEY_BYTES],
- const struct goldilocks_private_key_t *privkey
- ) __attribute__((nonnull(1,2),visibility ("default")));
-
- /**
- * @brief Extract the public key from a private key.
- *
- * This is essentially a memcpy from the public part of the privkey.
- *
- * @param [out] pubkey The extracted private key.
- * @param [in] privkey The private key.
- *
- * @retval GOLDI_EOK Success.
- * @retval GOLDI_ECORRUPT The private key is corrupt.
- */
- int
- goldilocks_private_to_public (
- struct goldilocks_public_key_t *pubkey,
- const struct goldilocks_private_key_t *privkey
- ) __attribute__((nonnull(1,2),visibility ("default")));
-
- /**
- * @brief Generate a Diffie-Hellman shared secret in constant time.
- *
- * This function uses some compile-time flags whose merit remains to
- * be decided.
- *
- * If the flag EXPERIMENT_ECDH_OBLITERATE_CT is set, prepend 40 bytes
- * of zeros to the secret before hashing. In the case that the other
- * party's key is detectably corrupt, instead the symmetric part
- * of the secret key is used to produce a pseudorandom value.
- *
- * If EXPERIMENT_ECDH_STIR_IN_PUBKEYS is set, the sum and product of
- * the two parties' public keys is prepended to the hash.
- *
- * In the current version, this function can safely be run even without
- * goldilocks_init(). But this property is not guaranteed for future
- * versions, so call it anyway.
- *
- * @warning This isn't even my final form!
- *
- * @param [out] shared The shared secret established with the other party.
- * @param [in] my_privkey My private key.
- * @param [in] your_pubkey The other party's public key.
- *
- * @retval GOLDI_EOK Success.
- * @retval GOLDI_ECORRUPT My key is corrupt.
- * @retval GOLDI_EINVAL The other party's key is corrupt.
- * @retval GOLDI_EUNINIT You must call goldilocks_init() first.
- */
- int
- goldilocks_shared_secret (
- uint8_t shared[GOLDI_SHARED_SECRET_BYTES],
- const struct goldilocks_private_key_t *my_privkey,
- const struct goldilocks_public_key_t *your_pubkey
- ) __attribute__((warn_unused_result,nonnull(1,2,3),visibility ("default")));
-
- /**
- * @brief Sign a message.
- *
- * The signature is deterministic, using the symmetric secret found in the
- * secret key to form a nonce.
- *
- * The technique used in signing is a modified Schnorr system, like EdDSA.
- *
- * @warning This isn't even my final form!
- *
- * @param [out] signature_out Space for the output signature.
- * @param [in] message The message to be signed.
- * @param [in] message_len The length of the message to be signed.
- * @param [in] privkey My private key.
- *
- * @retval GOLDI_EOK Success.
- * @retval GOLDI_ECORRUPT My key is corrupt.
- * @retval GOLDI_EUNINIT You must call goldilocks_init() first.
- */
- int
- goldilocks_sign (
- uint8_t signature_out[GOLDI_SIGNATURE_BYTES],
- const uint8_t *message,
- uint64_t message_len,
- const struct goldilocks_private_key_t *privkey
- ) __attribute__((nonnull(1,2,4),visibility ("default")));
-
- /**
- * @brief Verify a signature.
- *
- * This function is fairly strict. It will correctly detect when
- * the signature has the wrong cofactor component, or when the sig
- * values aren't less than p or q.
- *
- * Currently this function does not detect when the public key is weird,
- * eg 0, has cofactor, etc. As a result, a party with a bogus public
- * key could create signatures that succeed on some systems and fail on
- * others.
- *
- * @warning This isn't even my final form!
- *
- * @param [in] signature The signature.
- * @param [in] message The message to be verified.
- * @param [in] message_len The length of the message to be verified.
- * @param [in] pubkey The signer's public key.
- *
- * @retval GOLDI_EOK Success.
- * @retval GOLDI_EINVAL The public key or signature is corrupt.
- * @retval GOLDI_EUNINIT You must call goldilocks_init() first.
- */
- int
- goldilocks_verify (
- const uint8_t signature[GOLDI_SIGNATURE_BYTES],
- const uint8_t *message,
- uint64_t message_len,
- const struct goldilocks_public_key_t *pubkey
- ) __attribute__((warn_unused_result,nonnull(1,2,4),visibility ("default")));
-
- #if GOLDI_IMPLEMENT_PRECOMPUTED_KEYS
-
- /** A public key which has been expanded by precomputation for higher speed. */
- struct goldilocks_precomputed_public_key_t;
-
- /**
- * @brief Expand a public key by precomputation.
- *
- * @todo Give actual error returns, instead of ambiguous NULL.
- *
- * @warning This isn't even my final form!
- *
- * @param [in] pub The public key.
- * @retval NULL We ran out of memory, or the
- */
- struct goldilocks_precomputed_public_key_t *
- goldilocks_precompute_public_key (
- const struct goldilocks_public_key_t *pub
- ) __attribute__((warn_unused_result,nonnull(1),visibility ("default")));
-
- /**
- * @brief Overwrite an expanded public key with zeros, then destroy it.
- *
- * If the input is NULL, this function does nothing.
- *
- * @param [in] precom The public key.
- */
- void
- goldilocks_destroy_precomputed_public_key (
- struct goldilocks_precomputed_public_key_t *precom
- ) __attribute__((visibility ("default")));
-
- /**
- * @brief Verify a signature.
- *
- * This function is fairly strict. It will correctly detect when
- * the signature has the wrong cofactor component, or when the sig
- * values aren't less than p or q.
- *
- * @warning This isn't even my final form!
- *
- * @param [in] signature The signature.
- * @param [in] message The message to be verified.
- * @param [in] message_len The length of the message to be verified.
- * @param [in] pubkey The signer's public key, expanded by precomputation.
- *
- * @retval GOLDI_EOK Success.
- * @retval GOLDI_EINVAL The public key or signature is corrupt.
- * @retval GOLDI_EUNINIT You must call goldilocks_init() first.
- */
- int
- goldilocks_verify_precomputed (
- const uint8_t signature[GOLDI_SIGNATURE_BYTES],
- const uint8_t *message,
- uint64_t message_len,
- const struct goldilocks_precomputed_public_key_t *pubkey
- ) __attribute__((warn_unused_result,nonnull(1,2,4),visibility ("default")));
-
- /**
- * @brief Generate a Diffie-Hellman shared secret in constant time.
- * Uses a precomputation on the other party's public key for efficiency.
- *
- * This function uses some compile-time flags whose merit remains to
- * be decided.
- *
- * If the flag EXPERIMENT_ECDH_OBLITERATE_CT is set, prepend 40 bytes
- * of zeros to the secret before hashing. In the case that the other
- * party's key is detectably corrupt, instead the symmetric part
- * of the secret key is used to produce a pseudorandom value.
- *
- * If EXPERIMENT_ECDH_STIR_IN_PUBKEYS is set, the sum and product of
- * the two parties' public keys is prepended to the hash.
- *
- * In the current version, this function can safely be run even without
- * goldilocks_init(). But this property is not guaranteed for future
- * versions, so call it anyway.
- *
- * @warning This isn't even my final form!
- *
- * @param [out] shared The shared secret established with the other party.
- * @param [in] my_privkey My private key.
- * @param [in] your_pubkey The other party's precomputed public key.
- *
- * @retval GOLDI_EOK Success.
- * @retval GOLDI_ECORRUPT My key is corrupt.
- * @retval GOLDI_EINVAL The other party's key is corrupt.
- * @retval GOLDI_EUNINIT You must call goldilocks_init() first.
- */
- int
- goldilocks_shared_secret_precomputed (
- uint8_t shared[GOLDI_SHARED_SECRET_BYTES],
- const struct goldilocks_private_key_t *my_privkey,
- const struct goldilocks_precomputed_public_key_t *your_pubkey
- ) __attribute__((warn_unused_result,nonnull(1,2,3),visibility ("default")));
-
- #endif /* GOLDI_IMPLEMENT_PRECOMPUTED_KEYS */
-
- #ifdef __cplusplus
- }; /* extern "C" */
- #endif
-
- #endif /* __GOLDILOCKS_H__ */
|