1e1051a39Sopenharmony_ci#! /usr/bin/env perl
2e1051a39Sopenharmony_ci# Copyright 2013-2020 The OpenSSL Project Authors. All Rights Reserved.
3e1051a39Sopenharmony_ci# Copyright (c) 2012, Intel Corporation. All Rights Reserved.
4e1051a39Sopenharmony_ci#
5e1051a39Sopenharmony_ci# Licensed under the Apache License 2.0 (the "License").  You may not use
6e1051a39Sopenharmony_ci# this file except in compliance with the License.  You can obtain a copy
7e1051a39Sopenharmony_ci# in the file LICENSE in the source distribution or at
8e1051a39Sopenharmony_ci# https://www.openssl.org/source/license.html
9e1051a39Sopenharmony_ci#
10e1051a39Sopenharmony_ci# Originally written by Shay Gueron (1, 2), and Vlad Krasnov (1)
11e1051a39Sopenharmony_ci# (1) Intel Corporation, Israel Development Center, Haifa, Israel
12e1051a39Sopenharmony_ci# (2) University of Haifa, Israel
13e1051a39Sopenharmony_ci#
14e1051a39Sopenharmony_ci# References:
15e1051a39Sopenharmony_ci# [1] S. Gueron, V. Krasnov: "Software Implementation of Modular
16e1051a39Sopenharmony_ci#     Exponentiation,  Using Advanced Vector Instructions Architectures",
17e1051a39Sopenharmony_ci#     F. Ozbudak and F. Rodriguez-Henriquez (Eds.): WAIFI 2012, LNCS 7369,
18e1051a39Sopenharmony_ci#     pp. 119?135, 2012. Springer-Verlag Berlin Heidelberg 2012
19e1051a39Sopenharmony_ci# [2] S. Gueron: "Efficient Software Implementations of Modular
20e1051a39Sopenharmony_ci#     Exponentiation", Journal of Cryptographic Engineering 2:31-43 (2012).
21e1051a39Sopenharmony_ci# [3] S. Gueron, V. Krasnov: "Speeding up Big-numbers Squaring",IEEE
22e1051a39Sopenharmony_ci#     Proceedings of 9th International Conference on Information Technology:
23e1051a39Sopenharmony_ci#     New Generations (ITNG 2012), pp.821-823 (2012)
24e1051a39Sopenharmony_ci# [4] S. Gueron, V. Krasnov: "[PATCH] Efficient and side channel analysis
25e1051a39Sopenharmony_ci#     resistant 1024-bit modular exponentiation, for optimizing RSA2048
26e1051a39Sopenharmony_ci#     on AVX2 capable x86_64 platforms",
27e1051a39Sopenharmony_ci#     http://rt.openssl.org/Ticket/Display.html?id=2850&user=guest&pass=guest
28e1051a39Sopenharmony_ci#
29e1051a39Sopenharmony_ci# +13% improvement over original submission by <appro@openssl.org>
30e1051a39Sopenharmony_ci#
31e1051a39Sopenharmony_ci# rsa2048 sign/sec	OpenSSL 1.0.1	scalar(*)	this
32e1051a39Sopenharmony_ci# 2.3GHz Haswell	621		765/+23%	1113/+79%
33e1051a39Sopenharmony_ci# 2.3GHz Broadwell(**)	688		1200(***)/+74%	1120/+63%
34e1051a39Sopenharmony_ci#
35e1051a39Sopenharmony_ci# (*)	if system doesn't support AVX2, for reference purposes;
36e1051a39Sopenharmony_ci# (**)	scaled to 2.3GHz to simplify comparison;
37e1051a39Sopenharmony_ci# (***)	scalar AD*X code is faster than AVX2 and is preferred code
38e1051a39Sopenharmony_ci#	path for Broadwell;
39e1051a39Sopenharmony_ci
40e1051a39Sopenharmony_ci# $output is the last argument if it looks like a file (it has an extension)
41e1051a39Sopenharmony_ci# $flavour is the first argument if it doesn't look like a file
42e1051a39Sopenharmony_ci$output = $#ARGV >= 0 && $ARGV[$#ARGV] =~ m|\.\w+$| ? pop : undef;
43e1051a39Sopenharmony_ci$flavour = $#ARGV >= 0 && $ARGV[0] !~ m|\.| ? shift : undef;
44e1051a39Sopenharmony_ci
45e1051a39Sopenharmony_ci$win64=0; $win64=1 if ($flavour =~ /[nm]asm|mingw64/ || $output =~ /\.asm$/);
46e1051a39Sopenharmony_ci
47e1051a39Sopenharmony_ci$0 =~ m/(.*[\/\\])[^\/\\]+$/; $dir=$1;
48e1051a39Sopenharmony_ci( $xlate="${dir}x86_64-xlate.pl" and -f $xlate ) or
49e1051a39Sopenharmony_ci( $xlate="${dir}../../perlasm/x86_64-xlate.pl" and -f $xlate) or
50e1051a39Sopenharmony_cidie "can't locate x86_64-xlate.pl";
51e1051a39Sopenharmony_ci
52e1051a39Sopenharmony_ciif (`$ENV{CC} -Wa,-v -c -o /dev/null -x assembler /dev/null 2>&1`
53e1051a39Sopenharmony_ci		=~ /GNU assembler version ([2-9]\.[0-9]+)/) {
54e1051a39Sopenharmony_ci	$avx = ($1>=2.19) + ($1>=2.22);
55e1051a39Sopenharmony_ci	$addx = ($1>=2.23);
56e1051a39Sopenharmony_ci}
57e1051a39Sopenharmony_ci
58e1051a39Sopenharmony_ciif (!$avx && $win64 && ($flavour =~ /nasm/ || $ENV{ASM} =~ /nasm/) &&
59e1051a39Sopenharmony_ci	    `nasm -v 2>&1` =~ /NASM version ([2-9]\.[0-9]+)/) {
60e1051a39Sopenharmony_ci	$avx = ($1>=2.09) + ($1>=2.10);
61e1051a39Sopenharmony_ci	$addx = ($1>=2.10);
62e1051a39Sopenharmony_ci}
63e1051a39Sopenharmony_ci
64e1051a39Sopenharmony_ciif (!$avx && $win64 && ($flavour =~ /masm/ || $ENV{ASM} =~ /ml64/) &&
65e1051a39Sopenharmony_ci	    `ml64 2>&1` =~ /Version ([0-9]+)\./) {
66e1051a39Sopenharmony_ci	$avx = ($1>=10) + ($1>=11);
67e1051a39Sopenharmony_ci	$addx = ($1>=11);
68e1051a39Sopenharmony_ci}
69e1051a39Sopenharmony_ci
70e1051a39Sopenharmony_ciif (!$avx && `$ENV{CC} -v 2>&1` =~ /((?:clang|LLVM) version|based on LLVM) ([0-9]+)\.([0-9]+)/) {
71e1051a39Sopenharmony_ci	my $ver = $2 + $3/100.0;	# 3.1->3.01, 3.10->3.10
72e1051a39Sopenharmony_ci	$avx = ($ver>=3.0) + ($ver>=3.01);
73e1051a39Sopenharmony_ci	$addx = ($ver>=3.03);
74e1051a39Sopenharmony_ci}
75e1051a39Sopenharmony_ci
76e1051a39Sopenharmony_ciopen OUT,"| \"$^X\" \"$xlate\" $flavour \"$output\""
77e1051a39Sopenharmony_ci    or die "can't call $xlate: $!";
78e1051a39Sopenharmony_ci*STDOUT = *OUT;
79e1051a39Sopenharmony_ci
80e1051a39Sopenharmony_ciif ($avx>1) {{{
81e1051a39Sopenharmony_ci{ # void AMS_WW(
82e1051a39Sopenharmony_cimy $rp="%rdi";	# BN_ULONG *rp,
83e1051a39Sopenharmony_cimy $ap="%rsi";	# const BN_ULONG *ap,
84e1051a39Sopenharmony_cimy $np="%rdx";	# const BN_ULONG *np,
85e1051a39Sopenharmony_cimy $n0="%ecx";	# const BN_ULONG n0,
86e1051a39Sopenharmony_cimy $rep="%r8d";	# int repeat);
87e1051a39Sopenharmony_ci
88e1051a39Sopenharmony_ci# The registers that hold the accumulated redundant result
89e1051a39Sopenharmony_ci# The AMM works on 1024 bit operands, and redundant word size is 29
90e1051a39Sopenharmony_ci# Therefore: ceil(1024/29)/4 = 9
91e1051a39Sopenharmony_cimy $ACC0="%ymm0";
92e1051a39Sopenharmony_cimy $ACC1="%ymm1";
93e1051a39Sopenharmony_cimy $ACC2="%ymm2";
94e1051a39Sopenharmony_cimy $ACC3="%ymm3";
95e1051a39Sopenharmony_cimy $ACC4="%ymm4";
96e1051a39Sopenharmony_cimy $ACC5="%ymm5";
97e1051a39Sopenharmony_cimy $ACC6="%ymm6";
98e1051a39Sopenharmony_cimy $ACC7="%ymm7";
99e1051a39Sopenharmony_cimy $ACC8="%ymm8";
100e1051a39Sopenharmony_cimy $ACC9="%ymm9";
101e1051a39Sopenharmony_ci# Registers that hold the broadcasted words of bp, currently used
102e1051a39Sopenharmony_cimy $B1="%ymm10";
103e1051a39Sopenharmony_cimy $B2="%ymm11";
104e1051a39Sopenharmony_ci# Registers that hold the broadcasted words of Y, currently used
105e1051a39Sopenharmony_cimy $Y1="%ymm12";
106e1051a39Sopenharmony_cimy $Y2="%ymm13";
107e1051a39Sopenharmony_ci# Helper registers
108e1051a39Sopenharmony_cimy $TEMP1="%ymm14";
109e1051a39Sopenharmony_cimy $AND_MASK="%ymm15";
110e1051a39Sopenharmony_ci# alu registers that hold the first words of the ACC
111e1051a39Sopenharmony_cimy $r0="%r9";
112e1051a39Sopenharmony_cimy $r1="%r10";
113e1051a39Sopenharmony_cimy $r2="%r11";
114e1051a39Sopenharmony_cimy $r3="%r12";
115e1051a39Sopenharmony_ci
116e1051a39Sopenharmony_cimy $i="%r14d";			# loop counter
117e1051a39Sopenharmony_cimy $tmp = "%r15";
118e1051a39Sopenharmony_ci
119e1051a39Sopenharmony_cimy $FrameSize=32*18+32*8;	# place for A^2 and 2*A
120e1051a39Sopenharmony_ci
121e1051a39Sopenharmony_cimy $aap=$r0;
122e1051a39Sopenharmony_cimy $tp0="%rbx";
123e1051a39Sopenharmony_cimy $tp1=$r3;
124e1051a39Sopenharmony_cimy $tpa=$tmp;
125e1051a39Sopenharmony_ci
126e1051a39Sopenharmony_ci$np="%r13";			# reassigned argument
127e1051a39Sopenharmony_ci
128e1051a39Sopenharmony_ci$code.=<<___;
129e1051a39Sopenharmony_ci.text
130e1051a39Sopenharmony_ci
131e1051a39Sopenharmony_ci.globl	rsaz_1024_sqr_avx2
132e1051a39Sopenharmony_ci.type	rsaz_1024_sqr_avx2,\@function,5
133e1051a39Sopenharmony_ci.align	64
134e1051a39Sopenharmony_cirsaz_1024_sqr_avx2:		# 702 cycles, 14% faster than rsaz_1024_mul_avx2
135e1051a39Sopenharmony_ci.cfi_startproc
136e1051a39Sopenharmony_ci	lea	(%rsp), %rax
137e1051a39Sopenharmony_ci.cfi_def_cfa_register	%rax
138e1051a39Sopenharmony_ci	push	%rbx
139e1051a39Sopenharmony_ci.cfi_push	%rbx
140e1051a39Sopenharmony_ci	push	%rbp
141e1051a39Sopenharmony_ci.cfi_push	%rbp
142e1051a39Sopenharmony_ci	push	%r12
143e1051a39Sopenharmony_ci.cfi_push	%r12
144e1051a39Sopenharmony_ci	push	%r13
145e1051a39Sopenharmony_ci.cfi_push	%r13
146e1051a39Sopenharmony_ci	push	%r14
147e1051a39Sopenharmony_ci.cfi_push	%r14
148e1051a39Sopenharmony_ci	push	%r15
149e1051a39Sopenharmony_ci.cfi_push	%r15
150e1051a39Sopenharmony_ci	vzeroupper
151e1051a39Sopenharmony_ci___
152e1051a39Sopenharmony_ci$code.=<<___ if ($win64);
153e1051a39Sopenharmony_ci	lea	-0xa8(%rsp),%rsp
154e1051a39Sopenharmony_ci	vmovaps	%xmm6,-0xd8(%rax)
155e1051a39Sopenharmony_ci	vmovaps	%xmm7,-0xc8(%rax)
156e1051a39Sopenharmony_ci	vmovaps	%xmm8,-0xb8(%rax)
157e1051a39Sopenharmony_ci	vmovaps	%xmm9,-0xa8(%rax)
158e1051a39Sopenharmony_ci	vmovaps	%xmm10,-0x98(%rax)
159e1051a39Sopenharmony_ci	vmovaps	%xmm11,-0x88(%rax)
160e1051a39Sopenharmony_ci	vmovaps	%xmm12,-0x78(%rax)
161e1051a39Sopenharmony_ci	vmovaps	%xmm13,-0x68(%rax)
162e1051a39Sopenharmony_ci	vmovaps	%xmm14,-0x58(%rax)
163e1051a39Sopenharmony_ci	vmovaps	%xmm15,-0x48(%rax)
164e1051a39Sopenharmony_ci.Lsqr_1024_body:
165e1051a39Sopenharmony_ci___
166e1051a39Sopenharmony_ci$code.=<<___;
167e1051a39Sopenharmony_ci	mov	%rax,%rbp
168e1051a39Sopenharmony_ci.cfi_def_cfa_register	%rbp
169e1051a39Sopenharmony_ci	mov	%rdx, $np			# reassigned argument
170e1051a39Sopenharmony_ci	sub	\$$FrameSize, %rsp
171e1051a39Sopenharmony_ci	mov	$np, $tmp
172e1051a39Sopenharmony_ci	sub	\$-128, $rp			# size optimization
173e1051a39Sopenharmony_ci	sub	\$-128, $ap
174e1051a39Sopenharmony_ci	sub	\$-128, $np
175e1051a39Sopenharmony_ci
176e1051a39Sopenharmony_ci	and	\$4095, $tmp			# see if $np crosses page
177e1051a39Sopenharmony_ci	add	\$32*10, $tmp
178e1051a39Sopenharmony_ci	shr	\$12, $tmp
179e1051a39Sopenharmony_ci	vpxor	$ACC9,$ACC9,$ACC9
180e1051a39Sopenharmony_ci	jz	.Lsqr_1024_no_n_copy
181e1051a39Sopenharmony_ci
182e1051a39Sopenharmony_ci	# unaligned 256-bit load that crosses page boundary can
183e1051a39Sopenharmony_ci	# cause >2x performance degradation here, so if $np does
184e1051a39Sopenharmony_ci	# cross page boundary, copy it to stack and make sure stack
185e1051a39Sopenharmony_ci	# frame doesn't...
186e1051a39Sopenharmony_ci	sub		\$32*10,%rsp
187e1051a39Sopenharmony_ci	vmovdqu		32*0-128($np), $ACC0
188e1051a39Sopenharmony_ci	and		\$-2048, %rsp
189e1051a39Sopenharmony_ci	vmovdqu		32*1-128($np), $ACC1
190e1051a39Sopenharmony_ci	vmovdqu		32*2-128($np), $ACC2
191e1051a39Sopenharmony_ci	vmovdqu		32*3-128($np), $ACC3
192e1051a39Sopenharmony_ci	vmovdqu		32*4-128($np), $ACC4
193e1051a39Sopenharmony_ci	vmovdqu		32*5-128($np), $ACC5
194e1051a39Sopenharmony_ci	vmovdqu		32*6-128($np), $ACC6
195e1051a39Sopenharmony_ci	vmovdqu		32*7-128($np), $ACC7
196e1051a39Sopenharmony_ci	vmovdqu		32*8-128($np), $ACC8
197e1051a39Sopenharmony_ci	lea		$FrameSize+128(%rsp),$np
198e1051a39Sopenharmony_ci	vmovdqu		$ACC0, 32*0-128($np)
199e1051a39Sopenharmony_ci	vmovdqu		$ACC1, 32*1-128($np)
200e1051a39Sopenharmony_ci	vmovdqu		$ACC2, 32*2-128($np)
201e1051a39Sopenharmony_ci	vmovdqu		$ACC3, 32*3-128($np)
202e1051a39Sopenharmony_ci	vmovdqu		$ACC4, 32*4-128($np)
203e1051a39Sopenharmony_ci	vmovdqu		$ACC5, 32*5-128($np)
204e1051a39Sopenharmony_ci	vmovdqu		$ACC6, 32*6-128($np)
205e1051a39Sopenharmony_ci	vmovdqu		$ACC7, 32*7-128($np)
206e1051a39Sopenharmony_ci	vmovdqu		$ACC8, 32*8-128($np)
207e1051a39Sopenharmony_ci	vmovdqu		$ACC9, 32*9-128($np)	# $ACC9 is zero
208e1051a39Sopenharmony_ci
209e1051a39Sopenharmony_ci.Lsqr_1024_no_n_copy:
210e1051a39Sopenharmony_ci	and		\$-1024, %rsp
211e1051a39Sopenharmony_ci
212e1051a39Sopenharmony_ci	vmovdqu		32*1-128($ap), $ACC1
213e1051a39Sopenharmony_ci	vmovdqu		32*2-128($ap), $ACC2
214e1051a39Sopenharmony_ci	vmovdqu		32*3-128($ap), $ACC3
215e1051a39Sopenharmony_ci	vmovdqu		32*4-128($ap), $ACC4
216e1051a39Sopenharmony_ci	vmovdqu		32*5-128($ap), $ACC5
217e1051a39Sopenharmony_ci	vmovdqu		32*6-128($ap), $ACC6
218e1051a39Sopenharmony_ci	vmovdqu		32*7-128($ap), $ACC7
219e1051a39Sopenharmony_ci	vmovdqu		32*8-128($ap), $ACC8
220e1051a39Sopenharmony_ci
221e1051a39Sopenharmony_ci	lea	192(%rsp), $tp0			# 64+128=192
222e1051a39Sopenharmony_ci	vmovdqu	.Land_mask(%rip), $AND_MASK
223e1051a39Sopenharmony_ci	jmp	.LOOP_GRANDE_SQR_1024
224e1051a39Sopenharmony_ci
225e1051a39Sopenharmony_ci.align	32
226e1051a39Sopenharmony_ci.LOOP_GRANDE_SQR_1024:
227e1051a39Sopenharmony_ci	lea	32*18+128(%rsp), $aap		# size optimization
228e1051a39Sopenharmony_ci	lea	448(%rsp), $tp1			# 64+128+256=448
229e1051a39Sopenharmony_ci
230e1051a39Sopenharmony_ci	# the squaring is performed as described in Variant B of
231e1051a39Sopenharmony_ci	# "Speeding up Big-Number Squaring", so start by calculating
232e1051a39Sopenharmony_ci	# the A*2=A+A vector
233e1051a39Sopenharmony_ci	vpaddq		$ACC1, $ACC1, $ACC1
234e1051a39Sopenharmony_ci	 vpbroadcastq	32*0-128($ap), $B1
235e1051a39Sopenharmony_ci	vpaddq		$ACC2, $ACC2, $ACC2
236e1051a39Sopenharmony_ci	vmovdqa		$ACC1, 32*0-128($aap)
237e1051a39Sopenharmony_ci	vpaddq		$ACC3, $ACC3, $ACC3
238e1051a39Sopenharmony_ci	vmovdqa		$ACC2, 32*1-128($aap)
239e1051a39Sopenharmony_ci	vpaddq		$ACC4, $ACC4, $ACC4
240e1051a39Sopenharmony_ci	vmovdqa		$ACC3, 32*2-128($aap)
241e1051a39Sopenharmony_ci	vpaddq		$ACC5, $ACC5, $ACC5
242e1051a39Sopenharmony_ci	vmovdqa		$ACC4, 32*3-128($aap)
243e1051a39Sopenharmony_ci	vpaddq		$ACC6, $ACC6, $ACC6
244e1051a39Sopenharmony_ci	vmovdqa		$ACC5, 32*4-128($aap)
245e1051a39Sopenharmony_ci	vpaddq		$ACC7, $ACC7, $ACC7
246e1051a39Sopenharmony_ci	vmovdqa		$ACC6, 32*5-128($aap)
247e1051a39Sopenharmony_ci	vpaddq		$ACC8, $ACC8, $ACC8
248e1051a39Sopenharmony_ci	vmovdqa		$ACC7, 32*6-128($aap)
249e1051a39Sopenharmony_ci	vpxor		$ACC9, $ACC9, $ACC9
250e1051a39Sopenharmony_ci	vmovdqa		$ACC8, 32*7-128($aap)
251e1051a39Sopenharmony_ci
252e1051a39Sopenharmony_ci	vpmuludq	32*0-128($ap), $B1, $ACC0
253e1051a39Sopenharmony_ci	 vpbroadcastq	32*1-128($ap), $B2
254e1051a39Sopenharmony_ci	 vmovdqu	$ACC9, 32*9-192($tp0)	# zero upper half
255e1051a39Sopenharmony_ci	vpmuludq	$B1, $ACC1, $ACC1
256e1051a39Sopenharmony_ci	 vmovdqu	$ACC9, 32*10-448($tp1)
257e1051a39Sopenharmony_ci	vpmuludq	$B1, $ACC2, $ACC2
258e1051a39Sopenharmony_ci	 vmovdqu	$ACC9, 32*11-448($tp1)
259e1051a39Sopenharmony_ci	vpmuludq	$B1, $ACC3, $ACC3
260e1051a39Sopenharmony_ci	 vmovdqu	$ACC9, 32*12-448($tp1)
261e1051a39Sopenharmony_ci	vpmuludq	$B1, $ACC4, $ACC4
262e1051a39Sopenharmony_ci	 vmovdqu	$ACC9, 32*13-448($tp1)
263e1051a39Sopenharmony_ci	vpmuludq	$B1, $ACC5, $ACC5
264e1051a39Sopenharmony_ci	 vmovdqu	$ACC9, 32*14-448($tp1)
265e1051a39Sopenharmony_ci	vpmuludq	$B1, $ACC6, $ACC6
266e1051a39Sopenharmony_ci	 vmovdqu	$ACC9, 32*15-448($tp1)
267e1051a39Sopenharmony_ci	vpmuludq	$B1, $ACC7, $ACC7
268e1051a39Sopenharmony_ci	 vmovdqu	$ACC9, 32*16-448($tp1)
269e1051a39Sopenharmony_ci	vpmuludq	$B1, $ACC8, $ACC8
270e1051a39Sopenharmony_ci	 vpbroadcastq	32*2-128($ap), $B1
271e1051a39Sopenharmony_ci	 vmovdqu	$ACC9, 32*17-448($tp1)
272e1051a39Sopenharmony_ci
273e1051a39Sopenharmony_ci	mov	$ap, $tpa
274e1051a39Sopenharmony_ci	mov 	\$4, $i
275e1051a39Sopenharmony_ci	jmp	.Lsqr_entry_1024
276e1051a39Sopenharmony_ci___
277e1051a39Sopenharmony_ci$TEMP0=$Y1;
278e1051a39Sopenharmony_ci$TEMP2=$Y2;
279e1051a39Sopenharmony_ci$code.=<<___;
280e1051a39Sopenharmony_ci.align	32
281e1051a39Sopenharmony_ci.LOOP_SQR_1024:
282e1051a39Sopenharmony_ci	 vpbroadcastq	32*1-128($tpa), $B2
283e1051a39Sopenharmony_ci	vpmuludq	32*0-128($ap), $B1, $ACC0
284e1051a39Sopenharmony_ci	vpaddq		32*0-192($tp0), $ACC0, $ACC0
285e1051a39Sopenharmony_ci	vpmuludq	32*0-128($aap), $B1, $ACC1
286e1051a39Sopenharmony_ci	vpaddq		32*1-192($tp0), $ACC1, $ACC1
287e1051a39Sopenharmony_ci	vpmuludq	32*1-128($aap), $B1, $ACC2
288e1051a39Sopenharmony_ci	vpaddq		32*2-192($tp0), $ACC2, $ACC2
289e1051a39Sopenharmony_ci	vpmuludq	32*2-128($aap), $B1, $ACC3
290e1051a39Sopenharmony_ci	vpaddq		32*3-192($tp0), $ACC3, $ACC3
291e1051a39Sopenharmony_ci	vpmuludq	32*3-128($aap), $B1, $ACC4
292e1051a39Sopenharmony_ci	vpaddq		32*4-192($tp0), $ACC4, $ACC4
293e1051a39Sopenharmony_ci	vpmuludq	32*4-128($aap), $B1, $ACC5
294e1051a39Sopenharmony_ci	vpaddq		32*5-192($tp0), $ACC5, $ACC5
295e1051a39Sopenharmony_ci	vpmuludq	32*5-128($aap), $B1, $ACC6
296e1051a39Sopenharmony_ci	vpaddq		32*6-192($tp0), $ACC6, $ACC6
297e1051a39Sopenharmony_ci	vpmuludq	32*6-128($aap), $B1, $ACC7
298e1051a39Sopenharmony_ci	vpaddq		32*7-192($tp0), $ACC7, $ACC7
299e1051a39Sopenharmony_ci	vpmuludq	32*7-128($aap), $B1, $ACC8
300e1051a39Sopenharmony_ci	 vpbroadcastq	32*2-128($tpa), $B1
301e1051a39Sopenharmony_ci	vpaddq		32*8-192($tp0), $ACC8, $ACC8
302e1051a39Sopenharmony_ci.Lsqr_entry_1024:
303e1051a39Sopenharmony_ci	vmovdqu		$ACC0, 32*0-192($tp0)
304e1051a39Sopenharmony_ci	vmovdqu		$ACC1, 32*1-192($tp0)
305e1051a39Sopenharmony_ci
306e1051a39Sopenharmony_ci	vpmuludq	32*1-128($ap), $B2, $TEMP0
307e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC2, $ACC2
308e1051a39Sopenharmony_ci	vpmuludq	32*1-128($aap), $B2, $TEMP1
309e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC3, $ACC3
310e1051a39Sopenharmony_ci	vpmuludq	32*2-128($aap), $B2, $TEMP2
311e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC4, $ACC4
312e1051a39Sopenharmony_ci	vpmuludq	32*3-128($aap), $B2, $TEMP0
313e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC5, $ACC5
314e1051a39Sopenharmony_ci	vpmuludq	32*4-128($aap), $B2, $TEMP1
315e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC6, $ACC6
316e1051a39Sopenharmony_ci	vpmuludq	32*5-128($aap), $B2, $TEMP2
317e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC7, $ACC7
318e1051a39Sopenharmony_ci	vpmuludq	32*6-128($aap), $B2, $TEMP0
319e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC8, $ACC8
320e1051a39Sopenharmony_ci	vpmuludq	32*7-128($aap), $B2, $ACC0
321e1051a39Sopenharmony_ci	 vpbroadcastq	32*3-128($tpa), $B2
322e1051a39Sopenharmony_ci	vpaddq		32*9-192($tp0), $ACC0, $ACC0
323e1051a39Sopenharmony_ci
324e1051a39Sopenharmony_ci	vmovdqu		$ACC2, 32*2-192($tp0)
325e1051a39Sopenharmony_ci	vmovdqu		$ACC3, 32*3-192($tp0)
326e1051a39Sopenharmony_ci
327e1051a39Sopenharmony_ci	vpmuludq	32*2-128($ap), $B1, $TEMP2
328e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC4, $ACC4
329e1051a39Sopenharmony_ci	vpmuludq	32*2-128($aap), $B1, $TEMP0
330e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC5, $ACC5
331e1051a39Sopenharmony_ci	vpmuludq	32*3-128($aap), $B1, $TEMP1
332e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC6, $ACC6
333e1051a39Sopenharmony_ci	vpmuludq	32*4-128($aap), $B1, $TEMP2
334e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC7, $ACC7
335e1051a39Sopenharmony_ci	vpmuludq	32*5-128($aap), $B1, $TEMP0
336e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC8, $ACC8
337e1051a39Sopenharmony_ci	vpmuludq	32*6-128($aap), $B1, $TEMP1
338e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC0, $ACC0
339e1051a39Sopenharmony_ci	vpmuludq	32*7-128($aap), $B1, $ACC1
340e1051a39Sopenharmony_ci	 vpbroadcastq	32*4-128($tpa), $B1
341e1051a39Sopenharmony_ci	vpaddq		32*10-448($tp1), $ACC1, $ACC1
342e1051a39Sopenharmony_ci
343e1051a39Sopenharmony_ci	vmovdqu		$ACC4, 32*4-192($tp0)
344e1051a39Sopenharmony_ci	vmovdqu		$ACC5, 32*5-192($tp0)
345e1051a39Sopenharmony_ci
346e1051a39Sopenharmony_ci	vpmuludq	32*3-128($ap), $B2, $TEMP0
347e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC6, $ACC6
348e1051a39Sopenharmony_ci	vpmuludq	32*3-128($aap), $B2, $TEMP1
349e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC7, $ACC7
350e1051a39Sopenharmony_ci	vpmuludq	32*4-128($aap), $B2, $TEMP2
351e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC8, $ACC8
352e1051a39Sopenharmony_ci	vpmuludq	32*5-128($aap), $B2, $TEMP0
353e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC0, $ACC0
354e1051a39Sopenharmony_ci	vpmuludq	32*6-128($aap), $B2, $TEMP1
355e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC1, $ACC1
356e1051a39Sopenharmony_ci	vpmuludq	32*7-128($aap), $B2, $ACC2
357e1051a39Sopenharmony_ci	 vpbroadcastq	32*5-128($tpa), $B2
358e1051a39Sopenharmony_ci	vpaddq		32*11-448($tp1), $ACC2, $ACC2
359e1051a39Sopenharmony_ci
360e1051a39Sopenharmony_ci	vmovdqu		$ACC6, 32*6-192($tp0)
361e1051a39Sopenharmony_ci	vmovdqu		$ACC7, 32*7-192($tp0)
362e1051a39Sopenharmony_ci
363e1051a39Sopenharmony_ci	vpmuludq	32*4-128($ap), $B1, $TEMP0
364e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC8, $ACC8
365e1051a39Sopenharmony_ci	vpmuludq	32*4-128($aap), $B1, $TEMP1
366e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC0, $ACC0
367e1051a39Sopenharmony_ci	vpmuludq	32*5-128($aap), $B1, $TEMP2
368e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC1, $ACC1
369e1051a39Sopenharmony_ci	vpmuludq	32*6-128($aap), $B1, $TEMP0
370e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC2, $ACC2
371e1051a39Sopenharmony_ci	vpmuludq	32*7-128($aap), $B1, $ACC3
372e1051a39Sopenharmony_ci	 vpbroadcastq	32*6-128($tpa), $B1
373e1051a39Sopenharmony_ci	vpaddq		32*12-448($tp1), $ACC3, $ACC3
374e1051a39Sopenharmony_ci
375e1051a39Sopenharmony_ci	vmovdqu		$ACC8, 32*8-192($tp0)
376e1051a39Sopenharmony_ci	vmovdqu		$ACC0, 32*9-192($tp0)
377e1051a39Sopenharmony_ci	lea		8($tp0), $tp0
378e1051a39Sopenharmony_ci
379e1051a39Sopenharmony_ci	vpmuludq	32*5-128($ap), $B2, $TEMP2
380e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC1, $ACC1
381e1051a39Sopenharmony_ci	vpmuludq	32*5-128($aap), $B2, $TEMP0
382e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC2, $ACC2
383e1051a39Sopenharmony_ci	vpmuludq	32*6-128($aap), $B2, $TEMP1
384e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC3, $ACC3
385e1051a39Sopenharmony_ci	vpmuludq	32*7-128($aap), $B2, $ACC4
386e1051a39Sopenharmony_ci	 vpbroadcastq	32*7-128($tpa), $B2
387e1051a39Sopenharmony_ci	vpaddq		32*13-448($tp1), $ACC4, $ACC4
388e1051a39Sopenharmony_ci
389e1051a39Sopenharmony_ci	vmovdqu		$ACC1, 32*10-448($tp1)
390e1051a39Sopenharmony_ci	vmovdqu		$ACC2, 32*11-448($tp1)
391e1051a39Sopenharmony_ci
392e1051a39Sopenharmony_ci	vpmuludq	32*6-128($ap), $B1, $TEMP0
393e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC3, $ACC3
394e1051a39Sopenharmony_ci	vpmuludq	32*6-128($aap), $B1, $TEMP1
395e1051a39Sopenharmony_ci	 vpbroadcastq	32*8-128($tpa), $ACC0		# borrow $ACC0 for $B1
396e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC4, $ACC4
397e1051a39Sopenharmony_ci	vpmuludq	32*7-128($aap), $B1, $ACC5
398e1051a39Sopenharmony_ci	 vpbroadcastq	32*0+8-128($tpa), $B1		# for next iteration
399e1051a39Sopenharmony_ci	vpaddq		32*14-448($tp1), $ACC5, $ACC5
400e1051a39Sopenharmony_ci
401e1051a39Sopenharmony_ci	vmovdqu		$ACC3, 32*12-448($tp1)
402e1051a39Sopenharmony_ci	vmovdqu		$ACC4, 32*13-448($tp1)
403e1051a39Sopenharmony_ci	lea		8($tpa), $tpa
404e1051a39Sopenharmony_ci
405e1051a39Sopenharmony_ci	vpmuludq	32*7-128($ap), $B2, $TEMP0
406e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC5, $ACC5
407e1051a39Sopenharmony_ci	vpmuludq	32*7-128($aap), $B2, $ACC6
408e1051a39Sopenharmony_ci	vpaddq		32*15-448($tp1), $ACC6, $ACC6
409e1051a39Sopenharmony_ci
410e1051a39Sopenharmony_ci	vpmuludq	32*8-128($ap), $ACC0, $ACC7
411e1051a39Sopenharmony_ci	vmovdqu		$ACC5, 32*14-448($tp1)
412e1051a39Sopenharmony_ci	vpaddq		32*16-448($tp1), $ACC7, $ACC7
413e1051a39Sopenharmony_ci	vmovdqu		$ACC6, 32*15-448($tp1)
414e1051a39Sopenharmony_ci	vmovdqu		$ACC7, 32*16-448($tp1)
415e1051a39Sopenharmony_ci	lea		8($tp1), $tp1
416e1051a39Sopenharmony_ci
417e1051a39Sopenharmony_ci	dec	$i
418e1051a39Sopenharmony_ci	jnz	.LOOP_SQR_1024
419e1051a39Sopenharmony_ci___
420e1051a39Sopenharmony_ci$ZERO = $ACC9;
421e1051a39Sopenharmony_ci$TEMP0 = $B1;
422e1051a39Sopenharmony_ci$TEMP2 = $B2;
423e1051a39Sopenharmony_ci$TEMP3 = $Y1;
424e1051a39Sopenharmony_ci$TEMP4 = $Y2;
425e1051a39Sopenharmony_ci$code.=<<___;
426e1051a39Sopenharmony_ci	# we need to fix indices 32-39 to avoid overflow
427e1051a39Sopenharmony_ci	vmovdqu		32*8(%rsp), $ACC8		# 32*8-192($tp0),
428e1051a39Sopenharmony_ci	vmovdqu		32*9(%rsp), $ACC1		# 32*9-192($tp0)
429e1051a39Sopenharmony_ci	vmovdqu		32*10(%rsp), $ACC2		# 32*10-192($tp0)
430e1051a39Sopenharmony_ci	lea		192(%rsp), $tp0			# 64+128=192
431e1051a39Sopenharmony_ci
432e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC8, $TEMP1
433e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC8, $ACC8
434e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC1, $TEMP2
435e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC1, $ACC1
436e1051a39Sopenharmony_ci
437e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP1, $TEMP1
438e1051a39Sopenharmony_ci	vpxor		$ZERO, $ZERO, $ZERO
439e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP2, $TEMP2
440e1051a39Sopenharmony_ci
441e1051a39Sopenharmony_ci	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
442e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
443e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC8, $ACC8
444e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP2, $ZERO, $TEMP2
445e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC1, $ACC1
446e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC2, $ACC2
447e1051a39Sopenharmony_ci	vmovdqu		$ACC1, 32*9-192($tp0)
448e1051a39Sopenharmony_ci	vmovdqu		$ACC2, 32*10-192($tp0)
449e1051a39Sopenharmony_ci
450e1051a39Sopenharmony_ci	mov	(%rsp), %rax
451e1051a39Sopenharmony_ci	mov	8(%rsp), $r1
452e1051a39Sopenharmony_ci	mov	16(%rsp), $r2
453e1051a39Sopenharmony_ci	mov	24(%rsp), $r3
454e1051a39Sopenharmony_ci	vmovdqu	32*1(%rsp), $ACC1
455e1051a39Sopenharmony_ci	vmovdqu	32*2-192($tp0), $ACC2
456e1051a39Sopenharmony_ci	vmovdqu	32*3-192($tp0), $ACC3
457e1051a39Sopenharmony_ci	vmovdqu	32*4-192($tp0), $ACC4
458e1051a39Sopenharmony_ci	vmovdqu	32*5-192($tp0), $ACC5
459e1051a39Sopenharmony_ci	vmovdqu	32*6-192($tp0), $ACC6
460e1051a39Sopenharmony_ci	vmovdqu	32*7-192($tp0), $ACC7
461e1051a39Sopenharmony_ci
462e1051a39Sopenharmony_ci	mov	%rax, $r0
463e1051a39Sopenharmony_ci	imull	$n0, %eax
464e1051a39Sopenharmony_ci	and	\$0x1fffffff, %eax
465e1051a39Sopenharmony_ci	vmovd	%eax, $Y1
466e1051a39Sopenharmony_ci
467e1051a39Sopenharmony_ci	mov	%rax, %rdx
468e1051a39Sopenharmony_ci	imulq	-128($np), %rax
469e1051a39Sopenharmony_ci	 vpbroadcastq	$Y1, $Y1
470e1051a39Sopenharmony_ci	add	%rax, $r0
471e1051a39Sopenharmony_ci	mov	%rdx, %rax
472e1051a39Sopenharmony_ci	imulq	8-128($np), %rax
473e1051a39Sopenharmony_ci	shr	\$29, $r0
474e1051a39Sopenharmony_ci	add	%rax, $r1
475e1051a39Sopenharmony_ci	mov	%rdx, %rax
476e1051a39Sopenharmony_ci	imulq	16-128($np), %rax
477e1051a39Sopenharmony_ci	add	$r0, $r1
478e1051a39Sopenharmony_ci	add	%rax, $r2
479e1051a39Sopenharmony_ci	imulq	24-128($np), %rdx
480e1051a39Sopenharmony_ci	add	%rdx, $r3
481e1051a39Sopenharmony_ci
482e1051a39Sopenharmony_ci	mov	$r1, %rax
483e1051a39Sopenharmony_ci	imull	$n0, %eax
484e1051a39Sopenharmony_ci	and	\$0x1fffffff, %eax
485e1051a39Sopenharmony_ci
486e1051a39Sopenharmony_ci	mov \$9, $i
487e1051a39Sopenharmony_ci	jmp .LOOP_REDUCE_1024
488e1051a39Sopenharmony_ci
489e1051a39Sopenharmony_ci.align	32
490e1051a39Sopenharmony_ci.LOOP_REDUCE_1024:
491e1051a39Sopenharmony_ci	vmovd	%eax, $Y2
492e1051a39Sopenharmony_ci	vpbroadcastq	$Y2, $Y2
493e1051a39Sopenharmony_ci
494e1051a39Sopenharmony_ci	vpmuludq	32*1-128($np), $Y1, $TEMP0
495e1051a39Sopenharmony_ci	 mov	%rax, %rdx
496e1051a39Sopenharmony_ci	 imulq	-128($np), %rax
497e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC1, $ACC1
498e1051a39Sopenharmony_ci	 add	%rax, $r1
499e1051a39Sopenharmony_ci	vpmuludq	32*2-128($np), $Y1, $TEMP1
500e1051a39Sopenharmony_ci	 mov	%rdx, %rax
501e1051a39Sopenharmony_ci	 imulq	8-128($np), %rax
502e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC2, $ACC2
503e1051a39Sopenharmony_ci	vpmuludq	32*3-128($np), $Y1, $TEMP2
504e1051a39Sopenharmony_ci	 .byte	0x67
505e1051a39Sopenharmony_ci	 add	%rax, $r2
506e1051a39Sopenharmony_ci	 .byte	0x67
507e1051a39Sopenharmony_ci	 mov	%rdx, %rax
508e1051a39Sopenharmony_ci	 imulq	16-128($np), %rax
509e1051a39Sopenharmony_ci	 shr	\$29, $r1
510e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC3, $ACC3
511e1051a39Sopenharmony_ci	vpmuludq	32*4-128($np), $Y1, $TEMP0
512e1051a39Sopenharmony_ci	 add	%rax, $r3
513e1051a39Sopenharmony_ci	 add	$r1, $r2
514e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC4, $ACC4
515e1051a39Sopenharmony_ci	vpmuludq	32*5-128($np), $Y1, $TEMP1
516e1051a39Sopenharmony_ci	 mov	$r2, %rax
517e1051a39Sopenharmony_ci	 imull	$n0, %eax
518e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC5, $ACC5
519e1051a39Sopenharmony_ci	vpmuludq	32*6-128($np), $Y1, $TEMP2
520e1051a39Sopenharmony_ci	 and	\$0x1fffffff, %eax
521e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC6, $ACC6
522e1051a39Sopenharmony_ci	vpmuludq	32*7-128($np), $Y1, $TEMP0
523e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC7, $ACC7
524e1051a39Sopenharmony_ci	vpmuludq	32*8-128($np), $Y1, $TEMP1
525e1051a39Sopenharmony_ci	 vmovd	%eax, $Y1
526e1051a39Sopenharmony_ci	 #vmovdqu	32*1-8-128($np), $TEMP2		# moved below
527e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC8, $ACC8
528e1051a39Sopenharmony_ci	 #vmovdqu	32*2-8-128($np), $TEMP0		# moved below
529e1051a39Sopenharmony_ci	 vpbroadcastq	$Y1, $Y1
530e1051a39Sopenharmony_ci
531e1051a39Sopenharmony_ci	vpmuludq	32*1-8-128($np), $Y2, $TEMP2	# see above
532e1051a39Sopenharmony_ci	vmovdqu		32*3-8-128($np), $TEMP1
533e1051a39Sopenharmony_ci	 mov	%rax, %rdx
534e1051a39Sopenharmony_ci	 imulq	-128($np), %rax
535e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC1, $ACC1
536e1051a39Sopenharmony_ci	vpmuludq	32*2-8-128($np), $Y2, $TEMP0	# see above
537e1051a39Sopenharmony_ci	vmovdqu		32*4-8-128($np), $TEMP2
538e1051a39Sopenharmony_ci	 add	%rax, $r2
539e1051a39Sopenharmony_ci	 mov	%rdx, %rax
540e1051a39Sopenharmony_ci	 imulq	8-128($np), %rax
541e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC2, $ACC2
542e1051a39Sopenharmony_ci	 add	$r3, %rax
543e1051a39Sopenharmony_ci	 shr	\$29, $r2
544e1051a39Sopenharmony_ci	vpmuludq	$Y2, $TEMP1, $TEMP1
545e1051a39Sopenharmony_ci	vmovdqu		32*5-8-128($np), $TEMP0
546e1051a39Sopenharmony_ci	 add	$r2, %rax
547e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC3, $ACC3
548e1051a39Sopenharmony_ci	vpmuludq	$Y2, $TEMP2, $TEMP2
549e1051a39Sopenharmony_ci	vmovdqu		32*6-8-128($np), $TEMP1
550e1051a39Sopenharmony_ci	 .byte	0x67
551e1051a39Sopenharmony_ci	 mov	%rax, $r3
552e1051a39Sopenharmony_ci	 imull	$n0, %eax
553e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC4, $ACC4
554e1051a39Sopenharmony_ci	vpmuludq	$Y2, $TEMP0, $TEMP0
555e1051a39Sopenharmony_ci	.byte	0xc4,0x41,0x7e,0x6f,0x9d,0x58,0x00,0x00,0x00	# vmovdqu		32*7-8-128($np), $TEMP2
556e1051a39Sopenharmony_ci	 and	\$0x1fffffff, %eax
557e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC5, $ACC5
558e1051a39Sopenharmony_ci	vpmuludq	$Y2, $TEMP1, $TEMP1
559e1051a39Sopenharmony_ci	vmovdqu		32*8-8-128($np), $TEMP0
560e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC6, $ACC6
561e1051a39Sopenharmony_ci	vpmuludq	$Y2, $TEMP2, $TEMP2
562e1051a39Sopenharmony_ci	vmovdqu		32*9-8-128($np), $ACC9
563e1051a39Sopenharmony_ci	 vmovd	%eax, $ACC0			# borrow ACC0 for Y2
564e1051a39Sopenharmony_ci	 imulq	-128($np), %rax
565e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC7, $ACC7
566e1051a39Sopenharmony_ci	vpmuludq	$Y2, $TEMP0, $TEMP0
567e1051a39Sopenharmony_ci	 vmovdqu	32*1-16-128($np), $TEMP1
568e1051a39Sopenharmony_ci	 vpbroadcastq	$ACC0, $ACC0
569e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC8, $ACC8
570e1051a39Sopenharmony_ci	vpmuludq	$Y2, $ACC9, $ACC9
571e1051a39Sopenharmony_ci	 vmovdqu	32*2-16-128($np), $TEMP2
572e1051a39Sopenharmony_ci	 add	%rax, $r3
573e1051a39Sopenharmony_ci
574e1051a39Sopenharmony_ci___
575e1051a39Sopenharmony_ci($ACC0,$Y2)=($Y2,$ACC0);
576e1051a39Sopenharmony_ci$code.=<<___;
577e1051a39Sopenharmony_ci	 vmovdqu	32*1-24-128($np), $ACC0
578e1051a39Sopenharmony_ci	vpmuludq	$Y1, $TEMP1, $TEMP1
579e1051a39Sopenharmony_ci	vmovdqu		32*3-16-128($np), $TEMP0
580e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC1, $ACC1
581e1051a39Sopenharmony_ci	 vpmuludq	$Y2, $ACC0, $ACC0
582e1051a39Sopenharmony_ci	vpmuludq	$Y1, $TEMP2, $TEMP2
583e1051a39Sopenharmony_ci	.byte	0xc4,0x41,0x7e,0x6f,0xb5,0xf0,0xff,0xff,0xff	# vmovdqu		32*4-16-128($np), $TEMP1
584e1051a39Sopenharmony_ci	 vpaddq		$ACC1, $ACC0, $ACC0
585e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC2, $ACC2
586e1051a39Sopenharmony_ci	vpmuludq	$Y1, $TEMP0, $TEMP0
587e1051a39Sopenharmony_ci	vmovdqu		32*5-16-128($np), $TEMP2
588e1051a39Sopenharmony_ci	 .byte	0x67
589e1051a39Sopenharmony_ci	 vmovq		$ACC0, %rax
590e1051a39Sopenharmony_ci	 vmovdqu	$ACC0, (%rsp)		# transfer $r0-$r3
591e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC3, $ACC3
592e1051a39Sopenharmony_ci	vpmuludq	$Y1, $TEMP1, $TEMP1
593e1051a39Sopenharmony_ci	vmovdqu		32*6-16-128($np), $TEMP0
594e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC4, $ACC4
595e1051a39Sopenharmony_ci	vpmuludq	$Y1, $TEMP2, $TEMP2
596e1051a39Sopenharmony_ci	vmovdqu		32*7-16-128($np), $TEMP1
597e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC5, $ACC5
598e1051a39Sopenharmony_ci	vpmuludq	$Y1, $TEMP0, $TEMP0
599e1051a39Sopenharmony_ci	vmovdqu		32*8-16-128($np), $TEMP2
600e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC6, $ACC6
601e1051a39Sopenharmony_ci	vpmuludq	$Y1, $TEMP1, $TEMP1
602e1051a39Sopenharmony_ci	 shr	\$29, $r3
603e1051a39Sopenharmony_ci	vmovdqu		32*9-16-128($np), $TEMP0
604e1051a39Sopenharmony_ci	 add	$r3, %rax
605e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC7, $ACC7
606e1051a39Sopenharmony_ci	vpmuludq	$Y1, $TEMP2, $TEMP2
607e1051a39Sopenharmony_ci	 #vmovdqu	32*2-24-128($np), $TEMP1	# moved below
608e1051a39Sopenharmony_ci	 mov	%rax, $r0
609e1051a39Sopenharmony_ci	 imull	$n0, %eax
610e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC8, $ACC8
611e1051a39Sopenharmony_ci	vpmuludq	$Y1, $TEMP0, $TEMP0
612e1051a39Sopenharmony_ci	 and	\$0x1fffffff, %eax
613e1051a39Sopenharmony_ci	 vmovd	%eax, $Y1
614e1051a39Sopenharmony_ci	 vmovdqu	32*3-24-128($np), $TEMP2
615e1051a39Sopenharmony_ci	.byte	0x67
616e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC9, $ACC9
617e1051a39Sopenharmony_ci	 vpbroadcastq	$Y1, $Y1
618e1051a39Sopenharmony_ci
619e1051a39Sopenharmony_ci	vpmuludq	32*2-24-128($np), $Y2, $TEMP1	# see above
620e1051a39Sopenharmony_ci	vmovdqu		32*4-24-128($np), $TEMP0
621e1051a39Sopenharmony_ci	 mov	%rax, %rdx
622e1051a39Sopenharmony_ci	 imulq	-128($np), %rax
623e1051a39Sopenharmony_ci	 mov	8(%rsp), $r1
624e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC2, $ACC1
625e1051a39Sopenharmony_ci	vpmuludq	$Y2, $TEMP2, $TEMP2
626e1051a39Sopenharmony_ci	vmovdqu		32*5-24-128($np), $TEMP1
627e1051a39Sopenharmony_ci	 add	%rax, $r0
628e1051a39Sopenharmony_ci	 mov	%rdx, %rax
629e1051a39Sopenharmony_ci	 imulq	8-128($np), %rax
630e1051a39Sopenharmony_ci	 .byte	0x67
631e1051a39Sopenharmony_ci	 shr	\$29, $r0
632e1051a39Sopenharmony_ci	 mov	16(%rsp), $r2
633e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC3, $ACC2
634e1051a39Sopenharmony_ci	vpmuludq	$Y2, $TEMP0, $TEMP0
635e1051a39Sopenharmony_ci	vmovdqu		32*6-24-128($np), $TEMP2
636e1051a39Sopenharmony_ci	 add	%rax, $r1
637e1051a39Sopenharmony_ci	 mov	%rdx, %rax
638e1051a39Sopenharmony_ci	 imulq	16-128($np), %rax
639e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC4, $ACC3
640e1051a39Sopenharmony_ci	vpmuludq	$Y2, $TEMP1, $TEMP1
641e1051a39Sopenharmony_ci	vmovdqu		32*7-24-128($np), $TEMP0
642e1051a39Sopenharmony_ci	 imulq	24-128($np), %rdx		# future $r3
643e1051a39Sopenharmony_ci	 add	%rax, $r2
644e1051a39Sopenharmony_ci	 lea	($r0,$r1), %rax
645e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC5, $ACC4
646e1051a39Sopenharmony_ci	vpmuludq	$Y2, $TEMP2, $TEMP2
647e1051a39Sopenharmony_ci	vmovdqu		32*8-24-128($np), $TEMP1
648e1051a39Sopenharmony_ci	 mov	%rax, $r1
649e1051a39Sopenharmony_ci	 imull	$n0, %eax
650e1051a39Sopenharmony_ci	vpmuludq	$Y2, $TEMP0, $TEMP0
651e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC6, $ACC5
652e1051a39Sopenharmony_ci	vmovdqu		32*9-24-128($np), $TEMP2
653e1051a39Sopenharmony_ci	 and	\$0x1fffffff, %eax
654e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC7, $ACC6
655e1051a39Sopenharmony_ci	vpmuludq	$Y2, $TEMP1, $TEMP1
656e1051a39Sopenharmony_ci	 add	24(%rsp), %rdx
657e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC8, $ACC7
658e1051a39Sopenharmony_ci	vpmuludq	$Y2, $TEMP2, $TEMP2
659e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC9, $ACC8
660e1051a39Sopenharmony_ci	 vmovq	$r3, $ACC9
661e1051a39Sopenharmony_ci	 mov	%rdx, $r3
662e1051a39Sopenharmony_ci
663e1051a39Sopenharmony_ci	dec	$i
664e1051a39Sopenharmony_ci	jnz	.LOOP_REDUCE_1024
665e1051a39Sopenharmony_ci___
666e1051a39Sopenharmony_ci($ACC0,$Y2)=($Y2,$ACC0);
667e1051a39Sopenharmony_ci$code.=<<___;
668e1051a39Sopenharmony_ci	lea	448(%rsp), $tp1			# size optimization
669e1051a39Sopenharmony_ci	vpaddq	$ACC9, $Y2, $ACC0
670e1051a39Sopenharmony_ci	vpxor	$ZERO, $ZERO, $ZERO
671e1051a39Sopenharmony_ci
672e1051a39Sopenharmony_ci	vpaddq		32*9-192($tp0), $ACC0, $ACC0
673e1051a39Sopenharmony_ci	vpaddq		32*10-448($tp1), $ACC1, $ACC1
674e1051a39Sopenharmony_ci	vpaddq		32*11-448($tp1), $ACC2, $ACC2
675e1051a39Sopenharmony_ci	vpaddq		32*12-448($tp1), $ACC3, $ACC3
676e1051a39Sopenharmony_ci	vpaddq		32*13-448($tp1), $ACC4, $ACC4
677e1051a39Sopenharmony_ci	vpaddq		32*14-448($tp1), $ACC5, $ACC5
678e1051a39Sopenharmony_ci	vpaddq		32*15-448($tp1), $ACC6, $ACC6
679e1051a39Sopenharmony_ci	vpaddq		32*16-448($tp1), $ACC7, $ACC7
680e1051a39Sopenharmony_ci	vpaddq		32*17-448($tp1), $ACC8, $ACC8
681e1051a39Sopenharmony_ci
682e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC0, $TEMP1
683e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC0, $ACC0
684e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC1, $TEMP2
685e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC1, $ACC1
686e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC2, $TEMP3
687e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP1, $TEMP1
688e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC2, $ACC2
689e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC3, $TEMP4
690e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP2, $TEMP2
691e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC3, $ACC3
692e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP3, $TEMP3
693e1051a39Sopenharmony_ci
694e1051a39Sopenharmony_ci	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
695e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP4, $TEMP4
696e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
697e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC0, $ACC0
698e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
699e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC1, $ACC1
700e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
701e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC2, $ACC2
702e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP4, $ZERO, $TEMP4
703e1051a39Sopenharmony_ci	vpaddq		$TEMP3, $ACC3, $ACC3
704e1051a39Sopenharmony_ci	vpaddq		$TEMP4, $ACC4, $ACC4
705e1051a39Sopenharmony_ci
706e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC0, $TEMP1
707e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC0, $ACC0
708e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC1, $TEMP2
709e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC1, $ACC1
710e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC2, $TEMP3
711e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP1, $TEMP1
712e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC2, $ACC2
713e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC3, $TEMP4
714e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP2, $TEMP2
715e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC3, $ACC3
716e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP3, $TEMP3
717e1051a39Sopenharmony_ci
718e1051a39Sopenharmony_ci	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
719e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP4, $TEMP4
720e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
721e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC0, $ACC0
722e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
723e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC1, $ACC1
724e1051a39Sopenharmony_ci	vmovdqu		$ACC0, 32*0-128($rp)
725e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
726e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC2, $ACC2
727e1051a39Sopenharmony_ci	vmovdqu		$ACC1, 32*1-128($rp)
728e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP4, $ZERO, $TEMP4
729e1051a39Sopenharmony_ci	vpaddq		$TEMP3, $ACC3, $ACC3
730e1051a39Sopenharmony_ci	vmovdqu		$ACC2, 32*2-128($rp)
731e1051a39Sopenharmony_ci	vpaddq		$TEMP4, $ACC4, $ACC4
732e1051a39Sopenharmony_ci	vmovdqu		$ACC3, 32*3-128($rp)
733e1051a39Sopenharmony_ci___
734e1051a39Sopenharmony_ci$TEMP5=$ACC0;
735e1051a39Sopenharmony_ci$code.=<<___;
736e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC4, $TEMP1
737e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC4, $ACC4
738e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC5, $TEMP2
739e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC5, $ACC5
740e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC6, $TEMP3
741e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP1, $TEMP1
742e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC6, $ACC6
743e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC7, $TEMP4
744e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP2, $TEMP2
745e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC7, $ACC7
746e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC8, $TEMP5
747e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP3, $TEMP3
748e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC8, $ACC8
749e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP4, $TEMP4
750e1051a39Sopenharmony_ci
751e1051a39Sopenharmony_ci	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
752e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP5, $TEMP5
753e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
754e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC4, $ACC4
755e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
756e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC5, $ACC5
757e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
758e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC6, $ACC6
759e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP4, $TEMP5, $TEMP4
760e1051a39Sopenharmony_ci	vpaddq		$TEMP3, $ACC7, $ACC7
761e1051a39Sopenharmony_ci	vpaddq		$TEMP4, $ACC8, $ACC8
762e1051a39Sopenharmony_ci
763e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC4, $TEMP1
764e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC4, $ACC4
765e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC5, $TEMP2
766e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC5, $ACC5
767e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC6, $TEMP3
768e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP1, $TEMP1
769e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC6, $ACC6
770e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC7, $TEMP4
771e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP2, $TEMP2
772e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC7, $ACC7
773e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC8, $TEMP5
774e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP3, $TEMP3
775e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC8, $ACC8
776e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP4, $TEMP4
777e1051a39Sopenharmony_ci
778e1051a39Sopenharmony_ci	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
779e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP5, $TEMP5
780e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
781e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC4, $ACC4
782e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
783e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC5, $ACC5
784e1051a39Sopenharmony_ci	vmovdqu		$ACC4, 32*4-128($rp)
785e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
786e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC6, $ACC6
787e1051a39Sopenharmony_ci	vmovdqu		$ACC5, 32*5-128($rp)
788e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP4, $TEMP5, $TEMP4
789e1051a39Sopenharmony_ci	vpaddq		$TEMP3, $ACC7, $ACC7
790e1051a39Sopenharmony_ci	vmovdqu		$ACC6, 32*6-128($rp)
791e1051a39Sopenharmony_ci	vpaddq		$TEMP4, $ACC8, $ACC8
792e1051a39Sopenharmony_ci	vmovdqu		$ACC7, 32*7-128($rp)
793e1051a39Sopenharmony_ci	vmovdqu		$ACC8, 32*8-128($rp)
794e1051a39Sopenharmony_ci
795e1051a39Sopenharmony_ci	mov	$rp, $ap
796e1051a39Sopenharmony_ci	dec	$rep
797e1051a39Sopenharmony_ci	jne	.LOOP_GRANDE_SQR_1024
798e1051a39Sopenharmony_ci
799e1051a39Sopenharmony_ci	vzeroall
800e1051a39Sopenharmony_ci	mov	%rbp, %rax
801e1051a39Sopenharmony_ci.cfi_def_cfa_register	%rax
802e1051a39Sopenharmony_ci___
803e1051a39Sopenharmony_ci$code.=<<___ if ($win64);
804e1051a39Sopenharmony_ci.Lsqr_1024_in_tail:
805e1051a39Sopenharmony_ci	movaps	-0xd8(%rax),%xmm6
806e1051a39Sopenharmony_ci	movaps	-0xc8(%rax),%xmm7
807e1051a39Sopenharmony_ci	movaps	-0xb8(%rax),%xmm8
808e1051a39Sopenharmony_ci	movaps	-0xa8(%rax),%xmm9
809e1051a39Sopenharmony_ci	movaps	-0x98(%rax),%xmm10
810e1051a39Sopenharmony_ci	movaps	-0x88(%rax),%xmm11
811e1051a39Sopenharmony_ci	movaps	-0x78(%rax),%xmm12
812e1051a39Sopenharmony_ci	movaps	-0x68(%rax),%xmm13
813e1051a39Sopenharmony_ci	movaps	-0x58(%rax),%xmm14
814e1051a39Sopenharmony_ci	movaps	-0x48(%rax),%xmm15
815e1051a39Sopenharmony_ci___
816e1051a39Sopenharmony_ci$code.=<<___;
817e1051a39Sopenharmony_ci	mov	-48(%rax),%r15
818e1051a39Sopenharmony_ci.cfi_restore	%r15
819e1051a39Sopenharmony_ci	mov	-40(%rax),%r14
820e1051a39Sopenharmony_ci.cfi_restore	%r14
821e1051a39Sopenharmony_ci	mov	-32(%rax),%r13
822e1051a39Sopenharmony_ci.cfi_restore	%r13
823e1051a39Sopenharmony_ci	mov	-24(%rax),%r12
824e1051a39Sopenharmony_ci.cfi_restore	%r12
825e1051a39Sopenharmony_ci	mov	-16(%rax),%rbp
826e1051a39Sopenharmony_ci.cfi_restore	%rbp
827e1051a39Sopenharmony_ci	mov	-8(%rax),%rbx
828e1051a39Sopenharmony_ci.cfi_restore	%rbx
829e1051a39Sopenharmony_ci	lea	(%rax),%rsp		# restore %rsp
830e1051a39Sopenharmony_ci.cfi_def_cfa_register	%rsp
831e1051a39Sopenharmony_ci.Lsqr_1024_epilogue:
832e1051a39Sopenharmony_ci	ret
833e1051a39Sopenharmony_ci.cfi_endproc
834e1051a39Sopenharmony_ci.size	rsaz_1024_sqr_avx2,.-rsaz_1024_sqr_avx2
835e1051a39Sopenharmony_ci___
836e1051a39Sopenharmony_ci}
837e1051a39Sopenharmony_ci
838e1051a39Sopenharmony_ci{ # void AMM_WW(
839e1051a39Sopenharmony_cimy $rp="%rdi";	# BN_ULONG *rp,
840e1051a39Sopenharmony_cimy $ap="%rsi";	# const BN_ULONG *ap,
841e1051a39Sopenharmony_cimy $bp="%rdx";	# const BN_ULONG *bp,
842e1051a39Sopenharmony_cimy $np="%rcx";	# const BN_ULONG *np,
843e1051a39Sopenharmony_cimy $n0="%r8d";	# unsigned int n0);
844e1051a39Sopenharmony_ci
845e1051a39Sopenharmony_ci# The registers that hold the accumulated redundant result
846e1051a39Sopenharmony_ci# The AMM works on 1024 bit operands, and redundant word size is 29
847e1051a39Sopenharmony_ci# Therefore: ceil(1024/29)/4 = 9
848e1051a39Sopenharmony_cimy $ACC0="%ymm0";
849e1051a39Sopenharmony_cimy $ACC1="%ymm1";
850e1051a39Sopenharmony_cimy $ACC2="%ymm2";
851e1051a39Sopenharmony_cimy $ACC3="%ymm3";
852e1051a39Sopenharmony_cimy $ACC4="%ymm4";
853e1051a39Sopenharmony_cimy $ACC5="%ymm5";
854e1051a39Sopenharmony_cimy $ACC6="%ymm6";
855e1051a39Sopenharmony_cimy $ACC7="%ymm7";
856e1051a39Sopenharmony_cimy $ACC8="%ymm8";
857e1051a39Sopenharmony_cimy $ACC9="%ymm9";
858e1051a39Sopenharmony_ci
859e1051a39Sopenharmony_ci# Registers that hold the broadcasted words of multiplier, currently used
860e1051a39Sopenharmony_cimy $Bi="%ymm10";
861e1051a39Sopenharmony_cimy $Yi="%ymm11";
862e1051a39Sopenharmony_ci
863e1051a39Sopenharmony_ci# Helper registers
864e1051a39Sopenharmony_cimy $TEMP0=$ACC0;
865e1051a39Sopenharmony_cimy $TEMP1="%ymm12";
866e1051a39Sopenharmony_cimy $TEMP2="%ymm13";
867e1051a39Sopenharmony_cimy $ZERO="%ymm14";
868e1051a39Sopenharmony_cimy $AND_MASK="%ymm15";
869e1051a39Sopenharmony_ci
870e1051a39Sopenharmony_ci# alu registers that hold the first words of the ACC
871e1051a39Sopenharmony_cimy $r0="%r9";
872e1051a39Sopenharmony_cimy $r1="%r10";
873e1051a39Sopenharmony_cimy $r2="%r11";
874e1051a39Sopenharmony_cimy $r3="%r12";
875e1051a39Sopenharmony_ci
876e1051a39Sopenharmony_cimy $i="%r14d";
877e1051a39Sopenharmony_cimy $tmp="%r15";
878e1051a39Sopenharmony_ci
879e1051a39Sopenharmony_ci$bp="%r13";	# reassigned argument
880e1051a39Sopenharmony_ci
881e1051a39Sopenharmony_ci$code.=<<___;
882e1051a39Sopenharmony_ci.globl	rsaz_1024_mul_avx2
883e1051a39Sopenharmony_ci.type	rsaz_1024_mul_avx2,\@function,5
884e1051a39Sopenharmony_ci.align	64
885e1051a39Sopenharmony_cirsaz_1024_mul_avx2:
886e1051a39Sopenharmony_ci.cfi_startproc
887e1051a39Sopenharmony_ci	lea	(%rsp), %rax
888e1051a39Sopenharmony_ci.cfi_def_cfa_register	%rax
889e1051a39Sopenharmony_ci	push	%rbx
890e1051a39Sopenharmony_ci.cfi_push	%rbx
891e1051a39Sopenharmony_ci	push	%rbp
892e1051a39Sopenharmony_ci.cfi_push	%rbp
893e1051a39Sopenharmony_ci	push	%r12
894e1051a39Sopenharmony_ci.cfi_push	%r12
895e1051a39Sopenharmony_ci	push	%r13
896e1051a39Sopenharmony_ci.cfi_push	%r13
897e1051a39Sopenharmony_ci	push	%r14
898e1051a39Sopenharmony_ci.cfi_push	%r14
899e1051a39Sopenharmony_ci	push	%r15
900e1051a39Sopenharmony_ci.cfi_push	%r15
901e1051a39Sopenharmony_ci___
902e1051a39Sopenharmony_ci$code.=<<___ if ($win64);
903e1051a39Sopenharmony_ci	vzeroupper
904e1051a39Sopenharmony_ci	lea	-0xa8(%rsp),%rsp
905e1051a39Sopenharmony_ci	vmovaps	%xmm6,-0xd8(%rax)
906e1051a39Sopenharmony_ci	vmovaps	%xmm7,-0xc8(%rax)
907e1051a39Sopenharmony_ci	vmovaps	%xmm8,-0xb8(%rax)
908e1051a39Sopenharmony_ci	vmovaps	%xmm9,-0xa8(%rax)
909e1051a39Sopenharmony_ci	vmovaps	%xmm10,-0x98(%rax)
910e1051a39Sopenharmony_ci	vmovaps	%xmm11,-0x88(%rax)
911e1051a39Sopenharmony_ci	vmovaps	%xmm12,-0x78(%rax)
912e1051a39Sopenharmony_ci	vmovaps	%xmm13,-0x68(%rax)
913e1051a39Sopenharmony_ci	vmovaps	%xmm14,-0x58(%rax)
914e1051a39Sopenharmony_ci	vmovaps	%xmm15,-0x48(%rax)
915e1051a39Sopenharmony_ci.Lmul_1024_body:
916e1051a39Sopenharmony_ci___
917e1051a39Sopenharmony_ci$code.=<<___;
918e1051a39Sopenharmony_ci	mov	%rax,%rbp
919e1051a39Sopenharmony_ci.cfi_def_cfa_register	%rbp
920e1051a39Sopenharmony_ci	vzeroall
921e1051a39Sopenharmony_ci	mov	%rdx, $bp	# reassigned argument
922e1051a39Sopenharmony_ci	sub	\$64,%rsp
923e1051a39Sopenharmony_ci
924e1051a39Sopenharmony_ci	# unaligned 256-bit load that crosses page boundary can
925e1051a39Sopenharmony_ci	# cause severe performance degradation here, so if $ap does
926e1051a39Sopenharmony_ci	# cross page boundary, swap it with $bp [meaning that caller
927e1051a39Sopenharmony_ci	# is advised to lay down $ap and $bp next to each other, so
928e1051a39Sopenharmony_ci	# that only one can cross page boundary].
929e1051a39Sopenharmony_ci	.byte	0x67,0x67
930e1051a39Sopenharmony_ci	mov	$ap, $tmp
931e1051a39Sopenharmony_ci	and	\$4095, $tmp
932e1051a39Sopenharmony_ci	add	\$32*10, $tmp
933e1051a39Sopenharmony_ci	shr	\$12, $tmp
934e1051a39Sopenharmony_ci	mov	$ap, $tmp
935e1051a39Sopenharmony_ci	cmovnz	$bp, $ap
936e1051a39Sopenharmony_ci	cmovnz	$tmp, $bp
937e1051a39Sopenharmony_ci
938e1051a39Sopenharmony_ci	mov	$np, $tmp
939e1051a39Sopenharmony_ci	sub	\$-128,$ap	# size optimization
940e1051a39Sopenharmony_ci	sub	\$-128,$np
941e1051a39Sopenharmony_ci	sub	\$-128,$rp
942e1051a39Sopenharmony_ci
943e1051a39Sopenharmony_ci	and	\$4095, $tmp	# see if $np crosses page
944e1051a39Sopenharmony_ci	add	\$32*10, $tmp
945e1051a39Sopenharmony_ci	.byte	0x67,0x67
946e1051a39Sopenharmony_ci	shr	\$12, $tmp
947e1051a39Sopenharmony_ci	jz	.Lmul_1024_no_n_copy
948e1051a39Sopenharmony_ci
949e1051a39Sopenharmony_ci	# unaligned 256-bit load that crosses page boundary can
950e1051a39Sopenharmony_ci	# cause severe performance degradation here, so if $np does
951e1051a39Sopenharmony_ci	# cross page boundary, copy it to stack and make sure stack
952e1051a39Sopenharmony_ci	# frame doesn't...
953e1051a39Sopenharmony_ci	sub		\$32*10,%rsp
954e1051a39Sopenharmony_ci	vmovdqu		32*0-128($np), $ACC0
955e1051a39Sopenharmony_ci	and		\$-512, %rsp
956e1051a39Sopenharmony_ci	vmovdqu		32*1-128($np), $ACC1
957e1051a39Sopenharmony_ci	vmovdqu		32*2-128($np), $ACC2
958e1051a39Sopenharmony_ci	vmovdqu		32*3-128($np), $ACC3
959e1051a39Sopenharmony_ci	vmovdqu		32*4-128($np), $ACC4
960e1051a39Sopenharmony_ci	vmovdqu		32*5-128($np), $ACC5
961e1051a39Sopenharmony_ci	vmovdqu		32*6-128($np), $ACC6
962e1051a39Sopenharmony_ci	vmovdqu		32*7-128($np), $ACC7
963e1051a39Sopenharmony_ci	vmovdqu		32*8-128($np), $ACC8
964e1051a39Sopenharmony_ci	lea		64+128(%rsp),$np
965e1051a39Sopenharmony_ci	vmovdqu		$ACC0, 32*0-128($np)
966e1051a39Sopenharmony_ci	vpxor		$ACC0, $ACC0, $ACC0
967e1051a39Sopenharmony_ci	vmovdqu		$ACC1, 32*1-128($np)
968e1051a39Sopenharmony_ci	vpxor		$ACC1, $ACC1, $ACC1
969e1051a39Sopenharmony_ci	vmovdqu		$ACC2, 32*2-128($np)
970e1051a39Sopenharmony_ci	vpxor		$ACC2, $ACC2, $ACC2
971e1051a39Sopenharmony_ci	vmovdqu		$ACC3, 32*3-128($np)
972e1051a39Sopenharmony_ci	vpxor		$ACC3, $ACC3, $ACC3
973e1051a39Sopenharmony_ci	vmovdqu		$ACC4, 32*4-128($np)
974e1051a39Sopenharmony_ci	vpxor		$ACC4, $ACC4, $ACC4
975e1051a39Sopenharmony_ci	vmovdqu		$ACC5, 32*5-128($np)
976e1051a39Sopenharmony_ci	vpxor		$ACC5, $ACC5, $ACC5
977e1051a39Sopenharmony_ci	vmovdqu		$ACC6, 32*6-128($np)
978e1051a39Sopenharmony_ci	vpxor		$ACC6, $ACC6, $ACC6
979e1051a39Sopenharmony_ci	vmovdqu		$ACC7, 32*7-128($np)
980e1051a39Sopenharmony_ci	vpxor		$ACC7, $ACC7, $ACC7
981e1051a39Sopenharmony_ci	vmovdqu		$ACC8, 32*8-128($np)
982e1051a39Sopenharmony_ci	vmovdqa		$ACC0, $ACC8
983e1051a39Sopenharmony_ci	vmovdqu		$ACC9, 32*9-128($np)	# $ACC9 is zero after vzeroall
984e1051a39Sopenharmony_ci.Lmul_1024_no_n_copy:
985e1051a39Sopenharmony_ci	and	\$-64,%rsp
986e1051a39Sopenharmony_ci
987e1051a39Sopenharmony_ci	mov	($bp), %rbx
988e1051a39Sopenharmony_ci	vpbroadcastq ($bp), $Bi
989e1051a39Sopenharmony_ci	vmovdqu	$ACC0, (%rsp)			# clear top of stack
990e1051a39Sopenharmony_ci	xor	$r0, $r0
991e1051a39Sopenharmony_ci	.byte	0x67
992e1051a39Sopenharmony_ci	xor	$r1, $r1
993e1051a39Sopenharmony_ci	xor	$r2, $r2
994e1051a39Sopenharmony_ci	xor	$r3, $r3
995e1051a39Sopenharmony_ci
996e1051a39Sopenharmony_ci	vmovdqu	.Land_mask(%rip), $AND_MASK
997e1051a39Sopenharmony_ci	mov	\$9, $i
998e1051a39Sopenharmony_ci	vmovdqu	$ACC9, 32*9-128($rp)		# $ACC9 is zero after vzeroall
999e1051a39Sopenharmony_ci	jmp	.Loop_mul_1024
1000e1051a39Sopenharmony_ci
1001e1051a39Sopenharmony_ci.align	32
1002e1051a39Sopenharmony_ci.Loop_mul_1024:
1003e1051a39Sopenharmony_ci	 vpsrlq		\$29, $ACC3, $ACC9		# correct $ACC3(*)
1004e1051a39Sopenharmony_ci	mov	%rbx, %rax
1005e1051a39Sopenharmony_ci	imulq	-128($ap), %rax
1006e1051a39Sopenharmony_ci	add	$r0, %rax
1007e1051a39Sopenharmony_ci	mov	%rbx, $r1
1008e1051a39Sopenharmony_ci	imulq	8-128($ap), $r1
1009e1051a39Sopenharmony_ci	add	8(%rsp), $r1
1010e1051a39Sopenharmony_ci
1011e1051a39Sopenharmony_ci	mov	%rax, $r0
1012e1051a39Sopenharmony_ci	imull	$n0, %eax
1013e1051a39Sopenharmony_ci	and	\$0x1fffffff, %eax
1014e1051a39Sopenharmony_ci
1015e1051a39Sopenharmony_ci	 mov	%rbx, $r2
1016e1051a39Sopenharmony_ci	 imulq	16-128($ap), $r2
1017e1051a39Sopenharmony_ci	 add	16(%rsp), $r2
1018e1051a39Sopenharmony_ci
1019e1051a39Sopenharmony_ci	 mov	%rbx, $r3
1020e1051a39Sopenharmony_ci	 imulq	24-128($ap), $r3
1021e1051a39Sopenharmony_ci	 add	24(%rsp), $r3
1022e1051a39Sopenharmony_ci	vpmuludq	32*1-128($ap),$Bi,$TEMP0
1023e1051a39Sopenharmony_ci	 vmovd		%eax, $Yi
1024e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC1,$ACC1
1025e1051a39Sopenharmony_ci	vpmuludq	32*2-128($ap),$Bi,$TEMP1
1026e1051a39Sopenharmony_ci	 vpbroadcastq	$Yi, $Yi
1027e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC2,$ACC2
1028e1051a39Sopenharmony_ci	vpmuludq	32*3-128($ap),$Bi,$TEMP2
1029e1051a39Sopenharmony_ci	 vpand		$AND_MASK, $ACC3, $ACC3		# correct $ACC3
1030e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC3,$ACC3
1031e1051a39Sopenharmony_ci	vpmuludq	32*4-128($ap),$Bi,$TEMP0
1032e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC4,$ACC4
1033e1051a39Sopenharmony_ci	vpmuludq	32*5-128($ap),$Bi,$TEMP1
1034e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC5,$ACC5
1035e1051a39Sopenharmony_ci	vpmuludq	32*6-128($ap),$Bi,$TEMP2
1036e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC6,$ACC6
1037e1051a39Sopenharmony_ci	vpmuludq	32*7-128($ap),$Bi,$TEMP0
1038e1051a39Sopenharmony_ci	 vpermq		\$0x93, $ACC9, $ACC9		# correct $ACC3
1039e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC7,$ACC7
1040e1051a39Sopenharmony_ci	vpmuludq	32*8-128($ap),$Bi,$TEMP1
1041e1051a39Sopenharmony_ci	 vpbroadcastq	8($bp), $Bi
1042e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC8,$ACC8
1043e1051a39Sopenharmony_ci
1044e1051a39Sopenharmony_ci	mov	%rax,%rdx
1045e1051a39Sopenharmony_ci	imulq	-128($np),%rax
1046e1051a39Sopenharmony_ci	add	%rax,$r0
1047e1051a39Sopenharmony_ci	mov	%rdx,%rax
1048e1051a39Sopenharmony_ci	imulq	8-128($np),%rax
1049e1051a39Sopenharmony_ci	add	%rax,$r1
1050e1051a39Sopenharmony_ci	mov	%rdx,%rax
1051e1051a39Sopenharmony_ci	imulq	16-128($np),%rax
1052e1051a39Sopenharmony_ci	add	%rax,$r2
1053e1051a39Sopenharmony_ci	shr	\$29, $r0
1054e1051a39Sopenharmony_ci	imulq	24-128($np),%rdx
1055e1051a39Sopenharmony_ci	add	%rdx,$r3
1056e1051a39Sopenharmony_ci	add	$r0, $r1
1057e1051a39Sopenharmony_ci
1058e1051a39Sopenharmony_ci	vpmuludq	32*1-128($np),$Yi,$TEMP2
1059e1051a39Sopenharmony_ci	 vmovq		$Bi, %rbx
1060e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC1,$ACC1
1061e1051a39Sopenharmony_ci	vpmuludq	32*2-128($np),$Yi,$TEMP0
1062e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC2,$ACC2
1063e1051a39Sopenharmony_ci	vpmuludq	32*3-128($np),$Yi,$TEMP1
1064e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC3,$ACC3
1065e1051a39Sopenharmony_ci	vpmuludq	32*4-128($np),$Yi,$TEMP2
1066e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC4,$ACC4
1067e1051a39Sopenharmony_ci	vpmuludq	32*5-128($np),$Yi,$TEMP0
1068e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC5,$ACC5
1069e1051a39Sopenharmony_ci	vpmuludq	32*6-128($np),$Yi,$TEMP1
1070e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC6,$ACC6
1071e1051a39Sopenharmony_ci	vpmuludq	32*7-128($np),$Yi,$TEMP2
1072e1051a39Sopenharmony_ci	 vpblendd	\$3, $ZERO, $ACC9, $TEMP1	# correct $ACC3
1073e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC7,$ACC7
1074e1051a39Sopenharmony_ci	vpmuludq	32*8-128($np),$Yi,$TEMP0
1075e1051a39Sopenharmony_ci	 vpaddq		$TEMP1, $ACC3, $ACC3		# correct $ACC3
1076e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC8,$ACC8
1077e1051a39Sopenharmony_ci
1078e1051a39Sopenharmony_ci	mov	%rbx, %rax
1079e1051a39Sopenharmony_ci	imulq	-128($ap),%rax
1080e1051a39Sopenharmony_ci	add	%rax,$r1
1081e1051a39Sopenharmony_ci	 vmovdqu	-8+32*1-128($ap),$TEMP1
1082e1051a39Sopenharmony_ci	mov	%rbx, %rax
1083e1051a39Sopenharmony_ci	imulq	8-128($ap),%rax
1084e1051a39Sopenharmony_ci	add	%rax,$r2
1085e1051a39Sopenharmony_ci	 vmovdqu	-8+32*2-128($ap),$TEMP2
1086e1051a39Sopenharmony_ci
1087e1051a39Sopenharmony_ci	mov	$r1, %rax
1088e1051a39Sopenharmony_ci	 vpblendd	\$0xfc, $ZERO, $ACC9, $ACC9	# correct $ACC3
1089e1051a39Sopenharmony_ci	imull	$n0, %eax
1090e1051a39Sopenharmony_ci	 vpaddq		$ACC9,$ACC4,$ACC4		# correct $ACC3
1091e1051a39Sopenharmony_ci	and	\$0x1fffffff, %eax
1092e1051a39Sopenharmony_ci
1093e1051a39Sopenharmony_ci	 imulq	16-128($ap),%rbx
1094e1051a39Sopenharmony_ci	 add	%rbx,$r3
1095e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP1,$TEMP1
1096e1051a39Sopenharmony_ci	 vmovd		%eax, $Yi
1097e1051a39Sopenharmony_ci	vmovdqu		-8+32*3-128($ap),$TEMP0
1098e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC1,$ACC1
1099e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP2,$TEMP2
1100e1051a39Sopenharmony_ci	 vpbroadcastq	$Yi, $Yi
1101e1051a39Sopenharmony_ci	vmovdqu		-8+32*4-128($ap),$TEMP1
1102e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC2,$ACC2
1103e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP0,$TEMP0
1104e1051a39Sopenharmony_ci	vmovdqu		-8+32*5-128($ap),$TEMP2
1105e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC3,$ACC3
1106e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP1,$TEMP1
1107e1051a39Sopenharmony_ci	vmovdqu		-8+32*6-128($ap),$TEMP0
1108e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC4,$ACC4
1109e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP2,$TEMP2
1110e1051a39Sopenharmony_ci	vmovdqu		-8+32*7-128($ap),$TEMP1
1111e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC5,$ACC5
1112e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP0,$TEMP0
1113e1051a39Sopenharmony_ci	vmovdqu		-8+32*8-128($ap),$TEMP2
1114e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC6,$ACC6
1115e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP1,$TEMP1
1116e1051a39Sopenharmony_ci	vmovdqu		-8+32*9-128($ap),$ACC9
1117e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC7,$ACC7
1118e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP2,$TEMP2
1119e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC8,$ACC8
1120e1051a39Sopenharmony_ci	vpmuludq	$Bi,$ACC9,$ACC9
1121e1051a39Sopenharmony_ci	 vpbroadcastq	16($bp), $Bi
1122e1051a39Sopenharmony_ci
1123e1051a39Sopenharmony_ci	mov	%rax,%rdx
1124e1051a39Sopenharmony_ci	imulq	-128($np),%rax
1125e1051a39Sopenharmony_ci	add	%rax,$r1
1126e1051a39Sopenharmony_ci	 vmovdqu	-8+32*1-128($np),$TEMP0
1127e1051a39Sopenharmony_ci	mov	%rdx,%rax
1128e1051a39Sopenharmony_ci	imulq	8-128($np),%rax
1129e1051a39Sopenharmony_ci	add	%rax,$r2
1130e1051a39Sopenharmony_ci	 vmovdqu	-8+32*2-128($np),$TEMP1
1131e1051a39Sopenharmony_ci	shr	\$29, $r1
1132e1051a39Sopenharmony_ci	imulq	16-128($np),%rdx
1133e1051a39Sopenharmony_ci	add	%rdx,$r3
1134e1051a39Sopenharmony_ci	add	$r1, $r2
1135e1051a39Sopenharmony_ci
1136e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP0,$TEMP0
1137e1051a39Sopenharmony_ci	 vmovq		$Bi, %rbx
1138e1051a39Sopenharmony_ci	vmovdqu		-8+32*3-128($np),$TEMP2
1139e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC1,$ACC1
1140e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP1,$TEMP1
1141e1051a39Sopenharmony_ci	vmovdqu		-8+32*4-128($np),$TEMP0
1142e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC2,$ACC2
1143e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP2,$TEMP2
1144e1051a39Sopenharmony_ci	vmovdqu		-8+32*5-128($np),$TEMP1
1145e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC3,$ACC3
1146e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP0,$TEMP0
1147e1051a39Sopenharmony_ci	vmovdqu		-8+32*6-128($np),$TEMP2
1148e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC4,$ACC4
1149e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP1,$TEMP1
1150e1051a39Sopenharmony_ci	vmovdqu		-8+32*7-128($np),$TEMP0
1151e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC5,$ACC5
1152e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP2,$TEMP2
1153e1051a39Sopenharmony_ci	vmovdqu		-8+32*8-128($np),$TEMP1
1154e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC6,$ACC6
1155e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP0,$TEMP0
1156e1051a39Sopenharmony_ci	vmovdqu		-8+32*9-128($np),$TEMP2
1157e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC7,$ACC7
1158e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP1,$TEMP1
1159e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC8,$ACC8
1160e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP2,$TEMP2
1161e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC9,$ACC9
1162e1051a39Sopenharmony_ci
1163e1051a39Sopenharmony_ci	 vmovdqu	-16+32*1-128($ap),$TEMP0
1164e1051a39Sopenharmony_ci	mov	%rbx,%rax
1165e1051a39Sopenharmony_ci	imulq	-128($ap),%rax
1166e1051a39Sopenharmony_ci	add	$r2,%rax
1167e1051a39Sopenharmony_ci
1168e1051a39Sopenharmony_ci	 vmovdqu	-16+32*2-128($ap),$TEMP1
1169e1051a39Sopenharmony_ci	mov	%rax,$r2
1170e1051a39Sopenharmony_ci	imull	$n0, %eax
1171e1051a39Sopenharmony_ci	and	\$0x1fffffff, %eax
1172e1051a39Sopenharmony_ci
1173e1051a39Sopenharmony_ci	 imulq	8-128($ap),%rbx
1174e1051a39Sopenharmony_ci	 add	%rbx,$r3
1175e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP0,$TEMP0
1176e1051a39Sopenharmony_ci	 vmovd		%eax, $Yi
1177e1051a39Sopenharmony_ci	vmovdqu		-16+32*3-128($ap),$TEMP2
1178e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC1,$ACC1
1179e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP1,$TEMP1
1180e1051a39Sopenharmony_ci	 vpbroadcastq	$Yi, $Yi
1181e1051a39Sopenharmony_ci	vmovdqu		-16+32*4-128($ap),$TEMP0
1182e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC2,$ACC2
1183e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP2,$TEMP2
1184e1051a39Sopenharmony_ci	vmovdqu		-16+32*5-128($ap),$TEMP1
1185e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC3,$ACC3
1186e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP0,$TEMP0
1187e1051a39Sopenharmony_ci	vmovdqu		-16+32*6-128($ap),$TEMP2
1188e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC4,$ACC4
1189e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP1,$TEMP1
1190e1051a39Sopenharmony_ci	vmovdqu		-16+32*7-128($ap),$TEMP0
1191e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC5,$ACC5
1192e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP2,$TEMP2
1193e1051a39Sopenharmony_ci	vmovdqu		-16+32*8-128($ap),$TEMP1
1194e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC6,$ACC6
1195e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP0,$TEMP0
1196e1051a39Sopenharmony_ci	vmovdqu		-16+32*9-128($ap),$TEMP2
1197e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC7,$ACC7
1198e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP1,$TEMP1
1199e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC8,$ACC8
1200e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP2,$TEMP2
1201e1051a39Sopenharmony_ci	 vpbroadcastq	24($bp), $Bi
1202e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC9,$ACC9
1203e1051a39Sopenharmony_ci
1204e1051a39Sopenharmony_ci	 vmovdqu	-16+32*1-128($np),$TEMP0
1205e1051a39Sopenharmony_ci	mov	%rax,%rdx
1206e1051a39Sopenharmony_ci	imulq	-128($np),%rax
1207e1051a39Sopenharmony_ci	add	%rax,$r2
1208e1051a39Sopenharmony_ci	 vmovdqu	-16+32*2-128($np),$TEMP1
1209e1051a39Sopenharmony_ci	imulq	8-128($np),%rdx
1210e1051a39Sopenharmony_ci	add	%rdx,$r3
1211e1051a39Sopenharmony_ci	shr	\$29, $r2
1212e1051a39Sopenharmony_ci
1213e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP0,$TEMP0
1214e1051a39Sopenharmony_ci	 vmovq		$Bi, %rbx
1215e1051a39Sopenharmony_ci	vmovdqu		-16+32*3-128($np),$TEMP2
1216e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC1,$ACC1
1217e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP1,$TEMP1
1218e1051a39Sopenharmony_ci	vmovdqu		-16+32*4-128($np),$TEMP0
1219e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC2,$ACC2
1220e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP2,$TEMP2
1221e1051a39Sopenharmony_ci	vmovdqu		-16+32*5-128($np),$TEMP1
1222e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC3,$ACC3
1223e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP0,$TEMP0
1224e1051a39Sopenharmony_ci	vmovdqu		-16+32*6-128($np),$TEMP2
1225e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC4,$ACC4
1226e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP1,$TEMP1
1227e1051a39Sopenharmony_ci	vmovdqu		-16+32*7-128($np),$TEMP0
1228e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC5,$ACC5
1229e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP2,$TEMP2
1230e1051a39Sopenharmony_ci	vmovdqu		-16+32*8-128($np),$TEMP1
1231e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC6,$ACC6
1232e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP0,$TEMP0
1233e1051a39Sopenharmony_ci	vmovdqu		-16+32*9-128($np),$TEMP2
1234e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC7,$ACC7
1235e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP1,$TEMP1
1236e1051a39Sopenharmony_ci	 vmovdqu	-24+32*1-128($ap),$TEMP0
1237e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC8,$ACC8
1238e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP2,$TEMP2
1239e1051a39Sopenharmony_ci	 vmovdqu	-24+32*2-128($ap),$TEMP1
1240e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC9,$ACC9
1241e1051a39Sopenharmony_ci
1242e1051a39Sopenharmony_ci	add	$r2, $r3
1243e1051a39Sopenharmony_ci	imulq	-128($ap),%rbx
1244e1051a39Sopenharmony_ci	add	%rbx,$r3
1245e1051a39Sopenharmony_ci
1246e1051a39Sopenharmony_ci	mov	$r3, %rax
1247e1051a39Sopenharmony_ci	imull	$n0, %eax
1248e1051a39Sopenharmony_ci	and	\$0x1fffffff, %eax
1249e1051a39Sopenharmony_ci
1250e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP0,$TEMP0
1251e1051a39Sopenharmony_ci	 vmovd		%eax, $Yi
1252e1051a39Sopenharmony_ci	vmovdqu		-24+32*3-128($ap),$TEMP2
1253e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC1,$ACC1
1254e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP1,$TEMP1
1255e1051a39Sopenharmony_ci	 vpbroadcastq	$Yi, $Yi
1256e1051a39Sopenharmony_ci	vmovdqu		-24+32*4-128($ap),$TEMP0
1257e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC2,$ACC2
1258e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP2,$TEMP2
1259e1051a39Sopenharmony_ci	vmovdqu		-24+32*5-128($ap),$TEMP1
1260e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC3,$ACC3
1261e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP0,$TEMP0
1262e1051a39Sopenharmony_ci	vmovdqu		-24+32*6-128($ap),$TEMP2
1263e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC4,$ACC4
1264e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP1,$TEMP1
1265e1051a39Sopenharmony_ci	vmovdqu		-24+32*7-128($ap),$TEMP0
1266e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC5,$ACC5
1267e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP2,$TEMP2
1268e1051a39Sopenharmony_ci	vmovdqu		-24+32*8-128($ap),$TEMP1
1269e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC6,$ACC6
1270e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP0,$TEMP0
1271e1051a39Sopenharmony_ci	vmovdqu		-24+32*9-128($ap),$TEMP2
1272e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC7,$ACC7
1273e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP1,$TEMP1
1274e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC8,$ACC8
1275e1051a39Sopenharmony_ci	vpmuludq	$Bi,$TEMP2,$TEMP2
1276e1051a39Sopenharmony_ci	 vpbroadcastq	32($bp), $Bi
1277e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC9,$ACC9
1278e1051a39Sopenharmony_ci	 add		\$32, $bp			# $bp++
1279e1051a39Sopenharmony_ci
1280e1051a39Sopenharmony_ci	vmovdqu		-24+32*1-128($np),$TEMP0
1281e1051a39Sopenharmony_ci	imulq	-128($np),%rax
1282e1051a39Sopenharmony_ci	add	%rax,$r3
1283e1051a39Sopenharmony_ci	shr	\$29, $r3
1284e1051a39Sopenharmony_ci
1285e1051a39Sopenharmony_ci	vmovdqu		-24+32*2-128($np),$TEMP1
1286e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP0,$TEMP0
1287e1051a39Sopenharmony_ci	 vmovq		$Bi, %rbx
1288e1051a39Sopenharmony_ci	vmovdqu		-24+32*3-128($np),$TEMP2
1289e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC1,$ACC0		# $ACC0==$TEMP0
1290e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP1,$TEMP1
1291e1051a39Sopenharmony_ci	 vmovdqu	$ACC0, (%rsp)			# transfer $r0-$r3
1292e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC2,$ACC1
1293e1051a39Sopenharmony_ci	vmovdqu		-24+32*4-128($np),$TEMP0
1294e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP2,$TEMP2
1295e1051a39Sopenharmony_ci	vmovdqu		-24+32*5-128($np),$TEMP1
1296e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC3,$ACC2
1297e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP0,$TEMP0
1298e1051a39Sopenharmony_ci	vmovdqu		-24+32*6-128($np),$TEMP2
1299e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC4,$ACC3
1300e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP1,$TEMP1
1301e1051a39Sopenharmony_ci	vmovdqu		-24+32*7-128($np),$TEMP0
1302e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC5,$ACC4
1303e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP2,$TEMP2
1304e1051a39Sopenharmony_ci	vmovdqu		-24+32*8-128($np),$TEMP1
1305e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC6,$ACC5
1306e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP0,$TEMP0
1307e1051a39Sopenharmony_ci	vmovdqu		-24+32*9-128($np),$TEMP2
1308e1051a39Sopenharmony_ci	 mov	$r3, $r0
1309e1051a39Sopenharmony_ci	vpaddq		$TEMP0,$ACC7,$ACC6
1310e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP1,$TEMP1
1311e1051a39Sopenharmony_ci	 add	(%rsp), $r0
1312e1051a39Sopenharmony_ci	vpaddq		$TEMP1,$ACC8,$ACC7
1313e1051a39Sopenharmony_ci	vpmuludq	$Yi,$TEMP2,$TEMP2
1314e1051a39Sopenharmony_ci	 vmovq	$r3, $TEMP1
1315e1051a39Sopenharmony_ci	vpaddq		$TEMP2,$ACC9,$ACC8
1316e1051a39Sopenharmony_ci
1317e1051a39Sopenharmony_ci	dec	$i
1318e1051a39Sopenharmony_ci	jnz	.Loop_mul_1024
1319e1051a39Sopenharmony_ci___
1320e1051a39Sopenharmony_ci
1321e1051a39Sopenharmony_ci# (*)	Original implementation was correcting ACC1-ACC3 for overflow
1322e1051a39Sopenharmony_ci#	after 7 loop runs, or after 28 iterations, or 56 additions.
1323e1051a39Sopenharmony_ci#	But as we underutilize resources, it's possible to correct in
1324e1051a39Sopenharmony_ci#	each iteration with marginal performance loss. But then, as
1325e1051a39Sopenharmony_ci#	we do it in each iteration, we can correct less digits, and
1326e1051a39Sopenharmony_ci#	avoid performance penalties completely.
1327e1051a39Sopenharmony_ci
1328e1051a39Sopenharmony_ci$TEMP0 = $ACC9;
1329e1051a39Sopenharmony_ci$TEMP3 = $Bi;
1330e1051a39Sopenharmony_ci$TEMP4 = $Yi;
1331e1051a39Sopenharmony_ci$code.=<<___;
1332e1051a39Sopenharmony_ci	vpaddq		(%rsp), $TEMP1, $ACC0
1333e1051a39Sopenharmony_ci
1334e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC0, $TEMP1
1335e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC0, $ACC0
1336e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC1, $TEMP2
1337e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC1, $ACC1
1338e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC2, $TEMP3
1339e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP1, $TEMP1
1340e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC2, $ACC2
1341e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC3, $TEMP4
1342e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP2, $TEMP2
1343e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC3, $ACC3
1344e1051a39Sopenharmony_ci
1345e1051a39Sopenharmony_ci	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
1346e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP3, $TEMP3
1347e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
1348e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP4, $TEMP4
1349e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC0, $ACC0
1350e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
1351e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC1, $ACC1
1352e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
1353e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC2, $ACC2
1354e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP4, $ZERO, $TEMP4
1355e1051a39Sopenharmony_ci	vpaddq		$TEMP3, $ACC3, $ACC3
1356e1051a39Sopenharmony_ci	vpaddq		$TEMP4, $ACC4, $ACC4
1357e1051a39Sopenharmony_ci
1358e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC0, $TEMP1
1359e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC0, $ACC0
1360e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC1, $TEMP2
1361e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC1, $ACC1
1362e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC2, $TEMP3
1363e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP1, $TEMP1
1364e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC2, $ACC2
1365e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC3, $TEMP4
1366e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP2, $TEMP2
1367e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC3, $ACC3
1368e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP3, $TEMP3
1369e1051a39Sopenharmony_ci
1370e1051a39Sopenharmony_ci	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
1371e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP4, $TEMP4
1372e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
1373e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC0, $ACC0
1374e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
1375e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC1, $ACC1
1376e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
1377e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC2, $ACC2
1378e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP4, $ZERO, $TEMP4
1379e1051a39Sopenharmony_ci	vpaddq		$TEMP3, $ACC3, $ACC3
1380e1051a39Sopenharmony_ci	vpaddq		$TEMP4, $ACC4, $ACC4
1381e1051a39Sopenharmony_ci
1382e1051a39Sopenharmony_ci	vmovdqu		$ACC0, 0-128($rp)
1383e1051a39Sopenharmony_ci	vmovdqu		$ACC1, 32-128($rp)
1384e1051a39Sopenharmony_ci	vmovdqu		$ACC2, 64-128($rp)
1385e1051a39Sopenharmony_ci	vmovdqu		$ACC3, 96-128($rp)
1386e1051a39Sopenharmony_ci___
1387e1051a39Sopenharmony_ci
1388e1051a39Sopenharmony_ci$TEMP5=$ACC0;
1389e1051a39Sopenharmony_ci$code.=<<___;
1390e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC4, $TEMP1
1391e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC4, $ACC4
1392e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC5, $TEMP2
1393e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC5, $ACC5
1394e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC6, $TEMP3
1395e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP1, $TEMP1
1396e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC6, $ACC6
1397e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC7, $TEMP4
1398e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP2, $TEMP2
1399e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC7, $ACC7
1400e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC8, $TEMP5
1401e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP3, $TEMP3
1402e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC8, $ACC8
1403e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP4, $TEMP4
1404e1051a39Sopenharmony_ci
1405e1051a39Sopenharmony_ci	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
1406e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP5, $TEMP5
1407e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
1408e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC4, $ACC4
1409e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
1410e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC5, $ACC5
1411e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
1412e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC6, $ACC6
1413e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP4, $TEMP5, $TEMP4
1414e1051a39Sopenharmony_ci	vpaddq		$TEMP3, $ACC7, $ACC7
1415e1051a39Sopenharmony_ci	vpaddq		$TEMP4, $ACC8, $ACC8
1416e1051a39Sopenharmony_ci
1417e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC4, $TEMP1
1418e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC4, $ACC4
1419e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC5, $TEMP2
1420e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC5, $ACC5
1421e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC6, $TEMP3
1422e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP1, $TEMP1
1423e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC6, $ACC6
1424e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC7, $TEMP4
1425e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP2, $TEMP2
1426e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC7, $ACC7
1427e1051a39Sopenharmony_ci	vpsrlq		\$29, $ACC8, $TEMP5
1428e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP3, $TEMP3
1429e1051a39Sopenharmony_ci	vpand		$AND_MASK, $ACC8, $ACC8
1430e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP4, $TEMP4
1431e1051a39Sopenharmony_ci
1432e1051a39Sopenharmony_ci	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
1433e1051a39Sopenharmony_ci	vpermq		\$0x93, $TEMP5, $TEMP5
1434e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
1435e1051a39Sopenharmony_ci	vpaddq		$TEMP0, $ACC4, $ACC4
1436e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
1437e1051a39Sopenharmony_ci	vpaddq		$TEMP1, $ACC5, $ACC5
1438e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
1439e1051a39Sopenharmony_ci	vpaddq		$TEMP2, $ACC6, $ACC6
1440e1051a39Sopenharmony_ci	vpblendd	\$3, $TEMP4, $TEMP5, $TEMP4
1441e1051a39Sopenharmony_ci	vpaddq		$TEMP3, $ACC7, $ACC7
1442e1051a39Sopenharmony_ci	vpaddq		$TEMP4, $ACC8, $ACC8
1443e1051a39Sopenharmony_ci
1444e1051a39Sopenharmony_ci	vmovdqu		$ACC4, 128-128($rp)
1445e1051a39Sopenharmony_ci	vmovdqu		$ACC5, 160-128($rp)
1446e1051a39Sopenharmony_ci	vmovdqu		$ACC6, 192-128($rp)
1447e1051a39Sopenharmony_ci	vmovdqu		$ACC7, 224-128($rp)
1448e1051a39Sopenharmony_ci	vmovdqu		$ACC8, 256-128($rp)
1449e1051a39Sopenharmony_ci	vzeroupper
1450e1051a39Sopenharmony_ci
1451e1051a39Sopenharmony_ci	mov	%rbp, %rax
1452e1051a39Sopenharmony_ci.cfi_def_cfa_register	%rax
1453e1051a39Sopenharmony_ci___
1454e1051a39Sopenharmony_ci$code.=<<___ if ($win64);
1455e1051a39Sopenharmony_ci.Lmul_1024_in_tail:
1456e1051a39Sopenharmony_ci	movaps	-0xd8(%rax),%xmm6
1457e1051a39Sopenharmony_ci	movaps	-0xc8(%rax),%xmm7
1458e1051a39Sopenharmony_ci	movaps	-0xb8(%rax),%xmm8
1459e1051a39Sopenharmony_ci	movaps	-0xa8(%rax),%xmm9
1460e1051a39Sopenharmony_ci	movaps	-0x98(%rax),%xmm10
1461e1051a39Sopenharmony_ci	movaps	-0x88(%rax),%xmm11
1462e1051a39Sopenharmony_ci	movaps	-0x78(%rax),%xmm12
1463e1051a39Sopenharmony_ci	movaps	-0x68(%rax),%xmm13
1464e1051a39Sopenharmony_ci	movaps	-0x58(%rax),%xmm14
1465e1051a39Sopenharmony_ci	movaps	-0x48(%rax),%xmm15
1466e1051a39Sopenharmony_ci___
1467e1051a39Sopenharmony_ci$code.=<<___;
1468e1051a39Sopenharmony_ci	mov	-48(%rax),%r15
1469e1051a39Sopenharmony_ci.cfi_restore	%r15
1470e1051a39Sopenharmony_ci	mov	-40(%rax),%r14
1471e1051a39Sopenharmony_ci.cfi_restore	%r14
1472e1051a39Sopenharmony_ci	mov	-32(%rax),%r13
1473e1051a39Sopenharmony_ci.cfi_restore	%r13
1474e1051a39Sopenharmony_ci	mov	-24(%rax),%r12
1475e1051a39Sopenharmony_ci.cfi_restore	%r12
1476e1051a39Sopenharmony_ci	mov	-16(%rax),%rbp
1477e1051a39Sopenharmony_ci.cfi_restore	%rbp
1478e1051a39Sopenharmony_ci	mov	-8(%rax),%rbx
1479e1051a39Sopenharmony_ci.cfi_restore	%rbx
1480e1051a39Sopenharmony_ci	lea	(%rax),%rsp		# restore %rsp
1481e1051a39Sopenharmony_ci.cfi_def_cfa_register	%rsp
1482e1051a39Sopenharmony_ci.Lmul_1024_epilogue:
1483e1051a39Sopenharmony_ci	ret
1484e1051a39Sopenharmony_ci.cfi_endproc
1485e1051a39Sopenharmony_ci.size	rsaz_1024_mul_avx2,.-rsaz_1024_mul_avx2
1486e1051a39Sopenharmony_ci___
1487e1051a39Sopenharmony_ci}
1488e1051a39Sopenharmony_ci{
1489e1051a39Sopenharmony_cimy ($out,$inp) = $win64 ? ("%rcx","%rdx") : ("%rdi","%rsi");
1490e1051a39Sopenharmony_cimy @T = map("%r$_",(8..11));
1491e1051a39Sopenharmony_ci
1492e1051a39Sopenharmony_ci$code.=<<___;
1493e1051a39Sopenharmony_ci.globl	rsaz_1024_red2norm_avx2
1494e1051a39Sopenharmony_ci.type	rsaz_1024_red2norm_avx2,\@abi-omnipotent
1495e1051a39Sopenharmony_ci.align	32
1496e1051a39Sopenharmony_cirsaz_1024_red2norm_avx2:
1497e1051a39Sopenharmony_ci.cfi_startproc
1498e1051a39Sopenharmony_ci	sub	\$-128,$inp	# size optimization
1499e1051a39Sopenharmony_ci	xor	%rax,%rax
1500e1051a39Sopenharmony_ci___
1501e1051a39Sopenharmony_ci
1502e1051a39Sopenharmony_cifor ($j=0,$i=0; $i<16; $i++) {
1503e1051a39Sopenharmony_ci    my $k=0;
1504e1051a39Sopenharmony_ci    while (29*$j<64*($i+1)) {	# load data till boundary
1505e1051a39Sopenharmony_ci	$code.="	mov	`8*$j-128`($inp), @T[0]\n";
1506e1051a39Sopenharmony_ci	$j++; $k++; push(@T,shift(@T));
1507e1051a39Sopenharmony_ci    }
1508e1051a39Sopenharmony_ci    $l=$k;
1509e1051a39Sopenharmony_ci    while ($k>1) {		# shift loaded data but last value
1510e1051a39Sopenharmony_ci	$code.="	shl	\$`29*($j-$k)`,@T[-$k]\n";
1511e1051a39Sopenharmony_ci	$k--;
1512e1051a39Sopenharmony_ci    }
1513e1051a39Sopenharmony_ci    $code.=<<___;		# shift last value
1514e1051a39Sopenharmony_ci	mov	@T[-1], @T[0]
1515e1051a39Sopenharmony_ci	shl	\$`29*($j-1)`, @T[-1]
1516e1051a39Sopenharmony_ci	shr	\$`-29*($j-1)`, @T[0]
1517e1051a39Sopenharmony_ci___
1518e1051a39Sopenharmony_ci    while ($l) {		# accumulate all values
1519e1051a39Sopenharmony_ci	$code.="	add	@T[-$l], %rax\n";
1520e1051a39Sopenharmony_ci	$l--;
1521e1051a39Sopenharmony_ci    }
1522e1051a39Sopenharmony_ci	$code.=<<___;
1523e1051a39Sopenharmony_ci	adc	\$0, @T[0]	# consume eventual carry
1524e1051a39Sopenharmony_ci	mov	%rax, 8*$i($out)
1525e1051a39Sopenharmony_ci	mov	@T[0], %rax
1526e1051a39Sopenharmony_ci___
1527e1051a39Sopenharmony_ci    push(@T,shift(@T));
1528e1051a39Sopenharmony_ci}
1529e1051a39Sopenharmony_ci$code.=<<___;
1530e1051a39Sopenharmony_ci	ret
1531e1051a39Sopenharmony_ci.cfi_endproc
1532e1051a39Sopenharmony_ci.size	rsaz_1024_red2norm_avx2,.-rsaz_1024_red2norm_avx2
1533e1051a39Sopenharmony_ci
1534e1051a39Sopenharmony_ci.globl	rsaz_1024_norm2red_avx2
1535e1051a39Sopenharmony_ci.type	rsaz_1024_norm2red_avx2,\@abi-omnipotent
1536e1051a39Sopenharmony_ci.align	32
1537e1051a39Sopenharmony_cirsaz_1024_norm2red_avx2:
1538e1051a39Sopenharmony_ci.cfi_startproc
1539e1051a39Sopenharmony_ci	sub	\$-128,$out	# size optimization
1540e1051a39Sopenharmony_ci	mov	($inp),@T[0]
1541e1051a39Sopenharmony_ci	mov	\$0x1fffffff,%eax
1542e1051a39Sopenharmony_ci___
1543e1051a39Sopenharmony_cifor ($j=0,$i=0; $i<16; $i++) {
1544e1051a39Sopenharmony_ci    $code.="	mov	`8*($i+1)`($inp),@T[1]\n"	if ($i<15);
1545e1051a39Sopenharmony_ci    $code.="	xor	@T[1],@T[1]\n"			if ($i==15);
1546e1051a39Sopenharmony_ci    my $k=1;
1547e1051a39Sopenharmony_ci    while (29*($j+1)<64*($i+1)) {
1548e1051a39Sopenharmony_ci    	$code.=<<___;
1549e1051a39Sopenharmony_ci	mov	@T[0],@T[-$k]
1550e1051a39Sopenharmony_ci	shr	\$`29*$j`,@T[-$k]
1551e1051a39Sopenharmony_ci	and	%rax,@T[-$k]				# &0x1fffffff
1552e1051a39Sopenharmony_ci	mov	@T[-$k],`8*$j-128`($out)
1553e1051a39Sopenharmony_ci___
1554e1051a39Sopenharmony_ci	$j++; $k++;
1555e1051a39Sopenharmony_ci    }
1556e1051a39Sopenharmony_ci    $code.=<<___;
1557e1051a39Sopenharmony_ci	shrd	\$`29*$j`,@T[1],@T[0]
1558e1051a39Sopenharmony_ci	and	%rax,@T[0]
1559e1051a39Sopenharmony_ci	mov	@T[0],`8*$j-128`($out)
1560e1051a39Sopenharmony_ci___
1561e1051a39Sopenharmony_ci    $j++;
1562e1051a39Sopenharmony_ci    push(@T,shift(@T));
1563e1051a39Sopenharmony_ci}
1564e1051a39Sopenharmony_ci$code.=<<___;
1565e1051a39Sopenharmony_ci	mov	@T[0],`8*$j-128`($out)			# zero
1566e1051a39Sopenharmony_ci	mov	@T[0],`8*($j+1)-128`($out)
1567e1051a39Sopenharmony_ci	mov	@T[0],`8*($j+2)-128`($out)
1568e1051a39Sopenharmony_ci	mov	@T[0],`8*($j+3)-128`($out)
1569e1051a39Sopenharmony_ci	ret
1570e1051a39Sopenharmony_ci.cfi_endproc
1571e1051a39Sopenharmony_ci.size	rsaz_1024_norm2red_avx2,.-rsaz_1024_norm2red_avx2
1572e1051a39Sopenharmony_ci___
1573e1051a39Sopenharmony_ci}
1574e1051a39Sopenharmony_ci{
1575e1051a39Sopenharmony_cimy ($out,$inp,$power) = $win64 ? ("%rcx","%rdx","%r8d") : ("%rdi","%rsi","%edx");
1576e1051a39Sopenharmony_ci
1577e1051a39Sopenharmony_ci$code.=<<___;
1578e1051a39Sopenharmony_ci.globl	rsaz_1024_scatter5_avx2
1579e1051a39Sopenharmony_ci.type	rsaz_1024_scatter5_avx2,\@abi-omnipotent
1580e1051a39Sopenharmony_ci.align	32
1581e1051a39Sopenharmony_cirsaz_1024_scatter5_avx2:
1582e1051a39Sopenharmony_ci.cfi_startproc
1583e1051a39Sopenharmony_ci	vzeroupper
1584e1051a39Sopenharmony_ci	vmovdqu	.Lscatter_permd(%rip),%ymm5
1585e1051a39Sopenharmony_ci	shl	\$4,$power
1586e1051a39Sopenharmony_ci	lea	($out,$power),$out
1587e1051a39Sopenharmony_ci	mov	\$9,%eax
1588e1051a39Sopenharmony_ci	jmp	.Loop_scatter_1024
1589e1051a39Sopenharmony_ci
1590e1051a39Sopenharmony_ci.align	32
1591e1051a39Sopenharmony_ci.Loop_scatter_1024:
1592e1051a39Sopenharmony_ci	vmovdqu		($inp),%ymm0
1593e1051a39Sopenharmony_ci	lea		32($inp),$inp
1594e1051a39Sopenharmony_ci	vpermd		%ymm0,%ymm5,%ymm0
1595e1051a39Sopenharmony_ci	vmovdqu		%xmm0,($out)
1596e1051a39Sopenharmony_ci	lea		16*32($out),$out
1597e1051a39Sopenharmony_ci	dec	%eax
1598e1051a39Sopenharmony_ci	jnz	.Loop_scatter_1024
1599e1051a39Sopenharmony_ci
1600e1051a39Sopenharmony_ci	vzeroupper
1601e1051a39Sopenharmony_ci	ret
1602e1051a39Sopenharmony_ci.cfi_endproc
1603e1051a39Sopenharmony_ci.size	rsaz_1024_scatter5_avx2,.-rsaz_1024_scatter5_avx2
1604e1051a39Sopenharmony_ci
1605e1051a39Sopenharmony_ci.globl	rsaz_1024_gather5_avx2
1606e1051a39Sopenharmony_ci.type	rsaz_1024_gather5_avx2,\@abi-omnipotent
1607e1051a39Sopenharmony_ci.align	32
1608e1051a39Sopenharmony_cirsaz_1024_gather5_avx2:
1609e1051a39Sopenharmony_ci.cfi_startproc
1610e1051a39Sopenharmony_ci	vzeroupper
1611e1051a39Sopenharmony_ci	mov	%rsp,%r11
1612e1051a39Sopenharmony_ci.cfi_def_cfa_register	%r11
1613e1051a39Sopenharmony_ci___
1614e1051a39Sopenharmony_ci$code.=<<___ if ($win64);
1615e1051a39Sopenharmony_ci	lea	-0x88(%rsp),%rax
1616e1051a39Sopenharmony_ci.LSEH_begin_rsaz_1024_gather5:
1617e1051a39Sopenharmony_ci	# I can't trust assembler to use specific encoding:-(
1618e1051a39Sopenharmony_ci	.byte	0x48,0x8d,0x60,0xe0		# lea	-0x20(%rax),%rsp
1619e1051a39Sopenharmony_ci	.byte	0xc5,0xf8,0x29,0x70,0xe0	# vmovaps %xmm6,-0x20(%rax)
1620e1051a39Sopenharmony_ci	.byte	0xc5,0xf8,0x29,0x78,0xf0	# vmovaps %xmm7,-0x10(%rax)
1621e1051a39Sopenharmony_ci	.byte	0xc5,0x78,0x29,0x40,0x00	# vmovaps %xmm8,0(%rax)
1622e1051a39Sopenharmony_ci	.byte	0xc5,0x78,0x29,0x48,0x10	# vmovaps %xmm9,0x10(%rax)
1623e1051a39Sopenharmony_ci	.byte	0xc5,0x78,0x29,0x50,0x20	# vmovaps %xmm10,0x20(%rax)
1624e1051a39Sopenharmony_ci	.byte	0xc5,0x78,0x29,0x58,0x30	# vmovaps %xmm11,0x30(%rax)
1625e1051a39Sopenharmony_ci	.byte	0xc5,0x78,0x29,0x60,0x40	# vmovaps %xmm12,0x40(%rax)
1626e1051a39Sopenharmony_ci	.byte	0xc5,0x78,0x29,0x68,0x50	# vmovaps %xmm13,0x50(%rax)
1627e1051a39Sopenharmony_ci	.byte	0xc5,0x78,0x29,0x70,0x60	# vmovaps %xmm14,0x60(%rax)
1628e1051a39Sopenharmony_ci	.byte	0xc5,0x78,0x29,0x78,0x70	# vmovaps %xmm15,0x70(%rax)
1629e1051a39Sopenharmony_ci___
1630e1051a39Sopenharmony_ci$code.=<<___;
1631e1051a39Sopenharmony_ci	lea	-0x100(%rsp),%rsp
1632e1051a39Sopenharmony_ci	and	\$-32, %rsp
1633e1051a39Sopenharmony_ci	lea	.Linc(%rip), %r10
1634e1051a39Sopenharmony_ci	lea	-128(%rsp),%rax			# control u-op density
1635e1051a39Sopenharmony_ci
1636e1051a39Sopenharmony_ci	vmovd		$power, %xmm4
1637e1051a39Sopenharmony_ci	vmovdqa		(%r10),%ymm0
1638e1051a39Sopenharmony_ci	vmovdqa		32(%r10),%ymm1
1639e1051a39Sopenharmony_ci	vmovdqa		64(%r10),%ymm5
1640e1051a39Sopenharmony_ci	vpbroadcastd	%xmm4,%ymm4
1641e1051a39Sopenharmony_ci
1642e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm0, %ymm2
1643e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm0, %ymm0
1644e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm1, %ymm3
1645e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm1, %ymm1
1646e1051a39Sopenharmony_ci	vmovdqa		%ymm0, 32*0+128(%rax)
1647e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm2, %ymm0
1648e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm2, %ymm2
1649e1051a39Sopenharmony_ci	vmovdqa		%ymm1, 32*1+128(%rax)
1650e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm3, %ymm1
1651e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm3, %ymm3
1652e1051a39Sopenharmony_ci	vmovdqa		%ymm2, 32*2+128(%rax)
1653e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm0, %ymm2
1654e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm0, %ymm0
1655e1051a39Sopenharmony_ci	vmovdqa		%ymm3, 32*3+128(%rax)
1656e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm1, %ymm3
1657e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm1, %ymm1
1658e1051a39Sopenharmony_ci	vmovdqa		%ymm0, 32*4+128(%rax)
1659e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm2, %ymm8
1660e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm2, %ymm2
1661e1051a39Sopenharmony_ci	vmovdqa		%ymm1, 32*5+128(%rax)
1662e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm3, %ymm9
1663e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm3, %ymm3
1664e1051a39Sopenharmony_ci	vmovdqa		%ymm2, 32*6+128(%rax)
1665e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm8, %ymm10
1666e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm8, %ymm8
1667e1051a39Sopenharmony_ci	vmovdqa		%ymm3, 32*7+128(%rax)
1668e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm9, %ymm11
1669e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm9, %ymm9
1670e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm10, %ymm12
1671e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm10, %ymm10
1672e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm11, %ymm13
1673e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm11, %ymm11
1674e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm12, %ymm14
1675e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm12, %ymm12
1676e1051a39Sopenharmony_ci	vpaddd		%ymm5, %ymm13, %ymm15
1677e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm13, %ymm13
1678e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm14, %ymm14
1679e1051a39Sopenharmony_ci	vpcmpeqd	%ymm4, %ymm15, %ymm15
1680e1051a39Sopenharmony_ci
1681e1051a39Sopenharmony_ci	vmovdqa	-32(%r10),%ymm7			# .Lgather_permd
1682e1051a39Sopenharmony_ci	lea	128($inp), $inp
1683e1051a39Sopenharmony_ci	mov	\$9,$power
1684e1051a39Sopenharmony_ci
1685e1051a39Sopenharmony_ci.Loop_gather_1024:
1686e1051a39Sopenharmony_ci	vmovdqa		32*0-128($inp),	%ymm0
1687e1051a39Sopenharmony_ci	vmovdqa		32*1-128($inp),	%ymm1
1688e1051a39Sopenharmony_ci	vmovdqa		32*2-128($inp),	%ymm2
1689e1051a39Sopenharmony_ci	vmovdqa		32*3-128($inp),	%ymm3
1690e1051a39Sopenharmony_ci	vpand		32*0+128(%rax),	%ymm0,	%ymm0
1691e1051a39Sopenharmony_ci	vpand		32*1+128(%rax),	%ymm1,	%ymm1
1692e1051a39Sopenharmony_ci	vpand		32*2+128(%rax),	%ymm2,	%ymm2
1693e1051a39Sopenharmony_ci	vpor		%ymm0, %ymm1, %ymm4
1694e1051a39Sopenharmony_ci	vpand		32*3+128(%rax),	%ymm3,	%ymm3
1695e1051a39Sopenharmony_ci	vmovdqa		32*4-128($inp),	%ymm0
1696e1051a39Sopenharmony_ci	vmovdqa		32*5-128($inp),	%ymm1
1697e1051a39Sopenharmony_ci	vpor		%ymm2, %ymm3, %ymm5
1698e1051a39Sopenharmony_ci	vmovdqa		32*6-128($inp),	%ymm2
1699e1051a39Sopenharmony_ci	vmovdqa		32*7-128($inp),	%ymm3
1700e1051a39Sopenharmony_ci	vpand		32*4+128(%rax),	%ymm0,	%ymm0
1701e1051a39Sopenharmony_ci	vpand		32*5+128(%rax),	%ymm1,	%ymm1
1702e1051a39Sopenharmony_ci	vpand		32*6+128(%rax),	%ymm2,	%ymm2
1703e1051a39Sopenharmony_ci	vpor		%ymm0, %ymm4, %ymm4
1704e1051a39Sopenharmony_ci	vpand		32*7+128(%rax),	%ymm3,	%ymm3
1705e1051a39Sopenharmony_ci	vpand		32*8-128($inp),	%ymm8,	%ymm0
1706e1051a39Sopenharmony_ci	vpor		%ymm1, %ymm5, %ymm5
1707e1051a39Sopenharmony_ci	vpand		32*9-128($inp),	%ymm9,	%ymm1
1708e1051a39Sopenharmony_ci	vpor		%ymm2, %ymm4, %ymm4
1709e1051a39Sopenharmony_ci	vpand		32*10-128($inp),%ymm10,	%ymm2
1710e1051a39Sopenharmony_ci	vpor		%ymm3, %ymm5, %ymm5
1711e1051a39Sopenharmony_ci	vpand		32*11-128($inp),%ymm11,	%ymm3
1712e1051a39Sopenharmony_ci	vpor		%ymm0, %ymm4, %ymm4
1713e1051a39Sopenharmony_ci	vpand		32*12-128($inp),%ymm12,	%ymm0
1714e1051a39Sopenharmony_ci	vpor		%ymm1, %ymm5, %ymm5
1715e1051a39Sopenharmony_ci	vpand		32*13-128($inp),%ymm13,	%ymm1
1716e1051a39Sopenharmony_ci	vpor		%ymm2, %ymm4, %ymm4
1717e1051a39Sopenharmony_ci	vpand		32*14-128($inp),%ymm14,	%ymm2
1718e1051a39Sopenharmony_ci	vpor		%ymm3, %ymm5, %ymm5
1719e1051a39Sopenharmony_ci	vpand		32*15-128($inp),%ymm15,	%ymm3
1720e1051a39Sopenharmony_ci	lea		32*16($inp), $inp
1721e1051a39Sopenharmony_ci	vpor		%ymm0, %ymm4, %ymm4
1722e1051a39Sopenharmony_ci	vpor		%ymm1, %ymm5, %ymm5
1723e1051a39Sopenharmony_ci	vpor		%ymm2, %ymm4, %ymm4
1724e1051a39Sopenharmony_ci	vpor		%ymm3, %ymm5, %ymm5
1725e1051a39Sopenharmony_ci
1726e1051a39Sopenharmony_ci	vpor		%ymm5, %ymm4, %ymm4
1727e1051a39Sopenharmony_ci	vextracti128	\$1, %ymm4, %xmm5	# upper half is cleared
1728e1051a39Sopenharmony_ci	vpor		%xmm4, %xmm5, %xmm5
1729e1051a39Sopenharmony_ci	vpermd		%ymm5,%ymm7,%ymm5
1730e1051a39Sopenharmony_ci	vmovdqu		%ymm5,($out)
1731e1051a39Sopenharmony_ci	lea		32($out),$out
1732e1051a39Sopenharmony_ci	dec	$power
1733e1051a39Sopenharmony_ci	jnz	.Loop_gather_1024
1734e1051a39Sopenharmony_ci
1735e1051a39Sopenharmony_ci	vpxor	%ymm0,%ymm0,%ymm0
1736e1051a39Sopenharmony_ci	vmovdqu	%ymm0,($out)
1737e1051a39Sopenharmony_ci	vzeroupper
1738e1051a39Sopenharmony_ci___
1739e1051a39Sopenharmony_ci$code.=<<___ if ($win64);
1740e1051a39Sopenharmony_ci	movaps	-0xa8(%r11),%xmm6
1741e1051a39Sopenharmony_ci	movaps	-0x98(%r11),%xmm7
1742e1051a39Sopenharmony_ci	movaps	-0x88(%r11),%xmm8
1743e1051a39Sopenharmony_ci	movaps	-0x78(%r11),%xmm9
1744e1051a39Sopenharmony_ci	movaps	-0x68(%r11),%xmm10
1745e1051a39Sopenharmony_ci	movaps	-0x58(%r11),%xmm11
1746e1051a39Sopenharmony_ci	movaps	-0x48(%r11),%xmm12
1747e1051a39Sopenharmony_ci	movaps	-0x38(%r11),%xmm13
1748e1051a39Sopenharmony_ci	movaps	-0x28(%r11),%xmm14
1749e1051a39Sopenharmony_ci	movaps	-0x18(%r11),%xmm15
1750e1051a39Sopenharmony_ci___
1751e1051a39Sopenharmony_ci$code.=<<___;
1752e1051a39Sopenharmony_ci	lea	(%r11),%rsp
1753e1051a39Sopenharmony_ci.cfi_def_cfa_register	%rsp
1754e1051a39Sopenharmony_ci	ret
1755e1051a39Sopenharmony_ci.cfi_endproc
1756e1051a39Sopenharmony_ci.LSEH_end_rsaz_1024_gather5:
1757e1051a39Sopenharmony_ci.size	rsaz_1024_gather5_avx2,.-rsaz_1024_gather5_avx2
1758e1051a39Sopenharmony_ci___
1759e1051a39Sopenharmony_ci}
1760e1051a39Sopenharmony_ci
1761e1051a39Sopenharmony_ci$code.=<<___;
1762e1051a39Sopenharmony_ci.extern	OPENSSL_ia32cap_P
1763e1051a39Sopenharmony_ci.globl	rsaz_avx2_eligible
1764e1051a39Sopenharmony_ci.type	rsaz_avx2_eligible,\@abi-omnipotent
1765e1051a39Sopenharmony_ci.align	32
1766e1051a39Sopenharmony_cirsaz_avx2_eligible:
1767e1051a39Sopenharmony_ci	mov	OPENSSL_ia32cap_P+8(%rip),%eax
1768e1051a39Sopenharmony_ci___
1769e1051a39Sopenharmony_ci$code.=<<___	if ($addx);
1770e1051a39Sopenharmony_ci	mov	\$`1<<8|1<<19`,%ecx
1771e1051a39Sopenharmony_ci	mov	\$0,%edx
1772e1051a39Sopenharmony_ci	and	%eax,%ecx
1773e1051a39Sopenharmony_ci	cmp	\$`1<<8|1<<19`,%ecx	# check for BMI2+AD*X
1774e1051a39Sopenharmony_ci	cmove	%edx,%eax
1775e1051a39Sopenharmony_ci___
1776e1051a39Sopenharmony_ci$code.=<<___;
1777e1051a39Sopenharmony_ci	and	\$`1<<5`,%eax
1778e1051a39Sopenharmony_ci	shr	\$5,%eax
1779e1051a39Sopenharmony_ci	ret
1780e1051a39Sopenharmony_ci.size	rsaz_avx2_eligible,.-rsaz_avx2_eligible
1781e1051a39Sopenharmony_ci
1782e1051a39Sopenharmony_ci.align	64
1783e1051a39Sopenharmony_ci.Land_mask:
1784e1051a39Sopenharmony_ci	.quad	0x1fffffff,0x1fffffff,0x1fffffff,0x1fffffff
1785e1051a39Sopenharmony_ci.Lscatter_permd:
1786e1051a39Sopenharmony_ci	.long	0,2,4,6,7,7,7,7
1787e1051a39Sopenharmony_ci.Lgather_permd:
1788e1051a39Sopenharmony_ci	.long	0,7,1,7,2,7,3,7
1789e1051a39Sopenharmony_ci.Linc:
1790e1051a39Sopenharmony_ci	.long	0,0,0,0, 1,1,1,1
1791e1051a39Sopenharmony_ci	.long	2,2,2,2, 3,3,3,3
1792e1051a39Sopenharmony_ci	.long	4,4,4,4, 4,4,4,4
1793e1051a39Sopenharmony_ci.align	64
1794e1051a39Sopenharmony_ci___
1795e1051a39Sopenharmony_ci
1796e1051a39Sopenharmony_ciif ($win64) {
1797e1051a39Sopenharmony_ci$rec="%rcx";
1798e1051a39Sopenharmony_ci$frame="%rdx";
1799e1051a39Sopenharmony_ci$context="%r8";
1800e1051a39Sopenharmony_ci$disp="%r9";
1801e1051a39Sopenharmony_ci
1802e1051a39Sopenharmony_ci$code.=<<___
1803e1051a39Sopenharmony_ci.extern	__imp_RtlVirtualUnwind
1804e1051a39Sopenharmony_ci.type	rsaz_se_handler,\@abi-omnipotent
1805e1051a39Sopenharmony_ci.align	16
1806e1051a39Sopenharmony_cirsaz_se_handler:
1807e1051a39Sopenharmony_ci	push	%rsi
1808e1051a39Sopenharmony_ci	push	%rdi
1809e1051a39Sopenharmony_ci	push	%rbx
1810e1051a39Sopenharmony_ci	push	%rbp
1811e1051a39Sopenharmony_ci	push	%r12
1812e1051a39Sopenharmony_ci	push	%r13
1813e1051a39Sopenharmony_ci	push	%r14
1814e1051a39Sopenharmony_ci	push	%r15
1815e1051a39Sopenharmony_ci	pushfq
1816e1051a39Sopenharmony_ci	sub	\$64,%rsp
1817e1051a39Sopenharmony_ci
1818e1051a39Sopenharmony_ci	mov	120($context),%rax	# pull context->Rax
1819e1051a39Sopenharmony_ci	mov	248($context),%rbx	# pull context->Rip
1820e1051a39Sopenharmony_ci
1821e1051a39Sopenharmony_ci	mov	8($disp),%rsi		# disp->ImageBase
1822e1051a39Sopenharmony_ci	mov	56($disp),%r11		# disp->HandlerData
1823e1051a39Sopenharmony_ci
1824e1051a39Sopenharmony_ci	mov	0(%r11),%r10d		# HandlerData[0]
1825e1051a39Sopenharmony_ci	lea	(%rsi,%r10),%r10	# prologue label
1826e1051a39Sopenharmony_ci	cmp	%r10,%rbx		# context->Rip<prologue label
1827e1051a39Sopenharmony_ci	jb	.Lcommon_seh_tail
1828e1051a39Sopenharmony_ci
1829e1051a39Sopenharmony_ci	mov	4(%r11),%r10d		# HandlerData[1]
1830e1051a39Sopenharmony_ci	lea	(%rsi,%r10),%r10	# epilogue label
1831e1051a39Sopenharmony_ci	cmp	%r10,%rbx		# context->Rip>=epilogue label
1832e1051a39Sopenharmony_ci	jae	.Lcommon_seh_tail
1833e1051a39Sopenharmony_ci
1834e1051a39Sopenharmony_ci	mov	160($context),%rbp	# pull context->Rbp
1835e1051a39Sopenharmony_ci
1836e1051a39Sopenharmony_ci	mov	8(%r11),%r10d		# HandlerData[2]
1837e1051a39Sopenharmony_ci	lea	(%rsi,%r10),%r10	# "in tail" label
1838e1051a39Sopenharmony_ci	cmp	%r10,%rbx		# context->Rip>="in tail" label
1839e1051a39Sopenharmony_ci	cmovc	%rbp,%rax
1840e1051a39Sopenharmony_ci
1841e1051a39Sopenharmony_ci	mov	-48(%rax),%r15
1842e1051a39Sopenharmony_ci	mov	-40(%rax),%r14
1843e1051a39Sopenharmony_ci	mov	-32(%rax),%r13
1844e1051a39Sopenharmony_ci	mov	-24(%rax),%r12
1845e1051a39Sopenharmony_ci	mov	-16(%rax),%rbp
1846e1051a39Sopenharmony_ci	mov	-8(%rax),%rbx
1847e1051a39Sopenharmony_ci	mov	%r15,240($context)
1848e1051a39Sopenharmony_ci	mov	%r14,232($context)
1849e1051a39Sopenharmony_ci	mov	%r13,224($context)
1850e1051a39Sopenharmony_ci	mov	%r12,216($context)
1851e1051a39Sopenharmony_ci	mov	%rbp,160($context)
1852e1051a39Sopenharmony_ci	mov	%rbx,144($context)
1853e1051a39Sopenharmony_ci
1854e1051a39Sopenharmony_ci	lea	-0xd8(%rax),%rsi	# %xmm save area
1855e1051a39Sopenharmony_ci	lea	512($context),%rdi	# & context.Xmm6
1856e1051a39Sopenharmony_ci	mov	\$20,%ecx		# 10*sizeof(%xmm0)/sizeof(%rax)
1857e1051a39Sopenharmony_ci	.long	0xa548f3fc		# cld; rep movsq
1858e1051a39Sopenharmony_ci
1859e1051a39Sopenharmony_ci.Lcommon_seh_tail:
1860e1051a39Sopenharmony_ci	mov	8(%rax),%rdi
1861e1051a39Sopenharmony_ci	mov	16(%rax),%rsi
1862e1051a39Sopenharmony_ci	mov	%rax,152($context)	# restore context->Rsp
1863e1051a39Sopenharmony_ci	mov	%rsi,168($context)	# restore context->Rsi
1864e1051a39Sopenharmony_ci	mov	%rdi,176($context)	# restore context->Rdi
1865e1051a39Sopenharmony_ci
1866e1051a39Sopenharmony_ci	mov	40($disp),%rdi		# disp->ContextRecord
1867e1051a39Sopenharmony_ci	mov	$context,%rsi		# context
1868e1051a39Sopenharmony_ci	mov	\$154,%ecx		# sizeof(CONTEXT)
1869e1051a39Sopenharmony_ci	.long	0xa548f3fc		# cld; rep movsq
1870e1051a39Sopenharmony_ci
1871e1051a39Sopenharmony_ci	mov	$disp,%rsi
1872e1051a39Sopenharmony_ci	xor	%rcx,%rcx		# arg1, UNW_FLAG_NHANDLER
1873e1051a39Sopenharmony_ci	mov	8(%rsi),%rdx		# arg2, disp->ImageBase
1874e1051a39Sopenharmony_ci	mov	0(%rsi),%r8		# arg3, disp->ControlPc
1875e1051a39Sopenharmony_ci	mov	16(%rsi),%r9		# arg4, disp->FunctionEntry
1876e1051a39Sopenharmony_ci	mov	40(%rsi),%r10		# disp->ContextRecord
1877e1051a39Sopenharmony_ci	lea	56(%rsi),%r11		# &disp->HandlerData
1878e1051a39Sopenharmony_ci	lea	24(%rsi),%r12		# &disp->EstablisherFrame
1879e1051a39Sopenharmony_ci	mov	%r10,32(%rsp)		# arg5
1880e1051a39Sopenharmony_ci	mov	%r11,40(%rsp)		# arg6
1881e1051a39Sopenharmony_ci	mov	%r12,48(%rsp)		# arg7
1882e1051a39Sopenharmony_ci	mov	%rcx,56(%rsp)		# arg8, (NULL)
1883e1051a39Sopenharmony_ci	call	*__imp_RtlVirtualUnwind(%rip)
1884e1051a39Sopenharmony_ci
1885e1051a39Sopenharmony_ci	mov	\$1,%eax		# ExceptionContinueSearch
1886e1051a39Sopenharmony_ci	add	\$64,%rsp
1887e1051a39Sopenharmony_ci	popfq
1888e1051a39Sopenharmony_ci	pop	%r15
1889e1051a39Sopenharmony_ci	pop	%r14
1890e1051a39Sopenharmony_ci	pop	%r13
1891e1051a39Sopenharmony_ci	pop	%r12
1892e1051a39Sopenharmony_ci	pop	%rbp
1893e1051a39Sopenharmony_ci	pop	%rbx
1894e1051a39Sopenharmony_ci	pop	%rdi
1895e1051a39Sopenharmony_ci	pop	%rsi
1896e1051a39Sopenharmony_ci	ret
1897e1051a39Sopenharmony_ci.size	rsaz_se_handler,.-rsaz_se_handler
1898e1051a39Sopenharmony_ci
1899e1051a39Sopenharmony_ci.section	.pdata
1900e1051a39Sopenharmony_ci.align	4
1901e1051a39Sopenharmony_ci	.rva	.LSEH_begin_rsaz_1024_sqr_avx2
1902e1051a39Sopenharmony_ci	.rva	.LSEH_end_rsaz_1024_sqr_avx2
1903e1051a39Sopenharmony_ci	.rva	.LSEH_info_rsaz_1024_sqr_avx2
1904e1051a39Sopenharmony_ci
1905e1051a39Sopenharmony_ci	.rva	.LSEH_begin_rsaz_1024_mul_avx2
1906e1051a39Sopenharmony_ci	.rva	.LSEH_end_rsaz_1024_mul_avx2
1907e1051a39Sopenharmony_ci	.rva	.LSEH_info_rsaz_1024_mul_avx2
1908e1051a39Sopenharmony_ci
1909e1051a39Sopenharmony_ci	.rva	.LSEH_begin_rsaz_1024_gather5
1910e1051a39Sopenharmony_ci	.rva	.LSEH_end_rsaz_1024_gather5
1911e1051a39Sopenharmony_ci	.rva	.LSEH_info_rsaz_1024_gather5
1912e1051a39Sopenharmony_ci.section	.xdata
1913e1051a39Sopenharmony_ci.align	8
1914e1051a39Sopenharmony_ci.LSEH_info_rsaz_1024_sqr_avx2:
1915e1051a39Sopenharmony_ci	.byte	9,0,0,0
1916e1051a39Sopenharmony_ci	.rva	rsaz_se_handler
1917e1051a39Sopenharmony_ci	.rva	.Lsqr_1024_body,.Lsqr_1024_epilogue,.Lsqr_1024_in_tail
1918e1051a39Sopenharmony_ci	.long	0
1919e1051a39Sopenharmony_ci.LSEH_info_rsaz_1024_mul_avx2:
1920e1051a39Sopenharmony_ci	.byte	9,0,0,0
1921e1051a39Sopenharmony_ci	.rva	rsaz_se_handler
1922e1051a39Sopenharmony_ci	.rva	.Lmul_1024_body,.Lmul_1024_epilogue,.Lmul_1024_in_tail
1923e1051a39Sopenharmony_ci	.long	0
1924e1051a39Sopenharmony_ci.LSEH_info_rsaz_1024_gather5:
1925e1051a39Sopenharmony_ci	.byte	0x01,0x36,0x17,0x0b
1926e1051a39Sopenharmony_ci	.byte	0x36,0xf8,0x09,0x00	# vmovaps 0x90(rsp),xmm15
1927e1051a39Sopenharmony_ci	.byte	0x31,0xe8,0x08,0x00	# vmovaps 0x80(rsp),xmm14
1928e1051a39Sopenharmony_ci	.byte	0x2c,0xd8,0x07,0x00	# vmovaps 0x70(rsp),xmm13
1929e1051a39Sopenharmony_ci	.byte	0x27,0xc8,0x06,0x00	# vmovaps 0x60(rsp),xmm12
1930e1051a39Sopenharmony_ci	.byte	0x22,0xb8,0x05,0x00	# vmovaps 0x50(rsp),xmm11
1931e1051a39Sopenharmony_ci	.byte	0x1d,0xa8,0x04,0x00	# vmovaps 0x40(rsp),xmm10
1932e1051a39Sopenharmony_ci	.byte	0x18,0x98,0x03,0x00	# vmovaps 0x30(rsp),xmm9
1933e1051a39Sopenharmony_ci	.byte	0x13,0x88,0x02,0x00	# vmovaps 0x20(rsp),xmm8
1934e1051a39Sopenharmony_ci	.byte	0x0e,0x78,0x01,0x00	# vmovaps 0x10(rsp),xmm7
1935e1051a39Sopenharmony_ci	.byte	0x09,0x68,0x00,0x00	# vmovaps 0x00(rsp),xmm6
1936e1051a39Sopenharmony_ci	.byte	0x04,0x01,0x15,0x00	# sub	  rsp,0xa8
1937e1051a39Sopenharmony_ci	.byte	0x00,0xb3,0x00,0x00	# set_frame r11
1938e1051a39Sopenharmony_ci___
1939e1051a39Sopenharmony_ci}
1940e1051a39Sopenharmony_ci
1941e1051a39Sopenharmony_ciforeach (split("\n",$code)) {
1942e1051a39Sopenharmony_ci	s/\`([^\`]*)\`/eval($1)/ge;
1943e1051a39Sopenharmony_ci
1944e1051a39Sopenharmony_ci	s/\b(sh[rl]d?\s+\$)(-?[0-9]+)/$1.$2%64/ge		or
1945e1051a39Sopenharmony_ci
1946e1051a39Sopenharmony_ci	s/\b(vmov[dq])\b(.+)%ymm([0-9]+)/$1$2%xmm$3/go		or
1947e1051a39Sopenharmony_ci	s/\b(vmovdqu)\b(.+)%x%ymm([0-9]+)/$1$2%xmm$3/go		or
1948e1051a39Sopenharmony_ci	s/\b(vpinsr[qd])\b(.+)%ymm([0-9]+)/$1$2%xmm$3/go	or
1949e1051a39Sopenharmony_ci	s/\b(vpextr[qd])\b(.+)%ymm([0-9]+)/$1$2%xmm$3/go	or
1950e1051a39Sopenharmony_ci	s/\b(vpbroadcast[qd]\s+)%ymm([0-9]+)/$1%xmm$2/go;
1951e1051a39Sopenharmony_ci	print $_,"\n";
1952e1051a39Sopenharmony_ci}
1953e1051a39Sopenharmony_ci
1954e1051a39Sopenharmony_ci}}} else {{{
1955e1051a39Sopenharmony_ciprint <<___;	# assembler is too old
1956e1051a39Sopenharmony_ci.text
1957e1051a39Sopenharmony_ci
1958e1051a39Sopenharmony_ci.globl	rsaz_avx2_eligible
1959e1051a39Sopenharmony_ci.type	rsaz_avx2_eligible,\@abi-omnipotent
1960e1051a39Sopenharmony_cirsaz_avx2_eligible:
1961e1051a39Sopenharmony_ci	xor	%eax,%eax
1962e1051a39Sopenharmony_ci	ret
1963e1051a39Sopenharmony_ci.size	rsaz_avx2_eligible,.-rsaz_avx2_eligible
1964e1051a39Sopenharmony_ci
1965e1051a39Sopenharmony_ci.globl	rsaz_1024_sqr_avx2
1966e1051a39Sopenharmony_ci.globl	rsaz_1024_mul_avx2
1967e1051a39Sopenharmony_ci.globl	rsaz_1024_norm2red_avx2
1968e1051a39Sopenharmony_ci.globl	rsaz_1024_red2norm_avx2
1969e1051a39Sopenharmony_ci.globl	rsaz_1024_scatter5_avx2
1970e1051a39Sopenharmony_ci.globl	rsaz_1024_gather5_avx2
1971e1051a39Sopenharmony_ci.type	rsaz_1024_sqr_avx2,\@abi-omnipotent
1972e1051a39Sopenharmony_cirsaz_1024_sqr_avx2:
1973e1051a39Sopenharmony_cirsaz_1024_mul_avx2:
1974e1051a39Sopenharmony_cirsaz_1024_norm2red_avx2:
1975e1051a39Sopenharmony_cirsaz_1024_red2norm_avx2:
1976e1051a39Sopenharmony_cirsaz_1024_scatter5_avx2:
1977e1051a39Sopenharmony_cirsaz_1024_gather5_avx2:
1978e1051a39Sopenharmony_ci	.byte	0x0f,0x0b	# ud2
1979e1051a39Sopenharmony_ci	ret
1980e1051a39Sopenharmony_ci.size	rsaz_1024_sqr_avx2,.-rsaz_1024_sqr_avx2
1981e1051a39Sopenharmony_ci___
1982e1051a39Sopenharmony_ci}}}
1983e1051a39Sopenharmony_ci
1984e1051a39Sopenharmony_ciclose STDOUT or die "error closing STDOUT: $!";
1985