Shannon.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443
  1. #include "Shannon.h"
  2. // #include <bit>
  3. #include <stdint.h> // for uint32_t
  4. #include <limits.h> // for CHAR_BIT
  5. // #define NDEBUG
  6. #include <assert.h>
  7. static inline uint32_t rotl(uint32_t n, unsigned int c)
  8. {
  9. const unsigned int mask = (CHAR_BIT * sizeof(n) - 1); // assumes width is a power of 2.
  10. // assert ( (c<=mask) &&"rotate by type width or more");
  11. c &= mask;
  12. return (n << c) | (n >> ((-c) & mask));
  13. }
  14. static inline uint32_t rotr(uint32_t n, unsigned int c)
  15. {
  16. const unsigned int mask = (CHAR_BIT * sizeof(n) - 1);
  17. // assert ( (c<=mask) &&"rotate by type width or more");
  18. c &= mask;
  19. return (n >> c) | (n << ((-c) & mask));
  20. }
  21. uint32_t Shannon::sbox1(uint32_t w)
  22. {
  23. w ^= rotl(w, 5) | rotl(w, 7);
  24. w ^= rotl(w, 19) | rotl(w, 22);
  25. return w;
  26. }
  27. uint32_t Shannon::sbox2(uint32_t w)
  28. {
  29. w ^= rotl(w, 7) | rotl(w, 22);
  30. w ^= rotl(w, 5) | rotl(w, 19);
  31. return w;
  32. }
  33. void Shannon::cycle()
  34. {
  35. uint32_t t;
  36. int i;
  37. /* nonlinear feedback function */
  38. t = this->R[12] ^ this->R[13] ^ this->konst;
  39. t = Shannon::sbox1(t) ^ rotl(this->R[0], 1);
  40. /* shift register */
  41. for (i = 1; i < N; ++i)
  42. this->R[i - 1] = this->R[i];
  43. this->R[N - 1] = t;
  44. t = Shannon::sbox2(this->R[2] ^ this->R[15]);
  45. this->R[0] ^= t;
  46. this->sbuf = t ^ this->R[8] ^ this->R[12];
  47. }
  48. void Shannon::crcfunc(uint32_t i)
  49. {
  50. uint32_t t;
  51. int j;
  52. /* Accumulate CRC of input */
  53. t = this->CRC[0] ^ this->CRC[2] ^ this->CRC[15] ^ i;
  54. for (j = 1; j < N; ++j)
  55. this->CRC[j - 1] = this->CRC[j];
  56. this->CRC[N - 1] = t;
  57. }
  58. void Shannon::macfunc(uint32_t i)
  59. {
  60. this->crcfunc(i);
  61. this->R[KEYP] ^= i;
  62. }
  63. void Shannon::initState()
  64. {
  65. int i;
  66. /* Register initialised to Fibonacci numbers; Counter zeroed. */
  67. this->R[0] = 1;
  68. this->R[1] = 1;
  69. for (i = 2; i < N; ++i)
  70. this->R[i] = this->R[i - 1] + this->R[i - 2];
  71. this->konst = Shannon::INITKONST;
  72. }
  73. void Shannon::saveState()
  74. {
  75. int i;
  76. for (i = 0; i < Shannon::N; ++i)
  77. this->initR[i] = this->R[i];
  78. }
  79. void Shannon::reloadState()
  80. {
  81. int i;
  82. for (i = 0; i < Shannon::N; ++i)
  83. this->R[i] = this->initR[i];
  84. }
  85. void Shannon::genkonst()
  86. {
  87. this->konst = this->R[0];
  88. }
  89. void Shannon::diffuse()
  90. {
  91. int i;
  92. for (i = 0; i < Shannon::FOLD; ++i)
  93. this->cycle();
  94. }
  95. #define Byte(x, i) ((uint32_t)(((x) >> (8 * (i))) & 0xFF))
  96. #define BYTE2WORD(b) ( \
  97. (((uint32_t)(b)[3] & 0xFF) << 24) | \
  98. (((uint32_t)(b)[2] & 0xFF) << 16) | \
  99. (((uint32_t)(b)[1] & 0xFF) << 8) | \
  100. (((uint32_t)(b)[0] & 0xFF)))
  101. #define WORD2BYTE(w, b) \
  102. { \
  103. (b)[3] = Byte(w, 3); \
  104. (b)[2] = Byte(w, 2); \
  105. (b)[1] = Byte(w, 1); \
  106. (b)[0] = Byte(w, 0); \
  107. }
  108. #define XORWORD(w, b) \
  109. { \
  110. (b)[3] ^= Byte(w, 3); \
  111. (b)[2] ^= Byte(w, 2); \
  112. (b)[1] ^= Byte(w, 1); \
  113. (b)[0] ^= Byte(w, 0); \
  114. }
  115. #define XORWORD(w, b) \
  116. { \
  117. (b)[3] ^= Byte(w, 3); \
  118. (b)[2] ^= Byte(w, 2); \
  119. (b)[1] ^= Byte(w, 1); \
  120. (b)[0] ^= Byte(w, 0); \
  121. }
  122. /* Load key material into the register
  123. */
  124. #define ADDKEY(k) \
  125. this->R[KEYP] ^= (k);
  126. void Shannon::loadKey(const std::vector<uint8_t> &key)
  127. {
  128. int i, j;
  129. uint32_t k;
  130. uint8_t xtra[4];
  131. size_t keylen = key.size();
  132. /* start folding in key */
  133. for (i = 0; i < (keylen & ~0x3); i += 4)
  134. {
  135. k = BYTE2WORD(&key[i]);
  136. ADDKEY(k);
  137. this->cycle();
  138. }
  139. /* if there were any extra key bytes, zero pad to a word */
  140. if (i < keylen)
  141. {
  142. for (j = 0 /* i unchanged */; i < keylen; ++i)
  143. xtra[j++] = key[i];
  144. for (/* j unchanged */; j < 4; ++j)
  145. xtra[j] = 0;
  146. k = BYTE2WORD(xtra);
  147. ADDKEY(k);
  148. this->cycle();
  149. }
  150. /* also fold in the length of the key */
  151. ADDKEY(keylen);
  152. this->cycle();
  153. /* save a copy of the register */
  154. for (i = 0; i < N; ++i)
  155. this->CRC[i] = this->R[i];
  156. /* now diffuse */
  157. this->diffuse();
  158. /* now xor the copy back -- makes key loading irreversible */
  159. for (i = 0; i < N; ++i)
  160. this->R[i] ^= this->CRC[i];
  161. }
  162. void Shannon::key(const std::vector<uint8_t> &key)
  163. {
  164. this->initState();
  165. this->loadKey(key);
  166. this->genkonst(); /* in case we proceed to stream generation */
  167. this->saveState();
  168. this->nbuf = 0;
  169. }
  170. void Shannon::nonce(const std::vector<uint8_t> &nonce)
  171. {
  172. this->reloadState();
  173. this->konst = Shannon::INITKONST;
  174. this->loadKey(nonce);
  175. this->genkonst();
  176. this->nbuf = 0;
  177. }
  178. void Shannon::stream(std::vector<uint8_t> &bufVec)
  179. {
  180. uint8_t *endbuf;
  181. size_t nbytes = bufVec.size();
  182. uint8_t *buf = bufVec.data();
  183. /* handle any previously buffered bytes */
  184. while (this->nbuf != 0 && nbytes != 0)
  185. {
  186. *buf++ ^= this->sbuf & 0xFF;
  187. this->sbuf >>= 8;
  188. this->nbuf -= 8;
  189. --nbytes;
  190. }
  191. /* handle whole words */
  192. endbuf = &buf[nbytes & ~((uint32_t)0x03)];
  193. while (buf < endbuf)
  194. {
  195. this->cycle();
  196. XORWORD(this->sbuf, buf);
  197. buf += 4;
  198. }
  199. /* handle any trailing bytes */
  200. nbytes &= 0x03;
  201. if (nbytes != 0)
  202. {
  203. this->cycle();
  204. this->nbuf = 32;
  205. while (this->nbuf != 0 && nbytes != 0)
  206. {
  207. *buf++ ^= this->sbuf & 0xFF;
  208. this->sbuf >>= 8;
  209. this->nbuf -= 8;
  210. --nbytes;
  211. }
  212. }
  213. }
  214. void Shannon::maconly(std::vector<uint8_t> &bufVec)
  215. {
  216. size_t nbytes = bufVec.size();
  217. uint8_t *buf = bufVec.data();
  218. uint8_t *endbuf;
  219. /* handle any previously buffered bytes */
  220. if (this->nbuf != 0)
  221. {
  222. while (this->nbuf != 0 && nbytes != 0)
  223. {
  224. this->mbuf ^= (*buf++) << (32 - this->nbuf);
  225. this->nbuf -= 8;
  226. --nbytes;
  227. }
  228. if (this->nbuf != 0) /* not a whole word yet */
  229. return;
  230. /* LFSR already cycled */
  231. this->macfunc(this->mbuf);
  232. }
  233. /* handle whole words */
  234. endbuf = &buf[nbytes & ~((uint32_t)0x03)];
  235. while (buf < endbuf)
  236. {
  237. this->cycle();
  238. this->macfunc(BYTE2WORD(buf));
  239. buf += 4;
  240. }
  241. /* handle any trailing bytes */
  242. nbytes &= 0x03;
  243. if (nbytes != 0)
  244. {
  245. this->cycle();
  246. this->mbuf = 0;
  247. this->nbuf = 32;
  248. while (this->nbuf != 0 && nbytes != 0)
  249. {
  250. this->mbuf ^= (*buf++) << (32 - this->nbuf);
  251. this->nbuf -= 8;
  252. --nbytes;
  253. }
  254. }
  255. }
  256. void Shannon::encrypt(std::vector<uint8_t> &bufVec)
  257. {
  258. size_t nbytes = bufVec.size();
  259. uint8_t *buf = bufVec.data();
  260. uint8_t *endbuf;
  261. uint32_t t = 0;
  262. /* handle any previously buffered bytes */
  263. if (this->nbuf != 0)
  264. {
  265. while (this->nbuf != 0 && nbytes != 0)
  266. {
  267. this->mbuf ^= *buf << (32 - this->nbuf);
  268. *buf ^= (this->sbuf >> (32 - this->nbuf)) & 0xFF;
  269. ++buf;
  270. this->nbuf -= 8;
  271. --nbytes;
  272. }
  273. if (this->nbuf != 0) /* not a whole word yet */
  274. return;
  275. /* LFSR already cycled */
  276. this->macfunc(this->mbuf);
  277. }
  278. /* handle whole words */
  279. endbuf = &buf[nbytes & ~((uint32_t)0x03)];
  280. while (buf < endbuf)
  281. {
  282. this->cycle();
  283. t = BYTE2WORD(buf);
  284. this->macfunc(t);
  285. t ^= this->sbuf;
  286. WORD2BYTE(t, buf);
  287. buf += 4;
  288. }
  289. /* handle any trailing bytes */
  290. nbytes &= 0x03;
  291. if (nbytes != 0)
  292. {
  293. this->cycle();
  294. this->mbuf = 0;
  295. this->nbuf = 32;
  296. while (this->nbuf != 0 && nbytes != 0)
  297. {
  298. this->mbuf ^= *buf << (32 - this->nbuf);
  299. *buf ^= (this->sbuf >> (32 - this->nbuf)) & 0xFF;
  300. ++buf;
  301. this->nbuf -= 8;
  302. --nbytes;
  303. }
  304. }
  305. }
  306. void Shannon::decrypt(std::vector<uint8_t> &bufVec)
  307. {
  308. size_t nbytes = bufVec.size();
  309. uint8_t *buf = bufVec.data();
  310. uint8_t *endbuf;
  311. uint32_t t = 0;
  312. /* handle any previously buffered bytes */
  313. if (this->nbuf != 0)
  314. {
  315. while (this->nbuf != 0 && nbytes != 0)
  316. {
  317. *buf ^= (this->sbuf >> (32 - this->nbuf)) & 0xFF;
  318. this->mbuf ^= *buf << (32 - this->nbuf);
  319. ++buf;
  320. this->nbuf -= 8;
  321. --nbytes;
  322. }
  323. if (this->nbuf != 0) /* not a whole word yet */
  324. return;
  325. /* LFSR already cycled */
  326. this->macfunc(this->mbuf);
  327. }
  328. /* handle whole words */
  329. endbuf = &buf[nbytes & ~((uint32_t)0x03)];
  330. while (buf < endbuf)
  331. {
  332. this->cycle();
  333. t = BYTE2WORD(buf) ^ this->sbuf;
  334. this->macfunc(t);
  335. WORD2BYTE(t, buf);
  336. buf += 4;
  337. }
  338. /* handle any trailing bytes */
  339. nbytes &= 0x03;
  340. if (nbytes != 0)
  341. {
  342. this->cycle();
  343. this->mbuf = 0;
  344. this->nbuf = 32;
  345. while (this->nbuf != 0 && nbytes != 0)
  346. {
  347. *buf ^= (this->sbuf >> (32 - this->nbuf)) & 0xFF;
  348. this->mbuf ^= *buf << (32 - this->nbuf);
  349. ++buf;
  350. this->nbuf -= 8;
  351. --nbytes;
  352. }
  353. }
  354. }
  355. void Shannon::finish(std::vector<uint8_t> &bufVec)
  356. {
  357. size_t nbytes = bufVec.size();
  358. uint8_t *buf = bufVec.data();
  359. int i;
  360. /* handle any previously buffered bytes */
  361. if (this->nbuf != 0)
  362. {
  363. /* LFSR already cycled */
  364. this->macfunc(this->mbuf);
  365. }
  366. /* perturb the MAC to mark end of input.
  367. * Note that only the stream register is updated, not the CRC. This is an
  368. * action that can't be duplicated by passing in plaintext, hence
  369. * defeating any kind of extension attack.
  370. */
  371. this->cycle();
  372. ADDKEY(INITKONST ^ (this->nbuf << 3));
  373. this->nbuf = 0;
  374. /* now add the CRC to the stream register and diffuse it */
  375. for (i = 0; i < N; ++i)
  376. this->R[i] ^= this->CRC[i];
  377. this->diffuse();
  378. /* produce output from the stream buffer */
  379. while (nbytes > 0)
  380. {
  381. this->cycle();
  382. if (nbytes >= 4)
  383. {
  384. WORD2BYTE(this->sbuf, buf);
  385. nbytes -= 4;
  386. buf += 4;
  387. }
  388. else
  389. {
  390. for (i = 0; i < nbytes; ++i)
  391. buf[i] = Byte(this->sbuf, i);
  392. break;
  393. }
  394. }
  395. }