PageRenderTime 44ms CodeModel.GetById 14ms RepoModel.GetById 0ms app.codeStats 0ms

/ext/hash/hash_gost.c

http://github.com/php/php-src
C | 327 lines | 272 code | 35 blank | 20 comment | 15 complexity | db8d4a81ea769260a3d08f142bd55260 MD5 | raw file
Possible License(s): BSD-2-Clause, BSD-3-Clause, MPL-2.0-no-copyleft-exception, LGPL-2.1
  1. /*
  2. +----------------------------------------------------------------------+
  3. | Copyright (c) The PHP Group |
  4. +----------------------------------------------------------------------+
  5. | This source file is subject to version 3.01 of the PHP license, |
  6. | that is bundled with this package in the file LICENSE, and is |
  7. | available through the world-wide-web at the following url: |
  8. | http://www.php.net/license/3_01.txt |
  9. | If you did not receive a copy of the PHP license and are unable to |
  10. | obtain it through the world-wide-web, please send a note to |
  11. | license@php.net so we can mail you a copy immediately. |
  12. +----------------------------------------------------------------------+
  13. | Authors: Michael Wallner <mike@php.net> |
  14. | Sara Golemon <pollita@php.net> |
  15. +----------------------------------------------------------------------+
  16. */
  17. #include "php_hash.h"
  18. #include "php_hash_gost.h"
  19. #include "php_hash_gost_tables.h"
  20. /* {{{ Gost()
  21. * derived from gost_compress() by Markku-Juhani Saarinen <mjos@ssh.fi>
  22. */
  23. #define round(tables, k1, k2) \
  24. t = (k1) + r; \
  25. l ^= tables[0][t & 0xff] ^ tables[1][(t >> 8) & 0xff] ^ \
  26. tables[2][(t >> 16) & 0xff] ^ tables[3][t >> 24]; \
  27. t = (k2) + l; \
  28. r ^= tables[0][t & 0xff] ^ tables[1][(t >> 8) & 0xff] ^ \
  29. tables[2][(t >> 16) & 0xff] ^ tables[3][t >> 24];
  30. #define R(tables, key, h, i, t, l, r) \
  31. r = h[i]; \
  32. l = h[i + 1]; \
  33. round(tables, key[0], key[1]) \
  34. round(tables, key[2], key[3]) \
  35. round(tables, key[4], key[5]) \
  36. round(tables, key[6], key[7]) \
  37. round(tables, key[0], key[1]) \
  38. round(tables, key[2], key[3]) \
  39. round(tables, key[4], key[5]) \
  40. round(tables, key[6], key[7]) \
  41. round(tables, key[0], key[1]) \
  42. round(tables, key[2], key[3]) \
  43. round(tables, key[4], key[5]) \
  44. round(tables, key[6], key[7]) \
  45. round(tables, key[7], key[6]) \
  46. round(tables, key[5], key[4]) \
  47. round(tables, key[3], key[2]) \
  48. round(tables, key[1], key[0]) \
  49. t = r; \
  50. r = l; \
  51. l = t; \
  52. #define X(w, u, v) \
  53. w[0] = u[0] ^ v[0]; \
  54. w[1] = u[1] ^ v[1]; \
  55. w[2] = u[2] ^ v[2]; \
  56. w[3] = u[3] ^ v[3]; \
  57. w[4] = u[4] ^ v[4]; \
  58. w[5] = u[5] ^ v[5]; \
  59. w[6] = u[6] ^ v[6]; \
  60. w[7] = u[7] ^ v[7];
  61. #define P(key, w) \
  62. key[0] = (w[0] & 0x000000ff) | ((w[2] & 0x000000ff) << 8) | \
  63. ((w[4] & 0x000000ff) << 16) | ((w[6] & 0x000000ff) << 24); \
  64. key[1] = ((w[0] & 0x0000ff00) >> 8) | (w[2] & 0x0000ff00) | \
  65. ((w[4] & 0x0000ff00) << 8) | ((w[6] & 0x0000ff00) << 16); \
  66. key[2] = ((w[0] & 0x00ff0000) >> 16) | ((w[2] & 0x00ff0000) >> 8) | \
  67. (w[4] & 0x00ff0000) | ((w[6] & 0x00ff0000) << 8); \
  68. key[3] = ((w[0] & 0xff000000) >> 24) | ((w[2] & 0xff000000) >> 16) | \
  69. ((w[4] & 0xff000000) >> 8) | (w[6] & 0xff000000); \
  70. key[4] = (w[1] & 0x000000ff) | ((w[3] & 0x000000ff) << 8) | \
  71. ((w[5] & 0x000000ff) << 16) | ((w[7] & 0x000000ff) << 24); \
  72. key[5] = ((w[1] & 0x0000ff00) >> 8) | (w[3] & 0x0000ff00) | \
  73. ((w[5] & 0x0000ff00) << 8) | ((w[7] & 0x0000ff00) << 16); \
  74. key[6] = ((w[1] & 0x00ff0000) >> 16) | ((w[3] & 0x00ff0000) >> 8) | \
  75. (w[5] & 0x00ff0000) | ((w[7] & 0x00ff0000) << 8); \
  76. key[7] = ((w[1] & 0xff000000) >> 24) | ((w[3] & 0xff000000) >> 16) | \
  77. ((w[5] & 0xff000000) >> 8) | (w[7] & 0xff000000);
  78. #define A(x, l, r) \
  79. l = x[0] ^ x[2]; \
  80. r = x[1] ^ x[3]; \
  81. x[0] = x[2]; \
  82. x[1] = x[3]; \
  83. x[2] = x[4]; \
  84. x[3] = x[5]; \
  85. x[4] = x[6]; \
  86. x[5] = x[7]; \
  87. x[6] = l; \
  88. x[7] = r;
  89. #define AA(x, l, r) \
  90. l = x[0]; \
  91. r = x[2]; \
  92. x[0] = x[4]; \
  93. x[2] = x[6]; \
  94. x[4] = l ^ r; \
  95. x[6] = x[0] ^ r; \
  96. l = x[1]; \
  97. r = x[3]; \
  98. x[1] = x[5]; \
  99. x[3] = x[7]; \
  100. x[5] = l ^ r; \
  101. x[7] = x[1] ^ r;
  102. #define C(x) \
  103. x[0] ^= 0xff00ff00; \
  104. x[1] ^= 0xff00ff00; \
  105. x[2] ^= 0x00ff00ff; \
  106. x[3] ^= 0x00ff00ff; \
  107. x[4] ^= 0x00ffff00; \
  108. x[5] ^= 0xff0000ff; \
  109. x[6] ^= 0x000000ff; \
  110. x[7] ^= 0xff00ffff;
  111. #define S(s, l, r) \
  112. s[i] = r; \
  113. s[i + 1] = l;
  114. #define SHIFT12(u, m, s) \
  115. u[0] = m[0] ^ s[6]; \
  116. u[1] = m[1] ^ s[7]; \
  117. u[2] = m[2] ^ (s[0] << 16) ^ (s[0] >> 16) ^ (s[0] & 0xffff) ^ \
  118. (s[1] & 0xffff) ^ (s[1] >> 16) ^ (s[2] << 16) ^ s[6] ^ (s[6] << 16) ^ \
  119. (s[7] & 0xffff0000) ^ (s[7] >> 16); \
  120. u[3] = m[3] ^ (s[0] & 0xffff) ^ (s[0] << 16) ^ (s[1] & 0xffff) ^ \
  121. (s[1] << 16) ^ (s[1] >> 16) ^ (s[2] << 16) ^ (s[2] >> 16) ^ \
  122. (s[3] << 16) ^ s[6] ^ (s[6] << 16) ^ (s[6] >> 16) ^ (s[7] & 0xffff) ^ \
  123. (s[7] << 16) ^ (s[7] >> 16); \
  124. u[4] = m[4] ^ \
  125. (s[0] & 0xffff0000) ^ (s[0] << 16) ^ (s[0] >> 16) ^ \
  126. (s[1] & 0xffff0000) ^ (s[1] >> 16) ^ (s[2] << 16) ^ (s[2] >> 16) ^ \
  127. (s[3] << 16) ^ (s[3] >> 16) ^ (s[4] << 16) ^ (s[6] << 16) ^ \
  128. (s[6] >> 16) ^(s[7] & 0xffff) ^ (s[7] << 16) ^ (s[7] >> 16); \
  129. u[5] = m[5] ^ (s[0] << 16) ^ (s[0] >> 16) ^ (s[0] & 0xffff0000) ^ \
  130. (s[1] & 0xffff) ^ s[2] ^ (s[2] >> 16) ^ (s[3] << 16) ^ (s[3] >> 16) ^ \
  131. (s[4] << 16) ^ (s[4] >> 16) ^ (s[5] << 16) ^ (s[6] << 16) ^ \
  132. (s[6] >> 16) ^ (s[7] & 0xffff0000) ^ (s[7] << 16) ^ (s[7] >> 16); \
  133. u[6] = m[6] ^ s[0] ^ (s[1] >> 16) ^ (s[2] << 16) ^ s[3] ^ (s[3] >> 16) ^ \
  134. (s[4] << 16) ^ (s[4] >> 16) ^ (s[5] << 16) ^ (s[5] >> 16) ^ s[6] ^ \
  135. (s[6] << 16) ^ (s[6] >> 16) ^ (s[7] << 16); \
  136. u[7] = m[7] ^ (s[0] & 0xffff0000) ^ (s[0] << 16) ^ (s[1] & 0xffff) ^ \
  137. (s[1] << 16) ^ (s[2] >> 16) ^ (s[3] << 16) ^ s[4] ^ (s[4] >> 16) ^ \
  138. (s[5] << 16) ^ (s[5] >> 16) ^ (s[6] >> 16) ^ (s[7] & 0xffff) ^ \
  139. (s[7] << 16) ^ (s[7] >> 16);
  140. #define SHIFT16(h, v, u) \
  141. v[0] = h[0] ^ (u[1] << 16) ^ (u[0] >> 16); \
  142. v[1] = h[1] ^ (u[2] << 16) ^ (u[1] >> 16); \
  143. v[2] = h[2] ^ (u[3] << 16) ^ (u[2] >> 16); \
  144. v[3] = h[3] ^ (u[4] << 16) ^ (u[3] >> 16); \
  145. v[4] = h[4] ^ (u[5] << 16) ^ (u[4] >> 16); \
  146. v[5] = h[5] ^ (u[6] << 16) ^ (u[5] >> 16); \
  147. v[6] = h[6] ^ (u[7] << 16) ^ (u[6] >> 16); \
  148. v[7] = h[7] ^ (u[0] & 0xffff0000) ^ (u[0] << 16) ^ (u[7] >> 16) ^ \
  149. (u[1] & 0xffff0000) ^ (u[1] << 16) ^ (u[6] << 16) ^ (u[7] & 0xffff0000);
  150. #define SHIFT61(h, v) \
  151. h[0] = (v[0] & 0xffff0000) ^ (v[0] << 16) ^ (v[0] >> 16) ^ (v[1] >> 16) ^ \
  152. (v[1] & 0xffff0000) ^ (v[2] << 16) ^ (v[3] >> 16) ^ (v[4] << 16) ^ \
  153. (v[5] >> 16) ^ v[5] ^ (v[6] >> 16) ^ (v[7] << 16) ^ (v[7] >> 16) ^ \
  154. (v[7] & 0xffff); \
  155. h[1] = (v[0] << 16) ^ (v[0] >> 16) ^ (v[0] & 0xffff0000) ^ (v[1] & 0xffff) ^ \
  156. v[2] ^ (v[2] >> 16) ^ (v[3] << 16) ^ (v[4] >> 16) ^ (v[5] << 16) ^ \
  157. (v[6] << 16) ^ v[6] ^ (v[7] & 0xffff0000) ^ (v[7] >> 16); \
  158. h[2] = (v[0] & 0xffff) ^ (v[0] << 16) ^ (v[1] << 16) ^ (v[1] >> 16) ^ \
  159. (v[1] & 0xffff0000) ^ (v[2] << 16) ^ (v[3] >> 16) ^ v[3] ^ (v[4] << 16) ^ \
  160. (v[5] >> 16) ^ v[6] ^ (v[6] >> 16) ^ (v[7] & 0xffff) ^ (v[7] << 16) ^ \
  161. (v[7] >> 16); \
  162. h[3] = (v[0] << 16) ^ (v[0] >> 16) ^ (v[0] & 0xffff0000) ^ \
  163. (v[1] & 0xffff0000) ^ (v[1] >> 16) ^ (v[2] << 16) ^ (v[2] >> 16) ^ v[2] ^ \
  164. (v[3] << 16) ^ (v[4] >> 16) ^ v[4] ^ (v[5] << 16) ^ (v[6] << 16) ^ \
  165. (v[7] & 0xffff) ^ (v[7] >> 16); \
  166. h[4] = (v[0] >> 16) ^ (v[1] << 16) ^ v[1] ^ (v[2] >> 16) ^ v[2] ^ \
  167. (v[3] << 16) ^ (v[3] >> 16) ^ v[3] ^ (v[4] << 16) ^ (v[5] >> 16) ^ \
  168. v[5] ^ (v[6] << 16) ^ (v[6] >> 16) ^ (v[7] << 16); \
  169. h[5] = (v[0] << 16) ^ (v[0] & 0xffff0000) ^ (v[1] << 16) ^ (v[1] >> 16) ^ \
  170. (v[1] & 0xffff0000) ^ (v[2] << 16) ^ v[2] ^ (v[3] >> 16) ^ v[3] ^ \
  171. (v[4] << 16) ^ (v[4] >> 16) ^ v[4] ^ (v[5] << 16) ^ (v[6] << 16) ^ \
  172. (v[6] >> 16) ^ v[6] ^ (v[7] << 16) ^ (v[7] >> 16) ^ (v[7] & 0xffff0000); \
  173. h[6] = v[0] ^ v[2] ^ (v[2] >> 16) ^ v[3] ^ (v[3] << 16) ^ v[4] ^ \
  174. (v[4] >> 16) ^ (v[5] << 16) ^ (v[5] >> 16) ^ v[5] ^ (v[6] << 16) ^ \
  175. (v[6] >> 16) ^ v[6] ^ (v[7] << 16) ^ v[7]; \
  176. h[7] = v[0] ^ (v[0] >> 16) ^ (v[1] << 16) ^ (v[1] >> 16) ^ (v[2] << 16) ^ \
  177. (v[3] >> 16) ^ v[3] ^ (v[4] << 16) ^ v[4] ^ (v[5] >> 16) ^ v[5] ^ \
  178. (v[6] << 16) ^ (v[6] >> 16) ^ (v[7] << 16) ^ v[7];
  179. #define PASS(tables) \
  180. X(w, u, v); \
  181. P(key, w); \
  182. R((tables), key, h, i, t, l, r); \
  183. S(s, l, r); \
  184. if (i != 6) { \
  185. A(u, l, r); \
  186. if (i == 2) { \
  187. C(u); \
  188. } \
  189. AA(v, l, r); \
  190. }
  191. static inline void Gost(PHP_GOST_CTX *context, uint32_t data[8])
  192. {
  193. int i;
  194. uint32_t l, r, t, key[8], u[8], v[8], w[8], s[8], *h = context->state, *m = data;
  195. memcpy(u, context->state, sizeof(u));
  196. memcpy(v, data, sizeof(v));
  197. for (i = 0; i < 8; i += 2) {
  198. PASS(*context->tables);
  199. }
  200. SHIFT12(u, m, s);
  201. SHIFT16(h, v, u);
  202. SHIFT61(h, v);
  203. }
  204. /* }}} */
  205. static inline void GostTransform(PHP_GOST_CTX *context, const unsigned char input[32])
  206. {
  207. int i, j;
  208. uint32_t data[8], temp = 0;
  209. for (i = 0, j = 0; i < 8; ++i, j += 4) {
  210. data[i] = ((uint32_t) input[j]) | (((uint32_t) input[j + 1]) << 8) |
  211. (((uint32_t) input[j + 2]) << 16) | (((uint32_t) input[j + 3]) << 24);
  212. context->state[i + 8] += data[i] + temp;
  213. temp = context->state[i + 8] < data[i] ? 1 : (context->state[i + 8] == data[i] ? temp : 0);
  214. }
  215. Gost(context, data);
  216. }
  217. PHP_HASH_API void PHP_GOSTInit(PHP_GOST_CTX *context)
  218. {
  219. memset(context, 0, sizeof(*context));
  220. context->tables = &tables_test;
  221. }
  222. PHP_HASH_API void PHP_GOSTInitCrypto(PHP_GOST_CTX *context)
  223. {
  224. PHP_GOSTInit(context);
  225. context->tables = &tables_crypto;
  226. }
  227. static const uint32_t MAX32 = 0xffffffffLU;
  228. PHP_HASH_API void PHP_GOSTUpdate(PHP_GOST_CTX *context, const unsigned char *input, size_t len)
  229. {
  230. if ((MAX32 - context->count[0]) < (len * 8)) {
  231. context->count[1]++;
  232. context->count[0] = MAX32 - context->count[0];
  233. context->count[0] = (len * 8) - context->count[0];
  234. } else {
  235. context->count[0] += len * 8;
  236. }
  237. if (context->length + len < 32) {
  238. memcpy(&context->buffer[context->length], input, len);
  239. context->length += (unsigned char)len;
  240. } else {
  241. size_t i = 0, r = (context->length + len) % 32;
  242. if (context->length) {
  243. i = 32 - context->length;
  244. memcpy(&context->buffer[context->length], input, i);
  245. GostTransform(context, context->buffer);
  246. }
  247. for (; i + 32 <= len; i += 32) {
  248. GostTransform(context, input + i);
  249. }
  250. memcpy(context->buffer, input + i, r);
  251. ZEND_SECURE_ZERO(&context->buffer[r], 32 - r);
  252. context->length = (unsigned char)r;
  253. }
  254. }
  255. PHP_HASH_API void PHP_GOSTFinal(unsigned char digest[32], PHP_GOST_CTX *context)
  256. {
  257. uint32_t i, j, l[8] = {0};
  258. if (context->length) {
  259. GostTransform(context, context->buffer);
  260. }
  261. memcpy(l, context->count, sizeof(context->count));
  262. Gost(context, l);
  263. memcpy(l, &context->state[8], sizeof(l));
  264. Gost(context, l);
  265. for (i = 0, j = 0; j < 32; i++, j += 4) {
  266. digest[j] = (unsigned char) (context->state[i] & 0xff);
  267. digest[j + 1] = (unsigned char) ((context->state[i] >> 8) & 0xff);
  268. digest[j + 2] = (unsigned char) ((context->state[i] >> 16) & 0xff);
  269. digest[j + 3] = (unsigned char) ((context->state[i] >> 24) & 0xff);
  270. }
  271. ZEND_SECURE_ZERO(context, sizeof(*context));
  272. }
  273. const php_hash_ops php_hash_gost_ops = {
  274. (php_hash_init_func_t) PHP_GOSTInit,
  275. (php_hash_update_func_t) PHP_GOSTUpdate,
  276. (php_hash_final_func_t) PHP_GOSTFinal,
  277. (php_hash_copy_func_t) php_hash_copy,
  278. 32,
  279. 32,
  280. sizeof(PHP_GOST_CTX),
  281. 1
  282. };
  283. const php_hash_ops php_hash_gost_crypto_ops = {
  284. (php_hash_init_func_t) PHP_GOSTInitCrypto,
  285. (php_hash_update_func_t) PHP_GOSTUpdate,
  286. (php_hash_final_func_t) PHP_GOSTFinal,
  287. (php_hash_copy_func_t) php_hash_copy,
  288. 32,
  289. 32,
  290. sizeof(PHP_GOST_CTX),
  291. 1
  292. };