Skip to content

Commit

Permalink
add sign/verify, swap openssl use with bitcoin implementation
Browse files Browse the repository at this point in the history
  • Loading branch information
samantehrani committed Jan 4, 2025
1 parent 1a7b205 commit 425dc5b
Show file tree
Hide file tree
Showing 4 changed files with 280 additions and 44 deletions.
215 changes: 197 additions & 18 deletions apps/arweave/c_src/secp256k1/secp256k1_nif.c
Original file line number Diff line number Diff line change
Expand Up @@ -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 <agl@google.com> 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)
6 changes: 3 additions & 3 deletions apps/arweave/src/ar_wallet.erl
Original file line number Diff line number Diff line change
Expand Up @@ -180,7 +180,7 @@ sign({{KeyAlg, PublicExpnt}, Priv, Pub}, Data)
}
);
sign({_, #'ECPrivateKey'{} = PrivateKey, _}, Data)->
ec_secp256k1:sign(Data, sha256, PrivateKey).
ec_secp256k1:sign(Data, PrivateKey).

%% @doc Verify that a signature is correct.
verify({_, Identifier}, Data, Sig) when is_binary(Identifier) ->
Expand All @@ -191,11 +191,11 @@ verify(Identifier, Data, Sig) when is_binary(Identifier) ->
rsa_pss:verify(
Data, sha256, Sig, rsa_pss:from_identifier(Identifier));
{?ECDSA_SIGN_ALG, secp256k1} ->
ec_secp256k1:verify(Data, sha256, Sig, ec_secp256k1:from_identifier(Identifier));
ec_secp256k1:verify(Data, Sig, ec_secp256k1:from_identifier(Identifier));
{invalid_prefix, _} -> invalid_identifier
end;
verify({_, {#'ECPoint'{}, {namedCurve, secp256k1}} = PublicKey}, Data, Sig) ->
ec_secp256k1:verify(Data, sha256, Sig, PublicKey).
ec_secp256k1:verify(Data, Sig, PublicKey).

-spec from_identifier(Identifier :: binary()) -> public_key:rsa_public_key() | public_key:ecdsa_public_key().
from_identifier(Identifier) ->
Expand Down
Loading

0 comments on commit 425dc5b

Please sign in to comment.