選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。
 
 
 
 
 
 

759 行
21 KiB

  1. #!/usr/bin/env perl
  2. #
  3. # ====================================================================
  4. # Written by Andy Polyakov <appro@openssl.org> for the OpenSSL
  5. # project. The module is, however, dual licensed under OpenSSL and
  6. # CRYPTOGAMS licenses depending on where you obtain it. For further
  7. # details see http://www.openssl.org/~appro/cryptogams/.
  8. # ====================================================================
  9. #
  10. # January 2015
  11. #
  12. # ChaCha20 for x86.
  13. #
  14. # Performance in cycles per byte out of large buffer.
  15. #
  16. # 1xIALU/gcc 4xSSSE3
  17. # Pentium 17.5/+80%
  18. # PIII 14.2/+60%
  19. # P4 18.6/+84%
  20. # Core2 9.56/+89% 4.83
  21. # Westmere 9.50/+45% 3.35
  22. # Sandy Bridge 10.5/+47% 3.20
  23. # Haswell 8.15/+50% 2.83
  24. # Silvermont 17.4/+36% 8.35
  25. # Sledgehammer 10.2/+54%
  26. # Bulldozer 13.4/+50% 4.38(*)
  27. #
  28. # (*) Bulldozer actually executes 4xXOP code path that delivers 3.55;
  29. #
  30. # Modified from upstream OpenSSL to remove the XOP code.
  31. $0 =~ m/(.*[\/\\])[^\/\\]+$/; $dir=$1;
  32. push(@INC,"${dir}","${dir}../../perlasm");
  33. require "x86asm.pl";
  34. $output=pop;
  35. open STDOUT,">$output";
  36. &asm_init($ARGV[0],"chacha-x86.pl",$ARGV[$#ARGV] eq "386");
  37. $xmm=$ymm=0;
  38. for (@ARGV) { $xmm=1 if (/-DOPENSSL_IA32_SSE2/); }
  39. $ymm=$xmm;
  40. $a="eax";
  41. ($b,$b_)=("ebx","ebp");
  42. ($c,$c_)=("ecx","esi");
  43. ($d,$d_)=("edx","edi");
  44. sub QUARTERROUND {
  45. my ($ai,$bi,$ci,$di,$i)=@_;
  46. my ($an,$bn,$cn,$dn)=map(($_&~3)+(($_+1)&3),($ai,$bi,$ci,$di)); # next
  47. my ($ap,$bp,$cp,$dp)=map(($_&~3)+(($_-1)&3),($ai,$bi,$ci,$di)); # previous
  48. # a b c d
  49. #
  50. # 0 4 8 12 < even round
  51. # 1 5 9 13
  52. # 2 6 10 14
  53. # 3 7 11 15
  54. # 0 5 10 15 < odd round
  55. # 1 6 11 12
  56. # 2 7 8 13
  57. # 3 4 9 14
  58. if ($i==0) {
  59. my $j=4;
  60. ($ap,$bp,$cp,$dp)=map(($_&~3)+(($_-$j--)&3),($ap,$bp,$cp,$dp));
  61. } elsif ($i==3) {
  62. my $j=0;
  63. ($an,$bn,$cn,$dn)=map(($_&~3)+(($_+$j++)&3),($an,$bn,$cn,$dn));
  64. } elsif ($i==4) {
  65. my $j=4;
  66. ($ap,$bp,$cp,$dp)=map(($_&~3)+(($_+$j--)&3),($ap,$bp,$cp,$dp));
  67. } elsif ($i==7) {
  68. my $j=0;
  69. ($an,$bn,$cn,$dn)=map(($_&~3)+(($_-$j++)&3),($an,$bn,$cn,$dn));
  70. }
  71. #&add ($a,$b); # see elsewhere
  72. &xor ($d,$a);
  73. &mov (&DWP(4*$cp,"esp"),$c_) if ($ai>0 && $ai<3);
  74. &rol ($d,16);
  75. &mov (&DWP(4*$bp,"esp"),$b_) if ($i!=0);
  76. &add ($c,$d);
  77. &mov ($c_,&DWP(4*$cn,"esp")) if ($ai>0 && $ai<3);
  78. &xor ($b,$c);
  79. &mov ($d_,&DWP(4*$dn,"esp")) if ($di!=$dn);
  80. &rol ($b,12);
  81. &mov ($b_,&DWP(4*$bn,"esp")) if ($i<7);
  82. &mov ($b_,&DWP(128,"esp")) if ($i==7); # loop counter
  83. &add ($a,$b);
  84. &xor ($d,$a);
  85. &mov (&DWP(4*$ai,"esp"),$a);
  86. &rol ($d,8);
  87. &mov ($a,&DWP(4*$an,"esp"));
  88. &add ($c,$d);
  89. &mov (&DWP(4*$di,"esp"),$d) if ($di!=$dn);
  90. &mov ($d_,$d) if ($di==$dn);
  91. &xor ($b,$c);
  92. &add ($a,$b_) if ($i<7); # elsewhere
  93. &rol ($b,7);
  94. ($b,$b_)=($b_,$b);
  95. ($c,$c_)=($c_,$c);
  96. ($d,$d_)=($d_,$d);
  97. }
  98. &static_label("ssse3_shortcut");
  99. &static_label("ssse3_data");
  100. &static_label("pic_point");
  101. &function_begin("ChaCha20_ctr32");
  102. &xor ("eax","eax");
  103. &cmp ("eax",&wparam(2)); # len==0?
  104. &je (&label("no_data"));
  105. if ($xmm) {
  106. &call (&label("pic_point"));
  107. &set_label("pic_point");
  108. &blindpop("eax");
  109. &picmeup("ebp","OPENSSL_ia32cap_P","eax",&label("pic_point"));
  110. &test (&DWP(0,"ebp"),1<<24); # test FXSR bit
  111. &jz (&label("x86"));
  112. &test (&DWP(4,"ebp"),1<<9); # test SSSE3 bit
  113. &jz (&label("x86"));
  114. &jmp (&label("ssse3_shortcut"));
  115. &set_label("x86");
  116. }
  117. &mov ("esi",&wparam(3)); # key
  118. &mov ("edi",&wparam(4)); # counter and nonce
  119. &stack_push(33);
  120. &mov ("eax",&DWP(4*0,"esi")); # copy key
  121. &mov ("ebx",&DWP(4*1,"esi"));
  122. &mov ("ecx",&DWP(4*2,"esi"));
  123. &mov ("edx",&DWP(4*3,"esi"));
  124. &mov (&DWP(64+4*4,"esp"),"eax");
  125. &mov (&DWP(64+4*5,"esp"),"ebx");
  126. &mov (&DWP(64+4*6,"esp"),"ecx");
  127. &mov (&DWP(64+4*7,"esp"),"edx");
  128. &mov ("eax",&DWP(4*4,"esi"));
  129. &mov ("ebx",&DWP(4*5,"esi"));
  130. &mov ("ecx",&DWP(4*6,"esi"));
  131. &mov ("edx",&DWP(4*7,"esi"));
  132. &mov (&DWP(64+4*8,"esp"),"eax");
  133. &mov (&DWP(64+4*9,"esp"),"ebx");
  134. &mov (&DWP(64+4*10,"esp"),"ecx");
  135. &mov (&DWP(64+4*11,"esp"),"edx");
  136. &mov ("eax",&DWP(4*0,"edi")); # copy counter and nonce
  137. &mov ("ebx",&DWP(4*1,"edi"));
  138. &mov ("ecx",&DWP(4*2,"edi"));
  139. &mov ("edx",&DWP(4*3,"edi"));
  140. &sub ("eax",1);
  141. &mov (&DWP(64+4*12,"esp"),"eax");
  142. &mov (&DWP(64+4*13,"esp"),"ebx");
  143. &mov (&DWP(64+4*14,"esp"),"ecx");
  144. &mov (&DWP(64+4*15,"esp"),"edx");
  145. &jmp (&label("entry"));
  146. &set_label("outer_loop",16);
  147. &mov (&wparam(1),$b); # save input
  148. &mov (&wparam(0),$a); # save output
  149. &mov (&wparam(2),$c); # save len
  150. &set_label("entry");
  151. &mov ($a,0x61707865);
  152. &mov (&DWP(4*1,"esp"),0x3320646e);
  153. &mov (&DWP(4*2,"esp"),0x79622d32);
  154. &mov (&DWP(4*3,"esp"),0x6b206574);
  155. &mov ($b, &DWP(64+4*5,"esp")); # copy key material
  156. &mov ($b_,&DWP(64+4*6,"esp"));
  157. &mov ($c, &DWP(64+4*10,"esp"));
  158. &mov ($c_,&DWP(64+4*11,"esp"));
  159. &mov ($d, &DWP(64+4*13,"esp"));
  160. &mov ($d_,&DWP(64+4*14,"esp"));
  161. &mov (&DWP(4*5,"esp"),$b);
  162. &mov (&DWP(4*6,"esp"),$b_);
  163. &mov (&DWP(4*10,"esp"),$c);
  164. &mov (&DWP(4*11,"esp"),$c_);
  165. &mov (&DWP(4*13,"esp"),$d);
  166. &mov (&DWP(4*14,"esp"),$d_);
  167. &mov ($b, &DWP(64+4*7,"esp"));
  168. &mov ($d_,&DWP(64+4*15,"esp"));
  169. &mov ($d, &DWP(64+4*12,"esp"));
  170. &mov ($b_,&DWP(64+4*4,"esp"));
  171. &mov ($c, &DWP(64+4*8,"esp"));
  172. &mov ($c_,&DWP(64+4*9,"esp"));
  173. &add ($d,1); # counter value
  174. &mov (&DWP(4*7,"esp"),$b);
  175. &mov (&DWP(4*15,"esp"),$d_);
  176. &mov (&DWP(64+4*12,"esp"),$d); # save counter value
  177. &mov ($b,10); # loop counter
  178. &jmp (&label("loop"));
  179. &set_label("loop",16);
  180. &add ($a,$b_); # elsewhere
  181. &mov (&DWP(128,"esp"),$b); # save loop counter
  182. &mov ($b,$b_);
  183. &QUARTERROUND(0, 4, 8, 12, 0);
  184. &QUARTERROUND(1, 5, 9, 13, 1);
  185. &QUARTERROUND(2, 6,10, 14, 2);
  186. &QUARTERROUND(3, 7,11, 15, 3);
  187. &QUARTERROUND(0, 5,10, 15, 4);
  188. &QUARTERROUND(1, 6,11, 12, 5);
  189. &QUARTERROUND(2, 7, 8, 13, 6);
  190. &QUARTERROUND(3, 4, 9, 14, 7);
  191. &dec ($b);
  192. &jnz (&label("loop"));
  193. &mov ($b,&wparam(2)); # load len
  194. &add ($a,0x61707865); # accumulate key material
  195. &add ($b_,&DWP(64+4*4,"esp"));
  196. &add ($c, &DWP(64+4*8,"esp"));
  197. &add ($c_,&DWP(64+4*9,"esp"));
  198. &cmp ($b,64);
  199. &jb (&label("tail"));
  200. &mov ($b,&wparam(1)); # load input pointer
  201. &add ($d, &DWP(64+4*12,"esp"));
  202. &add ($d_,&DWP(64+4*14,"esp"));
  203. &xor ($a, &DWP(4*0,$b)); # xor with input
  204. &xor ($b_,&DWP(4*4,$b));
  205. &mov (&DWP(4*0,"esp"),$a);
  206. &mov ($a,&wparam(0)); # load output pointer
  207. &xor ($c, &DWP(4*8,$b));
  208. &xor ($c_,&DWP(4*9,$b));
  209. &xor ($d, &DWP(4*12,$b));
  210. &xor ($d_,&DWP(4*14,$b));
  211. &mov (&DWP(4*4,$a),$b_); # write output
  212. &mov (&DWP(4*8,$a),$c);
  213. &mov (&DWP(4*9,$a),$c_);
  214. &mov (&DWP(4*12,$a),$d);
  215. &mov (&DWP(4*14,$a),$d_);
  216. &mov ($b_,&DWP(4*1,"esp"));
  217. &mov ($c, &DWP(4*2,"esp"));
  218. &mov ($c_,&DWP(4*3,"esp"));
  219. &mov ($d, &DWP(4*5,"esp"));
  220. &mov ($d_,&DWP(4*6,"esp"));
  221. &add ($b_,0x3320646e); # accumulate key material
  222. &add ($c, 0x79622d32);
  223. &add ($c_,0x6b206574);
  224. &add ($d, &DWP(64+4*5,"esp"));
  225. &add ($d_,&DWP(64+4*6,"esp"));
  226. &xor ($b_,&DWP(4*1,$b));
  227. &xor ($c, &DWP(4*2,$b));
  228. &xor ($c_,&DWP(4*3,$b));
  229. &xor ($d, &DWP(4*5,$b));
  230. &xor ($d_,&DWP(4*6,$b));
  231. &mov (&DWP(4*1,$a),$b_);
  232. &mov (&DWP(4*2,$a),$c);
  233. &mov (&DWP(4*3,$a),$c_);
  234. &mov (&DWP(4*5,$a),$d);
  235. &mov (&DWP(4*6,$a),$d_);
  236. &mov ($b_,&DWP(4*7,"esp"));
  237. &mov ($c, &DWP(4*10,"esp"));
  238. &mov ($c_,&DWP(4*11,"esp"));
  239. &mov ($d, &DWP(4*13,"esp"));
  240. &mov ($d_,&DWP(4*15,"esp"));
  241. &add ($b_,&DWP(64+4*7,"esp"));
  242. &add ($c, &DWP(64+4*10,"esp"));
  243. &add ($c_,&DWP(64+4*11,"esp"));
  244. &add ($d, &DWP(64+4*13,"esp"));
  245. &add ($d_,&DWP(64+4*15,"esp"));
  246. &xor ($b_,&DWP(4*7,$b));
  247. &xor ($c, &DWP(4*10,$b));
  248. &xor ($c_,&DWP(4*11,$b));
  249. &xor ($d, &DWP(4*13,$b));
  250. &xor ($d_,&DWP(4*15,$b));
  251. &lea ($b,&DWP(4*16,$b));
  252. &mov (&DWP(4*7,$a),$b_);
  253. &mov ($b_,&DWP(4*0,"esp"));
  254. &mov (&DWP(4*10,$a),$c);
  255. &mov ($c,&wparam(2)); # len
  256. &mov (&DWP(4*11,$a),$c_);
  257. &mov (&DWP(4*13,$a),$d);
  258. &mov (&DWP(4*15,$a),$d_);
  259. &mov (&DWP(4*0,$a),$b_);
  260. &lea ($a,&DWP(4*16,$a));
  261. &sub ($c,64);
  262. &jnz (&label("outer_loop"));
  263. &jmp (&label("done"));
  264. &set_label("tail");
  265. &add ($d, &DWP(64+4*12,"esp"));
  266. &add ($d_,&DWP(64+4*14,"esp"));
  267. &mov (&DWP(4*0,"esp"),$a);
  268. &mov (&DWP(4*4,"esp"),$b_);
  269. &mov (&DWP(4*8,"esp"),$c);
  270. &mov (&DWP(4*9,"esp"),$c_);
  271. &mov (&DWP(4*12,"esp"),$d);
  272. &mov (&DWP(4*14,"esp"),$d_);
  273. &mov ($b_,&DWP(4*1,"esp"));
  274. &mov ($c, &DWP(4*2,"esp"));
  275. &mov ($c_,&DWP(4*3,"esp"));
  276. &mov ($d, &DWP(4*5,"esp"));
  277. &mov ($d_,&DWP(4*6,"esp"));
  278. &add ($b_,0x3320646e); # accumulate key material
  279. &add ($c, 0x79622d32);
  280. &add ($c_,0x6b206574);
  281. &add ($d, &DWP(64+4*5,"esp"));
  282. &add ($d_,&DWP(64+4*6,"esp"));
  283. &mov (&DWP(4*1,"esp"),$b_);
  284. &mov (&DWP(4*2,"esp"),$c);
  285. &mov (&DWP(4*3,"esp"),$c_);
  286. &mov (&DWP(4*5,"esp"),$d);
  287. &mov (&DWP(4*6,"esp"),$d_);
  288. &mov ($b_,&DWP(4*7,"esp"));
  289. &mov ($c, &DWP(4*10,"esp"));
  290. &mov ($c_,&DWP(4*11,"esp"));
  291. &mov ($d, &DWP(4*13,"esp"));
  292. &mov ($d_,&DWP(4*15,"esp"));
  293. &add ($b_,&DWP(64+4*7,"esp"));
  294. &add ($c, &DWP(64+4*10,"esp"));
  295. &add ($c_,&DWP(64+4*11,"esp"));
  296. &add ($d, &DWP(64+4*13,"esp"));
  297. &add ($d_,&DWP(64+4*15,"esp"));
  298. &mov (&DWP(4*7,"esp"),$b_);
  299. &mov ($b_,&wparam(1)); # load input
  300. &mov (&DWP(4*10,"esp"),$c);
  301. &mov ($c,&wparam(0)); # load output
  302. &mov (&DWP(4*11,"esp"),$c_);
  303. &xor ($c_,$c_);
  304. &mov (&DWP(4*13,"esp"),$d);
  305. &mov (&DWP(4*15,"esp"),$d_);
  306. &xor ("eax","eax");
  307. &xor ("edx","edx");
  308. &set_label("tail_loop");
  309. &movb ("al",&BP(0,$c_,$b_));
  310. &movb ("dl",&BP(0,"esp",$c_));
  311. &lea ($c_,&DWP(1,$c_));
  312. &xor ("al","dl");
  313. &mov (&BP(-1,$c,$c_),"al");
  314. &dec ($b);
  315. &jnz (&label("tail_loop"));
  316. &set_label("done");
  317. &stack_pop(33);
  318. &set_label("no_data");
  319. &function_end("ChaCha20_ctr32");
  320. if ($xmm) {
  321. my ($xa,$xa_,$xb,$xb_,$xc,$xc_,$xd,$xd_)=map("xmm$_",(0..7));
  322. my ($out,$inp,$len)=("edi","esi","ecx");
  323. sub QUARTERROUND_SSSE3 {
  324. my ($ai,$bi,$ci,$di,$i)=@_;
  325. my ($an,$bn,$cn,$dn)=map(($_&~3)+(($_+1)&3),($ai,$bi,$ci,$di)); # next
  326. my ($ap,$bp,$cp,$dp)=map(($_&~3)+(($_-1)&3),($ai,$bi,$ci,$di)); # previous
  327. # a b c d
  328. #
  329. # 0 4 8 12 < even round
  330. # 1 5 9 13
  331. # 2 6 10 14
  332. # 3 7 11 15
  333. # 0 5 10 15 < odd round
  334. # 1 6 11 12
  335. # 2 7 8 13
  336. # 3 4 9 14
  337. if ($i==0) {
  338. my $j=4;
  339. ($ap,$bp,$cp,$dp)=map(($_&~3)+(($_-$j--)&3),($ap,$bp,$cp,$dp));
  340. } elsif ($i==3) {
  341. my $j=0;
  342. ($an,$bn,$cn,$dn)=map(($_&~3)+(($_+$j++)&3),($an,$bn,$cn,$dn));
  343. } elsif ($i==4) {
  344. my $j=4;
  345. ($ap,$bp,$cp,$dp)=map(($_&~3)+(($_+$j--)&3),($ap,$bp,$cp,$dp));
  346. } elsif ($i==7) {
  347. my $j=0;
  348. ($an,$bn,$cn,$dn)=map(($_&~3)+(($_-$j++)&3),($an,$bn,$cn,$dn));
  349. }
  350. #&paddd ($xa,$xb); # see elsewhere
  351. #&pxor ($xd,$xa); # see elsewhere
  352. &movdqa(&QWP(16*$cp-128,"ebx"),$xc_) if ($ai>0 && $ai<3);
  353. &pshufb ($xd,&QWP(0,"eax")); # rot16
  354. &movdqa(&QWP(16*$bp-128,"ebx"),$xb_) if ($i!=0);
  355. &paddd ($xc,$xd);
  356. &movdqa($xc_,&QWP(16*$cn-128,"ebx")) if ($ai>0 && $ai<3);
  357. &pxor ($xb,$xc);
  358. &movdqa($xb_,&QWP(16*$bn-128,"ebx")) if ($i<7);
  359. &movdqa ($xa_,$xb); # borrow as temporary
  360. &pslld ($xb,12);
  361. &psrld ($xa_,20);
  362. &por ($xb,$xa_);
  363. &movdqa($xa_,&QWP(16*$an-128,"ebx"));
  364. &paddd ($xa,$xb);
  365. &movdqa($xd_,&QWP(16*$dn-128,"ebx")) if ($di!=$dn);
  366. &pxor ($xd,$xa);
  367. &movdqa (&QWP(16*$ai-128,"ebx"),$xa);
  368. &pshufb ($xd,&QWP(16,"eax")); # rot8
  369. &paddd ($xc,$xd);
  370. &movdqa (&QWP(16*$di-128,"ebx"),$xd) if ($di!=$dn);
  371. &movdqa ($xd_,$xd) if ($di==$dn);
  372. &pxor ($xb,$xc);
  373. &paddd ($xa_,$xb_) if ($i<7); # elsewhere
  374. &movdqa ($xa,$xb); # borrow as temporary
  375. &pslld ($xb,7);
  376. &psrld ($xa,25);
  377. &pxor ($xd_,$xa_) if ($i<7); # elsewhere
  378. &por ($xb,$xa);
  379. ($xa,$xa_)=($xa_,$xa);
  380. ($xb,$xb_)=($xb_,$xb);
  381. ($xc,$xc_)=($xc_,$xc);
  382. ($xd,$xd_)=($xd_,$xd);
  383. }
  384. &function_begin("ChaCha20_ssse3");
  385. &set_label("ssse3_shortcut");
  386. &mov ($out,&wparam(0));
  387. &mov ($inp,&wparam(1));
  388. &mov ($len,&wparam(2));
  389. &mov ("edx",&wparam(3)); # key
  390. &mov ("ebx",&wparam(4)); # counter and nonce
  391. &mov ("ebp","esp");
  392. &stack_push (131);
  393. &and ("esp",-64);
  394. &mov (&DWP(512,"esp"),"ebp");
  395. &lea ("eax",&DWP(&label("ssse3_data")."-".
  396. &label("pic_point"),"eax"));
  397. &movdqu ("xmm3",&QWP(0,"ebx")); # counter and nonce
  398. &cmp ($len,64*4);
  399. &jb (&label("1x"));
  400. &mov (&DWP(512+4,"esp"),"edx"); # offload pointers
  401. &mov (&DWP(512+8,"esp"),"ebx");
  402. &sub ($len,64*4); # bias len
  403. &lea ("ebp",&DWP(256+128,"esp")); # size optimization
  404. &movdqu ("xmm7",&QWP(0,"edx")); # key
  405. &pshufd ("xmm0","xmm3",0x00);
  406. &pshufd ("xmm1","xmm3",0x55);
  407. &pshufd ("xmm2","xmm3",0xaa);
  408. &pshufd ("xmm3","xmm3",0xff);
  409. &paddd ("xmm0",&QWP(16*3,"eax")); # fix counters
  410. &pshufd ("xmm4","xmm7",0x00);
  411. &pshufd ("xmm5","xmm7",0x55);
  412. &psubd ("xmm0",&QWP(16*4,"eax"));
  413. &pshufd ("xmm6","xmm7",0xaa);
  414. &pshufd ("xmm7","xmm7",0xff);
  415. &movdqa (&QWP(16*12-128,"ebp"),"xmm0");
  416. &movdqa (&QWP(16*13-128,"ebp"),"xmm1");
  417. &movdqa (&QWP(16*14-128,"ebp"),"xmm2");
  418. &movdqa (&QWP(16*15-128,"ebp"),"xmm3");
  419. &movdqu ("xmm3",&QWP(16,"edx")); # key
  420. &movdqa (&QWP(16*4-128,"ebp"),"xmm4");
  421. &movdqa (&QWP(16*5-128,"ebp"),"xmm5");
  422. &movdqa (&QWP(16*6-128,"ebp"),"xmm6");
  423. &movdqa (&QWP(16*7-128,"ebp"),"xmm7");
  424. &movdqa ("xmm7",&QWP(16*2,"eax")); # sigma
  425. &lea ("ebx",&DWP(128,"esp")); # size optimization
  426. &pshufd ("xmm0","xmm3",0x00);
  427. &pshufd ("xmm1","xmm3",0x55);
  428. &pshufd ("xmm2","xmm3",0xaa);
  429. &pshufd ("xmm3","xmm3",0xff);
  430. &pshufd ("xmm4","xmm7",0x00);
  431. &pshufd ("xmm5","xmm7",0x55);
  432. &pshufd ("xmm6","xmm7",0xaa);
  433. &pshufd ("xmm7","xmm7",0xff);
  434. &movdqa (&QWP(16*8-128,"ebp"),"xmm0");
  435. &movdqa (&QWP(16*9-128,"ebp"),"xmm1");
  436. &movdqa (&QWP(16*10-128,"ebp"),"xmm2");
  437. &movdqa (&QWP(16*11-128,"ebp"),"xmm3");
  438. &movdqa (&QWP(16*0-128,"ebp"),"xmm4");
  439. &movdqa (&QWP(16*1-128,"ebp"),"xmm5");
  440. &movdqa (&QWP(16*2-128,"ebp"),"xmm6");
  441. &movdqa (&QWP(16*3-128,"ebp"),"xmm7");
  442. &lea ($inp,&DWP(128,$inp)); # size optimization
  443. &lea ($out,&DWP(128,$out)); # size optimization
  444. &jmp (&label("outer_loop"));
  445. &set_label("outer_loop",16);
  446. #&movdqa ("xmm0",&QWP(16*0-128,"ebp")); # copy key material
  447. &movdqa ("xmm1",&QWP(16*1-128,"ebp"));
  448. &movdqa ("xmm2",&QWP(16*2-128,"ebp"));
  449. &movdqa ("xmm3",&QWP(16*3-128,"ebp"));
  450. #&movdqa ("xmm4",&QWP(16*4-128,"ebp"));
  451. &movdqa ("xmm5",&QWP(16*5-128,"ebp"));
  452. &movdqa ("xmm6",&QWP(16*6-128,"ebp"));
  453. &movdqa ("xmm7",&QWP(16*7-128,"ebp"));
  454. #&movdqa (&QWP(16*0-128,"ebx"),"xmm0");
  455. &movdqa (&QWP(16*1-128,"ebx"),"xmm1");
  456. &movdqa (&QWP(16*2-128,"ebx"),"xmm2");
  457. &movdqa (&QWP(16*3-128,"ebx"),"xmm3");
  458. #&movdqa (&QWP(16*4-128,"ebx"),"xmm4");
  459. &movdqa (&QWP(16*5-128,"ebx"),"xmm5");
  460. &movdqa (&QWP(16*6-128,"ebx"),"xmm6");
  461. &movdqa (&QWP(16*7-128,"ebx"),"xmm7");
  462. #&movdqa ("xmm0",&QWP(16*8-128,"ebp"));
  463. #&movdqa ("xmm1",&QWP(16*9-128,"ebp"));
  464. &movdqa ("xmm2",&QWP(16*10-128,"ebp"));
  465. &movdqa ("xmm3",&QWP(16*11-128,"ebp"));
  466. &movdqa ("xmm4",&QWP(16*12-128,"ebp"));
  467. &movdqa ("xmm5",&QWP(16*13-128,"ebp"));
  468. &movdqa ("xmm6",&QWP(16*14-128,"ebp"));
  469. &movdqa ("xmm7",&QWP(16*15-128,"ebp"));
  470. &paddd ("xmm4",&QWP(16*4,"eax")); # counter value
  471. #&movdqa (&QWP(16*8-128,"ebx"),"xmm0");
  472. #&movdqa (&QWP(16*9-128,"ebx"),"xmm1");
  473. &movdqa (&QWP(16*10-128,"ebx"),"xmm2");
  474. &movdqa (&QWP(16*11-128,"ebx"),"xmm3");
  475. &movdqa (&QWP(16*12-128,"ebx"),"xmm4");
  476. &movdqa (&QWP(16*13-128,"ebx"),"xmm5");
  477. &movdqa (&QWP(16*14-128,"ebx"),"xmm6");
  478. &movdqa (&QWP(16*15-128,"ebx"),"xmm7");
  479. &movdqa (&QWP(16*12-128,"ebp"),"xmm4"); # save counter value
  480. &movdqa ($xa, &QWP(16*0-128,"ebp"));
  481. &movdqa ($xd, "xmm4");
  482. &movdqa ($xb_,&QWP(16*4-128,"ebp"));
  483. &movdqa ($xc, &QWP(16*8-128,"ebp"));
  484. &movdqa ($xc_,&QWP(16*9-128,"ebp"));
  485. &mov ("edx",10); # loop counter
  486. &nop ();
  487. &set_label("loop",16);
  488. &paddd ($xa,$xb_); # elsewhere
  489. &movdqa ($xb,$xb_);
  490. &pxor ($xd,$xa); # elsewhere
  491. &QUARTERROUND_SSSE3(0, 4, 8, 12, 0);
  492. &QUARTERROUND_SSSE3(1, 5, 9, 13, 1);
  493. &QUARTERROUND_SSSE3(2, 6,10, 14, 2);
  494. &QUARTERROUND_SSSE3(3, 7,11, 15, 3);
  495. &QUARTERROUND_SSSE3(0, 5,10, 15, 4);
  496. &QUARTERROUND_SSSE3(1, 6,11, 12, 5);
  497. &QUARTERROUND_SSSE3(2, 7, 8, 13, 6);
  498. &QUARTERROUND_SSSE3(3, 4, 9, 14, 7);
  499. &dec ("edx");
  500. &jnz (&label("loop"));
  501. &movdqa (&QWP(16*4-128,"ebx"),$xb_);
  502. &movdqa (&QWP(16*8-128,"ebx"),$xc);
  503. &movdqa (&QWP(16*9-128,"ebx"),$xc_);
  504. &movdqa (&QWP(16*12-128,"ebx"),$xd);
  505. &movdqa (&QWP(16*14-128,"ebx"),$xd_);
  506. my ($xa0,$xa1,$xa2,$xa3,$xt0,$xt1,$xt2,$xt3)=map("xmm$_",(0..7));
  507. #&movdqa ($xa0,&QWP(16*0-128,"ebx")); # it's there
  508. &movdqa ($xa1,&QWP(16*1-128,"ebx"));
  509. &movdqa ($xa2,&QWP(16*2-128,"ebx"));
  510. &movdqa ($xa3,&QWP(16*3-128,"ebx"));
  511. for($i=0;$i<256;$i+=64) {
  512. &paddd ($xa0,&QWP($i+16*0-128,"ebp")); # accumulate key material
  513. &paddd ($xa1,&QWP($i+16*1-128,"ebp"));
  514. &paddd ($xa2,&QWP($i+16*2-128,"ebp"));
  515. &paddd ($xa3,&QWP($i+16*3-128,"ebp"));
  516. &movdqa ($xt2,$xa0); # "de-interlace" data
  517. &punpckldq ($xa0,$xa1);
  518. &movdqa ($xt3,$xa2);
  519. &punpckldq ($xa2,$xa3);
  520. &punpckhdq ($xt2,$xa1);
  521. &punpckhdq ($xt3,$xa3);
  522. &movdqa ($xa1,$xa0);
  523. &punpcklqdq ($xa0,$xa2); # "a0"
  524. &movdqa ($xa3,$xt2);
  525. &punpcklqdq ($xt2,$xt3); # "a2"
  526. &punpckhqdq ($xa1,$xa2); # "a1"
  527. &punpckhqdq ($xa3,$xt3); # "a3"
  528. #($xa2,$xt2)=($xt2,$xa2);
  529. &movdqu ($xt0,&QWP(64*0-128,$inp)); # load input
  530. &movdqu ($xt1,&QWP(64*1-128,$inp));
  531. &movdqu ($xa2,&QWP(64*2-128,$inp));
  532. &movdqu ($xt3,&QWP(64*3-128,$inp));
  533. &lea ($inp,&QWP($i<192?16:(64*4-16*3),$inp));
  534. &pxor ($xt0,$xa0);
  535. &movdqa ($xa0,&QWP($i+16*4-128,"ebx")) if ($i<192);
  536. &pxor ($xt1,$xa1);
  537. &movdqa ($xa1,&QWP($i+16*5-128,"ebx")) if ($i<192);
  538. &pxor ($xt2,$xa2);
  539. &movdqa ($xa2,&QWP($i+16*6-128,"ebx")) if ($i<192);
  540. &pxor ($xt3,$xa3);
  541. &movdqa ($xa3,&QWP($i+16*7-128,"ebx")) if ($i<192);
  542. &movdqu (&QWP(64*0-128,$out),$xt0); # store output
  543. &movdqu (&QWP(64*1-128,$out),$xt1);
  544. &movdqu (&QWP(64*2-128,$out),$xt2);
  545. &movdqu (&QWP(64*3-128,$out),$xt3);
  546. &lea ($out,&QWP($i<192?16:(64*4-16*3),$out));
  547. }
  548. &sub ($len,64*4);
  549. &jnc (&label("outer_loop"));
  550. &add ($len,64*4);
  551. &jz (&label("done"));
  552. &mov ("ebx",&DWP(512+8,"esp")); # restore pointers
  553. &lea ($inp,&DWP(-128,$inp));
  554. &mov ("edx",&DWP(512+4,"esp"));
  555. &lea ($out,&DWP(-128,$out));
  556. &movd ("xmm2",&DWP(16*12-128,"ebp")); # counter value
  557. &movdqu ("xmm3",&QWP(0,"ebx"));
  558. &paddd ("xmm2",&QWP(16*6,"eax")); # +four
  559. &pand ("xmm3",&QWP(16*7,"eax"));
  560. &por ("xmm3","xmm2"); # counter value
  561. {
  562. my ($a,$b,$c,$d,$t,$t1,$rot16,$rot24)=map("xmm$_",(0..7));
  563. sub SSSE3ROUND { # critical path is 20 "SIMD ticks" per round
  564. &paddd ($a,$b);
  565. &pxor ($d,$a);
  566. &pshufb ($d,$rot16);
  567. &paddd ($c,$d);
  568. &pxor ($b,$c);
  569. &movdqa ($t,$b);
  570. &psrld ($b,20);
  571. &pslld ($t,12);
  572. &por ($b,$t);
  573. &paddd ($a,$b);
  574. &pxor ($d,$a);
  575. &pshufb ($d,$rot24);
  576. &paddd ($c,$d);
  577. &pxor ($b,$c);
  578. &movdqa ($t,$b);
  579. &psrld ($b,25);
  580. &pslld ($t,7);
  581. &por ($b,$t);
  582. }
  583. &set_label("1x");
  584. &movdqa ($a,&QWP(16*2,"eax")); # sigma
  585. &movdqu ($b,&QWP(0,"edx"));
  586. &movdqu ($c,&QWP(16,"edx"));
  587. #&movdqu ($d,&QWP(0,"ebx")); # already loaded
  588. &movdqa ($rot16,&QWP(0,"eax"));
  589. &movdqa ($rot24,&QWP(16,"eax"));
  590. &mov (&DWP(16*3,"esp"),"ebp");
  591. &movdqa (&QWP(16*0,"esp"),$a);
  592. &movdqa (&QWP(16*1,"esp"),$b);
  593. &movdqa (&QWP(16*2,"esp"),$c);
  594. &movdqa (&QWP(16*3,"esp"),$d);
  595. &mov ("edx",10);
  596. &jmp (&label("loop1x"));
  597. &set_label("outer1x",16);
  598. &movdqa ($d,&QWP(16*5,"eax")); # one
  599. &movdqa ($a,&QWP(16*0,"esp"));
  600. &movdqa ($b,&QWP(16*1,"esp"));
  601. &movdqa ($c,&QWP(16*2,"esp"));
  602. &paddd ($d,&QWP(16*3,"esp"));
  603. &mov ("edx",10);
  604. &movdqa (&QWP(16*3,"esp"),$d);
  605. &jmp (&label("loop1x"));
  606. &set_label("loop1x",16);
  607. &SSSE3ROUND();
  608. &pshufd ($c,$c,0b01001110);
  609. &pshufd ($b,$b,0b00111001);
  610. &pshufd ($d,$d,0b10010011);
  611. &nop ();
  612. &SSSE3ROUND();
  613. &pshufd ($c,$c,0b01001110);
  614. &pshufd ($b,$b,0b10010011);
  615. &pshufd ($d,$d,0b00111001);
  616. &dec ("edx");
  617. &jnz (&label("loop1x"));
  618. &paddd ($a,&QWP(16*0,"esp"));
  619. &paddd ($b,&QWP(16*1,"esp"));
  620. &paddd ($c,&QWP(16*2,"esp"));
  621. &paddd ($d,&QWP(16*3,"esp"));
  622. &cmp ($len,64);
  623. &jb (&label("tail"));
  624. &movdqu ($t,&QWP(16*0,$inp));
  625. &movdqu ($t1,&QWP(16*1,$inp));
  626. &pxor ($a,$t); # xor with input
  627. &movdqu ($t,&QWP(16*2,$inp));
  628. &pxor ($b,$t1);
  629. &movdqu ($t1,&QWP(16*3,$inp));
  630. &pxor ($c,$t);
  631. &pxor ($d,$t1);
  632. &lea ($inp,&DWP(16*4,$inp)); # inp+=64
  633. &movdqu (&QWP(16*0,$out),$a); # write output
  634. &movdqu (&QWP(16*1,$out),$b);
  635. &movdqu (&QWP(16*2,$out),$c);
  636. &movdqu (&QWP(16*3,$out),$d);
  637. &lea ($out,&DWP(16*4,$out)); # inp+=64
  638. &sub ($len,64);
  639. &jnz (&label("outer1x"));
  640. &jmp (&label("done"));
  641. &set_label("tail");
  642. &movdqa (&QWP(16*0,"esp"),$a);
  643. &movdqa (&QWP(16*1,"esp"),$b);
  644. &movdqa (&QWP(16*2,"esp"),$c);
  645. &movdqa (&QWP(16*3,"esp"),$d);
  646. &xor ("eax","eax");
  647. &xor ("edx","edx");
  648. &xor ("ebp","ebp");
  649. &set_label("tail_loop");
  650. &movb ("al",&BP(0,"esp","ebp"));
  651. &movb ("dl",&BP(0,$inp,"ebp"));
  652. &lea ("ebp",&DWP(1,"ebp"));
  653. &xor ("al","dl");
  654. &movb (&BP(-1,$out,"ebp"),"al");
  655. &dec ($len);
  656. &jnz (&label("tail_loop"));
  657. }
  658. &set_label("done");
  659. &mov ("esp",&DWP(512,"esp"));
  660. &function_end("ChaCha20_ssse3");
  661. &align (64);
  662. &set_label("ssse3_data");
  663. &data_byte(0x2,0x3,0x0,0x1, 0x6,0x7,0x4,0x5, 0xa,0xb,0x8,0x9, 0xe,0xf,0xc,0xd);
  664. &data_byte(0x3,0x0,0x1,0x2, 0x7,0x4,0x5,0x6, 0xb,0x8,0x9,0xa, 0xf,0xc,0xd,0xe);
  665. &data_word(0x61707865,0x3320646e,0x79622d32,0x6b206574);
  666. &data_word(0,1,2,3);
  667. &data_word(4,4,4,4);
  668. &data_word(1,0,0,0);
  669. &data_word(4,0,0,0);
  670. &data_word(0,-1,-1,-1);
  671. &align (64);
  672. }
  673. &asciz ("ChaCha20 for x86, CRYPTOGAMS by <appro\@openssl.org>");
  674. &asm_finish();
  675. close STDOUT;