Shannon.cpp 10 KB

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