https://github.com/CryptDB/cryptdb
Raw File
Tip revision: 7678bc98d3054f1418371779c6d1050cd1a88b2e authored by Raluca Ada Popa on 04 January 2014, 01:31:06 UTC
small changes to readme
Tip revision: 7678bc9
bn.hh
#pragma once

#include <stdexcept>
#include <ostream>
#include <openssl/bn.h>
#include <openssl/crypto.h>

#include <util/errstream.hh>

class _bignum_ctx {
 public:
    _bignum_ctx() { c = BN_CTX_new(); }
    ~_bignum_ctx() { BN_CTX_free(c); }
    BN_CTX *ctx() { return c; }

    static BN_CTX *the_ctx() {
        static _bignum_ctx cx;
        return cx.ctx();
    }

 private:
    BN_CTX *c;
};

class bignum {
 public:
    bignum() {
        BN_init(&b);
    }

    bignum(unsigned long v) {
        BN_init(&b);
        BN_set_word(&b, v);
    }

    bignum(const bignum &other) {
        BN_init(&b);
        throw_c(BN_copy(&b, other.bn()));
    }

    bignum(const uint8_t *buf, size_t nbytes) {
        BN_init(&b);
        throw_c(BN_bin2bn(buf, nbytes, &b));
    }

    bignum(const std::string &v) {
        BN_init(&b);
        throw_c(BN_bin2bn((uint8_t*) v.data(), v.size(), &b));
    }

    ~bignum() { BN_free(&b); }

    BIGNUM *bn() { return &b; }
    const BIGNUM *bn() const { return &b; }
    unsigned long word() const {
        unsigned long v = BN_get_word(&b);
        if (v == 0xffffffffL)
            throw std::runtime_error("out of range");
        return v;
    }

#define op(opname, func, args...)                               \
    bignum opname(const bignum &mod) {                          \
        bignum res;                                             \
        throw_c(1 == func(res.bn(), &b, mod.bn(), ##args));      \
        return res;                                             \
    }

    op(operator+, BN_add)
    op(operator-, BN_sub)
    op(operator%, BN_mod, _bignum_ctx::the_ctx())
    op(operator*, BN_mul, _bignum_ctx::the_ctx())
#undef op

#define pred(predname, cmp)                                     \
    bool predname(const bignum &other) {                        \
        return BN_cmp(&b, other.bn()) cmp;                      \
    }

    pred(operator<,  <  0)
    pred(operator<=, <= 0)
    pred(operator>,  >  0)
    pred(operator>=, >= 0)
    pred(operator==, == 0)
#undef pred

    bignum invmod(const bignum &mod) {
        bignum r;
        throw_c(BN_mod_inverse(r.bn(), &b, mod.bn(), _bignum_ctx::the_ctx()));
        return r;
    }

 private:
    BIGNUM b;
};

static inline std::ostream&
operator<<(std::ostream &out, const bignum &bn)
{
    char *s = BN_bn2dec(bn.bn());
    out << s;
    OPENSSL_free(s);
    return out;
}
back to top