4GAES.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414
  1. /**
  2. * author wulianwei
  3. * title AES加密功能
  4. */
  5. #define _CRT_SECURE_NO_WARNINGS
  6. #include "4GAES.h"
  7. #include <stdio.h>
  8. #include "string.h"
  9. /*aes_small.c*/
  10. //辅助矩阵
  11. /*s盒矩阵:The AES Substitution Table*/// 256 位的密匙256 位支持长度为32 个字符
  12. static const unsigned char sbox[256] = { //static:内部变量 const:只读,不可变常量
  13. 0x63,0x7c,0x77,0x7b,0xf2,0x6b,0x6f,0xc5,
  14. 0x30,0x01,0x67,0x2b,0xfe,0xd7,0xab,0x76,
  15. 0xca,0x82,0xc9,0x7d,0xfa,0x59,0x47,0xf0,
  16. 0xad,0xd4,0xa2,0xaf,0x9c,0xa4,0x72,0xc0,
  17. 0xb7,0xfd,0x93,0x26,0x36,0x3f,0xf7,0xcc,
  18. 0x34,0xa5,0xe5,0xf1,0x71,0xd8,0x31,0x15,
  19. 0x04,0xc7,0x23,0xc3,0x18,0x96,0x05,0x9a,
  20. 0x07,0x12,0x80,0xe2,0xeb,0x27,0xb2,0x75,
  21. 0x09,0x83,0x2c,0x1a,0x1b,0x6e,0x5a,0xa0,
  22. 0x52,0x3b,0xd6,0xb3,0x29,0xe3,0x2f,0x84,
  23. 0x53,0xd1,0x00,0xed,0x20,0xfc,0xb1,0x5b,
  24. 0x6a,0xcb,0xbe,0x39,0x4a,0x4c,0x58,0xcf,
  25. 0xd0,0xef,0xaa,0xfb,0x43,0x4d,0x33,0x85,
  26. 0x45,0xf9,0x02,0x7f,0x50,0x3c,0x9f,0xa8,
  27. 0x51,0xa3,0x40,0x8f,0x92,0x9d,0x38,0xf5,
  28. 0xbc,0xb6,0xda,0x21,0x10,0xff,0xf3,0xd2,
  29. 0xcd,0x0c,0x13,0xec,0x5f,0x97,0x44,0x17,
  30. 0xc4,0xa7,0x7e,0x3d,0x64,0x5d,0x19,0x73,
  31. 0x60,0x81,0x4f,0xdc,0x22,0x2a,0x90,0x88,
  32. 0x46,0xee,0xb8,0x14,0xde,0x5e,0x0b,0xdb,
  33. 0xe0,0x32,0x3a,0x0a,0x49,0x06,0x24,0x5c,
  34. 0xc2,0xd3,0xac,0x62,0x91,0x95,0xe4,0x79,
  35. 0xe7,0xc8,0x37,0x6d,0x8d,0xd5,0x4e,0xa9,
  36. 0x6c,0x56,0xf4,0xea,0x65,0x7a,0xae,0x08,
  37. 0xba,0x78,0x25,0x2e,0x1c,0xa6,0xb4,0xc6,
  38. 0xe8,0xdd,0x74,0x1f,0x4b,0xbd,0x8b,0x8a,
  39. 0x70,0x3e,0xb5,0x66,0x48,0x03,0xf6,0x0e,
  40. 0x61,0x35,0x57,0xb9,0x86,0xc1,0x1d,0x9e,
  41. 0xe1,0xf8,0x98,0x11,0x69,0xd9,0x8e,0x94,
  42. 0x9b,0x1e,0x87,0xe9,0xce,0x55,0x28,0xdf,
  43. 0x8c,0xa1,0x89,0x0d,0xbf,0xe6,0x42,0x68,
  44. 0x41,0x99,0x2d,0x0f,0xb0,0x54,0xbb,0x16,
  45. };
  46. //逆向S 盒矩阵
  47. static const unsigned char contrary_sbox[256] = {
  48. 0x52,0x09,0x6a,0xd5,0x30,0x36,0xa5,0x38,
  49. 0xbf,0x40,0xa3,0x9e,0x81,0xf3,0xd7,0xfb,
  50. 0x7c,0xe3,0x39,0x82,0x9b,0x2f,0xff,0x87,
  51. 0x34,0x8e,0x43,0x44,0xc4,0xde,0xe9,0xcb,
  52. 0x54,0x7b,0x94,0x32,0xa6,0xc2,0x23,0x3d,
  53. 0xee,0x4c,0x95,0x0b,0x42,0xfa,0xc3,0x4e,//0x4e
  54. 0x08,0x2e,0xa1,0x66,0x28,0xd9,0x24,0xb2,
  55. 0x76,0x5b,0xa2,0x49,0x6d,0x8b,0xd1,0x25,
  56. 0x72,0xf8,0xf6,0x64,0x86,0x68,0x98,0x16,
  57. 0xd4,0xa4,0x5c,0xcc,0x5d,0x65,0xb6,0x92,
  58. 0x6c,0x70,0x48,0x50,0xfd,0xed,0xb9,0xda,
  59. 0x5e,0x15,0x46,0x57,0xa7,0x8d,0x9d,0x84,
  60. 0x90,0xd8,0xab,0x00,0x8c,0xbc,0xd3,0x0a,
  61. 0xf7,0xe4,0x58,0x05,0xb8,0xb3,0x45,0x06,
  62. 0xd0,0x2c,0x1e,0x8f,0xca,0x3f,0x0f,0x02,
  63. 0xc1,0xaf,0xbd,0x03,0x01,0x13,0x8a,0x6b,
  64. 0x3a,0x91,0x11,0x41,0x4f,0x67,0xdc,0xea,
  65. 0x97,0xf2,0xcf,0xce,0xf0,0xb4,0xe6,0x73,
  66. 0x96,0xac,0x74,0x22,0xe7,0xad,0x35,0x85,
  67. 0xe2,0xf9,0x37,0xe8,0x1c,0x75,0xdf,0x6e,
  68. 0x47,0xf1,0x1a,0x71,0x1d,0x29,0xc5,0x89,
  69. 0x6f,0xb7,0x62,0x0e,0xaa,0x18,0xbe,0x1b,
  70. 0xfc,0x56,0x3e,0x4b,0xc6,0xd2,0x79,0x20,
  71. 0x9a,0xdb,0xc0,0xfe,0x78,0xcd,0x5a,0xf4,
  72. 0x1f,0xdd,0xa8,0x33,0x88,0x07,0xc7,0x31,
  73. 0xb1,0x12,0x10,0x59,0x27,0x80,0xec,0x5f,
  74. 0x60,0x51,0x7f,0xa9,0x19,0xb5,0x4a,0x0d,
  75. 0x2d,0xe5,0x7a,0x9f,0x93,0xc9,0x9c,0xef,
  76. 0xa0,0xe0,0x3b,0x4d,0xae,0x2a,0xf5,0xb0,
  77. 0xc8,0xeb,0xbb,0x3c,0x83,0x53,0x99,0x61,
  78. 0x17,0x2b,0x04,0x7e,0xba,0x77,0xd6,0x26,
  79. 0xe1,0x69,0x14,0x63,0x55,0x21,0x0c,0x7d,
  80. };
  81. /*轮常量表 The key schedule rcon table*/
  82. static const unsigned char Rcon[10] = {
  83. 0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80,0x1b,0x36 };
  84. //辅助函数
  85. /*有限域*2乘法 The x2time() function */
  86. static unsigned char x2time(unsigned char x)
  87. {
  88. if (x & 0x80)
  89. {
  90. return (((x << 1) ^ 0x1B) & 0xFF);
  91. }
  92. return x << 1;
  93. }
  94. /*有限域*3乘法 The x2time() function */
  95. static unsigned char x3time(unsigned char x)
  96. {
  97. return (x2time(x) ^ x);
  98. }
  99. /*有限域*4乘法 The x4time() function */
  100. static unsigned char x4time(unsigned char x)
  101. {
  102. return (x2time(x2time(x)));
  103. }
  104. /*有限域*8乘法 The x8time() function */
  105. static unsigned char x8time(unsigned char x)
  106. {
  107. return (x2time(x2time(x2time(x))));
  108. }
  109. /*有限域9乘法 The x9time() function */
  110. static unsigned char x9time(unsigned char x) //9:1001
  111. {
  112. return (x8time(x) ^ x);
  113. }
  114. /*有限域*B乘法 The xBtime() function */
  115. static unsigned char xBtime(unsigned char x) //B:1011
  116. {
  117. return (x8time(x) ^ x2time(x) ^ x);
  118. }
  119. /*有限域*D乘法 The xDtime() function */
  120. static unsigned char xDtime(unsigned char x) //D:1101
  121. {
  122. return (x8time(x) ^ x4time(x) ^ x);
  123. }
  124. /*有限域*E乘法 The xEtime() function */
  125. static unsigned char xEtime(unsigned char x) //E:1110
  126. {
  127. return (x8time(x) ^ x4time(x) ^ x2time(x));
  128. }
  129. /****************************************************************************************************************/
  130. /*第三类操作:列混合操作 MixColumns: Process the entire block*/
  131. static void MixColumns(unsigned char *col)//列混合
  132. {
  133. unsigned char tmp[4];
  134. int i;
  135. for (i = 0; i < 4; i++, col += 4) //col代表一列的基地址,col+4:下一列的基地址
  136. {
  137. /*
  138. xt[0]=x2time(col[0]);
  139. xt[1]=x2time(col[1]);
  140. xt[2]=x2time(col[2]);
  141. xt[3]=x2time(col[3]);
  142. //xt[n]代表*2 xt[n]^col[n]代表*3 col[n]代表*1
  143. tmp[0]=(xt[0])^(xt[1]^col[1])^col[2]^col[3]; //2 3 1 1
  144. tmp[1]=col[0]^(xt[1])^(xt[2]^col[2])^col[3]; //1 2 3 1
  145. tmp[2]=col[0]^col[1]^(xt[2])^(xt[3]^col[3]); //1 1 2 3
  146. tmp[3]=(xt[0]^col[0])^col[1]^col[2]^(xt[3]); //3 1 1 2
  147. */
  148. tmp[0] = x2time(col[0]) ^ x3time(col[1]) ^ col[2] ^ col[3]; //2 3 1 1
  149. tmp[1] = col[0] ^ x2time(col[1]) ^ x3time(col[2]) ^ col[3]; //1 2 3 1
  150. tmp[2] = col[0] ^ col[1] ^ x2time(col[2]) ^ x3time(col[3]); //1 1 2 3
  151. tmp[3] = x3time(col[0]) ^ col[1] ^ col[2] ^ x2time(col[3]); //3 1 1 2
  152. //修改后的值 直接在原矩阵上修改
  153. col[0] = tmp[0];
  154. col[1] = tmp[1];
  155. col[2] = tmp[2];
  156. col[3] = tmp[3];
  157. }
  158. }
  159. //逆向列混淆
  160. static void Contrary_MixColumns(unsigned char *col)
  161. {
  162. unsigned char tmp[4];
  163. int x;
  164. for (x = 0; x < 4; x++, col += 4)
  165. {
  166. /*
  167. xt2[0]=x2time(col[0]);
  168. xt2[1]=x2time(col[1]);
  169. xt2[2]=x2time(col[2]);
  170. xt2[3]=x2time(col[3]);
  171. xt4[0]=x2time(xt2[0]);
  172. xt4[1]=x2time(xt2[1]);
  173. xt4[2]=x2time(xt2[2]);
  174. xt4[3]=x2time(xt2[3]);
  175. xt8[0]=x2time(xt4[0]);
  176. xt8[1]=x2time(xt4[1]);
  177. xt8[2]=x2time(xt4[2]);
  178. xt8[3]=x2time(xt4[3]);
  179. tmp[0]=xt8[0]^xt4[0]^xt2[0]^xt8[1]^xt2[1]^col[1]^xt8[2]^xt4[2]^col[2]^xt8[3]^col[3];
  180. tmp[1]=xt8[0]^col[0]^xt8[1]^xt4[1]^xt2[1]^xt8[2]^xt2[2]^col[2]^xt8[3]^xt4[3]^col[3];
  181. tmp[2]=xt8[0]^xt4[0]^col[0]^xt8[1]^col[1]^xt8[2]^xt4[2]^xt2[2]^xt8[3]^xt2[3]^col[3];
  182. tmp[3]=xt8[0]^xt2[0]^col[0]^xt8[1]^xt4[1]^col[1]^xt8[2]^col[2]^xt8[3]^xt4[3]^xt2[3];
  183. */
  184. tmp[0] = xEtime(col[0]) ^ xBtime(col[1]) ^ xDtime(col[2]) ^ x9time(col[3]);
  185. tmp[1] = x9time(col[0]) ^ xEtime(col[1]) ^ xBtime(col[2]) ^ xDtime(col[3]);
  186. tmp[2] = xDtime(col[0]) ^ x9time(col[1]) ^ xEtime(col[2]) ^ xBtime(col[3]);
  187. tmp[3] = xBtime(col[0]) ^ xDtime(col[1]) ^ x9time(col[2]) ^ xEtime(col[3]);
  188. col[0] = tmp[0];
  189. col[1] = tmp[1];
  190. col[2] = tmp[2];
  191. col[3] = tmp[3];
  192. }
  193. }
  194. /*第二类操作:行移位:行左循环移位 ShiftRows:Shifts the entire block*/
  195. static void ShiftRows(unsigned char *col)//正向行移位
  196. {
  197. /*
  198. 1 5 9 13 5 9 13 1
  199. 2 6 10 14 10 14 2 6
  200. 3 7 11 15 15 3 7 11
  201. 4 8 12 16 16 4 8 12
  202. */
  203. unsigned char t;
  204. /*1nd row*///左移1位
  205. t = col[1]; col[1] = col[5]; col[5] = col[9]; col[9] = col[13]; col[13] = t;
  206. /*2rd row*///左移2位,交换2次数字来实现
  207. t = col[2]; col[2] = col[10]; col[10] = t;
  208. t = col[6]; col[6] = col[14]; col[14] = t;
  209. /*3th row*///左移3位,相当于右移1次
  210. t = col[15]; col[15] = col[11]; col[11] = col[7]; col[7] = col[3]; col[3] = t;
  211. /*4th row*/ //第4行不移位
  212. }
  213. //逆向行移位
  214. static void Contrary_ShiftRows(unsigned char *col)
  215. {
  216. unsigned char t;
  217. /*1nd row*/
  218. t = col[13]; col[13] = col[9]; col[9] = col[5]; col[5] = col[1]; col[1] = t;
  219. /*2rd row*/
  220. t = col[2]; col[2] = col[10]; col[10] = t;
  221. t = col[6]; col[6] = col[14]; col[14] = t;
  222. /*3th row*/
  223. t = col[3]; col[3] = col[7]; col[7] = col[11]; col[11] = col[15]; col[15] = t;
  224. /*4th row*/ //第4行不移位
  225. }
  226. /*第一类操作:s盒字节代换替换 SubBytes*/
  227. static void SubBytes(unsigned char *col)//字节代换
  228. {
  229. int x;
  230. for (x = 0; x < 16; x++)
  231. {
  232. col[x] = sbox[col[x]];
  233. }
  234. }
  235. //逆向字节代换
  236. static void Contrary_SubBytes(unsigned char *col)
  237. {
  238. int x;
  239. for (x = 0; x < 16; x++)
  240. {
  241. col[x] = contrary_sbox[col[x]];
  242. }
  243. }
  244. /*第四类操作:轮密钥加 AddRoundKey*/
  245. static void AddRoundKey(unsigned char *col, unsigned char *expansionkey, int round)//密匙加
  246. {
  247. //扩展密钥:44*32bit =11*4* 4*8 = 16字节*11轮,每轮用16字节密钥
  248. //第0轮,只进行一次轮密钥加
  249. //第1-10轮,轮密钥加
  250. int x;
  251. for (x = 0; x < 16; x++) //每1轮操作:4*32bit密钥 = 16个字节密钥
  252. {
  253. col[x] ^= expansionkey[(round << 4) + x];
  254. }
  255. }
  256. /* AES加密总函数 10轮4类操作 Encrypt a single block with Nr Rounds(10,12,14)*/
  257. void AesEncrypt(unsigned char *blk, unsigned char *expansionkey, int Nr)//加密一个区块
  258. {
  259. //输入blk原文,直接在上面修改,输出blk密文
  260. //输入skey:
  261. //输入Nr = 10轮
  262. int round;
  263. //第1轮之前:轮密钥加
  264. AddRoundKey(blk, expansionkey, 0);
  265. //第1-9轮:4类操作:字节代换、行移位、列混合、轮密钥加
  266. for (round = 1; round <= (Nr - 1); round++)
  267. {
  268. SubBytes(blk); //输入16字节数组,直接在原数组上修改
  269. ShiftRows(blk); //输入16字节数组,直接在原数组上修改
  270. MixColumns(blk); //输入16字节数组,直接在原数组上修改
  271. AddRoundKey(blk, expansionkey, round);
  272. }
  273. //第10轮:不进行列混合
  274. SubBytes(blk);
  275. ShiftRows(blk);
  276. AddRoundKey(blk, expansionkey, Nr);
  277. }
  278. //AES 解密总函数
  279. void AesDecrypt(unsigned char *blk, unsigned char *expansionkey, int Nr)
  280. {
  281. int x;
  282. /* unsigned char *contrary_key=key;
  283. for(x=0;x<11;x++,key+=16)
  284. Contrary_MixColumns(key);*/
  285. AddRoundKey(blk, expansionkey, Nr);
  286. Contrary_ShiftRows(blk);
  287. Contrary_SubBytes(blk);
  288. for (x = (Nr - 1); x >= 1; x--)
  289. {
  290. AddRoundKey(blk, expansionkey, x);
  291. Contrary_MixColumns(blk);
  292. Contrary_ShiftRows(blk);
  293. Contrary_SubBytes(blk);
  294. }
  295. AddRoundKey(blk, expansionkey, 0);
  296. }
  297. /*//密钥编排,16字节--->44列32bit密钥生成--> 11组16字节:分别用于11轮 轮密钥加运算
  298. Schedule a secret key for use.
  299. *outkey[] must be 16*15 bytes in size
  300. *Nk==number of 32 bit words in the key,e.g.,4,6,8
  301. *Nr==number of rounds,e.g.,10,12,14
  302. */
  303. void ScheduleKey(unsigned char *inkey, unsigned char *outkey, int Nk, int Nr)//安排一个保密密钥使用
  304. {
  305. //inkey:初始16字节密钥key
  306. //outkey:11组*16字节扩展密钥expansionkey
  307. //Nk:4列
  308. //Nr:10轮round
  309. unsigned char temp[4], t;
  310. int x, i;
  311. /*copy the key*/
  312. //第0组:[0-3]直接拷贝
  313. for (i = 0; i < (4 * Nk); i++)
  314. {
  315. outkey[i] = inkey[i];
  316. }
  317. //第1-10组:[4-43]
  318. i = Nk;
  319. while (i < (4 * (Nr + 1))) //i=4~43 WORD 32bit的首字节地址,每一个4字节
  320. {//1次循环生成1个字节扩展密钥,4次循环生成一个WORD
  321. //temp:4字节数组:代表一个WORD密钥
  322. /*temp=w[i-1]*/
  323. //i不是4的倍数的时候
  324. //每个temp = 每个outkey32bit = 4字节
  325. for (x = 0; x < 4; x++)
  326. temp[x] = outkey[(4 * (i - 1)) + x]; //i:32bit的首字节地址
  327. //i是4的倍数的时候
  328. if (i%Nk == 0)
  329. {
  330. /*字循环:循环左移1字节 RotWord()*/
  331. t = temp[0]; temp[0] = temp[1]; temp[1] = temp[2]; temp[2] = temp[3]; temp[3] = t;
  332. /*字节代换:SubWord()*/
  333. for (x = 0; x < 4; x++)
  334. {
  335. temp[x] = sbox[temp[x]];
  336. }
  337. /*轮常量异或:Rcon[j]*/
  338. temp[0] ^= Rcon[(i / Nk) - 1];
  339. }
  340. //else if(Nk>6 && (i%Nk)==4) //Nk>6的算法不同,暂时用不到
  341. //{
  342. // /*SubWord*/
  343. // for(x=0;x<4;x++)
  344. // {
  345. // temp[x]=sbox[temp[x]];
  346. // }
  347. //}
  348. /*w[i] = w[i-4]^w[i-1]*/
  349. for (x = 0; x < 4; x++)
  350. {
  351. outkey[(4 * i) + x] = outkey[(4 * (i - Nk)) + x] ^ temp[x];
  352. }
  353. ++i;
  354. }
  355. }
  356. /**
  357. * ECB 模式加密, 返回加密长度
  358. */
  359. int ECBAesEncrypt(unsigned char *blk, unsigned char *expansionkey, int Nr)
  360. {
  361. int k=0;
  362. int len =fillSrcData((char*)blk);
  363. for(k=0;k<len/AES_BLOCK_SIZE;k++)
  364. {
  365. AesEncrypt(blk+16*k, expansionkey, Nr); //2、AES 加密
  366. }
  367. return len;
  368. }
  369. /**
  370. * ECB 模式解密
  371. */
  372. void ECBAesDecrypt(unsigned char *blk,int len, unsigned char *expansionkey, int Nr)
  373. {
  374. int k=0;
  375. for(k=0;k<len/AES_BLOCK_SIZE;k++)
  376. {
  377. AesDecrypt(blk+16*k, expansionkey, Nr); //2、AES 解密
  378. }
  379. cutSrcData((char*)blk);
  380. }
  381. /**
  382. * 填充源码,返回填充后的数据长度
  383. */
  384. int fillSrcData(char* data)
  385. {
  386. int left= 0;
  387. int len = strlen(data);
  388. if(len%AES_BLOCK_SIZE != 0)
  389. {
  390. left = AES_BLOCK_SIZE-strlen(data)%AES_BLOCK_SIZE;
  391. memset(data+strlen(data),left,left);
  392. len+=left;
  393. }
  394. return len;
  395. }
  396. /**
  397. * 去除源码无效数据
  398. */
  399. void cutSrcData(char* data)
  400. {
  401. int i= 0;
  402. int size = strlen(data);
  403. for(i=size-1;data[i]>0&&data[i]<AES_BLOCK_SIZE;i--)
  404. {
  405. data[i]=0;
  406. }
  407. }