oSHA256.cpp 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. /*====================================================================*
  2. *
  3. * oSHA256.cpp - oSHA256 class definition;
  4. *
  5. * implement 256-bit encryption according to FIPS180-2 sec 5.3.2 by
  6. * encoding variable-length input into fixed-length, 32 byte digest;
  7. *
  8. * Motley Tools by Charles Maier <cmaier@cmassoc.net>;
  9. * Copyright 2001-2006 by Charles Maier Associates;
  10. * Licensed under the Internet Software Consortium License;
  11. *
  12. *--------------------------------------------------------------------*/
  13. #ifndef oSHA256_SOURCE
  14. #define oSHA256_SOURCE
  15. /*====================================================================*
  16. * system header files;
  17. *--------------------------------------------------------------------*/
  18. #include <stdint.h>
  19. #include <iostream>
  20. /*====================================================================*
  21. * custom header files;
  22. *--------------------------------------------------------------------*/
  23. #include "../classes/oSHA256.hpp"
  24. /*====================================================================*
  25. * pseudo functions;
  26. *--------------------------------------------------------------------*/
  27. #define GET_UINT32(n,b,i) \
  28. n = ((uint32_t)(b)[(i)+0]<<24) \
  29. | ((uint32_t)(b)[(i)+1]<<16) \
  30. | ((uint32_t)(b)[(i)+2]<< 8) \
  31. | ((uint32_t)(b)[(i)+3]<< 0);
  32. #define PUT_UINT32(n,b,i) \
  33. (b)[(i)+0]=(uint8_t)((n)>>24); \
  34. (b)[(i)+1]=(uint8_t)((n)>>16); \
  35. (b)[(i)+2]=(uint8_t)((n)>> 8); \
  36. (b)[(i)+3]=(uint8_t)((n)>> 0);
  37. /*====================================================================*
  38. * class constants;
  39. *--------------------------------------------------------------------*/
  40. unsigned oSHA256::DigestLength = 32;
  41. /*====================================================================*
  42. *
  43. * oSHA256 & Write (void const * memory, size_t length);
  44. *
  45. * write a variable-length data block to the SHA256 digest; behave
  46. * like function write() but return no value and never fail;
  47. *
  48. * Read standard FIPS180-2 sec 5.3.2 for an explanation;
  49. *
  50. *--------------------------------------------------------------------*/
  51. oSHA256 & oSHA256::Write (void const * memory, size_t extent)
  52. {
  53. if (extent)
  54. {
  55. byte * buffer = (byte *)(memory);
  56. uint32_t left = this->mcount [0] & 0x3F;
  57. uint32_t fill = oSHA256_BUFFER_LENGTH - left;
  58. this->mcount [0] += (uint32_t)(extent);
  59. this->mcount [0] &= 0xFFFFFFFF;
  60. if (this->mcount [0] < extent)
  61. {
  62. this->mcount [1]++;
  63. }
  64. if ((left) && (extent >= fill))
  65. {
  66. std::memcpy (this->mblock + left, buffer, (uint16_t) (fill));
  67. this->Block (this->mblock);
  68. extent -= fill;
  69. buffer += fill;
  70. left = 0;
  71. }
  72. while (extent >= oSHA256_BUFFER_LENGTH)
  73. {
  74. oSHA256::Block (buffer);
  75. extent -= oSHA256_BUFFER_LENGTH;
  76. buffer += oSHA256_BUFFER_LENGTH;
  77. }
  78. if (extent)
  79. {
  80. std::memcpy (this->mblock + left, buffer, (uint16_t) (extent));
  81. }
  82. }
  83. return (*this);
  84. }
  85. /*====================================================================*
  86. *
  87. * oSHA256 & Block (const byte buffer []);
  88. *
  89. * merge a 64 byte data block into an SHA256 digest;
  90. *
  91. * Read standard FIPS180-2 sec 5.3.2 for an explanation;
  92. *
  93. *--------------------------------------------------------------------*/
  94. #define SHR(word,bits) ((word & 0xFFFFFFFF) >> bits)
  95. #define ROTR(word,bits) (SHR(word,bits) | (word << (32 - bits)))
  96. oSHA256 & oSHA256::Block (const byte buffer [])
  97. {
  98. static const uint32_t K [oSHA256_BUFFER_LENGTH] =
  99. {
  100. 0x428A2F98,
  101. 0x71374491,
  102. 0xB5C0FBCF,
  103. 0xE9B5DBA5,
  104. 0x3956C25B,
  105. 0x59F111F1,
  106. 0x923F82A4,
  107. 0xAB1C5ED5,
  108. 0xD807AA98,
  109. 0x12835B01,
  110. 0x243185BE,
  111. 0x550C7DC3,
  112. 0x72BE5D74,
  113. 0x80DEB1FE,
  114. 0x9BDC06A7,
  115. 0xC19BF174,
  116. 0xE49B69C1,
  117. 0xEFBE4786,
  118. 0x0FC19DC6,
  119. 0x240CA1CC,
  120. 0x2DE92C6F,
  121. 0x4A7484AA,
  122. 0x5CB0A9DC,
  123. 0x76F988DA,
  124. 0x983E5152,
  125. 0xA831C66D,
  126. 0xB00327C8,
  127. 0xBF597FC7,
  128. 0xC6E00BF3,
  129. 0xD5A79147,
  130. 0x06CA6351,
  131. 0x14292967,
  132. 0x27B70A85,
  133. 0x2E1B2138,
  134. 0x4D2C6DFC,
  135. 0x53380D13,
  136. 0x650A7354,
  137. 0x766A0ABB,
  138. 0x81C2C92E,
  139. 0x92722C85,
  140. 0xA2BFE8A1,
  141. 0xA81A664B,
  142. 0xC24B8B70,
  143. 0xC76C51A3,
  144. 0xD192E819,
  145. 0xD6990624,
  146. 0xF40E3585,
  147. 0x106AA070,
  148. 0x19A4C116,
  149. 0x1E376C08,
  150. 0x2748774C,
  151. 0x34B0BCB5,
  152. 0x391C0CB3,
  153. 0x4ED8AA4A,
  154. 0x5B9CCA4F,
  155. 0x682E6FF3,
  156. 0x748F82EE,
  157. 0x78A5636F,
  158. 0x84C87814,
  159. 0x8CC70208,
  160. 0x90BEFFFA,
  161. 0xA4506CEB,
  162. 0xBEF9A3F7,
  163. 0xC67178F2
  164. };
  165. signed word;
  166. signed pass;
  167. uint32_t H [oSHA256_DIGEST_LENGTH];
  168. uint32_t W [oSHA256_BUFFER_LENGTH];
  169. for (word = 0; word < 16; word++)
  170. {
  171. GET_UINT32 (W [word], buffer, word * sizeof (uint32_t));
  172. }
  173. while (word < oSHA256_BUFFER_LENGTH)
  174. {
  175. uint32_t s0 = ROTR (W [word-15], 7) ^ ROTR (W [word-15], 18) ^ SHR (W [word-15], 3);
  176. uint32_t s1 = ROTR (W [word- 2], 17) ^ ROTR (W [word- 2], 19) ^ SHR (W [word- 2], 10);
  177. W [word] = W [word - 16] + s0 + W [word - 7] + s1;
  178. word++;
  179. }
  180. for (word = 0; word < oSHA256_HASH_SIZE; word++)
  181. {
  182. H [word] = this->mstate [word];
  183. }
  184. for (pass = 0; pass < oSHA256_BUFFER_LENGTH; pass++)
  185. {
  186. uint32_t s2 = ROTR (H [0], 2) ^ ROTR (H [0], 13) ^ ROTR (H [0], 22);
  187. uint32_t maj = (H [0] & H [1]) ^ (H [0] & H [2]) ^ (H [1] & H [2]);
  188. uint32_t t2 = s2 + maj;
  189. uint32_t s3 = ROTR (H [4], 6) ^ ROTR (H [4], 11) ^ ROTR (H [4], 25);
  190. uint32_t ch = (H [4] & H [5]) ^ ((~ H [4]) & H [6]);
  191. uint32_t t1 = H [7] + s3 + ch + K [pass] + W [pass];
  192. for (word = oSHA256_DIGEST_LENGTH - 1; word > 0; word--)
  193. {
  194. H [word] = H [word-1];
  195. }
  196. H [0] = t1 + t2;
  197. H [4] += t1;
  198. }
  199. for (word = 0; word < oSHA256_DIGEST_LENGTH; word++)
  200. {
  201. this->mstate [word] += H [word];
  202. }
  203. return (*this);
  204. }
  205. /*====================================================================*
  206. *
  207. * oSHA256 & Fetch (byte diget []);
  208. *
  209. * read the SHA256 digest; this function works like function read()
  210. * but the function returns no value and the digest buffer is fixed
  211. * length;
  212. *
  213. * to start a digest, use the Reset method; to write data to
  214. * the digest use the Write method;
  215. *
  216. * Read standard FIPS180-2 sec 5.3.2 for an explanation;
  217. *
  218. *--------------------------------------------------------------------*/
  219. oSHA256 & oSHA256::Fetch (byte digest [])
  220. {
  221. unsigned word;
  222. unsigned char msglen [8];
  223. uint32_t upper = (this->mcount [0] >> 29) | (this->mcount [1] << 3);
  224. uint32_t lower = (this->mcount [0] << 3);
  225. uint32_t final = (this->mcount [0] & 0x3F);
  226. uint32_t extra = (final < 56)? (56 - final): (120 - final);
  227. PUT_UINT32 (upper, msglen, 0);
  228. PUT_UINT32 (lower, msglen, 4);
  229. oSHA256::Write (this->mextra, extra);
  230. oSHA256::Write (msglen, sizeof (msglen));
  231. for (word = 0; word < oSHA256_DIGEST_LENGTH; word++)
  232. {
  233. PUT_UINT32 (this->mstate [word], digest, word * sizeof (uint32_t));
  234. }
  235. oSHA256::Reset ();
  236. return (*this);
  237. }
  238. /*====================================================================*
  239. *
  240. * oSHA256 & Reset ()
  241. *
  242. *--------------------------------------------------------------------*/
  243. oSHA256 & oSHA256::Reset (void)
  244. {
  245. this->mstate [0] = 0x6A09E667;
  246. this->mstate [1] = 0xBB67AE85;
  247. this->mstate [2] = 0x3C6EF372;
  248. this->mstate [3] = 0xA54FF53A;
  249. this->mstate [4] = 0x510E527F;
  250. this->mstate [5] = 0x9B05688C;
  251. this->mstate [6] = 0x1F83D9AB;
  252. this->mstate [7] = 0x5BE0CD19;
  253. std::memset (this->mcount, 0, oSHA256_LEFT_SIZE * sizeof (uint32_t));
  254. std::memset (this->mblock, 0, oSHA256_BUFFER_LENGTH);
  255. std::memset (this->mextra, 0, oSHA256_BUFFER_LENGTH);
  256. this->mextra [0] = 0x80;
  257. return (*this);
  258. }
  259. /*====================================================================*
  260. *
  261. * oSHA256 ()
  262. *
  263. *--------------------------------------------------------------------*/
  264. oSHA256::oSHA256 ()
  265. {
  266. this->mcount = new uint32_t [oSHA256_LEFT_SIZE];
  267. this->mstate = new uint32_t [oSHA256_HASH_SIZE];
  268. this->mblock = new uint8_t [oSHA256_BUFFER_LENGTH];
  269. this->mextra = new uint8_t [oSHA256_BUFFER_LENGTH];
  270. oSHA256::Reset ();
  271. return;
  272. }
  273. /*====================================================================*
  274. *
  275. * void ~osha256 ()
  276. *
  277. *--------------------------------------------------------------------*/
  278. oSHA256::~oSHA256 ()
  279. {
  280. delete [] this->mstate;
  281. delete [] this->mcount;
  282. delete [] this->mblock;
  283. delete [] this->mextra;
  284. return;
  285. }
  286. /*====================================================================*
  287. * end definition;
  288. *--------------------------------------------------------------------*/
  289. #endif