mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-23 00:43:02 +01:00
SHA2 Refactoring
This commit is contained in:
parent
b0c5f20341
commit
40bc3c37d2
File diff suppressed because it is too large
Load Diff
@ -18,7 +18,7 @@
|
||||
* 3. Neither the name of the project nor the names of its contributors
|
||||
* may be used to endorse or promote products derived from this software
|
||||
* without specific prior written permission.
|
||||
*
|
||||
*
|
||||
* Updated for use in Firebird by Tony Whyman <tony@mwasoftware.co.uk>
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE PROJECT AND CONTRIBUTORS ``AS IS'' AND
|
||||
@ -33,14 +33,14 @@
|
||||
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||
* SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
|
||||
/*
|
||||
* This update is intended to make available the SHA-2 family of message
|
||||
* digests as C++ classes for use in Firebird. sha224, sha256, sha384 and
|
||||
* digests as C++ classes for use in Firebird. sha224, sha256, sha384 and
|
||||
* sha512 are each implemented as separate classes. The class methods are
|
||||
* intended to be as similar as possible to the legacy class sha1 in order
|
||||
* to facilitate straightforward replacement.
|
||||
*
|
||||
*
|
||||
* This implementation also comes with a NIST compliancy test for each
|
||||
* digest. This is enabled by building with the NIST_COMPLIANCY_TESTS symbol
|
||||
* defined.
|
||||
@ -71,15 +71,15 @@
|
||||
#define SHA224_BLOCK_SIZE SHA256_BLOCK_SIZE
|
||||
|
||||
namespace Firebird {
|
||||
|
||||
|
||||
/* This template function provides a simple one line means of computing a SHA-2
|
||||
* digest from an arbitrary length message.
|
||||
*/
|
||||
|
||||
template<class SHA>void get_digest(const unsigned char *message, size_t len, unsigned char *digest)
|
||||
|
||||
template<class SHA>void get_digest(const unsigned char* message, size_t len, unsigned char* digest)
|
||||
{
|
||||
SHA sha;
|
||||
sha.process(len, message);
|
||||
SHA sha;
|
||||
sha.process(len, message);
|
||||
sha.getHash(digest);
|
||||
}
|
||||
|
||||
@ -100,171 +100,173 @@ template<class SHA> void hashBased64(Firebird::string& hash, const Firebird::str
|
||||
/* The sha2_base class is an abstract class that is the ancestor for all
|
||||
* the SHA-2 classes. It defines all public methods for the classes and
|
||||
* a common model of use.
|
||||
*
|
||||
*
|
||||
* When instatiated a SHA-2 class is already initialized for use. The message
|
||||
* for which a digest is required is then fed to the class using one of
|
||||
* the "process" methods, either as a single action or accumulatively.
|
||||
*
|
||||
*
|
||||
* When the entire message has been input, the resulting digest is returned
|
||||
* by a "getHash" method. Calling "getHash" also clears the digest and
|
||||
* by a "getHash" method. Calling "getHash" also clears the digest and
|
||||
* re-initializes the SHA-2 generator ready to compute a new digest.
|
||||
*
|
||||
*
|
||||
* A SHA-2 generator can be cleared down and re-initialized at any time
|
||||
* by calling the "reset" method.
|
||||
*/
|
||||
|
||||
class sha224_traits;
|
||||
class sha256_traits;
|
||||
class sha384_traits;
|
||||
class sha512_traits;
|
||||
|
||||
class sha2_types
|
||||
{
|
||||
public:
|
||||
typedef unsigned char uint8;
|
||||
typedef unsigned int uint32;
|
||||
typedef unsigned long long uint64;
|
||||
};
|
||||
|
||||
template<class SHA_TRAITS>
|
||||
class sha2_base : public GlobalStorage {
|
||||
#else
|
||||
class sha2_base {
|
||||
#endif
|
||||
private:
|
||||
sha2_base(const sha2_base&);
|
||||
sha2_base& operator = (const sha2_base&);
|
||||
|
||||
public:
|
||||
sha2_base() {};
|
||||
sha2_base()
|
||||
{
|
||||
SHA_TRAITS::sha_init(&m_ctx);
|
||||
}
|
||||
|
||||
virtual ~sha2_base() {};
|
||||
|
||||
virtual const unsigned int get_DigestSize()=0;
|
||||
virtual const unsigned int get_BlockSize()=0;
|
||||
|
||||
typedef unsigned char uint8;
|
||||
typedef unsigned int uint32;
|
||||
typedef unsigned long long uint64;
|
||||
|
||||
protected:
|
||||
virtual void sha_init() {};
|
||||
virtual void sha_update(const unsigned char *message, unsigned int len)=0;
|
||||
virtual void sha_final(unsigned char *digest)=0;
|
||||
|
||||
|
||||
public:
|
||||
void reset() {sha_init();};
|
||||
|
||||
void reset()
|
||||
{
|
||||
SHA_TRAITS::sha_init(&m_ctx);
|
||||
}
|
||||
|
||||
void process(size_t length, const void* bytes)
|
||||
{
|
||||
sha_update(static_cast<const unsigned char*>(bytes), length);
|
||||
SHA_TRAITS::sha_update(&m_ctx, static_cast<const unsigned char*>(bytes), length);
|
||||
}
|
||||
|
||||
void process(size_t length, const unsigned char* message)
|
||||
{
|
||||
sha_update(message, length);
|
||||
SHA_TRAITS::sha_update(&m_ctx, message, length);
|
||||
}
|
||||
|
||||
void process(const char* str)
|
||||
{
|
||||
process(strlen(str), str);
|
||||
SHA_TRAITS::sha_update(&m_ctx, reinterpret_cast<const unsigned char*>(str), strlen(str));
|
||||
}
|
||||
|
||||
void getHash(unsigned char* digest)
|
||||
{
|
||||
SHA_TRAITS::sha_final(&m_ctx, digest);
|
||||
SHA_TRAITS::sha_init(&m_ctx);
|
||||
}
|
||||
|
||||
void getHash(unsigned char *digest);
|
||||
|
||||
#ifndef NIST_COMPLIANCY_TESTS
|
||||
void process(const UCharBuffer& bytes)
|
||||
{
|
||||
process(bytes.getCount(), bytes.begin());
|
||||
SHA_TRAITS::sha_update(&m_ctx, bytes.begin(), bytes.getCount());
|
||||
}
|
||||
|
||||
void getHash(UCharBuffer& h)
|
||||
{
|
||||
SHA_TRAITS::sha_final(&m_ctx, h.getBuffer(SHA_TRAITS::get_DigestSize()));
|
||||
SHA_TRAITS::sha_init(&m_ctx);
|
||||
}
|
||||
|
||||
void getHash(UCharBuffer& h);
|
||||
#endif
|
||||
|
||||
|
||||
private:
|
||||
typename SHA_TRAITS::sha_ctx m_ctx;
|
||||
};
|
||||
|
||||
class sha256 : public sha2_base {
|
||||
public:
|
||||
sha256();
|
||||
const unsigned int get_DigestSize() {return SHA256_DIGEST_SIZE;};
|
||||
const unsigned int get_BlockSize() {return SHA256_BLOCK_SIZE;};
|
||||
|
||||
protected:
|
||||
typedef struct {
|
||||
unsigned int tot_len;
|
||||
unsigned int len;
|
||||
unsigned char block[2 * SHA256_BLOCK_SIZE];
|
||||
uint32 h[8];
|
||||
} sha256_ctx;
|
||||
|
||||
private:
|
||||
void sha256_init(sha256_ctx * ctx);
|
||||
void sha256_update(sha256_ctx *ctx, const unsigned char *message,
|
||||
unsigned int len);
|
||||
void sha256_final(sha256_ctx *ctx, unsigned char *digest);
|
||||
|
||||
protected:
|
||||
sha256_ctx ctx;
|
||||
|
||||
void sha256_transf(sha256_ctx * ctx, const unsigned char *message,
|
||||
unsigned int block_nb);
|
||||
void sha_init() {sha256_init(&ctx);};
|
||||
void sha_update(const unsigned char *message, unsigned int len) {sha256_update(&ctx,message,len);};
|
||||
void sha_final(unsigned char *digest) {sha256_final(&ctx,digest);};
|
||||
typedef sha2_base<sha224_traits> sha224;
|
||||
typedef sha2_base<sha256_traits> sha256;
|
||||
typedef sha2_base<sha384_traits> sha384;
|
||||
typedef sha2_base<sha512_traits> sha512;
|
||||
|
||||
struct sha256_ctx {
|
||||
unsigned int tot_len;
|
||||
unsigned int len;
|
||||
unsigned char block[2 * SHA256_BLOCK_SIZE];
|
||||
sha2_types::uint32 h[8];
|
||||
|
||||
void transf(const unsigned char* message, unsigned int block_nb);
|
||||
};
|
||||
|
||||
class sha224 : public sha256 {
|
||||
class sha256_traits: private sha2_types {
|
||||
public:
|
||||
sha224();
|
||||
const unsigned int get_DigestSize() {return SHA224_DIGEST_SIZE;};
|
||||
const unsigned int get_BlockSize() {return SHA224_BLOCK_SIZE;};
|
||||
|
||||
private:
|
||||
typedef sha256_ctx sha224_ctx;
|
||||
void sha224_init(sha224_ctx *ctx);
|
||||
void sha224_update(sha224_ctx *ctx, const unsigned char *message,
|
||||
unsigned int len);
|
||||
void sha224_final(sha224_ctx *ctx, unsigned char *digest);
|
||||
typedef sha256_ctx sha_ctx;
|
||||
|
||||
protected:
|
||||
void sha_init() {sha224_init(&ctx);};
|
||||
void sha_update(const unsigned char *message, unsigned int len) {sha224_update(&ctx, message, len);};
|
||||
void sha_final(unsigned char *digest) {sha224_final(&ctx,digest);};
|
||||
public:
|
||||
static unsigned int get_DigestSize() {return SHA256_DIGEST_SIZE;};
|
||||
|
||||
static void sha_init(sha_ctx* ctx);
|
||||
|
||||
static void sha_update(sha_ctx* ctx, const unsigned char* message, unsigned int len);
|
||||
|
||||
static void sha_final(sha_ctx *ctx, unsigned char* digest);
|
||||
};
|
||||
|
||||
class sha512 : public sha2_base {
|
||||
class sha224_traits: private sha2_types {
|
||||
public:
|
||||
sha512();
|
||||
const unsigned int get_DigestSize() {return SHA512_DIGEST_SIZE;};
|
||||
const unsigned int get_BlockSize() {return SHA512_BLOCK_SIZE;};
|
||||
|
||||
protected:
|
||||
typedef struct {
|
||||
unsigned int tot_len;
|
||||
unsigned int len;
|
||||
unsigned char block[2 * SHA512_BLOCK_SIZE];
|
||||
uint64 h[8];
|
||||
} sha512_ctx;
|
||||
|
||||
private:
|
||||
void sha512_init(sha512_ctx *ctx);
|
||||
void sha512_update(sha512_ctx *ctx, const unsigned char *message,
|
||||
unsigned int len);
|
||||
void sha512_final(sha512_ctx *ctx, unsigned char *digest);
|
||||
protected:
|
||||
|
||||
sha512_ctx ctx;
|
||||
typedef sha256_ctx sha_ctx;
|
||||
|
||||
void sha512_transf(sha512_ctx *ctx, const unsigned char *message,
|
||||
unsigned int block_nb);
|
||||
void sha_init() {sha512_init(&ctx);};
|
||||
void sha_update(const unsigned char *message, unsigned int len) {sha512_update(&ctx, message, len);};
|
||||
void sha_final(unsigned char *digest) {sha512_final(&ctx, digest);};
|
||||
public:
|
||||
static unsigned int get_DigestSize() {return SHA224_DIGEST_SIZE;};
|
||||
|
||||
static void sha_init(sha_ctx* ctx);
|
||||
|
||||
static void sha_update(sha_ctx* ctx, const unsigned char* message, unsigned int len);
|
||||
|
||||
static void sha_final(sha_ctx* ctx, unsigned char* digest);
|
||||
};
|
||||
|
||||
struct sha512_ctx{
|
||||
unsigned int tot_len;
|
||||
unsigned int len;
|
||||
unsigned char block[2 * SHA512_BLOCK_SIZE];
|
||||
sha2_types::uint64 h[8];
|
||||
|
||||
void transf(const unsigned char* message, unsigned int block_nb);
|
||||
|
||||
};
|
||||
|
||||
class sha384 : public sha512 {
|
||||
class sha512_traits: private sha2_types {
|
||||
public:
|
||||
sha384();
|
||||
const unsigned int get_DigestSize() {return SHA384_DIGEST_SIZE;};
|
||||
const unsigned int get_BlockSize() {return SHA384_BLOCK_SIZE;};
|
||||
|
||||
private:
|
||||
typedef sha512_ctx sha384_ctx;
|
||||
void sha384_init(sha384_ctx *ctx);
|
||||
void sha384_update(sha384_ctx *ctx, const unsigned char *message,
|
||||
unsigned int len);
|
||||
void sha384_final(sha384_ctx *ctx, unsigned char *digest);
|
||||
typedef sha512_ctx sha_ctx;
|
||||
|
||||
protected:
|
||||
void sha_init() {sha384_init(&ctx);};
|
||||
void sha_update(const unsigned char *message, unsigned int len) {sha384_update(&ctx,message,len);};
|
||||
void sha_final(unsigned char *digest) {sha384_final(&ctx, digest);};
|
||||
public:
|
||||
static unsigned int get_DigestSize() {return SHA512_DIGEST_SIZE;};
|
||||
|
||||
static void sha_init(sha_ctx* ctx);
|
||||
|
||||
static void sha_update(sha_ctx* ctx, const unsigned char* message, unsigned int len);
|
||||
|
||||
static void sha_final(sha_ctx* ctx, unsigned char* digest);
|
||||
};
|
||||
|
||||
class sha384_traits: private sha2_types {
|
||||
public:
|
||||
typedef sha512_ctx sha_ctx;
|
||||
|
||||
public:
|
||||
static unsigned int get_DigestSize() {return SHA384_DIGEST_SIZE;};
|
||||
|
||||
static void sha_init(sha_ctx* ctx);
|
||||
|
||||
static void sha_update(sha_ctx* ctx, const unsigned char* message, unsigned int len);
|
||||
|
||||
static void sha_final(sha_ctx* ctx, unsigned char* digest);
|
||||
};
|
||||
|
||||
} //Firebird
|
||||
|
||||
#endif /* !_SHA2_H */
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user