프로젝트/암호화 모듈
[프로젝트-CRYPTO] HASH 구현하기
-=HaeJuK=-
2025. 1. 2. 16:02
반응형
#git URL
git clone https://github.com/HaeJuk-Lab/crypto.git
/******************************************************************************
* _ _ _ _____ _ _ _ _ __ _ _ ____ ____
* | | | | / \ | ____| | | | | | |/ / | | / \ | __ ) ___|
* | |_| | / _ \ | _| _ | | | | | ' / | | / _ \ | _ \___ \
* | _ |/ ___ \| |__| |_| | |_| | . \ | |___ / ___ \| |_) |__) |
* |_| |_/_/ \_\_____\___/ \___/|_|\_\ |_____/_/ \_\____/____/
*
* Copyright (c) HAEJUK LABS All Rights Reserved.
*
*******************************************************************************/
/**
@file hxcSHA.hpp
@brief
*/
#ifndef __HXC_SHA_HPP__
#define __HXC_SHA_HPP__
// SHA1 컨텍스트 구조체
struct hxsSHA1_CTX
{
uint32_t state[5]; // 상태 변수 (160비트)
uint64_t count; // 처리된 비트 수
uint8_t buffer[64]; // 데이터 버퍼 (512비트)
hxsSHA1_CTX()
{
Reset();
}
// 컨텍스트 초기화
void Reset()
{
state[0] = 0x67452301;
state[1] = 0xEFCDAB89;
state[2] = 0x98BADCFE;
state[3] = 0x10325476;
state[4] = 0xC3D2E1F0;
count = 0;
std::memset( buffer, 0, sizeof( buffer ) );
}
};
struct hxsSHA256_CTX
{
uint32_t state[8]; // 상태 변수 (256비트)
uint64_t count; // 처리된 비트 수
uint8_t buffer[64]; // 데이터 버퍼 (512비트)
hxsSHA256_CTX()
{
Reset();
}
void Reset()
{
state[0] = 0x6A09E667;
state[1] = 0xBB67AE85;
state[2] = 0x3C6EF372;
state[3] = 0xA54FF53A;
state[4] = 0x510E527F;
state[5] = 0x9B05688C;
state[6] = 0x1F83D9AB;
state[7] = 0x5BE0CD19;
count = 0;
std::memset( buffer, 0, sizeof( buffer ) );
}
};
// SHA384 컨텍스트 구조체
struct hxsSHA384_CTX
{
uint64_t state[8]; // 상태 변수 (384비트)
uint64_t count; // 처리된 비트 수
uint8_t buffer[128]; // 데이터 버퍼 (1024비트)
hxsSHA384_CTX()
{
Reset();
}
void Reset()
{
state[0] = 0xCBBB9D5DC1059ED8;
state[1] = 0x629A292A367CD507;
state[2] = 0x9159015A3070DD17;
state[3] = 0x152FECD8F70E5939;
state[4] = 0x67332667FFC00B31;
state[5] = 0x8EB44A8768581511;
state[6] = 0xDB0C2E0D64F98FA7;
state[7] = 0x47B5481DBEFA4FA4;
count = 0;
std::memset( buffer, 0, sizeof( buffer ) );
}
};
class hxcSHA
{
DECLARE_NO_COPY_CLASS( hxcSHA );
public:
hxcSHA() = default ;
virtual ~hxcSHA() = default;
virtual void Init() = 0;
virtual void Update( const uint8_t* data, size_t len ) = 0;
virtual void Final( uint8_t* hash ) = 0;
virtual size_t HashSize() const = 0;
};
#endif // !__HXC_SHA_HPP__
SHA1
/******************************************************************************
* _ _ _ _____ _ _ _ _ __ _ _ ____ ____
* | | | | / \ | ____| | | | | | |/ / | | / \ | __ ) ___|
* | |_| | / _ \ | _| _ | | | | | ' / | | / _ \ | _ \___ \
* | _ |/ ___ \| |__| |_| | |_| | . \ | |___ / ___ \| |_) |__) |
* |_| |_/_/ \_\_____\___/ \___/|_|\_\ |_____/_/ \_\____/____/
*
* Copyright (c) HAEJUK LABS All Rights Reserved.
*
*******************************************************************************/
/**
@file hxcSHA1.h
@brief
*/
#ifndef __HXC_SHA1_H__
#define __HXC_SHA1_H__
#include "hxPch.h"
#include "hxcSHA.hpp"
class hxcSHA1 : public hxcSHA
{
DECLARE_NO_COPY_CLASS( hxcSHA1 );
public:
hxcSHA1();
void Init() override;
void Update( const uint8_t* data, size_t len ) override;
void Final( uint8_t* hash ) override;
size_t HashSize() const override;
private:
void Transform( uint32_t state[5], const uint8_t buffer[64] );
private:
std::unique_ptr<hxsSHA1_CTX> ctx;
};
#endif //! __HXC_SHA1_H__
/******************************************************************************
* _ _ _ _____ _ _ _ _ __ _ _ ____ ____
* | | | | / \ | ____| | | | | | |/ / | | / \ | __ ) ___|
* | |_| | / _ \ | _| _ | | | | | ' / | | / _ \ | _ \___ \
* | _ |/ ___ \| |__| |_| | |_| | . \ | |___ / ___ \| |_) |__) |
* |_| |_/_/ \_\_____\___/ \___/|_|\_\ |_____/_/ \_\____/____/
*
* Copyright (c) HAEJUK LABS All Rights Reserved.
*
*******************************************************************************/
/**
@file hxcSHA1.cpp
@brief
*/
#include "hxcSHA1.h"
hxcSHA1::hxcSHA1()
: ctx( std::make_unique<hxsSHA1_CTX>() )
{
}
void hxcSHA1::Init()
{
ctx->Reset();
}
void hxcSHA1::Update( const uint8_t* data, size_t len )
{
size_t i, index, partLen;
index = (ctx->count / 8) % 64;
ctx->count += len * 8;
partLen = 64 - index;
// 데이터 처리
if( len >= partLen )
{
std::memcpy( &ctx->buffer[index], data, partLen );
Transform( ctx->state, ctx->buffer );
for( i = partLen; i + 63 < len; i += 64 )
{
Transform( ctx->state, &data[i] );
}
index = 0;
}
else {
i = 0;
}
std::memcpy( &ctx->buffer[index], &data[i], len - i );
}
void hxcSHA1::Final( uint8_t* hash )
{
uint8_t finalCount[8];
uint32_t i;
for( i = 0; i < 8; i++ )
{
finalCount[i] = static_cast<uint8_t>((ctx->count >> ((7 - i) * 8)) & 0xFF);
}
Update( reinterpret_cast<const uint8_t*>("\x80"), 1 );
while( (ctx->count / 8) % 64 != 56 )
{
Update( reinterpret_cast<const uint8_t*>("\0"), 1 );
}
Update( finalCount, 8 );
for( i = 0; i < 5; i++ )
{
hash[i * 4] = (ctx->state[i] >> 24) & 0xFF;
hash[i * 4 + 1] = (ctx->state[i] >> 16) & 0xFF;
hash[i * 4 + 2] = (ctx->state[i] >> 8) & 0xFF;
hash[i * 4 + 3] = ctx->state[i] & 0xFF;
}
}
size_t hxcSHA1::HashSize() const
{
return 20;
}
// SHA1 내부 라운드 함수
void hxcSHA1::Transform( uint32_t state[5], const uint8_t buffer[64] )
{
uint32_t a, b, c, d, e, t, W[80];
for( int i = 0; i < 16; i++ )
{
W[i] = (buffer[i * 4] << 24) |
(buffer[i * 4 + 1] << 16) |
(buffer[i * 4 + 2] << 8) |
(buffer[i * 4 + 3]);
}
for( int i = 16; i < 80; i++ )
{
W[i] = W[i - 3] ^ W[i - 8] ^ W[i - 14] ^ W[i - 16];
W[i] = (W[i] << 1) | (W[i] >> 31); // 좌측 순환
}
a = state[0];
b = state[1];
c = state[2];
d = state[3];
e = state[4];
// 메인 라운드
for( int i = 0; i < 80; i++ )
{
if( i < 20 )
{
t = ((b & c) | ((~b) & d)) + 0x5A827999;
}
else if( i < 40 )
{
t = (b ^ c ^ d) + 0x6ED9EBA1;
}
else if( i < 60 )
{
t = ((b & c) | (b & d) | (c & d)) + 0x8F1BBCDC;
}
else
{
t = (b ^ c ^ d) + 0xCA62C1D6;
}
t += (a << 5) | (a >> 27);
t += e + W[i];
e = d;
d = c;
c = (b << 30) | (b >> 2);
b = a;
a = t;
}
state[0] += a;
state[1] += b;
state[2] += c;
state[3] += d;
state[4] += e;
}
SHA256
/******************************************************************************
* _ _ _ _____ _ _ _ _ __ _ _ ____ ____
* | | | | / \ | ____| | | | | | |/ / | | / \ | __ ) ___|
* | |_| | / _ \ | _| _ | | | | | ' / | | / _ \ | _ \___ \
* | _ |/ ___ \| |__| |_| | |_| | . \ | |___ / ___ \| |_) |__) |
* |_| |_/_/ \_\_____\___/ \___/|_|\_\ |_____/_/ \_\____/____/
*
* Copyright (c) HAEJUK LABS All Rights Reserved.
*
*******************************************************************************/
/**
@file hxcSHA256.h
@brief
*/
#ifndef __HXC_SHA256_H__
#define __HXC_SHA256_H__
#include "hxPch.h"
#include "hxcSHA.hpp"
class hxcSHA256 : public hxcSHA
{
DECLARE_NO_COPY_CLASS( hxcSHA256 );
public:
hxcSHA256();
void Init() override;
void Update( const uint8_t* data, size_t len ) override;
void Final( uint8_t* hash ) override;
size_t HashSize() const override;
private:
void Transform( uint32_t state[8], const uint8_t buffer[64] );
private:
std::unique_ptr<hxsSHA256_CTX> ctx;
};
#endif //! __HXC_SHA256_H__
/******************************************************************************
* _ _ _ _____ _ _ _ _ __ _ _ ____ ____
* | | | | / \ | ____| | | | | | |/ / | | / \ | __ ) ___|
* | |_| | / _ \ | _| _ | | | | | ' / | | / _ \ | _ \___ \
* | _ |/ ___ \| |__| |_| | |_| | . \ | |___ / ___ \| |_) |__) |
* |_| |_/_/ \_\_____\___/ \___/|_|\_\ |_____/_/ \_\____/____/
*
* Copyright (c) HAEJUK LABS All Rights Reserved.
*
*******************************************************************************/
/**
@file hxcSHA256.cpp
@brief
*/
#include "hxcSHA256.h"
hxcSHA256::hxcSHA256()
: ctx( std::make_unique<hxsSHA256_CTX>() )
{
}
void hxcSHA256::Init()
{
ctx->Reset();
}
void hxcSHA256::Update( const uint8_t* data, size_t len )
{
size_t i, index, partLen;
index = (ctx->count / 8) % 64;
ctx->count += len * 8;
partLen = 64 - index;
if( len >= partLen )
{
memcpy( &ctx->buffer[index], data, partLen );
Transform( ctx->state, ctx->buffer );
for( i = partLen; i + 63 < len; i += 64 )
{
Transform( ctx->state, &data[i] );
}
index = 0;
}
else {
i = 0;
}
memcpy( &ctx->buffer[index], &data[i], len - i );
}
void hxcSHA256::Final( uint8_t* hash )
{
uint8_t finalCount[8];
uint32_t i;
for( i = 0; i < 8; i++ )
{
finalCount[i] = (ctx->count >> ((7 - i) * 8)) & 0xFF;
}
Update( (const uint8_t*)"\x80", 1 );
while( (ctx->count / 8) % 64 != 56 )
{
Update( (const uint8_t*)"\0", 1 );
}
Update( finalCount, 8 );
for( i = 0; i < 8; i++ )
{
hash[i * 4] = (ctx->state[i] >> 24) & 0xFF;
hash[i * 4 + 1] = (ctx->state[i] >> 16) & 0xFF;
hash[i * 4 + 2] = (ctx->state[i] >> 8) & 0xFF;
hash[i * 4 + 3] = ctx->state[i] & 0xFF;
}
}
size_t hxcSHA256::HashSize() const
{
return 32;
}
void hxcSHA256::Transform( uint32_t state[8], const uint8_t buffer[64] )
{
uint32_t W[64];
uint32_t a, b, c, d, e, f, g, h, t1, t2;
static const uint32_t K[64] =
{
0x428A2F98, 0x71374491, 0xB5C0FBCF, 0xE9B5DBA5, 0x3956C25B, 0x59F111F1, 0x923F82A4, 0xAB1C5ED5,
0xD807AA98, 0x12835B01, 0x243185BE, 0x550C7DC3, 0x72BE5D74, 0x80DEB1FE, 0x9BDC06A7, 0xC19BF174,
0xE49B69C1, 0xEFBE4786, 0x0FC19DC6, 0x240CA1CC, 0x2DE92C6F, 0x4A7484AA, 0x5CB0A9DC, 0x76F988DA,
0x983E5152, 0xA831C66D, 0xB00327C8, 0xBF597FC7, 0xC6E00BF3, 0xD5A79147, 0x06CA6351, 0x14292967,
0x27B70A85, 0x2E1B2138, 0x4D2C6DFC, 0x53380D13, 0x650A7354, 0x766A0ABB, 0x81C2C92E, 0x92722C85,
0xA2BFE8A1, 0xA81A664B, 0xC24B8B70, 0xC76C51A3, 0xD192E819, 0xD6990624, 0xF40E3585, 0x106AA070,
0x19A4C116, 0x1E376C08, 0x2748774C, 0x34B0BCB5, 0x391C0CB3, 0x4ED8AA4A, 0x5B9CCA4F, 0x682E6FF3,
0x748F82EE, 0x78A5636F, 0x84C87814, 0x8CC70208, 0x90BEFFFA, 0xA4506CEB, 0xBEF9A3F7, 0xC67178F2
};
for( int i = 0; i < 16; i++ )
{
W[i] = (buffer[i * 4] << 24) | (buffer[i * 4 + 1] << 16) | (buffer[i * 4 + 2] << 8) | buffer[i * 4 + 3];
}
for( int i = 16; i < 64; i++ )
{
W[i] = W[i - 16] + (W[i - 15] >> 7 | W[i - 15] << (32 - 7));
}
a = state[0]; b = state[1]; c = state[2]; d = state[3];
e = state[4]; f = state[5]; g = state[6]; h = state[7];
for( int i = 0; i < 64; i++ )
{
t1 = h + (e >> 6 | e << (32 - 6)) + ((e & f) ^ (~e & g)) + K[i] + W[i];
t2 = (a >> 2 | a << (32 - 2)) + ((a & b) ^ (a & c) ^ (b & c));
h = g; g = f; f = e; e = d + t1;
d = c; c = b; b = a; a = t1 + t2;
}
state[0] += a; state[1] += b; state[2] += c; state[3] += d;
state[4] += e; state[5] += f; state[6] += g; state[7] += h;
}
SHA384
/******************************************************************************
* _ _ _ _____ _ _ _ _ __ _ _ ____ ____
* | | | | / \ | ____| | | | | | |/ / | | / \ | __ ) ___|
* | |_| | / _ \ | _| _ | | | | | ' / | | / _ \ | _ \___ \
* | _ |/ ___ \| |__| |_| | |_| | . \ | |___ / ___ \| |_) |__) |
* |_| |_/_/ \_\_____\___/ \___/|_|\_\ |_____/_/ \_\____/____/
*
* Copyright (c) HAEJUK LABS All Rights Reserved.
*
*******************************************************************************/
/**
@file hxcSHA384.h
@brief
*/
#ifndef __HXC_SHA384_H__
#define __HXC_SHA384_H__
#include "hxPch.h"
#include "hxcSHA.hpp"
class hxcSHA384 : public hxcSHA
{
DECLARE_NO_COPY_CLASS( hxcSHA384 );
public:
hxcSHA384();
virtual ~hxcSHA384() = default;
void Init() override;
void Update( const uint8_t* data, size_t len ) override;
void Final( uint8_t* hash ) override;
size_t HashSize() const override;
private:
void Transform( uint64_t state[8], const uint8_t buffer[128] );
private:
std::unique_ptr<hxsSHA384_CTX> ctx;
};
#endif //! __HXC_SHA256_H__
/******************************************************************************
* _ _ _ _____ _ _ _ _ __ _ _ ____ ____
* | | | | / \ | ____| | | | | | |/ / | | / \ | __ ) ___|
* | |_| | / _ \ | _| _ | | | | | ' / | | / _ \ | _ \___ \
* | _ |/ ___ \| |__| |_| | |_| | . \ | |___ / ___ \| |_) |__) |
* |_| |_/_/ \_\_____\___/ \___/|_|\_\ |_____/_/ \_\____/____/
*
* Copyright (c) HAEJUK LABS All Rights Reserved.
*
*******************************************************************************/
/**
@file hxcSHA384.cpp
@brief
*/
#include "hxcSHA384.h"
hxcSHA384::hxcSHA384()
: ctx( std::make_unique<hxsSHA384_CTX>() )
{
}
void hxcSHA384::Init()
{
ctx->Reset();
}
void hxcSHA384::Update( const uint8_t* data, size_t len )
{
size_t i, index, partLen;
index = (ctx->count / 8) % 128;
ctx->count += len * 8;
partLen = 128 - index;
if( len >= partLen ) {
memcpy( &ctx->buffer[index], data, partLen );
Transform( ctx->state, ctx->buffer );
for( i = partLen; i + 127 < len; i += 128 ) {
Transform( ctx->state, &data[i] );
}
index = 0;
}
else {
i = 0;
}
memcpy( &ctx->buffer[index], &data[i], len - i );
}
void hxcSHA384::Final( uint8_t* hash )
{
uint8_t finalCount[16];
uint32_t i;
for( i = 0; i < 16; i++ )
{
finalCount[i] = (ctx->count >> ((15 - i) * 8)) & 0xFF;
}
Update( (const uint8_t*)"\x80", 1 );
while( (ctx->count / 8) % 128 != 112 )
{
Update( (const uint8_t*)"\0", 1 );
}
Update( finalCount, 16 );
for( i = 0; i < 6; i++ )
{
hash[i * 8] = (ctx->state[i] >> 56) & 0xFF;
hash[i * 8 + 1] = (ctx->state[i] >> 48) & 0xFF;
hash[i * 8 + 2] = (ctx->state[i] >> 40) & 0xFF;
hash[i * 8 + 3] = (ctx->state[i] >> 32) & 0xFF;
hash[i * 8 + 4] = (ctx->state[i] >> 24) & 0xFF;
hash[i * 8 + 5] = (ctx->state[i] >> 16) & 0xFF;
hash[i * 8 + 6] = (ctx->state[i] >> 8) & 0xFF;
hash[i * 8 + 7] = ctx->state[i] & 0xFF;
}
}
size_t hxcSHA384::HashSize() const
{
return 48;
}
void hxcSHA384::Transform( uint64_t state[8], const uint8_t buffer[128] )
{
uint64_t W[80];
uint64_t a, b, c, d, e, f, g, h, t1, t2;
static const uint64_t K[80] = {
0x428A2F98D728AE22, 0x7137449123EF65CD, 0xB5C0FBCFEC4D3B2F, 0xE9B5DBA58189DBBC,
0x3956C25BF348B538, 0x59F111F1B605D019, 0x923F82A4AF194F9B, 0xAB1C5ED5DA6D8118,
0xD807AA98A3030242, 0x12835B0145706FBE, 0x243185BE4EE4B28C, 0x550C7DC3D5FFB4E2,
0x72BE5D74F27B896F, 0x80DEB1FE3B1696B1, 0x9BDC06A725C71235, 0xC19BF174CF692694,
0xE49B69C19EF14AD2, 0xEFBE4786384F25E3, 0x0FC19DC68B8CD5B5, 0x240CA1CC77AC9C65,
0x2DE92C6F592B0275, 0x4A7484AA6EA6E483, 0x5CB0A9DCBD41FBD4, 0x76F988DA831153B5,
0x983E5152EE66DFAB, 0xA831C66D2DB43210, 0xB00327C898FB213F, 0xBF597FC7BEEF0EE4,
0xC6E00BF33DA88FC2, 0xD5A79147930AA725, 0x06CA6351E003826F, 0x142929670A0E6E70,
0x27B70A8546D22FFC, 0x2E1B21385C26C926, 0x4D2C6DFC5AC42AED, 0x53380D139D95B3DF,
0x650A73548BAF63DE, 0x766A0ABB3C77B2A8, 0x81C2C92E47EDAEE6, 0x92722C851482353B,
0xA2BFE8A14CF10364, 0xA81A664BBC423001, 0xC24B8B70D0F89791, 0xC76C51A30654BE30,
0xD192E819D6EF5218, 0xD69906245565A910, 0xF40E35855771202A, 0x106AA07032BBD1B8,
0x19A4C116B8D2D0C8, 0x1E376C085141AB53, 0x2748774CDF8EEB99, 0x34B0BCB5E19B48A8,
0x391C0CB3C5C95A63, 0x4ED8AA4AE3418ACB, 0x5B9CCA4F7763E373, 0x682E6FF3D6B2B8A3,
0x748F82EE5DEFB2FC, 0x78A5636F43172F60, 0x84C87814A1F0AB72, 0x8CC702081A6439EC,
0x90BEFFFA23631E28, 0xA4506CEBDE82BDE9, 0xBEF9A3F7B2C67915, 0xC67178F2E372532B,
0xCA273ECEEA26619C, 0xD186B8C721C0C207, 0xEADA7DD6CDE0EB1E, 0xF57D4F7FEE6ED178,
0x06F067AA72176FBA, 0x0A637DC5A2C898A6, 0x113F9804BEF90DAE, 0x1B710B35131C471B,
0x28DB77F523047D84, 0x32CAAB7B40C72493, 0x3C9EBE0A15C9BEBC, 0x431D67C49C100D4C,
0x4CC5D4BECB3E42B6, 0x597F299CFC657E2A, 0x5FCB6FAB3AD6FAEC, 0x6C44198C4A475817
};
for( int i = 0; i < 16; i++ )
{
W[i] = ((uint64_t)buffer[i * 8] << 56) | ((uint64_t)buffer[i * 8 + 1] << 48) |
((uint64_t)buffer[i * 8 + 2] << 40) | ((uint64_t)buffer[i * 8 + 3] << 32) |
((uint64_t)buffer[i * 8 + 4] << 24) | ((uint64_t)buffer[i * 8 + 5] << 16) |
((uint64_t)buffer[i * 8 + 6] << 8) | (uint64_t)buffer[i * 8 + 7];
}
for( int i = 16; i < 80; i++ )
{
W[i] = W[i - 16] + (W[i - 15] >> 1 | W[i - 15] << (64 - 1));
}
a = state[0]; b = state[1]; c = state[2]; d = state[3];
e = state[4]; f = state[5]; g = state[6]; h = state[7];
for( int i = 0; i < 80; i++ )
{
t1 = h + (e >> 14 | e << (64 - 14)) + ((e & f) ^ (~e & g)) + K[i] + W[i];
t2 = (a >> 28 | a << (64 - 28)) + ((a & b) ^ (a & c) ^ (b & c));
h = g; g = f; f = e; e = d + t1;
d = c; c = b; b = a; a = t1 + t2;
}
state[0] += a; state[1] += b; state[2] += c; state[3] += d;
state[4] += e; state[5] += f; state[6] += g; state[7] += h;
}
728x90