QuickCppLib 0.10
Eliminate all the tedious hassle when making state-of-the-art C++ 14 - 23 libraries!
Loading...
Searching...
No Matches
quickcpplib::_xxx::algorithm::hash::sha256_hash Class Reference

SHA256 hash. More...

#include "hash.hpp"

Classes

union  result_type
 The result type of the hash. More...
 

Public Member Functions

constexpr sha256_hash () noexcept
 Initialise the hash.
 
template<class T , typename std::enable_if<(sizeof(T)==1), bool >::type = true, typename std::enable_if<(std::is_trivially_copyable< T >::value), bool >::type = true>
void add (const T *data, size_t bytes) noexcept
 Hash input.
 
result_type finalise () noexcept
 Finalise and return hash.
 

Static Public Member Functions

template<class T , typename std::enable_if<(sizeof(T)==1), bool >::type = true, typename std::enable_if<(std::is_trivially_copyable< T >::value), bool >::type = true>
static constexpr result_type hash (const T *data, size_t bytes) noexcept
 Single shot hash of a sequence of bytes.
 

Detailed Description

SHA256 hash.

Constructor & Destructor Documentation

◆ sha256_hash()

constexpr quickcpplib::_xxx::algorithm::hash::sha256_hash::sha256_hash ( )
inlineconstexprnoexcept

Initialise the hash.

795{}

Member Function Documentation

◆ add()

template<class T , typename std::enable_if<(sizeof(T)==1), bool >::type = true, typename std::enable_if<(std::is_trivially_copyable< T >::value), bool >::type = true>
void quickcpplib::_xxx::algorithm::hash::sha256_hash::add ( const T *  data,
size_t  bytes 
)
inlinenoexcept

Hash input.

801 {
802 static const uint32_t *const k = _k();
803 /*
804 * Note 1: All integers (expect indexes) are 32-bit unsigned integers and addition is calculated modulo 2^32.
805 * Note 2: For each round, there is one round constant k[i] and one entry in the message schedule array w[i], 0 = i = 63
806 * Note 3: The compression function uses 8 working variables, a through h
807 * Note 4: Big-endian convention is used when expressing the constants in this pseudocode,
808 * and when parsing message block data from bytes to words, for example,
809 * the first word of the input message "abc" after padding is 0x61626380
810 */
811 while(const uint8_t *p = _calc_chunk(data, bytes))
812 {
813 /* Note that if we ever do support CPUID detection for the Intel SHA extensions, there is a public domain
814 implementation of the SHA256 rounds at https://github.com/noloader/SHA-Intrinsics/blob/master/sha256-x86.c
815 */
816 alignas(32) uint32_t ah[8];
817
818 /* Initialize working variables to current hash value: */
819 memcpy(ah, _h.as_ints, sizeof(ah));
820
821 /*
822 * create a 64-entry message schedule array w[0..63] of 32-bit words
823 * copy chunk into first 16 words w[0..15] of the message schedule array
824 */
825 alignas(32) uint32_t w[64];
826 for(unsigned i = 0; i < 16; i++)
827 {
828 w[i] = ((uint32_t) p[0] << 24) | ((uint32_t) p[1] << 16) | ((uint32_t) p[2] << 8) | ((uint32_t) p[3]);
829 p += 4;
830 }
831 /* Extend the first 16 words into the remaining 48 words w[16..63] of the message schedule array: */
832 for(unsigned i = 16; i < 64; i++)
833 {
834 const uint32_t s0 = _right_rot(w[i - 15], 7) ^ _right_rot(w[i - 15], 18) ^ (w[i - 15] >> 3);
835 const uint32_t s1 = _right_rot(w[i - 2], 17) ^ _right_rot(w[i - 2], 19) ^ (w[i - 2] >> 10);
836 w[i] = w[i - 16] + s0 + w[i - 7] + s1;
837 }
838
839 /* Compression function main loop: */
840 for(unsigned i = 0; i < 64; i++)
841 {
842 const uint32_t s1 = _right_rot(ah[4], 6) ^ _right_rot(ah[4], 11) ^ _right_rot(ah[4], 25);
843 const uint32_t ch = (ah[4] & ah[5]) ^ (~ah[4] & ah[6]);
844 const uint32_t temp1 = ah[7] + s1 + ch + k[i] + w[i];
845 const uint32_t s0 = _right_rot(ah[0], 2) ^ _right_rot(ah[0], 13) ^ _right_rot(ah[0], 22);
846 const uint32_t maj = (ah[0] & ah[1]) ^ (ah[0] & ah[2]) ^ (ah[1] & ah[2]);
847 const uint32_t temp2 = s0 + maj;
848
849 ah[7] = ah[6];
850 ah[6] = ah[5];
851 ah[5] = ah[4];
852 ah[4] = ah[3] + temp1;
853 ah[3] = ah[2];
854 ah[2] = ah[1];
855 ah[1] = ah[0];
856 ah[0] = temp1 + temp2;
857 }
858
859 /* Add the compressed chunk to the current hash value: */
860 for(unsigned i = 0; i < 8; i++)
861 _h.as_ints[i] += ah[i];
862 }
863 }

◆ finalise()

result_type quickcpplib::_xxx::algorithm::hash::sha256_hash::finalise ( )
inlinenoexcept

Finalise and return hash.

867 {
868 /*
869 * Now:
870 * - either there is enough space left for the total length, and we can conclude,
871 * - or there is too little space left, and we have to pad the rest of this chunk with zeroes.
872 * In the latter case, we will conclude at the next invokation of this function.
873 */
874 size_t space_in_chunk = _CHUNK_SIZE - _chunkidx;
875 assert(space_in_chunk > 0);
876 _chunk[_chunkidx++] = 0x80;
877 space_in_chunk--;
878 if(space_in_chunk < _TOTAL_LEN_LEN)
879 {
880 memory::cmemset(_chunk + _chunkidx, (uint8_t) 0, space_in_chunk);
881 _chunkidx += space_in_chunk;
882 assert(_chunkidx == _CHUNK_SIZE);
883 add((const uint8_t *) nullptr, 0);
884 assert(_chunkidx == 0);
885 space_in_chunk = _CHUNK_SIZE - _chunkidx;
886 }
887 assert(space_in_chunk >= _TOTAL_LEN_LEN);
888
889 const size_t left = space_in_chunk - _TOTAL_LEN_LEN;
890 memory::cmemset(_chunk + _chunkidx, (uint8_t) 0, left);
891 _chunkidx += left;
892
893 /* Storing of len * 8 as a big endian 64-bit without overflow. */
894 size_t len = _total_len;
895 _chunk[_chunkidx + 7] = (uint8_t) ((len << 3) & 0xff);
896 len >>= 5;
897 for(int i = 6; i >= 0; i--)
898 {
899 _chunk[_chunkidx + i] = (uint8_t) (len & 0xff);
900 len >>= 8;
901 }
902 _chunkidx += 8;
903 assert(_chunkidx == _CHUNK_SIZE);
904 add((const uint8_t *) nullptr, 0);
905 assert(_chunkidx == 0);
906
907 result_type ret;
908 /* Produce the final hash value (big-endian): */
909 for(unsigned i = 0, j = 0; i < 8; i++)
910 {
911 ret.as_bytes[j++] = (uint8_t) ((_h.as_ints[i] >> 24) & 0xff);
912 ret.as_bytes[j++] = (uint8_t) ((_h.as_ints[i] >> 16) & 0xff);
913 ret.as_bytes[j++] = (uint8_t) ((_h.as_ints[i] >> 8) & 0xff);
914 ret.as_bytes[j++] = (uint8_t) (_h.as_ints[i] & 0xff);
915 }
916 _h = result_type{0x6a09e667, 0xbb67ae85, 0x3c6ef372, 0xa54ff53a, 0x510e527f, 0x9b05688c, 0x1f83d9ab, 0x5be0cd19};
917 _chunkidx = _total_len = 0;
918 return ret;
919 }
void add(const T *data, size_t bytes) noexcept
Hash input.
Definition hash.hpp:800
EXECINFO_DECL _Check_return_ _In_ size_t len
Definition execinfo_win64.h:55
constexpr T * cmemset(T *dst, T value, size_t num)
Definition memory.hpp:89

◆ hash()

template<class T , typename std::enable_if<(sizeof(T)==1), bool >::type = true, typename std::enable_if<(std::is_trivially_copyable< T >::value), bool >::type = true>
static constexpr result_type quickcpplib::_xxx::algorithm::hash::sha256_hash::hash ( const T *  data,
size_t  bytes 
)
inlinestaticconstexprnoexcept

Single shot hash of a sequence of bytes.

925 {
926 sha256_hash ret;
927 ret.add(data, bytes);
928 return ret.finalise();
929 }
constexpr sha256_hash() noexcept
Initialise the hash.
Definition hash.hpp:795

The documentation for this class was generated from the following file: