Blame crypto/bn/asm/rsaz-avx2.pl

Packit Service 084de1
#! /usr/bin/env perl
Packit Service 084de1
# Copyright 2013-2020 The OpenSSL Project Authors. All Rights Reserved.
Packit Service 084de1
# Copyright (c) 2012, Intel Corporation. All Rights Reserved.
Packit Service 084de1
#
Packit Service 084de1
# Licensed under the OpenSSL license (the "License").  You may not use
Packit Service 084de1
# this file except in compliance with the License.  You can obtain a copy
Packit Service 084de1
# in the file LICENSE in the source distribution or at
Packit Service 084de1
# https://www.openssl.org/source/license.html
Packit Service 084de1
#
Packit Service 084de1
# Originally written by Shay Gueron (1, 2), and Vlad Krasnov (1)
Packit Service 084de1
# (1) Intel Corporation, Israel Development Center, Haifa, Israel
Packit Service 084de1
# (2) University of Haifa, Israel
Packit Service 084de1
#
Packit Service 084de1
# References:
Packit Service 084de1
# [1] S. Gueron, V. Krasnov: "Software Implementation of Modular
Packit Service 084de1
#     Exponentiation,  Using Advanced Vector Instructions Architectures",
Packit Service 084de1
#     F. Ozbudak and F. Rodriguez-Henriquez (Eds.): WAIFI 2012, LNCS 7369,
Packit Service 084de1
#     pp. 119?135, 2012. Springer-Verlag Berlin Heidelberg 2012
Packit Service 084de1
# [2] S. Gueron: "Efficient Software Implementations of Modular
Packit Service 084de1
#     Exponentiation", Journal of Cryptographic Engineering 2:31-43 (2012).
Packit Service 084de1
# [3] S. Gueron, V. Krasnov: "Speeding up Big-numbers Squaring",IEEE
Packit Service 084de1
#     Proceedings of 9th International Conference on Information Technology:
Packit Service 084de1
#     New Generations (ITNG 2012), pp.821-823 (2012)
Packit Service 084de1
# [4] S. Gueron, V. Krasnov: "[PATCH] Efficient and side channel analysis
Packit Service 084de1
#     resistant 1024-bit modular exponentiation, for optimizing RSA2048
Packit Service 084de1
#     on AVX2 capable x86_64 platforms",
Packit Service 084de1
#     http://rt.openssl.org/Ticket/Display.html?id=2850&user=guest&pass=guest
Packit Service 084de1
#
Packit Service 084de1
# +13% improvement over original submission by <appro@openssl.org>
Packit Service 084de1
#
Packit Service 084de1
# rsa2048 sign/sec	OpenSSL 1.0.1	scalar(*)	this
Packit Service 084de1
# 2.3GHz Haswell	621		765/+23%	1113/+79%
Packit Service 084de1
# 2.3GHz Broadwell(**)	688		1200(***)/+74%	1120/+63%
Packit Service 084de1
#
Packit Service 084de1
# (*)	if system doesn't support AVX2, for reference purposes;
Packit Service 084de1
# (**)	scaled to 2.3GHz to simplify comparison;
Packit Service 084de1
# (***)	scalar AD*X code is faster than AVX2 and is preferred code
Packit Service 084de1
#	path for Broadwell;
Packit Service 084de1
Packit Service 084de1
$flavour = shift;
Packit Service 084de1
$output  = shift;
Packit Service 084de1
if ($flavour =~ /\./) { $output = $flavour; undef $flavour; }
Packit Service 084de1
Packit Service 084de1
$win64=0; $win64=1 if ($flavour =~ /[nm]asm|mingw64/ || $output =~ /\.asm$/);
Packit Service 084de1
Packit Service 084de1
$0 =~ m/(.*[\/\\])[^\/\\]+$/; $dir=$1;
Packit Service 084de1
( $xlate="${dir}x86_64-xlate.pl" and -f $xlate ) or
Packit Service 084de1
( $xlate="${dir}../../perlasm/x86_64-xlate.pl" and -f $xlate) or
Packit Service 084de1
die "can't locate x86_64-xlate.pl";
Packit Service 084de1
Packit Service 084de1
if (`$ENV{CC} -Wa,-v -c -o /dev/null -x assembler /dev/null 2>&1`
Packit Service 084de1
		=~ /GNU assembler version ([2-9]\.[0-9]+)/) {
Packit Service 084de1
	$avx = ($1>=2.19) + ($1>=2.22);
Packit Service 084de1
	$addx = ($1>=2.23);
Packit Service 084de1
}
Packit Service 084de1
Packit Service 084de1
if (!$avx && $win64 && ($flavour =~ /nasm/ || $ENV{ASM} =~ /nasm/) &&
Packit Service 084de1
	    `nasm -v 2>&1` =~ /NASM version ([2-9]\.[0-9]+)/) {
Packit Service 084de1
	$avx = ($1>=2.09) + ($1>=2.10);
Packit Service 084de1
	$addx = ($1>=2.10);
Packit Service 084de1
}
Packit Service 084de1
Packit Service 084de1
if (!$avx && $win64 && ($flavour =~ /masm/ || $ENV{ASM} =~ /ml64/) &&
Packit Service 084de1
	    `ml64 2>&1` =~ /Version ([0-9]+)\./) {
Packit Service 084de1
	$avx = ($1>=10) + ($1>=11);
Packit Service 084de1
	$addx = ($1>=11);
Packit Service 084de1
}
Packit Service 084de1
Packit Service 084de1
if (!$avx && `$ENV{CC} -v 2>&1` =~ /((?:^clang|LLVM) version|based on LLVM) ([0-9]+)\.([0-9]+)/) {
Packit Service 084de1
	my $ver = $2 + $3/100.0;	# 3.1->3.01, 3.10->3.10
Packit Service 084de1
	$avx = ($ver>=3.0) + ($ver>=3.01);
Packit Service 084de1
	$addx = ($ver>=3.03);
Packit Service 084de1
}
Packit Service 084de1
Packit Service 084de1
open OUT,"| \"$^X\" \"$xlate\" $flavour \"$output\"";
Packit Service 084de1
*STDOUT = *OUT;
Packit Service 084de1
Packit Service 084de1
if ($avx>1) {{{
Packit Service 084de1
{ # void AMS_WW(
Packit Service 084de1
my $rp="%rdi";	# BN_ULONG *rp,
Packit Service 084de1
my $ap="%rsi";	# const BN_ULONG *ap,
Packit Service 084de1
my $np="%rdx";	# const BN_ULONG *np,
Packit Service 084de1
my $n0="%ecx";	# const BN_ULONG n0,
Packit Service 084de1
my $rep="%r8d";	# int repeat);
Packit Service 084de1
Packit Service 084de1
# The registers that hold the accumulated redundant result
Packit Service 084de1
# The AMM works on 1024 bit operands, and redundant word size is 29
Packit Service 084de1
# Therefore: ceil(1024/29)/4 = 9
Packit Service 084de1
my $ACC0="%ymm0";
Packit Service 084de1
my $ACC1="%ymm1";
Packit Service 084de1
my $ACC2="%ymm2";
Packit Service 084de1
my $ACC3="%ymm3";
Packit Service 084de1
my $ACC4="%ymm4";
Packit Service 084de1
my $ACC5="%ymm5";
Packit Service 084de1
my $ACC6="%ymm6";
Packit Service 084de1
my $ACC7="%ymm7";
Packit Service 084de1
my $ACC8="%ymm8";
Packit Service 084de1
my $ACC9="%ymm9";
Packit Service 084de1
# Registers that hold the broadcasted words of bp, currently used
Packit Service 084de1
my $B1="%ymm10";
Packit Service 084de1
my $B2="%ymm11";
Packit Service 084de1
# Registers that hold the broadcasted words of Y, currently used
Packit Service 084de1
my $Y1="%ymm12";
Packit Service 084de1
my $Y2="%ymm13";
Packit Service 084de1
# Helper registers
Packit Service 084de1
my $TEMP1="%ymm14";
Packit Service 084de1
my $AND_MASK="%ymm15";
Packit Service 084de1
# alu registers that hold the first words of the ACC
Packit Service 084de1
my $r0="%r9";
Packit Service 084de1
my $r1="%r10";
Packit Service 084de1
my $r2="%r11";
Packit Service 084de1
my $r3="%r12";
Packit Service 084de1
Packit Service 084de1
my $i="%r14d";			# loop counter
Packit Service 084de1
my $tmp = "%r15";
Packit Service 084de1
Packit Service 084de1
my $FrameSize=32*18+32*8;	# place for A^2 and 2*A
Packit Service 084de1
Packit Service 084de1
my $aap=$r0;
Packit Service 084de1
my $tp0="%rbx";
Packit Service 084de1
my $tp1=$r3;
Packit Service 084de1
my $tpa=$tmp;
Packit Service 084de1
Packit Service 084de1
$np="%r13";			# reassigned argument
Packit Service 084de1
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
.text
Packit Service 084de1
Packit Service 084de1
.globl	rsaz_1024_sqr_avx2
Packit Service 084de1
.type	rsaz_1024_sqr_avx2,\@function,5
Packit Service 084de1
.align	64
Packit Service 084de1
rsaz_1024_sqr_avx2:		# 702 cycles, 14% faster than rsaz_1024_mul_avx2
Packit Service 084de1
.cfi_startproc
Packit Service 084de1
	lea	(%rsp), %rax
Packit Service 084de1
.cfi_def_cfa_register	%rax
Packit Service 084de1
	push	%rbx
Packit Service 084de1
.cfi_push	%rbx
Packit Service 084de1
	push	%rbp
Packit Service 084de1
.cfi_push	%rbp
Packit Service 084de1
	push	%r12
Packit Service 084de1
.cfi_push	%r12
Packit Service 084de1
	push	%r13
Packit Service 084de1
.cfi_push	%r13
Packit Service 084de1
	push	%r14
Packit Service 084de1
.cfi_push	%r14
Packit Service 084de1
	push	%r15
Packit Service 084de1
.cfi_push	%r15
Packit Service 084de1
	vzeroupper
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___ if ($win64);
Packit Service 084de1
	lea	-0xa8(%rsp),%rsp
Packit Service 084de1
	vmovaps	%xmm6,-0xd8(%rax)
Packit Service 084de1
	vmovaps	%xmm7,-0xc8(%rax)
Packit Service 084de1
	vmovaps	%xmm8,-0xb8(%rax)
Packit Service 084de1
	vmovaps	%xmm9,-0xa8(%rax)
Packit Service 084de1
	vmovaps	%xmm10,-0x98(%rax)
Packit Service 084de1
	vmovaps	%xmm11,-0x88(%rax)
Packit Service 084de1
	vmovaps	%xmm12,-0x78(%rax)
Packit Service 084de1
	vmovaps	%xmm13,-0x68(%rax)
Packit Service 084de1
	vmovaps	%xmm14,-0x58(%rax)
Packit Service 084de1
	vmovaps	%xmm15,-0x48(%rax)
Packit Service 084de1
.Lsqr_1024_body:
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	mov	%rax,%rbp
Packit Service 084de1
.cfi_def_cfa_register	%rbp
Packit Service 084de1
	mov	%rdx, $np			# reassigned argument
Packit Service 084de1
	sub	\$$FrameSize, %rsp
Packit Service 084de1
	mov	$np, $tmp
Packit Service 084de1
	sub	\$-128, $rp			# size optimization
Packit Service 084de1
	sub	\$-128, $ap
Packit Service 084de1
	sub	\$-128, $np
Packit Service 084de1
Packit Service 084de1
	and	\$4095, $tmp			# see if $np crosses page
Packit Service 084de1
	add	\$32*10, $tmp
Packit Service 084de1
	shr	\$12, $tmp
Packit Service 084de1
	vpxor	$ACC9,$ACC9,$ACC9
Packit Service 084de1
	jz	.Lsqr_1024_no_n_copy
Packit Service 084de1
Packit Service 084de1
	# unaligned 256-bit load that crosses page boundary can
Packit Service 084de1
	# cause >2x performance degradation here, so if $np does
Packit Service 084de1
	# cross page boundary, copy it to stack and make sure stack
Packit Service 084de1
	# frame doesn't...
Packit Service 084de1
	sub		\$32*10,%rsp
Packit Service 084de1
	vmovdqu		32*0-128($np), $ACC0
Packit Service 084de1
	and		\$-2048, %rsp
Packit Service 084de1
	vmovdqu		32*1-128($np), $ACC1
Packit Service 084de1
	vmovdqu		32*2-128($np), $ACC2
Packit Service 084de1
	vmovdqu		32*3-128($np), $ACC3
Packit Service 084de1
	vmovdqu		32*4-128($np), $ACC4
Packit Service 084de1
	vmovdqu		32*5-128($np), $ACC5
Packit Service 084de1
	vmovdqu		32*6-128($np), $ACC6
Packit Service 084de1
	vmovdqu		32*7-128($np), $ACC7
Packit Service 084de1
	vmovdqu		32*8-128($np), $ACC8
Packit Service 084de1
	lea		$FrameSize+128(%rsp),$np
Packit Service 084de1
	vmovdqu		$ACC0, 32*0-128($np)
Packit Service 084de1
	vmovdqu		$ACC1, 32*1-128($np)
Packit Service 084de1
	vmovdqu		$ACC2, 32*2-128($np)
Packit Service 084de1
	vmovdqu		$ACC3, 32*3-128($np)
Packit Service 084de1
	vmovdqu		$ACC4, 32*4-128($np)
Packit Service 084de1
	vmovdqu		$ACC5, 32*5-128($np)
Packit Service 084de1
	vmovdqu		$ACC6, 32*6-128($np)
Packit Service 084de1
	vmovdqu		$ACC7, 32*7-128($np)
Packit Service 084de1
	vmovdqu		$ACC8, 32*8-128($np)
Packit Service 084de1
	vmovdqu		$ACC9, 32*9-128($np)	# $ACC9 is zero
Packit Service 084de1
Packit Service 084de1
.Lsqr_1024_no_n_copy:
Packit Service 084de1
	and		\$-1024, %rsp
Packit Service 084de1
Packit Service 084de1
	vmovdqu		32*1-128($ap), $ACC1
Packit Service 084de1
	vmovdqu		32*2-128($ap), $ACC2
Packit Service 084de1
	vmovdqu		32*3-128($ap), $ACC3
Packit Service 084de1
	vmovdqu		32*4-128($ap), $ACC4
Packit Service 084de1
	vmovdqu		32*5-128($ap), $ACC5
Packit Service 084de1
	vmovdqu		32*6-128($ap), $ACC6
Packit Service 084de1
	vmovdqu		32*7-128($ap), $ACC7
Packit Service 084de1
	vmovdqu		32*8-128($ap), $ACC8
Packit Service 084de1
Packit Service 084de1
	lea	192(%rsp), $tp0			# 64+128=192
Packit Service 084de1
	vmovdqu	.Land_mask(%rip), $AND_MASK
Packit Service 084de1
	jmp	.LOOP_GRANDE_SQR_1024
Packit Service 084de1
Packit Service 084de1
.align	32
Packit Service 084de1
.LOOP_GRANDE_SQR_1024:
Packit Service 084de1
	lea	32*18+128(%rsp), $aap		# size optimization
Packit Service 084de1
	lea	448(%rsp), $tp1			# 64+128+256=448
Packit Service 084de1
Packit Service 084de1
	# the squaring is performed as described in Variant B of
Packit Service 084de1
	# "Speeding up Big-Number Squaring", so start by calculating
Packit Service 084de1
	# the A*2=A+A vector
Packit Service 084de1
	vpaddq		$ACC1, $ACC1, $ACC1
Packit Service 084de1
	 vpbroadcastq	32*0-128($ap), $B1
Packit Service 084de1
	vpaddq		$ACC2, $ACC2, $ACC2
Packit Service 084de1
	vmovdqa		$ACC1, 32*0-128($aap)
Packit Service 084de1
	vpaddq		$ACC3, $ACC3, $ACC3
Packit Service 084de1
	vmovdqa		$ACC2, 32*1-128($aap)
Packit Service 084de1
	vpaddq		$ACC4, $ACC4, $ACC4
Packit Service 084de1
	vmovdqa		$ACC3, 32*2-128($aap)
Packit Service 084de1
	vpaddq		$ACC5, $ACC5, $ACC5
Packit Service 084de1
	vmovdqa		$ACC4, 32*3-128($aap)
Packit Service 084de1
	vpaddq		$ACC6, $ACC6, $ACC6
Packit Service 084de1
	vmovdqa		$ACC5, 32*4-128($aap)
Packit Service 084de1
	vpaddq		$ACC7, $ACC7, $ACC7
Packit Service 084de1
	vmovdqa		$ACC6, 32*5-128($aap)
Packit Service 084de1
	vpaddq		$ACC8, $ACC8, $ACC8
Packit Service 084de1
	vmovdqa		$ACC7, 32*6-128($aap)
Packit Service 084de1
	vpxor		$ACC9, $ACC9, $ACC9
Packit Service 084de1
	vmovdqa		$ACC8, 32*7-128($aap)
Packit Service 084de1
Packit Service 084de1
	vpmuludq	32*0-128($ap), $B1, $ACC0
Packit Service 084de1
	 vpbroadcastq	32*1-128($ap), $B2
Packit Service 084de1
	 vmovdqu	$ACC9, 32*9-192($tp0)	# zero upper half
Packit Service 084de1
	vpmuludq	$B1, $ACC1, $ACC1
Packit Service 084de1
	 vmovdqu	$ACC9, 32*10-448($tp1)
Packit Service 084de1
	vpmuludq	$B1, $ACC2, $ACC2
Packit Service 084de1
	 vmovdqu	$ACC9, 32*11-448($tp1)
Packit Service 084de1
	vpmuludq	$B1, $ACC3, $ACC3
Packit Service 084de1
	 vmovdqu	$ACC9, 32*12-448($tp1)
Packit Service 084de1
	vpmuludq	$B1, $ACC4, $ACC4
Packit Service 084de1
	 vmovdqu	$ACC9, 32*13-448($tp1)
Packit Service 084de1
	vpmuludq	$B1, $ACC5, $ACC5
Packit Service 084de1
	 vmovdqu	$ACC9, 32*14-448($tp1)
Packit Service 084de1
	vpmuludq	$B1, $ACC6, $ACC6
Packit Service 084de1
	 vmovdqu	$ACC9, 32*15-448($tp1)
Packit Service 084de1
	vpmuludq	$B1, $ACC7, $ACC7
Packit Service 084de1
	 vmovdqu	$ACC9, 32*16-448($tp1)
Packit Service 084de1
	vpmuludq	$B1, $ACC8, $ACC8
Packit Service 084de1
	 vpbroadcastq	32*2-128($ap), $B1
Packit Service 084de1
	 vmovdqu	$ACC9, 32*17-448($tp1)
Packit Service 084de1
Packit Service 084de1
	mov	$ap, $tpa
Packit Service 084de1
	mov 	\$4, $i
Packit Service 084de1
	jmp	.Lsqr_entry_1024
Packit Service 084de1
___
Packit Service 084de1
$TEMP0=$Y1;
Packit Service 084de1
$TEMP2=$Y2;
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
.align	32
Packit Service 084de1
.LOOP_SQR_1024:
Packit Service 084de1
	 vpbroadcastq	32*1-128($tpa), $B2
Packit Service 084de1
	vpmuludq	32*0-128($ap), $B1, $ACC0
Packit Service 084de1
	vpaddq		32*0-192($tp0), $ACC0, $ACC0
Packit Service 084de1
	vpmuludq	32*0-128($aap), $B1, $ACC1
Packit Service 084de1
	vpaddq		32*1-192($tp0), $ACC1, $ACC1
Packit Service 084de1
	vpmuludq	32*1-128($aap), $B1, $ACC2
Packit Service 084de1
	vpaddq		32*2-192($tp0), $ACC2, $ACC2
Packit Service 084de1
	vpmuludq	32*2-128($aap), $B1, $ACC3
Packit Service 084de1
	vpaddq		32*3-192($tp0), $ACC3, $ACC3
Packit Service 084de1
	vpmuludq	32*3-128($aap), $B1, $ACC4
Packit Service 084de1
	vpaddq		32*4-192($tp0), $ACC4, $ACC4
Packit Service 084de1
	vpmuludq	32*4-128($aap), $B1, $ACC5
Packit Service 084de1
	vpaddq		32*5-192($tp0), $ACC5, $ACC5
Packit Service 084de1
	vpmuludq	32*5-128($aap), $B1, $ACC6
Packit Service 084de1
	vpaddq		32*6-192($tp0), $ACC6, $ACC6
Packit Service 084de1
	vpmuludq	32*6-128($aap), $B1, $ACC7
Packit Service 084de1
	vpaddq		32*7-192($tp0), $ACC7, $ACC7
Packit Service 084de1
	vpmuludq	32*7-128($aap), $B1, $ACC8
Packit Service 084de1
	 vpbroadcastq	32*2-128($tpa), $B1
Packit Service 084de1
	vpaddq		32*8-192($tp0), $ACC8, $ACC8
Packit Service 084de1
.Lsqr_entry_1024:
Packit Service 084de1
	vmovdqu		$ACC0, 32*0-192($tp0)
Packit Service 084de1
	vmovdqu		$ACC1, 32*1-192($tp0)
Packit Service 084de1
Packit Service 084de1
	vpmuludq	32*1-128($ap), $B2, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC2, $ACC2
Packit Service 084de1
	vpmuludq	32*1-128($aap), $B2, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP1, $ACC3, $ACC3
Packit Service 084de1
	vpmuludq	32*2-128($aap), $B2, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP2, $ACC4, $ACC4
Packit Service 084de1
	vpmuludq	32*3-128($aap), $B2, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC5, $ACC5
Packit Service 084de1
	vpmuludq	32*4-128($aap), $B2, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP1, $ACC6, $ACC6
Packit Service 084de1
	vpmuludq	32*5-128($aap), $B2, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP2, $ACC7, $ACC7
Packit Service 084de1
	vpmuludq	32*6-128($aap), $B2, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC8, $ACC8
Packit Service 084de1
	vpmuludq	32*7-128($aap), $B2, $ACC0
Packit Service 084de1
	 vpbroadcastq	32*3-128($tpa), $B2
Packit Service 084de1
	vpaddq		32*9-192($tp0), $ACC0, $ACC0
Packit Service 084de1
Packit Service 084de1
	vmovdqu		$ACC2, 32*2-192($tp0)
Packit Service 084de1
	vmovdqu		$ACC3, 32*3-192($tp0)
Packit Service 084de1
Packit Service 084de1
	vpmuludq	32*2-128($ap), $B1, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP2, $ACC4, $ACC4
Packit Service 084de1
	vpmuludq	32*2-128($aap), $B1, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC5, $ACC5
Packit Service 084de1
	vpmuludq	32*3-128($aap), $B1, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP1, $ACC6, $ACC6
Packit Service 084de1
	vpmuludq	32*4-128($aap), $B1, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP2, $ACC7, $ACC7
Packit Service 084de1
	vpmuludq	32*5-128($aap), $B1, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC8, $ACC8
Packit Service 084de1
	vpmuludq	32*6-128($aap), $B1, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP1, $ACC0, $ACC0
Packit Service 084de1
	vpmuludq	32*7-128($aap), $B1, $ACC1
Packit Service 084de1
	 vpbroadcastq	32*4-128($tpa), $B1
Packit Service 084de1
	vpaddq		32*10-448($tp1), $ACC1, $ACC1
Packit Service 084de1
Packit Service 084de1
	vmovdqu		$ACC4, 32*4-192($tp0)
Packit Service 084de1
	vmovdqu		$ACC5, 32*5-192($tp0)
Packit Service 084de1
Packit Service 084de1
	vpmuludq	32*3-128($ap), $B2, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC6, $ACC6
Packit Service 084de1
	vpmuludq	32*3-128($aap), $B2, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP1, $ACC7, $ACC7
Packit Service 084de1
	vpmuludq	32*4-128($aap), $B2, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP2, $ACC8, $ACC8
Packit Service 084de1
	vpmuludq	32*5-128($aap), $B2, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC0, $ACC0
Packit Service 084de1
	vpmuludq	32*6-128($aap), $B2, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP1, $ACC1, $ACC1
Packit Service 084de1
	vpmuludq	32*7-128($aap), $B2, $ACC2
Packit Service 084de1
	 vpbroadcastq	32*5-128($tpa), $B2
Packit Service 084de1
	vpaddq		32*11-448($tp1), $ACC2, $ACC2
Packit Service 084de1
Packit Service 084de1
	vmovdqu		$ACC6, 32*6-192($tp0)
Packit Service 084de1
	vmovdqu		$ACC7, 32*7-192($tp0)
Packit Service 084de1
Packit Service 084de1
	vpmuludq	32*4-128($ap), $B1, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC8, $ACC8
Packit Service 084de1
	vpmuludq	32*4-128($aap), $B1, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP1, $ACC0, $ACC0
Packit Service 084de1
	vpmuludq	32*5-128($aap), $B1, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP2, $ACC1, $ACC1
Packit Service 084de1
	vpmuludq	32*6-128($aap), $B1, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC2, $ACC2
Packit Service 084de1
	vpmuludq	32*7-128($aap), $B1, $ACC3
Packit Service 084de1
	 vpbroadcastq	32*6-128($tpa), $B1
Packit Service 084de1
	vpaddq		32*12-448($tp1), $ACC3, $ACC3
Packit Service 084de1
Packit Service 084de1
	vmovdqu		$ACC8, 32*8-192($tp0)
Packit Service 084de1
	vmovdqu		$ACC0, 32*9-192($tp0)
Packit Service 084de1
	lea		8($tp0), $tp0
Packit Service 084de1
Packit Service 084de1
	vpmuludq	32*5-128($ap), $B2, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP2, $ACC1, $ACC1
Packit Service 084de1
	vpmuludq	32*5-128($aap), $B2, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC2, $ACC2
Packit Service 084de1
	vpmuludq	32*6-128($aap), $B2, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP1, $ACC3, $ACC3
Packit Service 084de1
	vpmuludq	32*7-128($aap), $B2, $ACC4
Packit Service 084de1
	 vpbroadcastq	32*7-128($tpa), $B2
Packit Service 084de1
	vpaddq		32*13-448($tp1), $ACC4, $ACC4
Packit Service 084de1
Packit Service 084de1
	vmovdqu		$ACC1, 32*10-448($tp1)
Packit Service 084de1
	vmovdqu		$ACC2, 32*11-448($tp1)
Packit Service 084de1
Packit Service 084de1
	vpmuludq	32*6-128($ap), $B1, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC3, $ACC3
Packit Service 084de1
	vpmuludq	32*6-128($aap), $B1, $TEMP1
Packit Service 084de1
	 vpbroadcastq	32*8-128($tpa), $ACC0		# borrow $ACC0 for $B1
Packit Service 084de1
	vpaddq		$TEMP1, $ACC4, $ACC4
Packit Service 084de1
	vpmuludq	32*7-128($aap), $B1, $ACC5
Packit Service 084de1
	 vpbroadcastq	32*0+8-128($tpa), $B1		# for next iteration
Packit Service 084de1
	vpaddq		32*14-448($tp1), $ACC5, $ACC5
Packit Service 084de1
Packit Service 084de1
	vmovdqu		$ACC3, 32*12-448($tp1)
Packit Service 084de1
	vmovdqu		$ACC4, 32*13-448($tp1)
Packit Service 084de1
	lea		8($tpa), $tpa
Packit Service 084de1
Packit Service 084de1
	vpmuludq	32*7-128($ap), $B2, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC5, $ACC5
Packit Service 084de1
	vpmuludq	32*7-128($aap), $B2, $ACC6
Packit Service 084de1
	vpaddq		32*15-448($tp1), $ACC6, $ACC6
Packit Service 084de1
Packit Service 084de1
	vpmuludq	32*8-128($ap), $ACC0, $ACC7
Packit Service 084de1
	vmovdqu		$ACC5, 32*14-448($tp1)
Packit Service 084de1
	vpaddq		32*16-448($tp1), $ACC7, $ACC7
Packit Service 084de1
	vmovdqu		$ACC6, 32*15-448($tp1)
Packit Service 084de1
	vmovdqu		$ACC7, 32*16-448($tp1)
Packit Service 084de1
	lea		8($tp1), $tp1
Packit Service 084de1
Packit Service 084de1
	dec	$i
Packit Service 084de1
	jnz	.LOOP_SQR_1024
Packit Service 084de1
___
Packit Service 084de1
$ZERO = $ACC9;
Packit Service 084de1
$TEMP0 = $B1;
Packit Service 084de1
$TEMP2 = $B2;
Packit Service 084de1
$TEMP3 = $Y1;
Packit Service 084de1
$TEMP4 = $Y2;
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	# we need to fix indices 32-39 to avoid overflow
Packit Service 084de1
	vmovdqu		32*8(%rsp), $ACC8		# 32*8-192($tp0),
Packit Service 084de1
	vmovdqu		32*9(%rsp), $ACC1		# 32*9-192($tp0)
Packit Service 084de1
	vmovdqu		32*10(%rsp), $ACC2		# 32*10-192($tp0)
Packit Service 084de1
	lea		192(%rsp), $tp0			# 64+128=192
Packit Service 084de1
Packit Service 084de1
	vpsrlq		\$29, $ACC8, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC8, $ACC8
Packit Service 084de1
	vpsrlq		\$29, $ACC1, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC1, $ACC1
Packit Service 084de1
Packit Service 084de1
	vpermq		\$0x93, $TEMP1, $TEMP1
Packit Service 084de1
	vpxor		$ZERO, $ZERO, $ZERO
Packit Service 084de1
	vpermq		\$0x93, $TEMP2, $TEMP2
Packit Service 084de1
Packit Service 084de1
	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
Packit Service 084de1
	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP0, $ACC8, $ACC8
Packit Service 084de1
	vpblendd	\$3, $TEMP2, $ZERO, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP1, $ACC1, $ACC1
Packit Service 084de1
	vpaddq		$TEMP2, $ACC2, $ACC2
Packit Service 084de1
	vmovdqu		$ACC1, 32*9-192($tp0)
Packit Service 084de1
	vmovdqu		$ACC2, 32*10-192($tp0)
Packit Service 084de1
Packit Service 084de1
	mov	(%rsp), %rax
Packit Service 084de1
	mov	8(%rsp), $r1
Packit Service 084de1
	mov	16(%rsp), $r2
Packit Service 084de1
	mov	24(%rsp), $r3
Packit Service 084de1
	vmovdqu	32*1(%rsp), $ACC1
Packit Service 084de1
	vmovdqu	32*2-192($tp0), $ACC2
Packit Service 084de1
	vmovdqu	32*3-192($tp0), $ACC3
Packit Service 084de1
	vmovdqu	32*4-192($tp0), $ACC4
Packit Service 084de1
	vmovdqu	32*5-192($tp0), $ACC5
Packit Service 084de1
	vmovdqu	32*6-192($tp0), $ACC6
Packit Service 084de1
	vmovdqu	32*7-192($tp0), $ACC7
Packit Service 084de1
Packit Service 084de1
	mov	%rax, $r0
Packit Service 084de1
	imull	$n0, %eax
Packit Service 084de1
	and	\$0x1fffffff, %eax
Packit Service 084de1
	vmovd	%eax, $Y1
Packit Service 084de1
Packit Service 084de1
	mov	%rax, %rdx
Packit Service 084de1
	imulq	-128($np), %rax
Packit Service 084de1
	 vpbroadcastq	$Y1, $Y1
Packit Service 084de1
	add	%rax, $r0
Packit Service 084de1
	mov	%rdx, %rax
Packit Service 084de1
	imulq	8-128($np), %rax
Packit Service 084de1
	shr	\$29, $r0
Packit Service 084de1
	add	%rax, $r1
Packit Service 084de1
	mov	%rdx, %rax
Packit Service 084de1
	imulq	16-128($np), %rax
Packit Service 084de1
	add	$r0, $r1
Packit Service 084de1
	add	%rax, $r2
Packit Service 084de1
	imulq	24-128($np), %rdx
Packit Service 084de1
	add	%rdx, $r3
Packit Service 084de1
Packit Service 084de1
	mov	$r1, %rax
Packit Service 084de1
	imull	$n0, %eax
Packit Service 084de1
	and	\$0x1fffffff, %eax
Packit Service 084de1
Packit Service 084de1
	mov \$9, $i
Packit Service 084de1
	jmp .LOOP_REDUCE_1024
Packit Service 084de1
Packit Service 084de1
.align	32
Packit Service 084de1
.LOOP_REDUCE_1024:
Packit Service 084de1
	vmovd	%eax, $Y2
Packit Service 084de1
	vpbroadcastq	$Y2, $Y2
Packit Service 084de1
Packit Service 084de1
	vpmuludq	32*1-128($np), $Y1, $TEMP0
Packit Service 084de1
	 mov	%rax, %rdx
Packit Service 084de1
	 imulq	-128($np), %rax
Packit Service 084de1
	vpaddq		$TEMP0, $ACC1, $ACC1
Packit Service 084de1
	 add	%rax, $r1
Packit Service 084de1
	vpmuludq	32*2-128($np), $Y1, $TEMP1
Packit Service 084de1
	 mov	%rdx, %rax
Packit Service 084de1
	 imulq	8-128($np), %rax
Packit Service 084de1
	vpaddq		$TEMP1, $ACC2, $ACC2
Packit Service 084de1
	vpmuludq	32*3-128($np), $Y1, $TEMP2
Packit Service 084de1
	 .byte	0x67
Packit Service 084de1
	 add	%rax, $r2
Packit Service 084de1
	 .byte	0x67
Packit Service 084de1
	 mov	%rdx, %rax
Packit Service 084de1
	 imulq	16-128($np), %rax
Packit Service 084de1
	 shr	\$29, $r1
Packit Service 084de1
	vpaddq		$TEMP2, $ACC3, $ACC3
Packit Service 084de1
	vpmuludq	32*4-128($np), $Y1, $TEMP0
Packit Service 084de1
	 add	%rax, $r3
Packit Service 084de1
	 add	$r1, $r2
Packit Service 084de1
	vpaddq		$TEMP0, $ACC4, $ACC4
Packit Service 084de1
	vpmuludq	32*5-128($np), $Y1, $TEMP1
Packit Service 084de1
	 mov	$r2, %rax
Packit Service 084de1
	 imull	$n0, %eax
Packit Service 084de1
	vpaddq		$TEMP1, $ACC5, $ACC5
Packit Service 084de1
	vpmuludq	32*6-128($np), $Y1, $TEMP2
Packit Service 084de1
	 and	\$0x1fffffff, %eax
Packit Service 084de1
	vpaddq		$TEMP2, $ACC6, $ACC6
Packit Service 084de1
	vpmuludq	32*7-128($np), $Y1, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC7, $ACC7
Packit Service 084de1
	vpmuludq	32*8-128($np), $Y1, $TEMP1
Packit Service 084de1
	 vmovd	%eax, $Y1
Packit Service 084de1
	 #vmovdqu	32*1-8-128($np), $TEMP2		# moved below
Packit Service 084de1
	vpaddq		$TEMP1, $ACC8, $ACC8
Packit Service 084de1
	 #vmovdqu	32*2-8-128($np), $TEMP0		# moved below
Packit Service 084de1
	 vpbroadcastq	$Y1, $Y1
Packit Service 084de1
Packit Service 084de1
	vpmuludq	32*1-8-128($np), $Y2, $TEMP2	# see above
Packit Service 084de1
	vmovdqu		32*3-8-128($np), $TEMP1
Packit Service 084de1
	 mov	%rax, %rdx
Packit Service 084de1
	 imulq	-128($np), %rax
Packit Service 084de1
	vpaddq		$TEMP2, $ACC1, $ACC1
Packit Service 084de1
	vpmuludq	32*2-8-128($np), $Y2, $TEMP0	# see above
Packit Service 084de1
	vmovdqu		32*4-8-128($np), $TEMP2
Packit Service 084de1
	 add	%rax, $r2
Packit Service 084de1
	 mov	%rdx, %rax
Packit Service 084de1
	 imulq	8-128($np), %rax
Packit Service 084de1
	vpaddq		$TEMP0, $ACC2, $ACC2
Packit Service 084de1
	 add	$r3, %rax
Packit Service 084de1
	 shr	\$29, $r2
Packit Service 084de1
	vpmuludq	$Y2, $TEMP1, $TEMP1
Packit Service 084de1
	vmovdqu		32*5-8-128($np), $TEMP0
Packit Service 084de1
	 add	$r2, %rax
Packit Service 084de1
	vpaddq		$TEMP1, $ACC3, $ACC3
Packit Service 084de1
	vpmuludq	$Y2, $TEMP2, $TEMP2
Packit Service 084de1
	vmovdqu		32*6-8-128($np), $TEMP1
Packit Service 084de1
	 .byte	0x67
Packit Service 084de1
	 mov	%rax, $r3
Packit Service 084de1
	 imull	$n0, %eax
Packit Service 084de1
	vpaddq		$TEMP2, $ACC4, $ACC4
Packit Service 084de1
	vpmuludq	$Y2, $TEMP0, $TEMP0
Packit Service 084de1
	.byte	0xc4,0x41,0x7e,0x6f,0x9d,0x58,0x00,0x00,0x00	# vmovdqu		32*7-8-128($np), $TEMP2
Packit Service 084de1
	 and	\$0x1fffffff, %eax
Packit Service 084de1
	vpaddq		$TEMP0, $ACC5, $ACC5
Packit Service 084de1
	vpmuludq	$Y2, $TEMP1, $TEMP1
Packit Service 084de1
	vmovdqu		32*8-8-128($np), $TEMP0
Packit Service 084de1
	vpaddq		$TEMP1, $ACC6, $ACC6
Packit Service 084de1
	vpmuludq	$Y2, $TEMP2, $TEMP2
Packit Service 084de1
	vmovdqu		32*9-8-128($np), $ACC9
Packit Service 084de1
	 vmovd	%eax, $ACC0			# borrow ACC0 for Y2
Packit Service 084de1
	 imulq	-128($np), %rax
Packit Service 084de1
	vpaddq		$TEMP2, $ACC7, $ACC7
Packit Service 084de1
	vpmuludq	$Y2, $TEMP0, $TEMP0
Packit Service 084de1
	 vmovdqu	32*1-16-128($np), $TEMP1
Packit Service 084de1
	 vpbroadcastq	$ACC0, $ACC0
Packit Service 084de1
	vpaddq		$TEMP0, $ACC8, $ACC8
Packit Service 084de1
	vpmuludq	$Y2, $ACC9, $ACC9
Packit Service 084de1
	 vmovdqu	32*2-16-128($np), $TEMP2
Packit Service 084de1
	 add	%rax, $r3
Packit Service 084de1
Packit Service 084de1
___
Packit Service 084de1
($ACC0,$Y2)=($Y2,$ACC0);
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	 vmovdqu	32*1-24-128($np), $ACC0
Packit Service 084de1
	vpmuludq	$Y1, $TEMP1, $TEMP1
Packit Service 084de1
	vmovdqu		32*3-16-128($np), $TEMP0
Packit Service 084de1
	vpaddq		$TEMP1, $ACC1, $ACC1
Packit Service 084de1
	 vpmuludq	$Y2, $ACC0, $ACC0
Packit Service 084de1
	vpmuludq	$Y1, $TEMP2, $TEMP2
Packit Service 084de1
	.byte	0xc4,0x41,0x7e,0x6f,0xb5,0xf0,0xff,0xff,0xff	# vmovdqu		32*4-16-128($np), $TEMP1
Packit Service 084de1
	 vpaddq		$ACC1, $ACC0, $ACC0
Packit Service 084de1
	vpaddq		$TEMP2, $ACC2, $ACC2
Packit Service 084de1
	vpmuludq	$Y1, $TEMP0, $TEMP0
Packit Service 084de1
	vmovdqu		32*5-16-128($np), $TEMP2
Packit Service 084de1
	 .byte	0x67
Packit Service 084de1
	 vmovq		$ACC0, %rax
Packit Service 084de1
	 vmovdqu	$ACC0, (%rsp)		# transfer $r0-$r3
Packit Service 084de1
	vpaddq		$TEMP0, $ACC3, $ACC3
Packit Service 084de1
	vpmuludq	$Y1, $TEMP1, $TEMP1
Packit Service 084de1
	vmovdqu		32*6-16-128($np), $TEMP0
Packit Service 084de1
	vpaddq		$TEMP1, $ACC4, $ACC4
Packit Service 084de1
	vpmuludq	$Y1, $TEMP2, $TEMP2
Packit Service 084de1
	vmovdqu		32*7-16-128($np), $TEMP1
Packit Service 084de1
	vpaddq		$TEMP2, $ACC5, $ACC5
Packit Service 084de1
	vpmuludq	$Y1, $TEMP0, $TEMP0
Packit Service 084de1
	vmovdqu		32*8-16-128($np), $TEMP2
Packit Service 084de1
	vpaddq		$TEMP0, $ACC6, $ACC6
Packit Service 084de1
	vpmuludq	$Y1, $TEMP1, $TEMP1
Packit Service 084de1
	 shr	\$29, $r3
Packit Service 084de1
	vmovdqu		32*9-16-128($np), $TEMP0
Packit Service 084de1
	 add	$r3, %rax
Packit Service 084de1
	vpaddq		$TEMP1, $ACC7, $ACC7
Packit Service 084de1
	vpmuludq	$Y1, $TEMP2, $TEMP2
Packit Service 084de1
	 #vmovdqu	32*2-24-128($np), $TEMP1	# moved below
Packit Service 084de1
	 mov	%rax, $r0
Packit Service 084de1
	 imull	$n0, %eax
Packit Service 084de1
	vpaddq		$TEMP2, $ACC8, $ACC8
Packit Service 084de1
	vpmuludq	$Y1, $TEMP0, $TEMP0
Packit Service 084de1
	 and	\$0x1fffffff, %eax
Packit Service 084de1
	 vmovd	%eax, $Y1
Packit Service 084de1
	 vmovdqu	32*3-24-128($np), $TEMP2
Packit Service 084de1
	.byte	0x67
Packit Service 084de1
	vpaddq		$TEMP0, $ACC9, $ACC9
Packit Service 084de1
	 vpbroadcastq	$Y1, $Y1
Packit Service 084de1
Packit Service 084de1
	vpmuludq	32*2-24-128($np), $Y2, $TEMP1	# see above
Packit Service 084de1
	vmovdqu		32*4-24-128($np), $TEMP0
Packit Service 084de1
	 mov	%rax, %rdx
Packit Service 084de1
	 imulq	-128($np), %rax
Packit Service 084de1
	 mov	8(%rsp), $r1
Packit Service 084de1
	vpaddq		$TEMP1, $ACC2, $ACC1
Packit Service 084de1
	vpmuludq	$Y2, $TEMP2, $TEMP2
Packit Service 084de1
	vmovdqu		32*5-24-128($np), $TEMP1
Packit Service 084de1
	 add	%rax, $r0
Packit Service 084de1
	 mov	%rdx, %rax
Packit Service 084de1
	 imulq	8-128($np), %rax
Packit Service 084de1
	 .byte	0x67
Packit Service 084de1
	 shr	\$29, $r0
Packit Service 084de1
	 mov	16(%rsp), $r2
Packit Service 084de1
	vpaddq		$TEMP2, $ACC3, $ACC2
Packit Service 084de1
	vpmuludq	$Y2, $TEMP0, $TEMP0
Packit Service 084de1
	vmovdqu		32*6-24-128($np), $TEMP2
Packit Service 084de1
	 add	%rax, $r1
Packit Service 084de1
	 mov	%rdx, %rax
Packit Service 084de1
	 imulq	16-128($np), %rax
Packit Service 084de1
	vpaddq		$TEMP0, $ACC4, $ACC3
Packit Service 084de1
	vpmuludq	$Y2, $TEMP1, $TEMP1
Packit Service 084de1
	vmovdqu		32*7-24-128($np), $TEMP0
Packit Service 084de1
	 imulq	24-128($np), %rdx		# future $r3
Packit Service 084de1
	 add	%rax, $r2
Packit Service 084de1
	 lea	($r0,$r1), %rax
Packit Service 084de1
	vpaddq		$TEMP1, $ACC5, $ACC4
Packit Service 084de1
	vpmuludq	$Y2, $TEMP2, $TEMP2
Packit Service 084de1
	vmovdqu		32*8-24-128($np), $TEMP1
Packit Service 084de1
	 mov	%rax, $r1
Packit Service 084de1
	 imull	$n0, %eax
Packit Service 084de1
	vpmuludq	$Y2, $TEMP0, $TEMP0
Packit Service 084de1
	vpaddq		$TEMP2, $ACC6, $ACC5
Packit Service 084de1
	vmovdqu		32*9-24-128($np), $TEMP2
Packit Service 084de1
	 and	\$0x1fffffff, %eax
Packit Service 084de1
	vpaddq		$TEMP0, $ACC7, $ACC6
Packit Service 084de1
	vpmuludq	$Y2, $TEMP1, $TEMP1
Packit Service 084de1
	 add	24(%rsp), %rdx
Packit Service 084de1
	vpaddq		$TEMP1, $ACC8, $ACC7
Packit Service 084de1
	vpmuludq	$Y2, $TEMP2, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP2, $ACC9, $ACC8
Packit Service 084de1
	 vmovq	$r3, $ACC9
Packit Service 084de1
	 mov	%rdx, $r3
Packit Service 084de1
Packit Service 084de1
	dec	$i
Packit Service 084de1
	jnz	.LOOP_REDUCE_1024
Packit Service 084de1
___
Packit Service 084de1
($ACC0,$Y2)=($Y2,$ACC0);
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	lea	448(%rsp), $tp1			# size optimization
Packit Service 084de1
	vpaddq	$ACC9, $Y2, $ACC0
Packit Service 084de1
	vpxor	$ZERO, $ZERO, $ZERO
Packit Service 084de1
Packit Service 084de1
	vpaddq		32*9-192($tp0), $ACC0, $ACC0
Packit Service 084de1
	vpaddq		32*10-448($tp1), $ACC1, $ACC1
Packit Service 084de1
	vpaddq		32*11-448($tp1), $ACC2, $ACC2
Packit Service 084de1
	vpaddq		32*12-448($tp1), $ACC3, $ACC3
Packit Service 084de1
	vpaddq		32*13-448($tp1), $ACC4, $ACC4
Packit Service 084de1
	vpaddq		32*14-448($tp1), $ACC5, $ACC5
Packit Service 084de1
	vpaddq		32*15-448($tp1), $ACC6, $ACC6
Packit Service 084de1
	vpaddq		32*16-448($tp1), $ACC7, $ACC7
Packit Service 084de1
	vpaddq		32*17-448($tp1), $ACC8, $ACC8
Packit Service 084de1
Packit Service 084de1
	vpsrlq		\$29, $ACC0, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC0, $ACC0
Packit Service 084de1
	vpsrlq		\$29, $ACC1, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC1, $ACC1
Packit Service 084de1
	vpsrlq		\$29, $ACC2, $TEMP3
Packit Service 084de1
	vpermq		\$0x93, $TEMP1, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC2, $ACC2
Packit Service 084de1
	vpsrlq		\$29, $ACC3, $TEMP4
Packit Service 084de1
	vpermq		\$0x93, $TEMP2, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC3, $ACC3
Packit Service 084de1
	vpermq		\$0x93, $TEMP3, $TEMP3
Packit Service 084de1
Packit Service 084de1
	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
Packit Service 084de1
	vpermq		\$0x93, $TEMP4, $TEMP4
Packit Service 084de1
	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP0, $ACC0, $ACC0
Packit Service 084de1
	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP1, $ACC1, $ACC1
Packit Service 084de1
	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
Packit Service 084de1
	vpaddq		$TEMP2, $ACC2, $ACC2
Packit Service 084de1
	vpblendd	\$3, $TEMP4, $ZERO, $TEMP4
Packit Service 084de1
	vpaddq		$TEMP3, $ACC3, $ACC3
Packit Service 084de1
	vpaddq		$TEMP4, $ACC4, $ACC4
Packit Service 084de1
Packit Service 084de1
	vpsrlq		\$29, $ACC0, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC0, $ACC0
Packit Service 084de1
	vpsrlq		\$29, $ACC1, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC1, $ACC1
Packit Service 084de1
	vpsrlq		\$29, $ACC2, $TEMP3
Packit Service 084de1
	vpermq		\$0x93, $TEMP1, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC2, $ACC2
Packit Service 084de1
	vpsrlq		\$29, $ACC3, $TEMP4
Packit Service 084de1
	vpermq		\$0x93, $TEMP2, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC3, $ACC3
Packit Service 084de1
	vpermq		\$0x93, $TEMP3, $TEMP3
Packit Service 084de1
Packit Service 084de1
	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
Packit Service 084de1
	vpermq		\$0x93, $TEMP4, $TEMP4
Packit Service 084de1
	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP0, $ACC0, $ACC0
Packit Service 084de1
	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP1, $ACC1, $ACC1
Packit Service 084de1
	vmovdqu		$ACC0, 32*0-128($rp)
Packit Service 084de1
	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
Packit Service 084de1
	vpaddq		$TEMP2, $ACC2, $ACC2
Packit Service 084de1
	vmovdqu		$ACC1, 32*1-128($rp)
Packit Service 084de1
	vpblendd	\$3, $TEMP4, $ZERO, $TEMP4
Packit Service 084de1
	vpaddq		$TEMP3, $ACC3, $ACC3
Packit Service 084de1
	vmovdqu		$ACC2, 32*2-128($rp)
Packit Service 084de1
	vpaddq		$TEMP4, $ACC4, $ACC4
Packit Service 084de1
	vmovdqu		$ACC3, 32*3-128($rp)
Packit Service 084de1
___
Packit Service 084de1
$TEMP5=$ACC0;
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	vpsrlq		\$29, $ACC4, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC4, $ACC4
Packit Service 084de1
	vpsrlq		\$29, $ACC5, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC5, $ACC5
Packit Service 084de1
	vpsrlq		\$29, $ACC6, $TEMP3
Packit Service 084de1
	vpermq		\$0x93, $TEMP1, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC6, $ACC6
Packit Service 084de1
	vpsrlq		\$29, $ACC7, $TEMP4
Packit Service 084de1
	vpermq		\$0x93, $TEMP2, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC7, $ACC7
Packit Service 084de1
	vpsrlq		\$29, $ACC8, $TEMP5
Packit Service 084de1
	vpermq		\$0x93, $TEMP3, $TEMP3
Packit Service 084de1
	vpand		$AND_MASK, $ACC8, $ACC8
Packit Service 084de1
	vpermq		\$0x93, $TEMP4, $TEMP4
Packit Service 084de1
Packit Service 084de1
	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
Packit Service 084de1
	vpermq		\$0x93, $TEMP5, $TEMP5
Packit Service 084de1
	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP0, $ACC4, $ACC4
Packit Service 084de1
	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP1, $ACC5, $ACC5
Packit Service 084de1
	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
Packit Service 084de1
	vpaddq		$TEMP2, $ACC6, $ACC6
Packit Service 084de1
	vpblendd	\$3, $TEMP4, $TEMP5, $TEMP4
Packit Service 084de1
	vpaddq		$TEMP3, $ACC7, $ACC7
Packit Service 084de1
	vpaddq		$TEMP4, $ACC8, $ACC8
Packit Service 084de1
Packit Service 084de1
	vpsrlq		\$29, $ACC4, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC4, $ACC4
Packit Service 084de1
	vpsrlq		\$29, $ACC5, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC5, $ACC5
Packit Service 084de1
	vpsrlq		\$29, $ACC6, $TEMP3
Packit Service 084de1
	vpermq		\$0x93, $TEMP1, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC6, $ACC6
Packit Service 084de1
	vpsrlq		\$29, $ACC7, $TEMP4
Packit Service 084de1
	vpermq		\$0x93, $TEMP2, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC7, $ACC7
Packit Service 084de1
	vpsrlq		\$29, $ACC8, $TEMP5
Packit Service 084de1
	vpermq		\$0x93, $TEMP3, $TEMP3
Packit Service 084de1
	vpand		$AND_MASK, $ACC8, $ACC8
Packit Service 084de1
	vpermq		\$0x93, $TEMP4, $TEMP4
Packit Service 084de1
Packit Service 084de1
	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
Packit Service 084de1
	vpermq		\$0x93, $TEMP5, $TEMP5
Packit Service 084de1
	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP0, $ACC4, $ACC4
Packit Service 084de1
	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP1, $ACC5, $ACC5
Packit Service 084de1
	vmovdqu		$ACC4, 32*4-128($rp)
Packit Service 084de1
	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
Packit Service 084de1
	vpaddq		$TEMP2, $ACC6, $ACC6
Packit Service 084de1
	vmovdqu		$ACC5, 32*5-128($rp)
Packit Service 084de1
	vpblendd	\$3, $TEMP4, $TEMP5, $TEMP4
Packit Service 084de1
	vpaddq		$TEMP3, $ACC7, $ACC7
Packit Service 084de1
	vmovdqu		$ACC6, 32*6-128($rp)
Packit Service 084de1
	vpaddq		$TEMP4, $ACC8, $ACC8
Packit Service 084de1
	vmovdqu		$ACC7, 32*7-128($rp)
Packit Service 084de1
	vmovdqu		$ACC8, 32*8-128($rp)
Packit Service 084de1
Packit Service 084de1
	mov	$rp, $ap
Packit Service 084de1
	dec	$rep
Packit Service 084de1
	jne	.LOOP_GRANDE_SQR_1024
Packit Service 084de1
Packit Service 084de1
	vzeroall
Packit Service 084de1
	mov	%rbp, %rax
Packit Service 084de1
.cfi_def_cfa_register	%rax
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___ if ($win64);
Packit Service 084de1
.Lsqr_1024_in_tail:
Packit Service 084de1
	movaps	-0xd8(%rax),%xmm6
Packit Service 084de1
	movaps	-0xc8(%rax),%xmm7
Packit Service 084de1
	movaps	-0xb8(%rax),%xmm8
Packit Service 084de1
	movaps	-0xa8(%rax),%xmm9
Packit Service 084de1
	movaps	-0x98(%rax),%xmm10
Packit Service 084de1
	movaps	-0x88(%rax),%xmm11
Packit Service 084de1
	movaps	-0x78(%rax),%xmm12
Packit Service 084de1
	movaps	-0x68(%rax),%xmm13
Packit Service 084de1
	movaps	-0x58(%rax),%xmm14
Packit Service 084de1
	movaps	-0x48(%rax),%xmm15
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	mov	-48(%rax),%r15
Packit Service 084de1
.cfi_restore	%r15
Packit Service 084de1
	mov	-40(%rax),%r14
Packit Service 084de1
.cfi_restore	%r14
Packit Service 084de1
	mov	-32(%rax),%r13
Packit Service 084de1
.cfi_restore	%r13
Packit Service 084de1
	mov	-24(%rax),%r12
Packit Service 084de1
.cfi_restore	%r12
Packit Service 084de1
	mov	-16(%rax),%rbp
Packit Service 084de1
.cfi_restore	%rbp
Packit Service 084de1
	mov	-8(%rax),%rbx
Packit Service 084de1
.cfi_restore	%rbx
Packit Service 084de1
	lea	(%rax),%rsp		# restore %rsp
Packit Service 084de1
.cfi_def_cfa_register	%rsp
Packit Service 084de1
.Lsqr_1024_epilogue:
Packit Service 084de1
	ret
Packit Service 084de1
.cfi_endproc
Packit Service 084de1
.size	rsaz_1024_sqr_avx2,.-rsaz_1024_sqr_avx2
Packit Service 084de1
___
Packit Service 084de1
}
Packit Service 084de1
Packit Service 084de1
{ # void AMM_WW(
Packit Service 084de1
my $rp="%rdi";	# BN_ULONG *rp,
Packit Service 084de1
my $ap="%rsi";	# const BN_ULONG *ap,
Packit Service 084de1
my $bp="%rdx";	# const BN_ULONG *bp,
Packit Service 084de1
my $np="%rcx";	# const BN_ULONG *np,
Packit Service 084de1
my $n0="%r8d";	# unsigned int n0);
Packit Service 084de1
Packit Service 084de1
# The registers that hold the accumulated redundant result
Packit Service 084de1
# The AMM works on 1024 bit operands, and redundant word size is 29
Packit Service 084de1
# Therefore: ceil(1024/29)/4 = 9
Packit Service 084de1
my $ACC0="%ymm0";
Packit Service 084de1
my $ACC1="%ymm1";
Packit Service 084de1
my $ACC2="%ymm2";
Packit Service 084de1
my $ACC3="%ymm3";
Packit Service 084de1
my $ACC4="%ymm4";
Packit Service 084de1
my $ACC5="%ymm5";
Packit Service 084de1
my $ACC6="%ymm6";
Packit Service 084de1
my $ACC7="%ymm7";
Packit Service 084de1
my $ACC8="%ymm8";
Packit Service 084de1
my $ACC9="%ymm9";
Packit Service 084de1
Packit Service 084de1
# Registers that hold the broadcasted words of multiplier, currently used
Packit Service 084de1
my $Bi="%ymm10";
Packit Service 084de1
my $Yi="%ymm11";
Packit Service 084de1
Packit Service 084de1
# Helper registers
Packit Service 084de1
my $TEMP0=$ACC0;
Packit Service 084de1
my $TEMP1="%ymm12";
Packit Service 084de1
my $TEMP2="%ymm13";
Packit Service 084de1
my $ZERO="%ymm14";
Packit Service 084de1
my $AND_MASK="%ymm15";
Packit Service 084de1
Packit Service 084de1
# alu registers that hold the first words of the ACC
Packit Service 084de1
my $r0="%r9";
Packit Service 084de1
my $r1="%r10";
Packit Service 084de1
my $r2="%r11";
Packit Service 084de1
my $r3="%r12";
Packit Service 084de1
Packit Service 084de1
my $i="%r14d";
Packit Service 084de1
my $tmp="%r15";
Packit Service 084de1
Packit Service 084de1
$bp="%r13";	# reassigned argument
Packit Service 084de1
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
.globl	rsaz_1024_mul_avx2
Packit Service 084de1
.type	rsaz_1024_mul_avx2,\@function,5
Packit Service 084de1
.align	64
Packit Service 084de1
rsaz_1024_mul_avx2:
Packit Service 084de1
.cfi_startproc
Packit Service 084de1
	lea	(%rsp), %rax
Packit Service 084de1
.cfi_def_cfa_register	%rax
Packit Service 084de1
	push	%rbx
Packit Service 084de1
.cfi_push	%rbx
Packit Service 084de1
	push	%rbp
Packit Service 084de1
.cfi_push	%rbp
Packit Service 084de1
	push	%r12
Packit Service 084de1
.cfi_push	%r12
Packit Service 084de1
	push	%r13
Packit Service 084de1
.cfi_push	%r13
Packit Service 084de1
	push	%r14
Packit Service 084de1
.cfi_push	%r14
Packit Service 084de1
	push	%r15
Packit Service 084de1
.cfi_push	%r15
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___ if ($win64);
Packit Service 084de1
	vzeroupper
Packit Service 084de1
	lea	-0xa8(%rsp),%rsp
Packit Service 084de1
	vmovaps	%xmm6,-0xd8(%rax)
Packit Service 084de1
	vmovaps	%xmm7,-0xc8(%rax)
Packit Service 084de1
	vmovaps	%xmm8,-0xb8(%rax)
Packit Service 084de1
	vmovaps	%xmm9,-0xa8(%rax)
Packit Service 084de1
	vmovaps	%xmm10,-0x98(%rax)
Packit Service 084de1
	vmovaps	%xmm11,-0x88(%rax)
Packit Service 084de1
	vmovaps	%xmm12,-0x78(%rax)
Packit Service 084de1
	vmovaps	%xmm13,-0x68(%rax)
Packit Service 084de1
	vmovaps	%xmm14,-0x58(%rax)
Packit Service 084de1
	vmovaps	%xmm15,-0x48(%rax)
Packit Service 084de1
.Lmul_1024_body:
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	mov	%rax,%rbp
Packit Service 084de1
.cfi_def_cfa_register	%rbp
Packit Service 084de1
	vzeroall
Packit Service 084de1
	mov	%rdx, $bp	# reassigned argument
Packit Service 084de1
	sub	\$64,%rsp
Packit Service 084de1
Packit Service 084de1
	# unaligned 256-bit load that crosses page boundary can
Packit Service 084de1
	# cause severe performance degradation here, so if $ap does
Packit Service 084de1
	# cross page boundary, swap it with $bp [meaning that caller
Packit Service 084de1
	# is advised to lay down $ap and $bp next to each other, so
Packit Service 084de1
	# that only one can cross page boundary].
Packit Service 084de1
	.byte	0x67,0x67
Packit Service 084de1
	mov	$ap, $tmp
Packit Service 084de1
	and	\$4095, $tmp
Packit Service 084de1
	add	\$32*10, $tmp
Packit Service 084de1
	shr	\$12, $tmp
Packit Service 084de1
	mov	$ap, $tmp
Packit Service 084de1
	cmovnz	$bp, $ap
Packit Service 084de1
	cmovnz	$tmp, $bp
Packit Service 084de1
Packit Service 084de1
	mov	$np, $tmp
Packit Service 084de1
	sub	\$-128,$ap	# size optimization
Packit Service 084de1
	sub	\$-128,$np
Packit Service 084de1
	sub	\$-128,$rp
Packit Service 084de1
Packit Service 084de1
	and	\$4095, $tmp	# see if $np crosses page
Packit Service 084de1
	add	\$32*10, $tmp
Packit Service 084de1
	.byte	0x67,0x67
Packit Service 084de1
	shr	\$12, $tmp
Packit Service 084de1
	jz	.Lmul_1024_no_n_copy
Packit Service 084de1
Packit Service 084de1
	# unaligned 256-bit load that crosses page boundary can
Packit Service 084de1
	# cause severe performance degradation here, so if $np does
Packit Service 084de1
	# cross page boundary, copy it to stack and make sure stack
Packit Service 084de1
	# frame doesn't...
Packit Service 084de1
	sub		\$32*10,%rsp
Packit Service 084de1
	vmovdqu		32*0-128($np), $ACC0
Packit Service 084de1
	and		\$-512, %rsp
Packit Service 084de1
	vmovdqu		32*1-128($np), $ACC1
Packit Service 084de1
	vmovdqu		32*2-128($np), $ACC2
Packit Service 084de1
	vmovdqu		32*3-128($np), $ACC3
Packit Service 084de1
	vmovdqu		32*4-128($np), $ACC4
Packit Service 084de1
	vmovdqu		32*5-128($np), $ACC5
Packit Service 084de1
	vmovdqu		32*6-128($np), $ACC6
Packit Service 084de1
	vmovdqu		32*7-128($np), $ACC7
Packit Service 084de1
	vmovdqu		32*8-128($np), $ACC8
Packit Service 084de1
	lea		64+128(%rsp),$np
Packit Service 084de1
	vmovdqu		$ACC0, 32*0-128($np)
Packit Service 084de1
	vpxor		$ACC0, $ACC0, $ACC0
Packit Service 084de1
	vmovdqu		$ACC1, 32*1-128($np)
Packit Service 084de1
	vpxor		$ACC1, $ACC1, $ACC1
Packit Service 084de1
	vmovdqu		$ACC2, 32*2-128($np)
Packit Service 084de1
	vpxor		$ACC2, $ACC2, $ACC2
Packit Service 084de1
	vmovdqu		$ACC3, 32*3-128($np)
Packit Service 084de1
	vpxor		$ACC3, $ACC3, $ACC3
Packit Service 084de1
	vmovdqu		$ACC4, 32*4-128($np)
Packit Service 084de1
	vpxor		$ACC4, $ACC4, $ACC4
Packit Service 084de1
	vmovdqu		$ACC5, 32*5-128($np)
Packit Service 084de1
	vpxor		$ACC5, $ACC5, $ACC5
Packit Service 084de1
	vmovdqu		$ACC6, 32*6-128($np)
Packit Service 084de1
	vpxor		$ACC6, $ACC6, $ACC6
Packit Service 084de1
	vmovdqu		$ACC7, 32*7-128($np)
Packit Service 084de1
	vpxor		$ACC7, $ACC7, $ACC7
Packit Service 084de1
	vmovdqu		$ACC8, 32*8-128($np)
Packit Service 084de1
	vmovdqa		$ACC0, $ACC8
Packit Service 084de1
	vmovdqu		$ACC9, 32*9-128($np)	# $ACC9 is zero after vzeroall
Packit Service 084de1
.Lmul_1024_no_n_copy:
Packit Service 084de1
	and	\$-64,%rsp
Packit Service 084de1
Packit Service 084de1
	mov	($bp), %rbx
Packit Service 084de1
	vpbroadcastq ($bp), $Bi
Packit Service 084de1
	vmovdqu	$ACC0, (%rsp)			# clear top of stack
Packit Service 084de1
	xor	$r0, $r0
Packit Service 084de1
	.byte	0x67
Packit Service 084de1
	xor	$r1, $r1
Packit Service 084de1
	xor	$r2, $r2
Packit Service 084de1
	xor	$r3, $r3
Packit Service 084de1
Packit Service 084de1
	vmovdqu	.Land_mask(%rip), $AND_MASK
Packit Service 084de1
	mov	\$9, $i
Packit Service 084de1
	vmovdqu	$ACC9, 32*9-128($rp)		# $ACC9 is zero after vzeroall
Packit Service 084de1
	jmp	.Loop_mul_1024
Packit Service 084de1
Packit Service 084de1
.align	32
Packit Service 084de1
.Loop_mul_1024:
Packit Service 084de1
	 vpsrlq		\$29, $ACC3, $ACC9		# correct $ACC3(*)
Packit Service 084de1
	mov	%rbx, %rax
Packit Service 084de1
	imulq	-128($ap), %rax
Packit Service 084de1
	add	$r0, %rax
Packit Service 084de1
	mov	%rbx, $r1
Packit Service 084de1
	imulq	8-128($ap), $r1
Packit Service 084de1
	add	8(%rsp), $r1
Packit Service 084de1
Packit Service 084de1
	mov	%rax, $r0
Packit Service 084de1
	imull	$n0, %eax
Packit Service 084de1
	and	\$0x1fffffff, %eax
Packit Service 084de1
Packit Service 084de1
	 mov	%rbx, $r2
Packit Service 084de1
	 imulq	16-128($ap), $r2
Packit Service 084de1
	 add	16(%rsp), $r2
Packit Service 084de1
Packit Service 084de1
	 mov	%rbx, $r3
Packit Service 084de1
	 imulq	24-128($ap), $r3
Packit Service 084de1
	 add	24(%rsp), $r3
Packit Service 084de1
	vpmuludq	32*1-128($ap),$Bi,$TEMP0
Packit Service 084de1
	 vmovd		%eax, $Yi
Packit Service 084de1
	vpaddq		$TEMP0,$ACC1,$ACC1
Packit Service 084de1
	vpmuludq	32*2-128($ap),$Bi,$TEMP1
Packit Service 084de1
	 vpbroadcastq	$Yi, $Yi
Packit Service 084de1
	vpaddq		$TEMP1,$ACC2,$ACC2
Packit Service 084de1
	vpmuludq	32*3-128($ap),$Bi,$TEMP2
Packit Service 084de1
	 vpand		$AND_MASK, $ACC3, $ACC3		# correct $ACC3
Packit Service 084de1
	vpaddq		$TEMP2,$ACC3,$ACC3
Packit Service 084de1
	vpmuludq	32*4-128($ap),$Bi,$TEMP0
Packit Service 084de1
	vpaddq		$TEMP0,$ACC4,$ACC4
Packit Service 084de1
	vpmuludq	32*5-128($ap),$Bi,$TEMP1
Packit Service 084de1
	vpaddq		$TEMP1,$ACC5,$ACC5
Packit Service 084de1
	vpmuludq	32*6-128($ap),$Bi,$TEMP2
Packit Service 084de1
	vpaddq		$TEMP2,$ACC6,$ACC6
Packit Service 084de1
	vpmuludq	32*7-128($ap),$Bi,$TEMP0
Packit Service 084de1
	 vpermq		\$0x93, $ACC9, $ACC9		# correct $ACC3
Packit Service 084de1
	vpaddq		$TEMP0,$ACC7,$ACC7
Packit Service 084de1
	vpmuludq	32*8-128($ap),$Bi,$TEMP1
Packit Service 084de1
	 vpbroadcastq	8($bp), $Bi
Packit Service 084de1
	vpaddq		$TEMP1,$ACC8,$ACC8
Packit Service 084de1
Packit Service 084de1
	mov	%rax,%rdx
Packit Service 084de1
	imulq	-128($np),%rax
Packit Service 084de1
	add	%rax,$r0
Packit Service 084de1
	mov	%rdx,%rax
Packit Service 084de1
	imulq	8-128($np),%rax
Packit Service 084de1
	add	%rax,$r1
Packit Service 084de1
	mov	%rdx,%rax
Packit Service 084de1
	imulq	16-128($np),%rax
Packit Service 084de1
	add	%rax,$r2
Packit Service 084de1
	shr	\$29, $r0
Packit Service 084de1
	imulq	24-128($np),%rdx
Packit Service 084de1
	add	%rdx,$r3
Packit Service 084de1
	add	$r0, $r1
Packit Service 084de1
Packit Service 084de1
	vpmuludq	32*1-128($np),$Yi,$TEMP2
Packit Service 084de1
	 vmovq		$Bi, %rbx
Packit Service 084de1
	vpaddq		$TEMP2,$ACC1,$ACC1
Packit Service 084de1
	vpmuludq	32*2-128($np),$Yi,$TEMP0
Packit Service 084de1
	vpaddq		$TEMP0,$ACC2,$ACC2
Packit Service 084de1
	vpmuludq	32*3-128($np),$Yi,$TEMP1
Packit Service 084de1
	vpaddq		$TEMP1,$ACC3,$ACC3
Packit Service 084de1
	vpmuludq	32*4-128($np),$Yi,$TEMP2
Packit Service 084de1
	vpaddq		$TEMP2,$ACC4,$ACC4
Packit Service 084de1
	vpmuludq	32*5-128($np),$Yi,$TEMP0
Packit Service 084de1
	vpaddq		$TEMP0,$ACC5,$ACC5
Packit Service 084de1
	vpmuludq	32*6-128($np),$Yi,$TEMP1
Packit Service 084de1
	vpaddq		$TEMP1,$ACC6,$ACC6
Packit Service 084de1
	vpmuludq	32*7-128($np),$Yi,$TEMP2
Packit Service 084de1
	 vpblendd	\$3, $ZERO, $ACC9, $TEMP1	# correct $ACC3
Packit Service 084de1
	vpaddq		$TEMP2,$ACC7,$ACC7
Packit Service 084de1
	vpmuludq	32*8-128($np),$Yi,$TEMP0
Packit Service 084de1
	 vpaddq		$TEMP1, $ACC3, $ACC3		# correct $ACC3
Packit Service 084de1
	vpaddq		$TEMP0,$ACC8,$ACC8
Packit Service 084de1
Packit Service 084de1
	mov	%rbx, %rax
Packit Service 084de1
	imulq	-128($ap),%rax
Packit Service 084de1
	add	%rax,$r1
Packit Service 084de1
	 vmovdqu	-8+32*1-128($ap),$TEMP1
Packit Service 084de1
	mov	%rbx, %rax
Packit Service 084de1
	imulq	8-128($ap),%rax
Packit Service 084de1
	add	%rax,$r2
Packit Service 084de1
	 vmovdqu	-8+32*2-128($ap),$TEMP2
Packit Service 084de1
Packit Service 084de1
	mov	$r1, %rax
Packit Service 084de1
	 vpblendd	\$0xfc, $ZERO, $ACC9, $ACC9	# correct $ACC3
Packit Service 084de1
	imull	$n0, %eax
Packit Service 084de1
	 vpaddq		$ACC9,$ACC4,$ACC4		# correct $ACC3
Packit Service 084de1
	and	\$0x1fffffff, %eax
Packit Service 084de1
Packit Service 084de1
	 imulq	16-128($ap),%rbx
Packit Service 084de1
	 add	%rbx,$r3
Packit Service 084de1
	vpmuludq	$Bi,$TEMP1,$TEMP1
Packit Service 084de1
	 vmovd		%eax, $Yi
Packit Service 084de1
	vmovdqu		-8+32*3-128($ap),$TEMP0
Packit Service 084de1
	vpaddq		$TEMP1,$ACC1,$ACC1
Packit Service 084de1
	vpmuludq	$Bi,$TEMP2,$TEMP2
Packit Service 084de1
	 vpbroadcastq	$Yi, $Yi
Packit Service 084de1
	vmovdqu		-8+32*4-128($ap),$TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC2,$ACC2
Packit Service 084de1
	vpmuludq	$Bi,$TEMP0,$TEMP0
Packit Service 084de1
	vmovdqu		-8+32*5-128($ap),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC3,$ACC3
Packit Service 084de1
	vpmuludq	$Bi,$TEMP1,$TEMP1
Packit Service 084de1
	vmovdqu		-8+32*6-128($ap),$TEMP0
Packit Service 084de1
	vpaddq		$TEMP1,$ACC4,$ACC4
Packit Service 084de1
	vpmuludq	$Bi,$TEMP2,$TEMP2
Packit Service 084de1
	vmovdqu		-8+32*7-128($ap),$TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC5,$ACC5
Packit Service 084de1
	vpmuludq	$Bi,$TEMP0,$TEMP0
Packit Service 084de1
	vmovdqu		-8+32*8-128($ap),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC6,$ACC6
Packit Service 084de1
	vpmuludq	$Bi,$TEMP1,$TEMP1
Packit Service 084de1
	vmovdqu		-8+32*9-128($ap),$ACC9
Packit Service 084de1
	vpaddq		$TEMP1,$ACC7,$ACC7
Packit Service 084de1
	vpmuludq	$Bi,$TEMP2,$TEMP2
Packit Service 084de1
	vpaddq		$TEMP2,$ACC8,$ACC8
Packit Service 084de1
	vpmuludq	$Bi,$ACC9,$ACC9
Packit Service 084de1
	 vpbroadcastq	16($bp), $Bi
Packit Service 084de1
Packit Service 084de1
	mov	%rax,%rdx
Packit Service 084de1
	imulq	-128($np),%rax
Packit Service 084de1
	add	%rax,$r1
Packit Service 084de1
	 vmovdqu	-8+32*1-128($np),$TEMP0
Packit Service 084de1
	mov	%rdx,%rax
Packit Service 084de1
	imulq	8-128($np),%rax
Packit Service 084de1
	add	%rax,$r2
Packit Service 084de1
	 vmovdqu	-8+32*2-128($np),$TEMP1
Packit Service 084de1
	shr	\$29, $r1
Packit Service 084de1
	imulq	16-128($np),%rdx
Packit Service 084de1
	add	%rdx,$r3
Packit Service 084de1
	add	$r1, $r2
Packit Service 084de1
Packit Service 084de1
	vpmuludq	$Yi,$TEMP0,$TEMP0
Packit Service 084de1
	 vmovq		$Bi, %rbx
Packit Service 084de1
	vmovdqu		-8+32*3-128($np),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC1,$ACC1
Packit Service 084de1
	vpmuludq	$Yi,$TEMP1,$TEMP1
Packit Service 084de1
	vmovdqu		-8+32*4-128($np),$TEMP0
Packit Service 084de1
	vpaddq		$TEMP1,$ACC2,$ACC2
Packit Service 084de1
	vpmuludq	$Yi,$TEMP2,$TEMP2
Packit Service 084de1
	vmovdqu		-8+32*5-128($np),$TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC3,$ACC3
Packit Service 084de1
	vpmuludq	$Yi,$TEMP0,$TEMP0
Packit Service 084de1
	vmovdqu		-8+32*6-128($np),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC4,$ACC4
Packit Service 084de1
	vpmuludq	$Yi,$TEMP1,$TEMP1
Packit Service 084de1
	vmovdqu		-8+32*7-128($np),$TEMP0
Packit Service 084de1
	vpaddq		$TEMP1,$ACC5,$ACC5
Packit Service 084de1
	vpmuludq	$Yi,$TEMP2,$TEMP2
Packit Service 084de1
	vmovdqu		-8+32*8-128($np),$TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC6,$ACC6
Packit Service 084de1
	vpmuludq	$Yi,$TEMP0,$TEMP0
Packit Service 084de1
	vmovdqu		-8+32*9-128($np),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC7,$ACC7
Packit Service 084de1
	vpmuludq	$Yi,$TEMP1,$TEMP1
Packit Service 084de1
	vpaddq		$TEMP1,$ACC8,$ACC8
Packit Service 084de1
	vpmuludq	$Yi,$TEMP2,$TEMP2
Packit Service 084de1
	vpaddq		$TEMP2,$ACC9,$ACC9
Packit Service 084de1
Packit Service 084de1
	 vmovdqu	-16+32*1-128($ap),$TEMP0
Packit Service 084de1
	mov	%rbx,%rax
Packit Service 084de1
	imulq	-128($ap),%rax
Packit Service 084de1
	add	$r2,%rax
Packit Service 084de1
Packit Service 084de1
	 vmovdqu	-16+32*2-128($ap),$TEMP1
Packit Service 084de1
	mov	%rax,$r2
Packit Service 084de1
	imull	$n0, %eax
Packit Service 084de1
	and	\$0x1fffffff, %eax
Packit Service 084de1
Packit Service 084de1
	 imulq	8-128($ap),%rbx
Packit Service 084de1
	 add	%rbx,$r3
Packit Service 084de1
	vpmuludq	$Bi,$TEMP0,$TEMP0
Packit Service 084de1
	 vmovd		%eax, $Yi
Packit Service 084de1
	vmovdqu		-16+32*3-128($ap),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC1,$ACC1
Packit Service 084de1
	vpmuludq	$Bi,$TEMP1,$TEMP1
Packit Service 084de1
	 vpbroadcastq	$Yi, $Yi
Packit Service 084de1
	vmovdqu		-16+32*4-128($ap),$TEMP0
Packit Service 084de1
	vpaddq		$TEMP1,$ACC2,$ACC2
Packit Service 084de1
	vpmuludq	$Bi,$TEMP2,$TEMP2
Packit Service 084de1
	vmovdqu		-16+32*5-128($ap),$TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC3,$ACC3
Packit Service 084de1
	vpmuludq	$Bi,$TEMP0,$TEMP0
Packit Service 084de1
	vmovdqu		-16+32*6-128($ap),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC4,$ACC4
Packit Service 084de1
	vpmuludq	$Bi,$TEMP1,$TEMP1
Packit Service 084de1
	vmovdqu		-16+32*7-128($ap),$TEMP0
Packit Service 084de1
	vpaddq		$TEMP1,$ACC5,$ACC5
Packit Service 084de1
	vpmuludq	$Bi,$TEMP2,$TEMP2
Packit Service 084de1
	vmovdqu		-16+32*8-128($ap),$TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC6,$ACC6
Packit Service 084de1
	vpmuludq	$Bi,$TEMP0,$TEMP0
Packit Service 084de1
	vmovdqu		-16+32*9-128($ap),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC7,$ACC7
Packit Service 084de1
	vpmuludq	$Bi,$TEMP1,$TEMP1
Packit Service 084de1
	vpaddq		$TEMP1,$ACC8,$ACC8
Packit Service 084de1
	vpmuludq	$Bi,$TEMP2,$TEMP2
Packit Service 084de1
	 vpbroadcastq	24($bp), $Bi
Packit Service 084de1
	vpaddq		$TEMP2,$ACC9,$ACC9
Packit Service 084de1
Packit Service 084de1
	 vmovdqu	-16+32*1-128($np),$TEMP0
Packit Service 084de1
	mov	%rax,%rdx
Packit Service 084de1
	imulq	-128($np),%rax
Packit Service 084de1
	add	%rax,$r2
Packit Service 084de1
	 vmovdqu	-16+32*2-128($np),$TEMP1
Packit Service 084de1
	imulq	8-128($np),%rdx
Packit Service 084de1
	add	%rdx,$r3
Packit Service 084de1
	shr	\$29, $r2
Packit Service 084de1
Packit Service 084de1
	vpmuludq	$Yi,$TEMP0,$TEMP0
Packit Service 084de1
	 vmovq		$Bi, %rbx
Packit Service 084de1
	vmovdqu		-16+32*3-128($np),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC1,$ACC1
Packit Service 084de1
	vpmuludq	$Yi,$TEMP1,$TEMP1
Packit Service 084de1
	vmovdqu		-16+32*4-128($np),$TEMP0
Packit Service 084de1
	vpaddq		$TEMP1,$ACC2,$ACC2
Packit Service 084de1
	vpmuludq	$Yi,$TEMP2,$TEMP2
Packit Service 084de1
	vmovdqu		-16+32*5-128($np),$TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC3,$ACC3
Packit Service 084de1
	vpmuludq	$Yi,$TEMP0,$TEMP0
Packit Service 084de1
	vmovdqu		-16+32*6-128($np),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC4,$ACC4
Packit Service 084de1
	vpmuludq	$Yi,$TEMP1,$TEMP1
Packit Service 084de1
	vmovdqu		-16+32*7-128($np),$TEMP0
Packit Service 084de1
	vpaddq		$TEMP1,$ACC5,$ACC5
Packit Service 084de1
	vpmuludq	$Yi,$TEMP2,$TEMP2
Packit Service 084de1
	vmovdqu		-16+32*8-128($np),$TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC6,$ACC6
Packit Service 084de1
	vpmuludq	$Yi,$TEMP0,$TEMP0
Packit Service 084de1
	vmovdqu		-16+32*9-128($np),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC7,$ACC7
Packit Service 084de1
	vpmuludq	$Yi,$TEMP1,$TEMP1
Packit Service 084de1
	 vmovdqu	-24+32*1-128($ap),$TEMP0
Packit Service 084de1
	vpaddq		$TEMP1,$ACC8,$ACC8
Packit Service 084de1
	vpmuludq	$Yi,$TEMP2,$TEMP2
Packit Service 084de1
	 vmovdqu	-24+32*2-128($ap),$TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC9,$ACC9
Packit Service 084de1
Packit Service 084de1
	add	$r2, $r3
Packit Service 084de1
	imulq	-128($ap),%rbx
Packit Service 084de1
	add	%rbx,$r3
Packit Service 084de1
Packit Service 084de1
	mov	$r3, %rax
Packit Service 084de1
	imull	$n0, %eax
Packit Service 084de1
	and	\$0x1fffffff, %eax
Packit Service 084de1
Packit Service 084de1
	vpmuludq	$Bi,$TEMP0,$TEMP0
Packit Service 084de1
	 vmovd		%eax, $Yi
Packit Service 084de1
	vmovdqu		-24+32*3-128($ap),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC1,$ACC1
Packit Service 084de1
	vpmuludq	$Bi,$TEMP1,$TEMP1
Packit Service 084de1
	 vpbroadcastq	$Yi, $Yi
Packit Service 084de1
	vmovdqu		-24+32*4-128($ap),$TEMP0
Packit Service 084de1
	vpaddq		$TEMP1,$ACC2,$ACC2
Packit Service 084de1
	vpmuludq	$Bi,$TEMP2,$TEMP2
Packit Service 084de1
	vmovdqu		-24+32*5-128($ap),$TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC3,$ACC3
Packit Service 084de1
	vpmuludq	$Bi,$TEMP0,$TEMP0
Packit Service 084de1
	vmovdqu		-24+32*6-128($ap),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC4,$ACC4
Packit Service 084de1
	vpmuludq	$Bi,$TEMP1,$TEMP1
Packit Service 084de1
	vmovdqu		-24+32*7-128($ap),$TEMP0
Packit Service 084de1
	vpaddq		$TEMP1,$ACC5,$ACC5
Packit Service 084de1
	vpmuludq	$Bi,$TEMP2,$TEMP2
Packit Service 084de1
	vmovdqu		-24+32*8-128($ap),$TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC6,$ACC6
Packit Service 084de1
	vpmuludq	$Bi,$TEMP0,$TEMP0
Packit Service 084de1
	vmovdqu		-24+32*9-128($ap),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC7,$ACC7
Packit Service 084de1
	vpmuludq	$Bi,$TEMP1,$TEMP1
Packit Service 084de1
	vpaddq		$TEMP1,$ACC8,$ACC8
Packit Service 084de1
	vpmuludq	$Bi,$TEMP2,$TEMP2
Packit Service 084de1
	 vpbroadcastq	32($bp), $Bi
Packit Service 084de1
	vpaddq		$TEMP2,$ACC9,$ACC9
Packit Service 084de1
	 add		\$32, $bp			# $bp++
Packit Service 084de1
Packit Service 084de1
	vmovdqu		-24+32*1-128($np),$TEMP0
Packit Service 084de1
	imulq	-128($np),%rax
Packit Service 084de1
	add	%rax,$r3
Packit Service 084de1
	shr	\$29, $r3
Packit Service 084de1
Packit Service 084de1
	vmovdqu		-24+32*2-128($np),$TEMP1
Packit Service 084de1
	vpmuludq	$Yi,$TEMP0,$TEMP0
Packit Service 084de1
	 vmovq		$Bi, %rbx
Packit Service 084de1
	vmovdqu		-24+32*3-128($np),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC1,$ACC0		# $ACC0==$TEMP0
Packit Service 084de1
	vpmuludq	$Yi,$TEMP1,$TEMP1
Packit Service 084de1
	 vmovdqu	$ACC0, (%rsp)			# transfer $r0-$r3
Packit Service 084de1
	vpaddq		$TEMP1,$ACC2,$ACC1
Packit Service 084de1
	vmovdqu		-24+32*4-128($np),$TEMP0
Packit Service 084de1
	vpmuludq	$Yi,$TEMP2,$TEMP2
Packit Service 084de1
	vmovdqu		-24+32*5-128($np),$TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC3,$ACC2
Packit Service 084de1
	vpmuludq	$Yi,$TEMP0,$TEMP0
Packit Service 084de1
	vmovdqu		-24+32*6-128($np),$TEMP2
Packit Service 084de1
	vpaddq		$TEMP0,$ACC4,$ACC3
Packit Service 084de1
	vpmuludq	$Yi,$TEMP1,$TEMP1
Packit Service 084de1
	vmovdqu		-24+32*7-128($np),$TEMP0
Packit Service 084de1
	vpaddq		$TEMP1,$ACC5,$ACC4
Packit Service 084de1
	vpmuludq	$Yi,$TEMP2,$TEMP2
Packit Service 084de1
	vmovdqu		-24+32*8-128($np),$TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC6,$ACC5
Packit Service 084de1
	vpmuludq	$Yi,$TEMP0,$TEMP0
Packit Service 084de1
	vmovdqu		-24+32*9-128($np),$TEMP2
Packit Service 084de1
	 mov	$r3, $r0
Packit Service 084de1
	vpaddq		$TEMP0,$ACC7,$ACC6
Packit Service 084de1
	vpmuludq	$Yi,$TEMP1,$TEMP1
Packit Service 084de1
	 add	(%rsp), $r0
Packit Service 084de1
	vpaddq		$TEMP1,$ACC8,$ACC7
Packit Service 084de1
	vpmuludq	$Yi,$TEMP2,$TEMP2
Packit Service 084de1
	 vmovq	$r3, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP2,$ACC9,$ACC8
Packit Service 084de1
Packit Service 084de1
	dec	$i
Packit Service 084de1
	jnz	.Loop_mul_1024
Packit Service 084de1
___
Packit Service 084de1
Packit Service 084de1
# (*)	Original implementation was correcting ACC1-ACC3 for overflow
Packit Service 084de1
#	after 7 loop runs, or after 28 iterations, or 56 additions.
Packit Service 084de1
#	But as we underutilize resources, it's possible to correct in
Packit Service 084de1
#	each iteration with marginal performance loss. But then, as
Packit Service 084de1
#	we do it in each iteration, we can correct less digits, and
Packit Service 084de1
#	avoid performance penalties completely.
Packit Service 084de1
Packit Service 084de1
$TEMP0 = $ACC9;
Packit Service 084de1
$TEMP3 = $Bi;
Packit Service 084de1
$TEMP4 = $Yi;
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	vpaddq		(%rsp), $TEMP1, $ACC0
Packit Service 084de1
Packit Service 084de1
	vpsrlq		\$29, $ACC0, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC0, $ACC0
Packit Service 084de1
	vpsrlq		\$29, $ACC1, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC1, $ACC1
Packit Service 084de1
	vpsrlq		\$29, $ACC2, $TEMP3
Packit Service 084de1
	vpermq		\$0x93, $TEMP1, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC2, $ACC2
Packit Service 084de1
	vpsrlq		\$29, $ACC3, $TEMP4
Packit Service 084de1
	vpermq		\$0x93, $TEMP2, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC3, $ACC3
Packit Service 084de1
Packit Service 084de1
	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
Packit Service 084de1
	vpermq		\$0x93, $TEMP3, $TEMP3
Packit Service 084de1
	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
Packit Service 084de1
	vpermq		\$0x93, $TEMP4, $TEMP4
Packit Service 084de1
	vpaddq		$TEMP0, $ACC0, $ACC0
Packit Service 084de1
	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP1, $ACC1, $ACC1
Packit Service 084de1
	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
Packit Service 084de1
	vpaddq		$TEMP2, $ACC2, $ACC2
Packit Service 084de1
	vpblendd	\$3, $TEMP4, $ZERO, $TEMP4
Packit Service 084de1
	vpaddq		$TEMP3, $ACC3, $ACC3
Packit Service 084de1
	vpaddq		$TEMP4, $ACC4, $ACC4
Packit Service 084de1
Packit Service 084de1
	vpsrlq		\$29, $ACC0, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC0, $ACC0
Packit Service 084de1
	vpsrlq		\$29, $ACC1, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC1, $ACC1
Packit Service 084de1
	vpsrlq		\$29, $ACC2, $TEMP3
Packit Service 084de1
	vpermq		\$0x93, $TEMP1, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC2, $ACC2
Packit Service 084de1
	vpsrlq		\$29, $ACC3, $TEMP4
Packit Service 084de1
	vpermq		\$0x93, $TEMP2, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC3, $ACC3
Packit Service 084de1
	vpermq		\$0x93, $TEMP3, $TEMP3
Packit Service 084de1
Packit Service 084de1
	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
Packit Service 084de1
	vpermq		\$0x93, $TEMP4, $TEMP4
Packit Service 084de1
	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP0, $ACC0, $ACC0
Packit Service 084de1
	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP1, $ACC1, $ACC1
Packit Service 084de1
	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
Packit Service 084de1
	vpaddq		$TEMP2, $ACC2, $ACC2
Packit Service 084de1
	vpblendd	\$3, $TEMP4, $ZERO, $TEMP4
Packit Service 084de1
	vpaddq		$TEMP3, $ACC3, $ACC3
Packit Service 084de1
	vpaddq		$TEMP4, $ACC4, $ACC4
Packit Service 084de1
Packit Service 084de1
	vmovdqu		$ACC0, 0-128($rp)
Packit Service 084de1
	vmovdqu		$ACC1, 32-128($rp)
Packit Service 084de1
	vmovdqu		$ACC2, 64-128($rp)
Packit Service 084de1
	vmovdqu		$ACC3, 96-128($rp)
Packit Service 084de1
___
Packit Service 084de1
Packit Service 084de1
$TEMP5=$ACC0;
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	vpsrlq		\$29, $ACC4, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC4, $ACC4
Packit Service 084de1
	vpsrlq		\$29, $ACC5, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC5, $ACC5
Packit Service 084de1
	vpsrlq		\$29, $ACC6, $TEMP3
Packit Service 084de1
	vpermq		\$0x93, $TEMP1, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC6, $ACC6
Packit Service 084de1
	vpsrlq		\$29, $ACC7, $TEMP4
Packit Service 084de1
	vpermq		\$0x93, $TEMP2, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC7, $ACC7
Packit Service 084de1
	vpsrlq		\$29, $ACC8, $TEMP5
Packit Service 084de1
	vpermq		\$0x93, $TEMP3, $TEMP3
Packit Service 084de1
	vpand		$AND_MASK, $ACC8, $ACC8
Packit Service 084de1
	vpermq		\$0x93, $TEMP4, $TEMP4
Packit Service 084de1
Packit Service 084de1
	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
Packit Service 084de1
	vpermq		\$0x93, $TEMP5, $TEMP5
Packit Service 084de1
	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP0, $ACC4, $ACC4
Packit Service 084de1
	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP1, $ACC5, $ACC5
Packit Service 084de1
	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
Packit Service 084de1
	vpaddq		$TEMP2, $ACC6, $ACC6
Packit Service 084de1
	vpblendd	\$3, $TEMP4, $TEMP5, $TEMP4
Packit Service 084de1
	vpaddq		$TEMP3, $ACC7, $ACC7
Packit Service 084de1
	vpaddq		$TEMP4, $ACC8, $ACC8
Packit Service 084de1
Packit Service 084de1
	vpsrlq		\$29, $ACC4, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC4, $ACC4
Packit Service 084de1
	vpsrlq		\$29, $ACC5, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC5, $ACC5
Packit Service 084de1
	vpsrlq		\$29, $ACC6, $TEMP3
Packit Service 084de1
	vpermq		\$0x93, $TEMP1, $TEMP1
Packit Service 084de1
	vpand		$AND_MASK, $ACC6, $ACC6
Packit Service 084de1
	vpsrlq		\$29, $ACC7, $TEMP4
Packit Service 084de1
	vpermq		\$0x93, $TEMP2, $TEMP2
Packit Service 084de1
	vpand		$AND_MASK, $ACC7, $ACC7
Packit Service 084de1
	vpsrlq		\$29, $ACC8, $TEMP5
Packit Service 084de1
	vpermq		\$0x93, $TEMP3, $TEMP3
Packit Service 084de1
	vpand		$AND_MASK, $ACC8, $ACC8
Packit Service 084de1
	vpermq		\$0x93, $TEMP4, $TEMP4
Packit Service 084de1
Packit Service 084de1
	vpblendd	\$3, $ZERO, $TEMP1, $TEMP0
Packit Service 084de1
	vpermq		\$0x93, $TEMP5, $TEMP5
Packit Service 084de1
	vpblendd	\$3, $TEMP1, $TEMP2, $TEMP1
Packit Service 084de1
	vpaddq		$TEMP0, $ACC4, $ACC4
Packit Service 084de1
	vpblendd	\$3, $TEMP2, $TEMP3, $TEMP2
Packit Service 084de1
	vpaddq		$TEMP1, $ACC5, $ACC5
Packit Service 084de1
	vpblendd	\$3, $TEMP3, $TEMP4, $TEMP3
Packit Service 084de1
	vpaddq		$TEMP2, $ACC6, $ACC6
Packit Service 084de1
	vpblendd	\$3, $TEMP4, $TEMP5, $TEMP4
Packit Service 084de1
	vpaddq		$TEMP3, $ACC7, $ACC7
Packit Service 084de1
	vpaddq		$TEMP4, $ACC8, $ACC8
Packit Service 084de1
Packit Service 084de1
	vmovdqu		$ACC4, 128-128($rp)
Packit Service 084de1
	vmovdqu		$ACC5, 160-128($rp)
Packit Service 084de1
	vmovdqu		$ACC6, 192-128($rp)
Packit Service 084de1
	vmovdqu		$ACC7, 224-128($rp)
Packit Service 084de1
	vmovdqu		$ACC8, 256-128($rp)
Packit Service 084de1
	vzeroupper
Packit Service 084de1
Packit Service 084de1
	mov	%rbp, %rax
Packit Service 084de1
.cfi_def_cfa_register	%rax
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___ if ($win64);
Packit Service 084de1
.Lmul_1024_in_tail:
Packit Service 084de1
	movaps	-0xd8(%rax),%xmm6
Packit Service 084de1
	movaps	-0xc8(%rax),%xmm7
Packit Service 084de1
	movaps	-0xb8(%rax),%xmm8
Packit Service 084de1
	movaps	-0xa8(%rax),%xmm9
Packit Service 084de1
	movaps	-0x98(%rax),%xmm10
Packit Service 084de1
	movaps	-0x88(%rax),%xmm11
Packit Service 084de1
	movaps	-0x78(%rax),%xmm12
Packit Service 084de1
	movaps	-0x68(%rax),%xmm13
Packit Service 084de1
	movaps	-0x58(%rax),%xmm14
Packit Service 084de1
	movaps	-0x48(%rax),%xmm15
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	mov	-48(%rax),%r15
Packit Service 084de1
.cfi_restore	%r15
Packit Service 084de1
	mov	-40(%rax),%r14
Packit Service 084de1
.cfi_restore	%r14
Packit Service 084de1
	mov	-32(%rax),%r13
Packit Service 084de1
.cfi_restore	%r13
Packit Service 084de1
	mov	-24(%rax),%r12
Packit Service 084de1
.cfi_restore	%r12
Packit Service 084de1
	mov	-16(%rax),%rbp
Packit Service 084de1
.cfi_restore	%rbp
Packit Service 084de1
	mov	-8(%rax),%rbx
Packit Service 084de1
.cfi_restore	%rbx
Packit Service 084de1
	lea	(%rax),%rsp		# restore %rsp
Packit Service 084de1
.cfi_def_cfa_register	%rsp
Packit Service 084de1
.Lmul_1024_epilogue:
Packit Service 084de1
	ret
Packit Service 084de1
.cfi_endproc
Packit Service 084de1
.size	rsaz_1024_mul_avx2,.-rsaz_1024_mul_avx2
Packit Service 084de1
___
Packit Service 084de1
}
Packit Service 084de1
{
Packit Service 084de1
my ($out,$inp) = $win64 ? ("%rcx","%rdx") : ("%rdi","%rsi");
Packit Service 084de1
my @T = map("%r$_",(8..11));
Packit Service 084de1
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
.globl	rsaz_1024_red2norm_avx2
Packit Service 084de1
.type	rsaz_1024_red2norm_avx2,\@abi-omnipotent
Packit Service 084de1
.align	32
Packit Service 084de1
rsaz_1024_red2norm_avx2:
Packit Service 084de1
.cfi_startproc
Packit Service 084de1
	sub	\$-128,$inp	# size optimization
Packit Service 084de1
	xor	%rax,%rax
Packit Service 084de1
___
Packit Service 084de1
Packit Service 084de1
for ($j=0,$i=0; $i<16; $i++) {
Packit Service 084de1
    my $k=0;
Packit Service 084de1
    while (29*$j<64*($i+1)) {	# load data till boundary
Packit Service 084de1
	$code.="	mov	`8*$j-128`($inp), @T[0]\n";
Packit Service 084de1
	$j++; $k++; push(@T,shift(@T));
Packit Service 084de1
    }
Packit Service 084de1
    $l=$k;
Packit Service 084de1
    while ($k>1) {		# shift loaded data but last value
Packit Service 084de1
	$code.="	shl	\$`29*($j-$k)`,@T[-$k]\n";
Packit Service 084de1
	$k--;
Packit Service 084de1
    }
Packit Service 084de1
    $code.=<<___;		# shift last value
Packit Service 084de1
	mov	@T[-1], @T[0]
Packit Service 084de1
	shl	\$`29*($j-1)`, @T[-1]
Packit Service 084de1
	shr	\$`-29*($j-1)`, @T[0]
Packit Service 084de1
___
Packit Service 084de1
    while ($l) {		# accumulate all values
Packit Service 084de1
	$code.="	add	@T[-$l], %rax\n";
Packit Service 084de1
	$l--;
Packit Service 084de1
    }
Packit Service 084de1
	$code.=<<___;
Packit Service 084de1
	adc	\$0, @T[0]	# consume eventual carry
Packit Service 084de1
	mov	%rax, 8*$i($out)
Packit Service 084de1
	mov	@T[0], %rax
Packit Service 084de1
___
Packit Service 084de1
    push(@T,shift(@T));
Packit Service 084de1
}
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	ret
Packit Service 084de1
.cfi_endproc
Packit Service 084de1
.size	rsaz_1024_red2norm_avx2,.-rsaz_1024_red2norm_avx2
Packit Service 084de1
Packit Service 084de1
.globl	rsaz_1024_norm2red_avx2
Packit Service 084de1
.type	rsaz_1024_norm2red_avx2,\@abi-omnipotent
Packit Service 084de1
.align	32
Packit Service 084de1
rsaz_1024_norm2red_avx2:
Packit Service 084de1
.cfi_startproc
Packit Service 084de1
	sub	\$-128,$out	# size optimization
Packit Service 084de1
	mov	($inp),@T[0]
Packit Service 084de1
	mov	\$0x1fffffff,%eax
Packit Service 084de1
___
Packit Service 084de1
for ($j=0,$i=0; $i<16; $i++) {
Packit Service 084de1
    $code.="	mov	`8*($i+1)`($inp),@T[1]\n"	if ($i<15);
Packit Service 084de1
    $code.="	xor	@T[1],@T[1]\n"			if ($i==15);
Packit Service 084de1
    my $k=1;
Packit Service 084de1
    while (29*($j+1)<64*($i+1)) {
Packit Service 084de1
    	$code.=<<___;
Packit Service 084de1
	mov	@T[0],@T[-$k]
Packit Service 084de1
	shr	\$`29*$j`,@T[-$k]
Packit Service 084de1
	and	%rax,@T[-$k]				# &0x1fffffff
Packit Service 084de1
	mov	@T[-$k],`8*$j-128`($out)
Packit Service 084de1
___
Packit Service 084de1
	$j++; $k++;
Packit Service 084de1
    }
Packit Service 084de1
    $code.=<<___;
Packit Service 084de1
	shrd	\$`29*$j`,@T[1],@T[0]
Packit Service 084de1
	and	%rax,@T[0]
Packit Service 084de1
	mov	@T[0],`8*$j-128`($out)
Packit Service 084de1
___
Packit Service 084de1
    $j++;
Packit Service 084de1
    push(@T,shift(@T));
Packit Service 084de1
}
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	mov	@T[0],`8*$j-128`($out)			# zero
Packit Service 084de1
	mov	@T[0],`8*($j+1)-128`($out)
Packit Service 084de1
	mov	@T[0],`8*($j+2)-128`($out)
Packit Service 084de1
	mov	@T[0],`8*($j+3)-128`($out)
Packit Service 084de1
	ret
Packit Service 084de1
.cfi_endproc
Packit Service 084de1
.size	rsaz_1024_norm2red_avx2,.-rsaz_1024_norm2red_avx2
Packit Service 084de1
___
Packit Service 084de1
}
Packit Service 084de1
{
Packit Service 084de1
my ($out,$inp,$power) = $win64 ? ("%rcx","%rdx","%r8d") : ("%rdi","%rsi","%edx");
Packit Service 084de1
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
.globl	rsaz_1024_scatter5_avx2
Packit Service 084de1
.type	rsaz_1024_scatter5_avx2,\@abi-omnipotent
Packit Service 084de1
.align	32
Packit Service 084de1
rsaz_1024_scatter5_avx2:
Packit Service 084de1
.cfi_startproc
Packit Service 084de1
	vzeroupper
Packit Service 084de1
	vmovdqu	.Lscatter_permd(%rip),%ymm5
Packit Service 084de1
	shl	\$4,$power
Packit Service 084de1
	lea	($out,$power),$out
Packit Service 084de1
	mov	\$9,%eax
Packit Service 084de1
	jmp	.Loop_scatter_1024
Packit Service 084de1
Packit Service 084de1
.align	32
Packit Service 084de1
.Loop_scatter_1024:
Packit Service 084de1
	vmovdqu		($inp),%ymm0
Packit Service 084de1
	lea		32($inp),$inp
Packit Service 084de1
	vpermd		%ymm0,%ymm5,%ymm0
Packit Service 084de1
	vmovdqu		%xmm0,($out)
Packit Service 084de1
	lea		16*32($out),$out
Packit Service 084de1
	dec	%eax
Packit Service 084de1
	jnz	.Loop_scatter_1024
Packit Service 084de1
Packit Service 084de1
	vzeroupper
Packit Service 084de1
	ret
Packit Service 084de1
.cfi_endproc
Packit Service 084de1
.size	rsaz_1024_scatter5_avx2,.-rsaz_1024_scatter5_avx2
Packit Service 084de1
Packit Service 084de1
.globl	rsaz_1024_gather5_avx2
Packit Service 084de1
.type	rsaz_1024_gather5_avx2,\@abi-omnipotent
Packit Service 084de1
.align	32
Packit Service 084de1
rsaz_1024_gather5_avx2:
Packit Service 084de1
.cfi_startproc
Packit Service 084de1
	vzeroupper
Packit Service 084de1
	mov	%rsp,%r11
Packit Service 084de1
.cfi_def_cfa_register	%r11
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___ if ($win64);
Packit Service 084de1
	lea	-0x88(%rsp),%rax
Packit Service 084de1
.LSEH_begin_rsaz_1024_gather5:
Packit Service 084de1
	# I can't trust assembler to use specific encoding:-(
Packit Service 084de1
	.byte	0x48,0x8d,0x60,0xe0		# lea	-0x20(%rax),%rsp
Packit Service 084de1
	.byte	0xc5,0xf8,0x29,0x70,0xe0	# vmovaps %xmm6,-0x20(%rax)
Packit Service 084de1
	.byte	0xc5,0xf8,0x29,0x78,0xf0	# vmovaps %xmm7,-0x10(%rax)
Packit Service 084de1
	.byte	0xc5,0x78,0x29,0x40,0x00	# vmovaps %xmm8,0(%rax)
Packit Service 084de1
	.byte	0xc5,0x78,0x29,0x48,0x10	# vmovaps %xmm9,0x10(%rax)
Packit Service 084de1
	.byte	0xc5,0x78,0x29,0x50,0x20	# vmovaps %xmm10,0x20(%rax)
Packit Service 084de1
	.byte	0xc5,0x78,0x29,0x58,0x30	# vmovaps %xmm11,0x30(%rax)
Packit Service 084de1
	.byte	0xc5,0x78,0x29,0x60,0x40	# vmovaps %xmm12,0x40(%rax)
Packit Service 084de1
	.byte	0xc5,0x78,0x29,0x68,0x50	# vmovaps %xmm13,0x50(%rax)
Packit Service 084de1
	.byte	0xc5,0x78,0x29,0x70,0x60	# vmovaps %xmm14,0x60(%rax)
Packit Service 084de1
	.byte	0xc5,0x78,0x29,0x78,0x70	# vmovaps %xmm15,0x70(%rax)
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	lea	-0x100(%rsp),%rsp
Packit Service 084de1
	and	\$-32, %rsp
Packit Service 084de1
	lea	.Linc(%rip), %r10
Packit Service 084de1
	lea	-128(%rsp),%rax			# control u-op density
Packit Service 084de1
Packit Service 084de1
	vmovd		$power, %xmm4
Packit Service 084de1
	vmovdqa		(%r10),%ymm0
Packit Service 084de1
	vmovdqa		32(%r10),%ymm1
Packit Service 084de1
	vmovdqa		64(%r10),%ymm5
Packit Service 084de1
	vpbroadcastd	%xmm4,%ymm4
Packit Service 084de1
Packit Service 084de1
	vpaddd		%ymm5, %ymm0, %ymm2
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm0, %ymm0
Packit Service 084de1
	vpaddd		%ymm5, %ymm1, %ymm3
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm1, %ymm1
Packit Service 084de1
	vmovdqa		%ymm0, 32*0+128(%rax)
Packit Service 084de1
	vpaddd		%ymm5, %ymm2, %ymm0
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm2, %ymm2
Packit Service 084de1
	vmovdqa		%ymm1, 32*1+128(%rax)
Packit Service 084de1
	vpaddd		%ymm5, %ymm3, %ymm1
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm3, %ymm3
Packit Service 084de1
	vmovdqa		%ymm2, 32*2+128(%rax)
Packit Service 084de1
	vpaddd		%ymm5, %ymm0, %ymm2
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm0, %ymm0
Packit Service 084de1
	vmovdqa		%ymm3, 32*3+128(%rax)
Packit Service 084de1
	vpaddd		%ymm5, %ymm1, %ymm3
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm1, %ymm1
Packit Service 084de1
	vmovdqa		%ymm0, 32*4+128(%rax)
Packit Service 084de1
	vpaddd		%ymm5, %ymm2, %ymm8
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm2, %ymm2
Packit Service 084de1
	vmovdqa		%ymm1, 32*5+128(%rax)
Packit Service 084de1
	vpaddd		%ymm5, %ymm3, %ymm9
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm3, %ymm3
Packit Service 084de1
	vmovdqa		%ymm2, 32*6+128(%rax)
Packit Service 084de1
	vpaddd		%ymm5, %ymm8, %ymm10
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm8, %ymm8
Packit Service 084de1
	vmovdqa		%ymm3, 32*7+128(%rax)
Packit Service 084de1
	vpaddd		%ymm5, %ymm9, %ymm11
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm9, %ymm9
Packit Service 084de1
	vpaddd		%ymm5, %ymm10, %ymm12
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm10, %ymm10
Packit Service 084de1
	vpaddd		%ymm5, %ymm11, %ymm13
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm11, %ymm11
Packit Service 084de1
	vpaddd		%ymm5, %ymm12, %ymm14
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm12, %ymm12
Packit Service 084de1
	vpaddd		%ymm5, %ymm13, %ymm15
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm13, %ymm13
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm14, %ymm14
Packit Service 084de1
	vpcmpeqd	%ymm4, %ymm15, %ymm15
Packit Service 084de1
Packit Service 084de1
	vmovdqa	-32(%r10),%ymm7			# .Lgather_permd
Packit Service 084de1
	lea	128($inp), $inp
Packit Service 084de1
	mov	\$9,$power
Packit Service 084de1
Packit Service 084de1
.Loop_gather_1024:
Packit Service 084de1
	vmovdqa		32*0-128($inp),	%ymm0
Packit Service 084de1
	vmovdqa		32*1-128($inp),	%ymm1
Packit Service 084de1
	vmovdqa		32*2-128($inp),	%ymm2
Packit Service 084de1
	vmovdqa		32*3-128($inp),	%ymm3
Packit Service 084de1
	vpand		32*0+128(%rax),	%ymm0,	%ymm0
Packit Service 084de1
	vpand		32*1+128(%rax),	%ymm1,	%ymm1
Packit Service 084de1
	vpand		32*2+128(%rax),	%ymm2,	%ymm2
Packit Service 084de1
	vpor		%ymm0, %ymm1, %ymm4
Packit Service 084de1
	vpand		32*3+128(%rax),	%ymm3,	%ymm3
Packit Service 084de1
	vmovdqa		32*4-128($inp),	%ymm0
Packit Service 084de1
	vmovdqa		32*5-128($inp),	%ymm1
Packit Service 084de1
	vpor		%ymm2, %ymm3, %ymm5
Packit Service 084de1
	vmovdqa		32*6-128($inp),	%ymm2
Packit Service 084de1
	vmovdqa		32*7-128($inp),	%ymm3
Packit Service 084de1
	vpand		32*4+128(%rax),	%ymm0,	%ymm0
Packit Service 084de1
	vpand		32*5+128(%rax),	%ymm1,	%ymm1
Packit Service 084de1
	vpand		32*6+128(%rax),	%ymm2,	%ymm2
Packit Service 084de1
	vpor		%ymm0, %ymm4, %ymm4
Packit Service 084de1
	vpand		32*7+128(%rax),	%ymm3,	%ymm3
Packit Service 084de1
	vpand		32*8-128($inp),	%ymm8,	%ymm0
Packit Service 084de1
	vpor		%ymm1, %ymm5, %ymm5
Packit Service 084de1
	vpand		32*9-128($inp),	%ymm9,	%ymm1
Packit Service 084de1
	vpor		%ymm2, %ymm4, %ymm4
Packit Service 084de1
	vpand		32*10-128($inp),%ymm10,	%ymm2
Packit Service 084de1
	vpor		%ymm3, %ymm5, %ymm5
Packit Service 084de1
	vpand		32*11-128($inp),%ymm11,	%ymm3
Packit Service 084de1
	vpor		%ymm0, %ymm4, %ymm4
Packit Service 084de1
	vpand		32*12-128($inp),%ymm12,	%ymm0
Packit Service 084de1
	vpor		%ymm1, %ymm5, %ymm5
Packit Service 084de1
	vpand		32*13-128($inp),%ymm13,	%ymm1
Packit Service 084de1
	vpor		%ymm2, %ymm4, %ymm4
Packit Service 084de1
	vpand		32*14-128($inp),%ymm14,	%ymm2
Packit Service 084de1
	vpor		%ymm3, %ymm5, %ymm5
Packit Service 084de1
	vpand		32*15-128($inp),%ymm15,	%ymm3
Packit Service 084de1
	lea		32*16($inp), $inp
Packit Service 084de1
	vpor		%ymm0, %ymm4, %ymm4
Packit Service 084de1
	vpor		%ymm1, %ymm5, %ymm5
Packit Service 084de1
	vpor		%ymm2, %ymm4, %ymm4
Packit Service 084de1
	vpor		%ymm3, %ymm5, %ymm5
Packit Service 084de1
Packit Service 084de1
	vpor		%ymm5, %ymm4, %ymm4
Packit Service 084de1
	vextracti128	\$1, %ymm4, %xmm5	# upper half is cleared
Packit Service 084de1
	vpor		%xmm4, %xmm5, %xmm5
Packit Service 084de1
	vpermd		%ymm5,%ymm7,%ymm5
Packit Service 084de1
	vmovdqu		%ymm5,($out)
Packit Service 084de1
	lea		32($out),$out
Packit Service 084de1
	dec	$power
Packit Service 084de1
	jnz	.Loop_gather_1024
Packit Service 084de1
Packit Service 084de1
	vpxor	%ymm0,%ymm0,%ymm0
Packit Service 084de1
	vmovdqu	%ymm0,($out)
Packit Service 084de1
	vzeroupper
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___ if ($win64);
Packit Service 084de1
	movaps	-0xa8(%r11),%xmm6
Packit Service 084de1
	movaps	-0x98(%r11),%xmm7
Packit Service 084de1
	movaps	-0x88(%r11),%xmm8
Packit Service 084de1
	movaps	-0x78(%r11),%xmm9
Packit Service 084de1
	movaps	-0x68(%r11),%xmm10
Packit Service 084de1
	movaps	-0x58(%r11),%xmm11
Packit Service 084de1
	movaps	-0x48(%r11),%xmm12
Packit Service 084de1
	movaps	-0x38(%r11),%xmm13
Packit Service 084de1
	movaps	-0x28(%r11),%xmm14
Packit Service 084de1
	movaps	-0x18(%r11),%xmm15
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	lea	(%r11),%rsp
Packit Service 084de1
.cfi_def_cfa_register	%rsp
Packit Service 084de1
	ret
Packit Service 084de1
.cfi_endproc
Packit Service 084de1
.LSEH_end_rsaz_1024_gather5:
Packit Service 084de1
.size	rsaz_1024_gather5_avx2,.-rsaz_1024_gather5_avx2
Packit Service 084de1
___
Packit Service 084de1
}
Packit Service 084de1
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
.extern	OPENSSL_ia32cap_P
Packit Service 084de1
.globl	rsaz_avx2_eligible
Packit Service 084de1
.type	rsaz_avx2_eligible,\@abi-omnipotent
Packit Service 084de1
.align	32
Packit Service 084de1
rsaz_avx2_eligible:
Packit Service 084de1
	mov	OPENSSL_ia32cap_P+8(%rip),%eax
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___	if ($addx);
Packit Service 084de1
	mov	\$`1<<8|1<<19`,%ecx
Packit Service 084de1
	mov	\$0,%edx
Packit Service 084de1
	and	%eax,%ecx
Packit Service 084de1
	cmp	\$`1<<8|1<<19`,%ecx	# check for BMI2+AD*X
Packit Service 084de1
	cmove	%edx,%eax
Packit Service 084de1
___
Packit Service 084de1
$code.=<<___;
Packit Service 084de1
	and	\$`1<<5`,%eax
Packit Service 084de1
	shr	\$5,%eax
Packit Service 084de1
	ret
Packit Service 084de1
.size	rsaz_avx2_eligible,.-rsaz_avx2_eligible
Packit Service 084de1
Packit Service 084de1
.align	64
Packit Service 084de1
.Land_mask:
Packit Service 084de1
	.quad	0x1fffffff,0x1fffffff,0x1fffffff,0x1fffffff
Packit Service 084de1
.Lscatter_permd:
Packit Service 084de1
	.long	0,2,4,6,7,7,7,7
Packit Service 084de1
.Lgather_permd:
Packit Service 084de1
	.long	0,7,1,7,2,7,3,7
Packit Service 084de1
.Linc:
Packit Service 084de1
	.long	0,0,0,0, 1,1,1,1
Packit Service 084de1
	.long	2,2,2,2, 3,3,3,3
Packit Service 084de1
	.long	4,4,4,4, 4,4,4,4
Packit Service 084de1
.align	64
Packit Service 084de1
___
Packit Service 084de1
Packit Service 084de1
if ($win64) {
Packit Service 084de1
$rec="%rcx";
Packit Service 084de1
$frame="%rdx";
Packit Service 084de1
$context="%r8";
Packit Service 084de1
$disp="%r9";
Packit Service 084de1
Packit Service 084de1
$code.=<<___
Packit Service 084de1
.extern	__imp_RtlVirtualUnwind
Packit Service 084de1
.type	rsaz_se_handler,\@abi-omnipotent
Packit Service 084de1
.align	16
Packit Service 084de1
rsaz_se_handler:
Packit Service 084de1
	push	%rsi
Packit Service 084de1
	push	%rdi
Packit Service 084de1
	push	%rbx
Packit Service 084de1
	push	%rbp
Packit Service 084de1
	push	%r12
Packit Service 084de1
	push	%r13
Packit Service 084de1
	push	%r14
Packit Service 084de1
	push	%r15
Packit Service 084de1
	pushfq
Packit Service 084de1
	sub	\$64,%rsp
Packit Service 084de1
Packit Service 084de1
	mov	120($context),%rax	# pull context->Rax
Packit Service 084de1
	mov	248($context),%rbx	# pull context->Rip
Packit Service 084de1
Packit Service 084de1
	mov	8($disp),%rsi		# disp->ImageBase
Packit Service 084de1
	mov	56($disp),%r11		# disp->HandlerData
Packit Service 084de1
Packit Service 084de1
	mov	0(%r11),%r10d		# HandlerData[0]
Packit Service 084de1
	lea	(%rsi,%r10),%r10	# prologue label
Packit Service 084de1
	cmp	%r10,%rbx		# context->Rip
Packit Service 084de1
	jb	.Lcommon_seh_tail
Packit Service 084de1
Packit Service 084de1
	mov	4(%r11),%r10d		# HandlerData[1]
Packit Service 084de1
	lea	(%rsi,%r10),%r10	# epilogue label
Packit Service 084de1
	cmp	%r10,%rbx		# context->Rip>=epilogue label
Packit Service 084de1
	jae	.Lcommon_seh_tail
Packit Service 084de1
Packit Service 084de1
	mov	160($context),%rbp	# pull context->Rbp
Packit Service 084de1
Packit Service 084de1
	mov	8(%r11),%r10d		# HandlerData[2]
Packit Service 084de1
	lea	(%rsi,%r10),%r10	# "in tail" label
Packit Service 084de1
	cmp	%r10,%rbx		# context->Rip>="in tail" label
Packit Service 084de1
	cmovc	%rbp,%rax
Packit Service 084de1
Packit Service 084de1
	mov	-48(%rax),%r15
Packit Service 084de1
	mov	-40(%rax),%r14
Packit Service 084de1
	mov	-32(%rax),%r13
Packit Service 084de1
	mov	-24(%rax),%r12
Packit Service 084de1
	mov	-16(%rax),%rbp
Packit Service 084de1
	mov	-8(%rax),%rbx
Packit Service 084de1
	mov	%r15,240($context)
Packit Service 084de1
	mov	%r14,232($context)
Packit Service 084de1
	mov	%r13,224($context)
Packit Service 084de1
	mov	%r12,216($context)
Packit Service 084de1
	mov	%rbp,160($context)
Packit Service 084de1
	mov	%rbx,144($context)
Packit Service 084de1
Packit Service 084de1
	lea	-0xd8(%rax),%rsi	# %xmm save area
Packit Service 084de1
	lea	512($context),%rdi	# & context.Xmm6
Packit Service 084de1
	mov	\$20,%ecx		# 10*sizeof(%xmm0)/sizeof(%rax)
Packit Service 084de1
	.long	0xa548f3fc		# cld; rep movsq
Packit Service 084de1
Packit Service 084de1
.Lcommon_seh_tail:
Packit Service 084de1
	mov	8(%rax),%rdi
Packit Service 084de1
	mov	16(%rax),%rsi
Packit Service 084de1
	mov	%rax,152($context)	# restore context->Rsp
Packit Service 084de1
	mov	%rsi,168($context)	# restore context->Rsi
Packit Service 084de1
	mov	%rdi,176($context)	# restore context->Rdi
Packit Service 084de1
Packit Service 084de1
	mov	40($disp),%rdi		# disp->ContextRecord
Packit Service 084de1
	mov	$context,%rsi		# context
Packit Service 084de1
	mov	\$154,%ecx		# sizeof(CONTEXT)
Packit Service 084de1
	.long	0xa548f3fc		# cld; rep movsq
Packit Service 084de1
Packit Service 084de1
	mov	$disp,%rsi
Packit Service 084de1
	xor	%rcx,%rcx		# arg1, UNW_FLAG_NHANDLER
Packit Service 084de1
	mov	8(%rsi),%rdx		# arg2, disp->ImageBase
Packit Service 084de1
	mov	0(%rsi),%r8		# arg3, disp->ControlPc
Packit Service 084de1
	mov	16(%rsi),%r9		# arg4, disp->FunctionEntry
Packit Service 084de1
	mov	40(%rsi),%r10		# disp->ContextRecord
Packit Service 084de1
	lea	56(%rsi),%r11		# &disp->HandlerData
Packit Service 084de1
	lea	24(%rsi),%r12		# &disp->EstablisherFrame
Packit Service 084de1
	mov	%r10,32(%rsp)		# arg5
Packit Service 084de1
	mov	%r11,40(%rsp)		# arg6
Packit Service 084de1
	mov	%r12,48(%rsp)		# arg7
Packit Service 084de1
	mov	%rcx,56(%rsp)		# arg8, (NULL)
Packit Service 084de1
	call	*__imp_RtlVirtualUnwind(%rip)
Packit Service 084de1
Packit Service 084de1
	mov	\$1,%eax		# ExceptionContinueSearch
Packit Service 084de1
	add	\$64,%rsp
Packit Service 084de1
	popfq
Packit Service 084de1
	pop	%r15
Packit Service 084de1
	pop	%r14
Packit Service 084de1
	pop	%r13
Packit Service 084de1
	pop	%r12
Packit Service 084de1
	pop	%rbp
Packit Service 084de1
	pop	%rbx
Packit Service 084de1
	pop	%rdi
Packit Service 084de1
	pop	%rsi
Packit Service 084de1
	ret
Packit Service 084de1
.size	rsaz_se_handler,.-rsaz_se_handler
Packit Service 084de1
Packit Service 084de1
.section	.pdata
Packit Service 084de1
.align	4
Packit Service 084de1
	.rva	.LSEH_begin_rsaz_1024_sqr_avx2
Packit Service 084de1
	.rva	.LSEH_end_rsaz_1024_sqr_avx2
Packit Service 084de1
	.rva	.LSEH_info_rsaz_1024_sqr_avx2
Packit Service 084de1
Packit Service 084de1
	.rva	.LSEH_begin_rsaz_1024_mul_avx2
Packit Service 084de1
	.rva	.LSEH_end_rsaz_1024_mul_avx2
Packit Service 084de1
	.rva	.LSEH_info_rsaz_1024_mul_avx2
Packit Service 084de1
Packit Service 084de1
	.rva	.LSEH_begin_rsaz_1024_gather5
Packit Service 084de1
	.rva	.LSEH_end_rsaz_1024_gather5
Packit Service 084de1
	.rva	.LSEH_info_rsaz_1024_gather5
Packit Service 084de1
.section	.xdata
Packit Service 084de1
.align	8
Packit Service 084de1
.LSEH_info_rsaz_1024_sqr_avx2:
Packit Service 084de1
	.byte	9,0,0,0
Packit Service 084de1
	.rva	rsaz_se_handler
Packit Service 084de1
	.rva	.Lsqr_1024_body,.Lsqr_1024_epilogue,.Lsqr_1024_in_tail
Packit Service 084de1
	.long	0
Packit Service 084de1
.LSEH_info_rsaz_1024_mul_avx2:
Packit Service 084de1
	.byte	9,0,0,0
Packit Service 084de1
	.rva	rsaz_se_handler
Packit Service 084de1
	.rva	.Lmul_1024_body,.Lmul_1024_epilogue,.Lmul_1024_in_tail
Packit Service 084de1
	.long	0
Packit Service 084de1
.LSEH_info_rsaz_1024_gather5:
Packit Service 084de1
	.byte	0x01,0x36,0x17,0x0b
Packit Service 084de1
	.byte	0x36,0xf8,0x09,0x00	# vmovaps 0x90(rsp),xmm15
Packit Service 084de1
	.byte	0x31,0xe8,0x08,0x00	# vmovaps 0x80(rsp),xmm14
Packit Service 084de1
	.byte	0x2c,0xd8,0x07,0x00	# vmovaps 0x70(rsp),xmm13
Packit Service 084de1
	.byte	0x27,0xc8,0x06,0x00	# vmovaps 0x60(rsp),xmm12
Packit Service 084de1
	.byte	0x22,0xb8,0x05,0x00	# vmovaps 0x50(rsp),xmm11
Packit Service 084de1
	.byte	0x1d,0xa8,0x04,0x00	# vmovaps 0x40(rsp),xmm10
Packit Service 084de1
	.byte	0x18,0x98,0x03,0x00	# vmovaps 0x30(rsp),xmm9
Packit Service 084de1
	.byte	0x13,0x88,0x02,0x00	# vmovaps 0x20(rsp),xmm8
Packit Service 084de1
	.byte	0x0e,0x78,0x01,0x00	# vmovaps 0x10(rsp),xmm7
Packit Service 084de1
	.byte	0x09,0x68,0x00,0x00	# vmovaps 0x00(rsp),xmm6
Packit Service 084de1
	.byte	0x04,0x01,0x15,0x00	# sub	  rsp,0xa8
Packit Service 084de1
	.byte	0x00,0xb3,0x00,0x00	# set_frame r11
Packit Service 084de1
___
Packit Service 084de1
}
Packit Service 084de1
Packit Service 084de1
foreach (split("\n",$code)) {
Packit Service 084de1
	s/\`([^\`]*)\`/eval($1)/ge;
Packit Service 084de1
Packit Service 084de1
	s/\b(sh[rl]d?\s+\$)(-?[0-9]+)/$1.$2%64/ge		or
Packit Service 084de1
Packit Service 084de1
	s/\b(vmov[dq])\b(.+)%ymm([0-9]+)/$1$2%xmm$3/go		or
Packit Service 084de1
	s/\b(vmovdqu)\b(.+)%x%ymm([0-9]+)/$1$2%xmm$3/go		or
Packit Service 084de1
	s/\b(vpinsr[qd])\b(.+)%ymm([0-9]+)/$1$2%xmm$3/go	or
Packit Service 084de1
	s/\b(vpextr[qd])\b(.+)%ymm([0-9]+)/$1$2%xmm$3/go	or
Packit Service 084de1
	s/\b(vpbroadcast[qd]\s+)%ymm([0-9]+)/$1%xmm$2/go;
Packit Service 084de1
	print $_,"\n";
Packit Service 084de1
}
Packit Service 084de1
Packit Service 084de1
}}} else {{{
Packit Service 084de1
print <<___;	# assembler is too old
Packit Service 084de1
.text
Packit Service 084de1
Packit Service 084de1
.globl	rsaz_avx2_eligible
Packit Service 084de1
.type	rsaz_avx2_eligible,\@abi-omnipotent
Packit Service 084de1
rsaz_avx2_eligible:
Packit Service 084de1
	xor	%eax,%eax
Packit Service 084de1
	ret
Packit Service 084de1
.size	rsaz_avx2_eligible,.-rsaz_avx2_eligible
Packit Service 084de1
Packit Service 084de1
.globl	rsaz_1024_sqr_avx2
Packit Service 084de1
.globl	rsaz_1024_mul_avx2
Packit Service 084de1
.globl	rsaz_1024_norm2red_avx2
Packit Service 084de1
.globl	rsaz_1024_red2norm_avx2
Packit Service 084de1
.globl	rsaz_1024_scatter5_avx2
Packit Service 084de1
.globl	rsaz_1024_gather5_avx2
Packit Service 084de1
.type	rsaz_1024_sqr_avx2,\@abi-omnipotent
Packit Service 084de1
rsaz_1024_sqr_avx2:
Packit Service 084de1
rsaz_1024_mul_avx2:
Packit Service 084de1
rsaz_1024_norm2red_avx2:
Packit Service 084de1
rsaz_1024_red2norm_avx2:
Packit Service 084de1
rsaz_1024_scatter5_avx2:
Packit Service 084de1
rsaz_1024_gather5_avx2:
Packit Service 084de1
	.byte	0x0f,0x0b	# ud2
Packit Service 084de1
	ret
Packit Service 084de1
.size	rsaz_1024_sqr_avx2,.-rsaz_1024_sqr_avx2
Packit Service 084de1
___
Packit Service 084de1
}}}
Packit Service 084de1
Packit Service 084de1
close STDOUT or die "error closing STDOUT: $!";