niceBit

Unnamed repository; edit this file 'description' to name the repository.
Log | Files | Refs | README | LICENSE

commit 49579e775dc0a54aad6e939157a9e73f894671f9
parent 17360bff3e0f17a18d9368f3fd09da200563902a
Author: NicholasFarrow <nicholas.w.farrow@gmail.com>
Date:   Thu, 26 Mar 2020 23:16:54 +1100

Implemented wallet import format (WIF) conversion of private key, for easy import to wallet software

Diffstat:
Rcreate_pubkey.h -> createPubKey.h | 0
Alib/base58.c | 205+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Alib/libbase58.h | 23+++++++++++++++++++++++
AwalletImportFormat.h | 88+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
4 files changed, 316 insertions(+), 0 deletions(-)

diff --git a/create_pubkey.h b/createPubKey.h diff --git a/lib/base58.c b/lib/base58.c @@ -0,0 +1,205 @@ +/* + * Copyright 2012-2014 Luke Dashjr + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the standard MIT license. See COPYING for more details. + */ + +#ifndef WIN32 +#include <arpa/inet.h> +#else +#include <winsock2.h> +#endif + +#include <stdbool.h> +#include <stddef.h> +#include <stdint.h> +#include <string.h> + +#include "libbase58.h" + +bool (*b58_sha256_impl)(void *, const void *, size_t) = NULL; + +static const int8_t b58digits_map[] = { + -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1, + -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1, + -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1, + -1, 0, 1, 2, 3, 4, 5, 6, 7, 8,-1,-1,-1,-1,-1,-1, + -1, 9,10,11,12,13,14,15, 16,-1,17,18,19,20,21,-1, + 22,23,24,25,26,27,28,29, 30,31,32,-1,-1,-1,-1,-1, + -1,33,34,35,36,37,38,39, 40,41,42,43,-1,44,45,46, + 47,48,49,50,51,52,53,54, 55,56,57,-1,-1,-1,-1,-1, +}; + +typedef uint64_t b58_maxint_t; +typedef uint32_t b58_almostmaxint_t; +#define b58_almostmaxint_bits (sizeof(b58_almostmaxint_t) * 8) +static const b58_almostmaxint_t b58_almostmaxint_mask = ((((b58_maxint_t)1) << b58_almostmaxint_bits) - 1); + +bool b58tobin(void *bin, size_t *binszp, const char *b58, size_t b58sz) +{ + size_t binsz = *binszp; + const unsigned char *b58u = (void*)b58; + unsigned char *binu = bin; + size_t outisz = (binsz + sizeof(b58_almostmaxint_t) - 1) / sizeof(b58_almostmaxint_t); + b58_almostmaxint_t outi[outisz]; + b58_maxint_t t; + b58_almostmaxint_t c; + size_t i, j; + uint8_t bytesleft = binsz % sizeof(b58_almostmaxint_t); + b58_almostmaxint_t zeromask = bytesleft ? (b58_almostmaxint_mask << (bytesleft * 8)) : 0; + unsigned zerocount = 0; + + if (!b58sz) + b58sz = strlen(b58); + + for (i = 0; i < outisz; ++i) { + outi[i] = 0; + } + + // Leading zeros, just count + for (i = 0; i < b58sz && b58u[i] == '1'; ++i) + ++zerocount; + + for ( ; i < b58sz; ++i) + { + if (b58u[i] & 0x80) + // High-bit set on invalid digit + return false; + if (b58digits_map[b58u[i]] == -1) + // Invalid base58 digit + return false; + c = (unsigned)b58digits_map[b58u[i]]; + for (j = outisz; j--; ) + { + t = ((b58_maxint_t)outi[j]) * 58 + c; + c = t >> b58_almostmaxint_bits; + outi[j] = t & b58_almostmaxint_mask; + } + if (c) + // Output number too big (carry to the next int32) + return false; + if (outi[0] & zeromask) + // Output number too big (last int32 filled too far) + return false; + } + + j = 0; + if (bytesleft) { + for (i = bytesleft; i > 0; --i) { + *(binu++) = (outi[0] >> (8 * (i - 1))) & 0xff; + } + ++j; + } + + for (; j < outisz; ++j) + { + for (i = sizeof(*outi); i > 0; --i) { + *(binu++) = (outi[j] >> (8 * (i - 1))) & 0xff; + } + } + + // Count canonical base58 byte count + binu = bin; + for (i = 0; i < binsz; ++i) + { + if (binu[i]) + break; + --*binszp; + } + *binszp += zerocount; + + return true; +} + +static +bool my_dblsha256(void *hash, const void *data, size_t datasz) +{ + uint8_t buf[0x20]; + return b58_sha256_impl(buf, data, datasz) && b58_sha256_impl(hash, buf, sizeof(buf)); +} + +int b58check(const void *bin, size_t binsz, const char *base58str, size_t b58sz) +{ + unsigned char buf[32]; + const uint8_t *binc = bin; + unsigned i; + if (binsz < 4) + return -4; + if (!my_dblsha256(buf, bin, binsz - 4)) + return -2; + if (memcmp(&binc[binsz - 4], buf, 4)) + return -1; + + // Check number of zeros is correct AFTER verifying checksum (to avoid possibility of accessing base58str beyond the end) + for (i = 0; binc[i] == '\0' && base58str[i] == '1'; ++i) + {} // Just finding the end of zeros, nothing to do in loop + if (binc[i] == '\0' || base58str[i] == '1') + return -3; + + return binc[0]; +} + +static const char b58digits_ordered[] = "123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz"; + +bool b58enc(char *b58, size_t *b58sz, const void *data, size_t binsz) +{ + const uint8_t *bin = data; + int carry; + size_t i, j, high, zcount = 0; + size_t size; + + while (zcount < binsz && !bin[zcount]) + ++zcount; + + size = (binsz - zcount) * 138 / 100 + 1; + uint8_t buf[size]; + memset(buf, 0, size); + + for (i = zcount, high = size - 1; i < binsz; ++i, high = j) + { + for (carry = bin[i], j = size - 1; (j > high) || carry; --j) + { + carry += 256 * buf[j]; + buf[j] = carry % 58; + carry /= 58; + if (!j) { + // Otherwise j wraps to maxint which is > high + break; + } + } + } + + for (j = 0; j < size && !buf[j]; ++j); + + if (*b58sz <= zcount + size - j) + { + *b58sz = zcount + size - j + 1; + return false; + } + + if (zcount) + memset(b58, '1', zcount); + for (i = zcount; j < size; ++i, ++j) + b58[i] = b58digits_ordered[buf[j]]; + b58[i] = '\0'; + *b58sz = i + 1; + + return true; +} + +bool b58check_enc(char *b58c, size_t *b58c_sz, uint8_t ver, const void *data, size_t datasz) +{ + uint8_t buf[1 + datasz + 0x20]; + uint8_t *hash = &buf[1 + datasz]; + + buf[0] = ver; + memcpy(&buf[1], data, datasz); + if (!my_dblsha256(hash, buf, datasz + 1)) + { + *b58c_sz = 0; + return false; + } + + return b58enc(b58c, b58c_sz, buf, 1 + datasz + 4); +} diff --git a/lib/libbase58.h b/lib/libbase58.h @@ -0,0 +1,23 @@ +#ifndef LIBBASE58_H +#define LIBBASE58_H + +#include <stdbool.h> +#include <stddef.h> + +#ifdef __cplusplus +extern "C" { +#endif + +extern bool (*b58_sha256_impl)(void *, const void *, size_t); + +extern bool b58tobin(void *bin, size_t *binsz, const char *b58, size_t b58sz); +extern int b58check(const void *bin, size_t binsz, const char *b58, size_t b58sz); + +extern bool b58enc(char *b58, size_t *b58sz, const void *bin, size_t binsz); +extern bool b58check_enc(char *b58c, size_t *b58c_sz, uint8_t ver, const void *data, size_t datasz); + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/walletImportFormat.h b/walletImportFormat.h @@ -0,0 +1,88 @@ +#include <stdio.h> +#include <openssl/sha.h> +#include <openssl/ripemd.h> +#include "lib/base58.c" + +/* https://en.bitcoin.it/wiki/Wallet_import_format */ + +char *create_wif(const unsigned char *privatekey) { + unsigned char newKey[65]; + unsigned char SHAkey[65]; + unsigned char SHAkey2[65]; + unsigned char checksum[11]; + unsigned char combinedKey[75]; + + size_t combinedKeySize = 37; + size_t wifSize = 52; + char wif[52]; + + wifSize = 80; + + /* Add 0x80 byte in front */ + newKey[0] = 128; + for(int i=0; i<32; i++) { + newKey[i+1] = privatekey[i]; + } + + /* + for(int i=0; i<32; i++) { + printf("%02X", privatekey[i]); + } + printf("\n"); + + for(int i=0; i<33; i++) { + printf("%02X", newKey[i]); + } + printf("\n"); + */ + + /* Perform SHA-256 hash on the extended key */ + SHA256(newKey, 33, SHAkey); + /* + for(int i=0; i<32; i++) { + printf("%02X", SHAkey[i]); + } + printf("\n"); + */ + + /* Perform SHA-256 hash again on the result */ + SHA256(SHAkey, 32, SHAkey2); + /* + for(int i=0; i<32; i++) { + printf("%02X", SHAkey2[i]); + } + printf("\n"); + */ + + /* Checksum is first 4 bytes of 2nd SHA*/ + for(int i=0; i<4; i++) { + checksum[i] = SHAkey2[i]; + } + + /* + for(int i=0; i<4; i++) { + printf("%02X", checksum[i]); + } + printf("\n"); + */ + + /* Append checksum to end of 2nd SHA */ + for(int i=0; i<33; i++) { + combinedKey[i] = newKey[i]; + } + for(int i=0; i<4; i++) { + combinedKey[33+i] = checksum[i]; + } + /* + for(int i=0; i<37; i++) { + printf("%02X", combinedKey[i]); + } + printf("\n"); + */ + + /* Encode with base-58 */ + b58enc(wif, &wifSize, combinedKey, combinedKeySize); + puts(wif); + + return 0; +}