mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-26 03:03:22 -03:00
157 lines
3.7 KiB
C++
157 lines
3.7 KiB
C++
#ifndef _SECP256K1_NUM_OPENSSL_
|
|
#define _SECP256K1_NUM_OPENSSL_
|
|
|
|
#include <assert.h>
|
|
#include <string>
|
|
#include <string.h>
|
|
#include <openssl/bn.h>
|
|
#include <openssl/crypto.h>
|
|
|
|
namespace secp256k1 {
|
|
|
|
class Context {
|
|
private:
|
|
BN_CTX *bn_ctx;
|
|
bool root;
|
|
bool offspring;
|
|
|
|
public:
|
|
operator BN_CTX*() {
|
|
return bn_ctx;
|
|
}
|
|
|
|
Context() {
|
|
bn_ctx = BN_CTX_new();
|
|
BN_CTX_start(bn_ctx);
|
|
root = true;
|
|
offspring = false;
|
|
}
|
|
|
|
Context(Context &par) {
|
|
bn_ctx = par.bn_ctx;
|
|
root = false;
|
|
offspring = false;
|
|
par.offspring = true;
|
|
BN_CTX_start(bn_ctx);
|
|
}
|
|
|
|
~Context() {
|
|
BN_CTX_end(bn_ctx);
|
|
if (root)
|
|
BN_CTX_free(bn_ctx);
|
|
}
|
|
|
|
BIGNUM *Get() {
|
|
assert(offspring == false);
|
|
return BN_CTX_get(bn_ctx);
|
|
}
|
|
};
|
|
|
|
|
|
class Number {
|
|
private:
|
|
BIGNUM *bn;
|
|
Number(const Number &x) {}
|
|
public:
|
|
Number(Context &ctx) : bn(ctx.Get()) {}
|
|
Number(Context &ctx, const unsigned char *bin, int len) : bn(ctx.Get()) {
|
|
SetBytes(bin,len);
|
|
}
|
|
void SetNumber(const Number &x) {
|
|
BN_copy(bn, x.bn);
|
|
}
|
|
Number &operator=(const Number &x) {
|
|
BN_copy(bn, x.bn);
|
|
return *this;
|
|
}
|
|
void SetBytes(const unsigned char *bin, int len) {
|
|
BN_bin2bn(bin, len, bn);
|
|
}
|
|
void GetBytes(unsigned char *bin, int len) {
|
|
int size = BN_num_bytes(bn);
|
|
assert(size <= len);
|
|
memset(bin,0,len);
|
|
BN_bn2bin(bn, bin + size - len);
|
|
}
|
|
void SetInt(int x) {
|
|
BN_set_word(bn, x);
|
|
}
|
|
void SetModInverse(Context &ctx, const Number &x, const Number &m) {
|
|
BN_mod_inverse(bn, x.bn, m.bn, ctx);
|
|
}
|
|
void SetModMul(Context &ctx, const Number &a, const Number &b, const Number &m) {
|
|
BN_mod_mul(bn, a.bn, b.bn, m.bn, ctx);
|
|
}
|
|
void SetAdd(Context &ctx, const Number &a1, const Number &a2) {
|
|
BN_add(bn, a1.bn, a2.bn);
|
|
}
|
|
void SetSub(Context &ctx, const Number &a1, const Number &a2) {
|
|
BN_sub(bn, a1.bn, a2.bn);
|
|
}
|
|
void SetMult(Context &ctx, const Number &a1, const Number &a2) {
|
|
BN_mul(bn, a1.bn, a2.bn, ctx);
|
|
}
|
|
void SetDiv(Context &ctx, const Number &a1, const Number &a2) {
|
|
BN_div(bn, NULL, a1.bn, a2.bn, ctx);
|
|
}
|
|
void SetMod(Context &ctx, const Number &a, const Number &m) {
|
|
BN_nnmod(bn, a.bn, m.bn, ctx);
|
|
}
|
|
int Compare(const Number &a) const {
|
|
return BN_cmp(bn, a.bn);
|
|
}
|
|
int GetBits() const {
|
|
return BN_num_bits(bn);
|
|
}
|
|
// return the lowest (rightmost) bits bits, and rshift them away
|
|
int ShiftLowBits(Context &ctx, int bits) {
|
|
Context ct(ctx);
|
|
BIGNUM *tmp = ct.Get();
|
|
BN_copy(tmp, bn);
|
|
BN_mask_bits(tmp, bits);
|
|
int ret = BN_get_word(tmp);
|
|
BN_rshift(bn, bn, bits);
|
|
return ret;
|
|
}
|
|
// check whether number is 0,
|
|
bool IsZero() const {
|
|
return BN_is_zero(bn);
|
|
}
|
|
bool IsOdd() const {
|
|
return BN_is_odd(bn);
|
|
}
|
|
bool IsNeg() const {
|
|
return BN_is_negative(bn);
|
|
}
|
|
void Negate() {
|
|
BN_set_negative(bn, !IsNeg());
|
|
}
|
|
void Shift1() {
|
|
BN_rshift1(bn,bn);
|
|
}
|
|
void Inc() {
|
|
BN_add_word(bn,1);
|
|
}
|
|
void SetHex(const std::string &str) {
|
|
BN_hex2bn(&bn, str.c_str());
|
|
}
|
|
void SetPseudoRand(const Number &max) {
|
|
BN_pseudo_rand_range(bn, max.bn);
|
|
}
|
|
void SplitInto(Context &ctx, int bits, Number &low, Number &high) const {
|
|
BN_copy(low.bn, bn);
|
|
BN_mask_bits(low.bn, bits);
|
|
BN_rshift(high.bn, bn, bits);
|
|
}
|
|
|
|
std::string ToString() {
|
|
char *str = BN_bn2hex(bn);
|
|
std::string ret(str);
|
|
OPENSSL_free(str);
|
|
return ret;
|
|
}
|
|
};
|
|
|
|
}
|
|
|
|
#endif
|