forked from ArweaveTeam/arweave
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
add sign/verify, swap openssl use with bitcoin implementation
- Loading branch information
1 parent
1a7b205
commit 30c8ffa
Showing
4 changed files
with
280 additions
and
44 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -4,49 +4,228 @@ | |
|
||
#define SECP256K1_PUBKEY_UNCOMPRESSED_SIZE 65 | ||
#define SECP256K1_PUBKEY_COMPRESSED_SIZE 33 | ||
#define SECP256K1_SIGNATURE_SIZE 64 | ||
#define SECP256K1_PRIVKEY_SIZE 32 | ||
#define SECP256K1_CONTEXT_SEED_SIZE 32 | ||
#define SECP256K1_DIGEST_SIZE 32 | ||
|
||
static int secp256k1_load(ErlNifEnv* env, void** priv, ERL_NIF_TERM load_info) { | ||
return 0; | ||
} | ||
|
||
#if defined(_MSC_VER) | ||
// For SecureZeroMemory | ||
#include <Windows.h> | ||
#endif | ||
/* Cleanses memory to prevent leaking sensitive info. Won't be optimized out. */ | ||
static void secure_erase(void *ptr, size_t len) { | ||
#if defined(_MSC_VER) | ||
/* SecureZeroMemory is guaranteed not to be optimized out by MSVC. */ | ||
SecureZeroMemory(ptr, len); | ||
#elif defined(__GNUC__) | ||
/* We use a memory barrier that scares the compiler away from optimizing out the memset. | ||
* | ||
* Quoting Adam Langley <[email protected]> in commit ad1907fe73334d6c696c8539646c21b11178f20f | ||
* in BoringSSL (ISC License): | ||
* As best as we can tell, this is sufficient to break any optimisations that | ||
* might try to eliminate "superfluous" memsets. | ||
* This method used in memzero_explicit() the Linux kernel, too. Its advantage is that it is | ||
* pretty efficient, because the compiler can still implement the memset() efficiently, | ||
* just not remove it entirely. See "Dead Store Elimination (Still) Considered Harmful" by | ||
* Yang et al. (USENIX Security 2017) for more background. | ||
*/ | ||
memset(ptr, 0, len); | ||
__asm__ __volatile__("" : : "r"(ptr) : "memory"); | ||
#else | ||
void *(*volatile const volatile_memset)(void *, int, size_t) = memset; | ||
volatile_memset(ptr, 0, len); | ||
#endif | ||
} | ||
|
||
static ERL_NIF_TERM generate_key(ErlNifEnv* env, int argc, const ERL_NIF_TERM argv[]) { | ||
char *error = NULL; | ||
unsigned char seed[SECP256K1_CONTEXT_SEED_SIZE]; | ||
if (!RAND_bytes(seed, sizeof(seed))) { | ||
return error_tuple(env, "Failed to generate random seed for context."); | ||
} | ||
unsigned char privbytes[SECP256K1_PRIVKEY_SIZE]; | ||
unsigned char pubbytes[SECP256K1_PUBKEY_UNCOMPRESSED_SIZE]; | ||
secp256k1_pubkey pubkey; | ||
secp256k1_context* ctx = secp256k1_context_create(SECP256K1_CONTEXT_NONE); | ||
|
||
if (!RAND_priv_bytes(seed, sizeof(seed))) { | ||
error = "Failed to generate random seed for context."; | ||
goto cleanup; | ||
} | ||
|
||
if (!secp256k1_context_randomize(ctx, seed)) { | ||
return error_tuple(env, "Failed to randomize context."); | ||
error = "Failed to randomize context."; | ||
goto cleanup; | ||
} | ||
|
||
unsigned char privbytes[SECP256K1_PRIVKEY_SIZE]; | ||
if (!RAND_bytes(privbytes, sizeof(privbytes))) { | ||
return error_tuple(env, "Failed to generate random key."); | ||
if (!RAND_priv_bytes(privbytes, sizeof(privbytes))) { | ||
error = "Failed to generate random key."; | ||
goto cleanup; | ||
} | ||
|
||
if (!secp256k1_ec_seckey_verify(ctx, privbytes)) { | ||
return error_tuple(env, "Generated secret secp256k1 key is invalid."); | ||
error = "Generated secret secp256k1 key is invalid."; | ||
goto cleanup; | ||
} | ||
|
||
secp256k1_pubkey pubkey; | ||
if(!secp256k1_ec_pubkey_create(ctx, &pubkey, privbytes)) { | ||
return error_tuple(env, "Failed to populate public key."); | ||
error = "Failed to populate public key."; | ||
goto cleanup; | ||
} | ||
|
||
unsigned char pubbytes[SECP256K1_PUBKEY_UNCOMPRESSED_SIZE]; | ||
size_t l = sizeof(pubbytes); | ||
size_t l = SECP256K1_PUBKEY_UNCOMPRESSED_SIZE; | ||
if (!secp256k1_ec_pubkey_serialize(ctx, pubbytes, &l, &pubkey, SECP256K1_EC_UNCOMPRESSED)) { | ||
return error_tuple(env, "Failed to serialize public key."); | ||
error = "Failed to serialize public key."; | ||
goto cleanup; | ||
} | ||
|
||
ERL_NIF_TERM privkey_term = make_output_binary(env, privbytes, SECP256K1_PRIVKEY_SIZE); | ||
ERL_NIF_TERM pubkey_term = make_output_binary(env, pubbytes, SECP256K1_PUBKEY_UNCOMPRESSED_SIZE); | ||
|
||
cleanup: | ||
secp256k1_context_destroy(ctx); | ||
secure_erase(seed, sizeof(seed)); | ||
secure_erase(privbytes, sizeof(privbytes)); | ||
memset(pubbytes, 0, SECP256K1_PUBKEY_UNCOMPRESSED_SIZE); | ||
|
||
if (error) { | ||
return error_tuple(env, error); | ||
} | ||
return ok_tuple2(env, privkey_term, pubkey_term); | ||
|
||
} | ||
|
||
static ERL_NIF_TERM sign(ErlNifEnv *env, int argc, const ERL_NIF_TERM argv[]) { | ||
if (argc != 2) { | ||
return enif_make_badarg(env); | ||
} | ||
ErlNifBinary Digest, PrivateBytes; | ||
if (!enif_inspect_binary(env, argv[0], &Digest)) { | ||
return enif_make_badarg(env); | ||
} | ||
if (Digest.size != SECP256K1_DIGEST_SIZE) { | ||
return enif_make_badarg(env); | ||
} | ||
|
||
if (!enif_inspect_binary(env, argv[1], &PrivateBytes)) { | ||
return enif_make_badarg(env); | ||
} | ||
if (PrivateBytes.size != SECP256K1_PRIVKEY_SIZE) { | ||
return enif_make_badarg(env); | ||
} | ||
|
||
char *error = NULL; | ||
unsigned char seed[SECP256K1_CONTEXT_SEED_SIZE]; | ||
unsigned char digest[SECP256K1_DIGEST_SIZE]; | ||
unsigned char privbytes[SECP256K1_PRIVKEY_SIZE]; | ||
unsigned char signature[SECP256K1_SIGNATURE_SIZE]; | ||
secp256k1_ecdsa_signature s; | ||
secp256k1_context* ctx = secp256k1_context_create(SECP256K1_CONTEXT_NONE); | ||
|
||
memcpy(digest, Digest.data, SECP256K1_DIGEST_SIZE); | ||
memcpy(privbytes, PrivateBytes.data, SECP256K1_PRIVKEY_SIZE); | ||
|
||
if (!RAND_priv_bytes(seed, sizeof(seed))) { | ||
error = "Failed to generate random seed for context."; | ||
goto cleanup; | ||
} | ||
|
||
if (!secp256k1_context_randomize(ctx, seed)) { | ||
error = "Failed to randomize context."; | ||
goto cleanup; | ||
} | ||
|
||
if(!secp256k1_ecdsa_sign(ctx, &s, digest, privbytes, NULL, NULL)) { | ||
error = "Failed to create signature."; | ||
goto cleanup; | ||
} | ||
|
||
if(!secp256k1_ecdsa_signature_serialize_compact(ctx, signature, &s)) { | ||
error = "Failed to serialize signature."; | ||
goto cleanup; | ||
} | ||
|
||
ERL_NIF_TERM signature_term = make_output_binary(env, signature, SECP256K1_SIGNATURE_SIZE); | ||
|
||
cleanup: | ||
secp256k1_context_destroy(ctx); | ||
secure_erase(seed, sizeof(seed)); | ||
secure_erase(privbytes, sizeof(privbytes)); | ||
memset(signature, 0, SECP256K1_SIGNATURE_SIZE); | ||
|
||
if (error) { | ||
return error_tuple(env, error); | ||
} | ||
return ok_tuple(env, signature_term); | ||
} | ||
|
||
static ERL_NIF_TERM verify(ErlNifEnv *env, int argc, const ERL_NIF_TERM argv[]) { | ||
if (argc != 3) { | ||
return enif_make_badarg(env); | ||
} | ||
ErlNifBinary Digest, Signature, PublicBytes; | ||
if (!enif_inspect_binary(env, argv[0], &Digest)) { | ||
return enif_make_badarg(env); | ||
} | ||
if (Digest.size != SECP256K1_DIGEST_SIZE) { | ||
return enif_make_badarg(env); | ||
} | ||
|
||
if (!enif_inspect_binary(env, argv[1], &Signature)) { | ||
return enif_make_badarg(env); | ||
} | ||
if (Signature.size != SECP256K1_SIGNATURE_SIZE) { | ||
return enif_make_badarg(env); | ||
} | ||
if (!enif_inspect_binary(env, argv[2], &PublicBytes)) { | ||
return enif_make_badarg(env); | ||
} | ||
if (PublicBytes.size != SECP256K1_PUBKEY_COMPRESSED_SIZE && PublicBytes.size != SECP256K1_PUBKEY_UNCOMPRESSED_SIZE) { | ||
return enif_make_badarg(env); | ||
} | ||
|
||
char *error = NULL; | ||
unsigned char digest[SECP256K1_DIGEST_SIZE]; | ||
unsigned char signature[SECP256K1_SIGNATURE_SIZE]; | ||
unsigned char pubbytes[PublicBytes.size]; | ||
secp256k1_ecdsa_signature s; | ||
secp256k1_pubkey pubkey; | ||
|
||
memcpy(digest, Digest.data, SECP256K1_DIGEST_SIZE); | ||
memcpy(signature, Signature.data, SECP256K1_SIGNATURE_SIZE); | ||
memcpy(pubbytes, PublicBytes.data, PublicBytes.size); | ||
|
||
if (!secp256k1_ecdsa_signature_parse_compact(secp256k1_context_static, &s, signature)) { | ||
error = "Failed to deserialize/parse signature."; | ||
goto cleanup; | ||
} | ||
|
||
if (!secp256k1_ec_pubkey_parse(secp256k1_context_static, &pubkey, pubbytes, PublicBytes.size)) { | ||
error = "Failed to deserialize/parse public key."; | ||
goto cleanup; | ||
} | ||
|
||
ERL_NIF_TERM privkey_bin = make_output_binary(env, privbytes, SECP256K1_PRIVKEY_SIZE); | ||
ERL_NIF_TERM pubkey_bin = make_output_binary(env, pubbytes, SECP256K1_PUBKEY_UNCOMPRESSED_SIZE); | ||
return ok_tuple2(env, privkey_bin, pubkey_bin); | ||
int is_valid = secp256k1_ecdsa_verify(secp256k1_context_static, &s, digest, &pubkey); | ||
|
||
cleanup: | ||
memset(digest, 0, SECP256K1_DIGEST_SIZE); | ||
memset(pubbytes, 0, PublicBytes.size); | ||
memset(signature, 0, SECP256K1_SIGNATURE_SIZE); | ||
|
||
if (error) { | ||
return error_tuple(env, error); | ||
} | ||
if (is_valid) { | ||
return enif_make_atom(env, "true"); | ||
} | ||
return enif_make_atom(env, "false"); | ||
} | ||
|
||
static ErlNifFunc nif_funcs[] = { | ||
{"generate_key", 0, generate_key} | ||
{"generate_key", 0, generate_key}, | ||
{"sign", 2, sign}, | ||
{"verify", 3, verify}, | ||
}; | ||
|
||
ERL_NIF_INIT(secp256k1_nif, nif_funcs, secp256k1_load, NULL, NULL, NULL) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.