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

Packit c4476c
#! /usr/bin/env perl
Packit c4476c
# Copyright 2013-2020 The OpenSSL Project Authors. All Rights Reserved.
Packit c4476c
# Copyright (c) 2012, Intel Corporation. All Rights Reserved.
Packit c4476c
#
Packit c4476c
# Licensed under the OpenSSL license (the "License").  You may not use
Packit c4476c
# this file except in compliance with the License.  You can obtain a copy
Packit c4476c
# in the file LICENSE in the source distribution or at
Packit c4476c
# https://www.openssl.org/source/license.html
Packit c4476c
#
Packit c4476c
# Originally written by Shay Gueron (1, 2), and Vlad Krasnov (1)
Packit c4476c
# (1) Intel Corporation, Israel Development Center, Haifa, Israel
Packit c4476c
# (2) University of Haifa, Israel
Packit c4476c
#
Packit c4476c
# References:
Packit c4476c
# [1] S. Gueron, "Efficient Software Implementations of Modular
Packit c4476c
#     Exponentiation", http://eprint.iacr.org/2011/239
Packit c4476c
# [2] S. Gueron, V. Krasnov. "Speeding up Big-Numbers Squaring".
Packit c4476c
#     IEEE Proceedings of 9th International Conference on Information
Packit c4476c
#     Technology: New Generations (ITNG 2012), 821-823 (2012).
Packit c4476c
# [3] S. Gueron, Efficient Software Implementations of Modular Exponentiation
Packit c4476c
#     Journal of Cryptographic Engineering 2:31-43 (2012).
Packit c4476c
# [4] S. Gueron, V. Krasnov: "[PATCH] Efficient and side channel analysis
Packit c4476c
#     resistant 512-bit and 1024-bit modular exponentiation for optimizing
Packit c4476c
#     RSA1024 and RSA2048 on x86_64 platforms",
Packit c4476c
#     http://rt.openssl.org/Ticket/Display.html?id=2582&user=guest&pass=guest
Packit c4476c
#
Packit c4476c
# While original submission covers 512- and 1024-bit exponentiation,
Packit c4476c
# this module is limited to 512-bit version only (and as such
Packit c4476c
# accelerates RSA1024 sign). This is because improvement for longer
Packit c4476c
# keys is not high enough to justify the effort, highest measured
Packit c4476c
# was ~5% on Westmere. [This is relative to OpenSSL 1.0.2, upcoming
Packit c4476c
# for the moment of this writing!] Nor does this module implement
Packit c4476c
# "monolithic" complete exponentiation jumbo-subroutine, but adheres
Packit c4476c
# to more modular mixture of C and assembly. And it's optimized even
Packit c4476c
# for processors other than Intel Core family (see table below for
Packit c4476c
# improvement coefficients).
Packit c4476c
# 						<appro@openssl.org>
Packit c4476c
#
Packit c4476c
# RSA1024 sign/sec	this/original	|this/rsax(*)	this/fips(*)
Packit c4476c
#			----------------+---------------------------
Packit c4476c
# Opteron		+13%		|+5%		+20%
Packit c4476c
# Bulldozer		-0%		|-1%		+10%
Packit c4476c
# P4			+11%		|+7%		+8%
Packit c4476c
# Westmere		+5%		|+14%		+17%
Packit c4476c
# Sandy Bridge		+2%		|+12%		+29%
Packit c4476c
# Ivy Bridge		+1%		|+11%		+35%
Packit c4476c
# Haswell(**)		-0%		|+12%		+39%
Packit c4476c
# Atom			+13%		|+11%		+4%
Packit c4476c
# VIA Nano		+70%		|+9%		+25%
Packit c4476c
#
Packit c4476c
# (*)	rsax engine and fips numbers are presented for reference
Packit c4476c
#	purposes;
Packit c4476c
# (**)	MULX was attempted, but found to give only marginal improvement;
Packit c4476c
Packit c4476c
$flavour = shift;
Packit c4476c
$output  = shift;
Packit c4476c
if ($flavour =~ /\./) { $output = $flavour; undef $flavour; }
Packit c4476c
Packit c4476c
$win64=0; $win64=1 if ($flavour =~ /[nm]asm|mingw64/ || $output =~ /\.asm$/);
Packit c4476c
Packit c4476c
$0 =~ m/(.*[\/\\])[^\/\\]+$/; $dir=$1;
Packit c4476c
( $xlate="${dir}x86_64-xlate.pl" and -f $xlate ) or
Packit c4476c
( $xlate="${dir}../../perlasm/x86_64-xlate.pl" and -f $xlate) or
Packit c4476c
die "can't locate x86_64-xlate.pl";
Packit c4476c
Packit c4476c
open OUT,"| \"$^X\" \"$xlate\" $flavour \"$output\"";
Packit c4476c
*STDOUT=*OUT;
Packit c4476c
Packit c4476c
if (`$ENV{CC} -Wa,-v -c -o /dev/null -x assembler /dev/null 2>&1`
Packit c4476c
		=~ /GNU assembler version ([2-9]\.[0-9]+)/) {
Packit c4476c
	$addx = ($1>=2.23);
Packit c4476c
}
Packit c4476c
Packit c4476c
if (!$addx && $win64 && ($flavour =~ /nasm/ || $ENV{ASM} =~ /nasm/) &&
Packit c4476c
	    `nasm -v 2>&1` =~ /NASM version ([2-9]\.[0-9]+)/) {
Packit c4476c
	$addx = ($1>=2.10);
Packit c4476c
}
Packit c4476c
Packit c4476c
if (!$addx && $win64 && ($flavour =~ /masm/ || $ENV{ASM} =~ /ml64/) &&
Packit c4476c
	    `ml64 2>&1` =~ /Version ([0-9]+)\./) {
Packit c4476c
	$addx = ($1>=12);
Packit c4476c
}
Packit c4476c
Packit c4476c
if (!$addx && `$ENV{CC} -v 2>&1` =~ /((?:^clang|LLVM) version|.*based on LLVM) ([0-9]+)\.([0-9]+)/) {
Packit c4476c
	my $ver = $2 + $3/100.0;	# 3.1->3.01, 3.10->3.10
Packit c4476c
	$addx = ($ver>=3.03);
Packit c4476c
}
Packit c4476c
Packit c4476c
($out, $inp, $mod) = ("%rdi", "%rsi", "%rbp");	# common internal API
Packit c4476c
{
Packit c4476c
my ($out,$inp,$mod,$n0,$times) = ("%rdi","%rsi","%rdx","%rcx","%r8d");
Packit c4476c
Packit c4476c
$code.=<<___;
Packit c4476c
.text
Packit c4476c
Packit c4476c
.extern	OPENSSL_ia32cap_P
Packit c4476c
Packit c4476c
.globl	rsaz_512_sqr
Packit c4476c
.type	rsaz_512_sqr,\@function,5
Packit c4476c
.align	32
Packit c4476c
rsaz_512_sqr:				# 25-29% faster than rsaz_512_mul
Packit c4476c
.cfi_startproc
Packit c4476c
	push	%rbx
Packit c4476c
.cfi_push	%rbx
Packit c4476c
	push	%rbp
Packit c4476c
.cfi_push	%rbp
Packit c4476c
	push	%r12
Packit c4476c
.cfi_push	%r12
Packit c4476c
	push	%r13
Packit c4476c
.cfi_push	%r13
Packit c4476c
	push	%r14
Packit c4476c
.cfi_push	%r14
Packit c4476c
	push	%r15
Packit c4476c
.cfi_push	%r15
Packit c4476c
Packit c4476c
	subq	\$128+24, %rsp
Packit c4476c
.cfi_adjust_cfa_offset	128+24
Packit c4476c
.Lsqr_body:
Packit c4476c
	movq	$mod, %xmm1		# common off-load
Packit c4476c
	movq	($inp), %rdx
Packit c4476c
	movq	8($inp), %rax
Packit c4476c
	movq	$n0, 128(%rsp)
Packit c4476c
___
Packit c4476c
$code.=<<___ if ($addx);
Packit c4476c
	movl	\$0x80100,%r11d
Packit c4476c
	andl	OPENSSL_ia32cap_P+8(%rip),%r11d
Packit c4476c
	cmpl	\$0x80100,%r11d		# check for MULX and ADO/CX
Packit c4476c
	je	.Loop_sqrx
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
	jmp	.Loop_sqr
Packit c4476c
Packit c4476c
.align	32
Packit c4476c
.Loop_sqr:
Packit c4476c
	movl	$times,128+8(%rsp)
Packit c4476c
#first iteration
Packit c4476c
	movq	%rdx, %rbx		# 0($inp)
Packit c4476c
	mov	%rax, %rbp		# 8($inp)
Packit c4476c
	mulq	%rdx
Packit c4476c
	movq	%rax, %r8
Packit c4476c
	movq	16($inp), %rax
Packit c4476c
	movq	%rdx, %r9
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r9
Packit c4476c
	movq	24($inp), %rax
Packit c4476c
	movq	%rdx, %r10
Packit c4476c
	adcq	\$0, %r10
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r10
Packit c4476c
	movq	32($inp), %rax
Packit c4476c
	movq	%rdx, %r11
Packit c4476c
	adcq	\$0, %r11
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r11
Packit c4476c
	movq	40($inp), %rax
Packit c4476c
	movq	%rdx, %r12
Packit c4476c
	adcq	\$0, %r12
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r12
Packit c4476c
	movq	48($inp), %rax
Packit c4476c
	movq	%rdx, %r13
Packit c4476c
	adcq	\$0, %r13
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r13
Packit c4476c
	movq	56($inp), %rax
Packit c4476c
	movq	%rdx, %r14
Packit c4476c
	adcq	\$0, %r14
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r14
Packit c4476c
	movq	%rbx, %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
Packit c4476c
	xorq	%rcx,%rcx		# rcx:r8 = r8 << 1
Packit c4476c
	addq	%r8, %r8
Packit c4476c
	 movq	%rdx, %r15
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	mulq	%rax
Packit c4476c
	addq	%r8, %rdx
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	movq	%rax, (%rsp)
Packit c4476c
	movq	%rdx, 8(%rsp)
Packit c4476c
Packit c4476c
#second iteration
Packit c4476c
	movq	16($inp), %rax
Packit c4476c
	mulq	%rbp
Packit c4476c
	addq	%rax, %r10
Packit c4476c
	movq	24($inp), %rax
Packit c4476c
	movq	%rdx, %rbx
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	mulq	%rbp
Packit c4476c
	addq	%rax, %r11
Packit c4476c
	movq	32($inp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rbx, %r11
Packit c4476c
	movq	%rdx, %rbx
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	mulq	%rbp
Packit c4476c
	addq	%rax, %r12
Packit c4476c
	movq	40($inp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rbx, %r12
Packit c4476c
	movq	%rdx, %rbx
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	mulq	%rbp
Packit c4476c
	addq	%rax, %r13
Packit c4476c
	movq	48($inp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rbx, %r13
Packit c4476c
	movq	%rdx, %rbx
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	mulq	%rbp
Packit c4476c
	addq	%rax, %r14
Packit c4476c
	movq	56($inp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rbx, %r14
Packit c4476c
	movq	%rdx, %rbx
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	mulq	%rbp
Packit c4476c
	addq	%rax, %r15
Packit c4476c
	movq	%rbp, %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rbx, %r15
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
Packit c4476c
	xorq	%rbx, %rbx		# rbx:r10:r9 = r10:r9 << 1
Packit c4476c
	addq	%r9, %r9
Packit c4476c
	 movq	%rdx, %r8
Packit c4476c
	adcq	%r10, %r10
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	mulq	%rax
Packit c4476c
	# rcx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	addq	%rcx, %rax
Packit c4476c
	 movq	16($inp), %rbp
Packit c4476c
	addq	%rax, %r9
Packit c4476c
	 movq	24($inp), %rax
Packit c4476c
	adcq	%rdx, %r10
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	movq	%r9, 16(%rsp)
Packit c4476c
	movq	%r10, 24(%rsp)
Packit c4476c
Packit c4476c
#third iteration
Packit c4476c
	mulq	%rbp
Packit c4476c
	addq	%rax, %r12
Packit c4476c
	movq	32($inp), %rax
Packit c4476c
	movq	%rdx, %rcx
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	mulq	%rbp
Packit c4476c
	addq	%rax, %r13
Packit c4476c
	movq	40($inp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rcx, %r13
Packit c4476c
	movq	%rdx, %rcx
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	mulq	%rbp
Packit c4476c
	addq	%rax, %r14
Packit c4476c
	movq	48($inp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rcx, %r14
Packit c4476c
	movq	%rdx, %rcx
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	mulq	%rbp
Packit c4476c
	addq	%rax, %r15
Packit c4476c
	movq	56($inp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rcx, %r15
Packit c4476c
	movq	%rdx, %rcx
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	mulq	%rbp
Packit c4476c
	addq	%rax, %r8
Packit c4476c
	movq	%rbp, %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rcx, %r8
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
Packit c4476c
	xorq	%rcx, %rcx		# rcx:r12:r11 = r12:r11 << 1
Packit c4476c
	addq	%r11, %r11
Packit c4476c
	 movq	%rdx, %r9
Packit c4476c
	adcq	%r12, %r12
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	mulq	%rax
Packit c4476c
	# rbx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	addq	%rbx, %rax
Packit c4476c
	 movq	24($inp), %r10
Packit c4476c
	addq	%rax, %r11
Packit c4476c
	 movq	32($inp), %rax
Packit c4476c
	adcq	%rdx, %r12
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	movq	%r11, 32(%rsp)
Packit c4476c
	movq	%r12, 40(%rsp)
Packit c4476c
Packit c4476c
#fourth iteration
Packit c4476c
	mov	%rax, %r11		# 32($inp)
Packit c4476c
	mulq	%r10
Packit c4476c
	addq	%rax, %r14
Packit c4476c
	movq	40($inp), %rax
Packit c4476c
	movq	%rdx, %rbx
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	mov	%rax, %r12		# 40($inp)
Packit c4476c
	mulq	%r10
Packit c4476c
	addq	%rax, %r15
Packit c4476c
	movq	48($inp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rbx, %r15
Packit c4476c
	movq	%rdx, %rbx
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	mov	%rax, %rbp		# 48($inp)
Packit c4476c
	mulq	%r10
Packit c4476c
	addq	%rax, %r8
Packit c4476c
	movq	56($inp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rbx, %r8
Packit c4476c
	movq	%rdx, %rbx
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	mulq	%r10
Packit c4476c
	addq	%rax, %r9
Packit c4476c
	movq	%r10, %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rbx, %r9
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
Packit c4476c
	xorq	%rbx, %rbx		# rbx:r13:r14 = r13:r14 << 1
Packit c4476c
	addq	%r13, %r13
Packit c4476c
	 movq	%rdx, %r10
Packit c4476c
	adcq	%r14, %r14
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	mulq	%rax
Packit c4476c
	# rcx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	addq	%rcx, %rax
Packit c4476c
	addq	%rax, %r13
Packit c4476c
	 movq	%r12, %rax		# 40($inp)
Packit c4476c
	adcq	%rdx, %r14
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	movq	%r13, 48(%rsp)
Packit c4476c
	movq	%r14, 56(%rsp)
Packit c4476c
Packit c4476c
#fifth iteration
Packit c4476c
	mulq	%r11
Packit c4476c
	addq	%rax, %r8
Packit c4476c
	movq	%rbp, %rax		# 48($inp)
Packit c4476c
	movq	%rdx, %rcx
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	mulq	%r11
Packit c4476c
	addq	%rax, %r9
Packit c4476c
	movq	56($inp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rcx, %r9
Packit c4476c
	movq	%rdx, %rcx
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	mov	%rax, %r14		# 56($inp)
Packit c4476c
	mulq	%r11
Packit c4476c
	addq	%rax, %r10
Packit c4476c
	movq	%r11, %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rcx, %r10
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
Packit c4476c
	xorq	%rcx, %rcx		# rcx:r8:r15 = r8:r15 << 1
Packit c4476c
	addq	%r15, %r15
Packit c4476c
	 movq	%rdx, %r11
Packit c4476c
	adcq	%r8, %r8
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	mulq	%rax
Packit c4476c
	# rbx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	addq	%rbx, %rax
Packit c4476c
	addq	%rax, %r15
Packit c4476c
	 movq	%rbp, %rax		# 48($inp)
Packit c4476c
	adcq	%rdx, %r8
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	movq	%r15, 64(%rsp)
Packit c4476c
	movq	%r8, 72(%rsp)
Packit c4476c
Packit c4476c
#sixth iteration
Packit c4476c
	mulq	%r12
Packit c4476c
	addq	%rax, %r10
Packit c4476c
	movq	%r14, %rax		# 56($inp)
Packit c4476c
	movq	%rdx, %rbx
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	mulq	%r12
Packit c4476c
	addq	%rax, %r11
Packit c4476c
	movq	%r12, %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%rbx, %r11
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
Packit c4476c
	xorq	%rbx, %rbx		# rbx:r10:r9 = r10:r9 << 1
Packit c4476c
	addq	%r9, %r9
Packit c4476c
	 movq	%rdx, %r12
Packit c4476c
	adcq	%r10, %r10
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	mulq	%rax
Packit c4476c
	# rcx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	addq	%rcx, %rax
Packit c4476c
	addq	%rax, %r9
Packit c4476c
	 movq	%r14, %rax		# 56($inp)
Packit c4476c
	adcq	%rdx, %r10
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	movq	%r9, 80(%rsp)
Packit c4476c
	movq	%r10, 88(%rsp)
Packit c4476c
Packit c4476c
#seventh iteration
Packit c4476c
	mulq	%rbp
Packit c4476c
	addq	%rax, %r12
Packit c4476c
	movq	%rbp, %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
Packit c4476c
	xorq	%rcx, %rcx		# rcx:r12:r11 = r12:r11 << 1
Packit c4476c
	addq	%r11, %r11
Packit c4476c
	 movq	%rdx, %r13
Packit c4476c
	adcq	%r12, %r12
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	mulq	%rax
Packit c4476c
	# rbx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	addq	%rbx, %rax
Packit c4476c
	addq	%rax, %r11
Packit c4476c
	 movq	%r14, %rax		# 56($inp)
Packit c4476c
	adcq	%rdx, %r12
Packit c4476c
	adcq	\$0, %rcx
Packit c4476c
Packit c4476c
	movq	%r11, 96(%rsp)
Packit c4476c
	movq	%r12, 104(%rsp)
Packit c4476c
Packit c4476c
#eighth iteration
Packit c4476c
	xorq	%rbx, %rbx		# rbx:r13 = r13 << 1
Packit c4476c
	addq	%r13, %r13
Packit c4476c
	adcq	\$0, %rbx
Packit c4476c
Packit c4476c
	mulq	%rax
Packit c4476c
	# rcx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	addq	%rcx, %rax
Packit c4476c
	addq	%r13, %rax
Packit c4476c
	adcq	%rbx, %rdx
Packit c4476c
Packit c4476c
	movq	(%rsp), %r8
Packit c4476c
	movq	8(%rsp), %r9
Packit c4476c
	movq	16(%rsp), %r10
Packit c4476c
	movq	24(%rsp), %r11
Packit c4476c
	movq	32(%rsp), %r12
Packit c4476c
	movq	40(%rsp), %r13
Packit c4476c
	movq	48(%rsp), %r14
Packit c4476c
	movq	56(%rsp), %r15
Packit c4476c
	movq	%xmm1, %rbp
Packit c4476c
Packit c4476c
	movq	%rax, 112(%rsp)
Packit c4476c
	movq	%rdx, 120(%rsp)
Packit c4476c
Packit c4476c
	call	__rsaz_512_reduce
Packit c4476c
Packit c4476c
	addq	64(%rsp), %r8
Packit c4476c
	adcq	72(%rsp), %r9
Packit c4476c
	adcq	80(%rsp), %r10
Packit c4476c
	adcq	88(%rsp), %r11
Packit c4476c
	adcq	96(%rsp), %r12
Packit c4476c
	adcq	104(%rsp), %r13
Packit c4476c
	adcq	112(%rsp), %r14
Packit c4476c
	adcq	120(%rsp), %r15
Packit c4476c
	sbbq	%rcx, %rcx
Packit c4476c
Packit c4476c
	call	__rsaz_512_subtract
Packit c4476c
Packit c4476c
	movq	%r8, %rdx
Packit c4476c
	movq	%r9, %rax
Packit c4476c
	movl	128+8(%rsp), $times
Packit c4476c
	movq	$out, $inp
Packit c4476c
Packit c4476c
	decl	$times
Packit c4476c
	jnz	.Loop_sqr
Packit c4476c
___
Packit c4476c
if ($addx) {
Packit c4476c
$code.=<<___;
Packit c4476c
	jmp	.Lsqr_tail
Packit c4476c
Packit c4476c
.align	32
Packit c4476c
.Loop_sqrx:
Packit c4476c
	movl	$times,128+8(%rsp)
Packit c4476c
	movq	$out, %xmm0		# off-load
Packit c4476c
#first iteration
Packit c4476c
	mulx	%rax, %r8, %r9
Packit c4476c
	mov	%rax, %rbx
Packit c4476c
Packit c4476c
	mulx	16($inp), %rcx, %r10
Packit c4476c
	xor	%rbp, %rbp		# cf=0, of=0
Packit c4476c
Packit c4476c
	mulx	24($inp), %rax, %r11
Packit c4476c
	adcx	%rcx, %r9
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xf3,0xf6,0xa6,0x20,0x00,0x00,0x00	# mulx	32($inp), %rcx, %r12
Packit c4476c
	adcx	%rax, %r10
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xfb,0xf6,0xae,0x28,0x00,0x00,0x00	# mulx	40($inp), %rax, %r13
Packit c4476c
	adcx	%rcx, %r11
Packit c4476c
Packit c4476c
	mulx	48($inp), %rcx, %r14
Packit c4476c
	adcx	%rax, %r12
Packit c4476c
	adcx	%rcx, %r13
Packit c4476c
Packit c4476c
	mulx	56($inp), %rax, %r15
Packit c4476c
	adcx	%rax, %r14
Packit c4476c
	adcx	%rbp, %r15		# %rbp is 0
Packit c4476c
Packit c4476c
	mulx	%rdx, %rax, $out
Packit c4476c
	 mov	%rbx, %rdx		# 8($inp)
Packit c4476c
	xor	%rcx, %rcx
Packit c4476c
	adox	%r8, %r8
Packit c4476c
	adcx	$out, %r8
Packit c4476c
	adox	%rbp, %rcx
Packit c4476c
	adcx	%rbp, %rcx
Packit c4476c
Packit c4476c
	mov	%rax, (%rsp)
Packit c4476c
	mov	%r8, 8(%rsp)
Packit c4476c
Packit c4476c
#second iteration
Packit c4476c
	.byte	0xc4,0xe2,0xfb,0xf6,0x9e,0x10,0x00,0x00,0x00	# mulx	16($inp), %rax, %rbx
Packit c4476c
	adox	%rax, %r10
Packit c4476c
	adcx	%rbx, %r11
Packit c4476c
Packit c4476c
	mulx	24($inp), $out, %r8
Packit c4476c
	adox	$out, %r11
Packit c4476c
	.byte	0x66
Packit c4476c
	adcx	%r8, %r12
Packit c4476c
Packit c4476c
	mulx	32($inp), %rax, %rbx
Packit c4476c
	adox	%rax, %r12
Packit c4476c
	adcx	%rbx, %r13
Packit c4476c
Packit c4476c
	mulx	40($inp), $out, %r8
Packit c4476c
	adox	$out, %r13
Packit c4476c
	adcx	%r8, %r14
Packit c4476c
Packit c4476c
	.byte	0xc4,0xe2,0xfb,0xf6,0x9e,0x30,0x00,0x00,0x00	# mulx	48($inp), %rax, %rbx
Packit c4476c
	adox	%rax, %r14
Packit c4476c
	adcx	%rbx, %r15
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xc3,0xf6,0x86,0x38,0x00,0x00,0x00	# mulx	56($inp), $out, %r8
Packit c4476c
	adox	$out, %r15
Packit c4476c
	adcx	%rbp, %r8
Packit c4476c
	 mulx	%rdx, %rax, $out
Packit c4476c
	adox	%rbp, %r8
Packit c4476c
	 .byte	0x48,0x8b,0x96,0x10,0x00,0x00,0x00		# mov	16($inp), %rdx
Packit c4476c
Packit c4476c
	xor	%rbx, %rbx
Packit c4476c
	 adox	%r9, %r9
Packit c4476c
	# rcx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	adcx	%rcx, %rax
Packit c4476c
	adox	%r10, %r10
Packit c4476c
	adcx	%rax, %r9
Packit c4476c
	adox	%rbp, %rbx
Packit c4476c
	adcx	$out, %r10
Packit c4476c
	adcx	%rbp, %rbx
Packit c4476c
Packit c4476c
	mov	%r9, 16(%rsp)
Packit c4476c
	.byte	0x4c,0x89,0x94,0x24,0x18,0x00,0x00,0x00		# mov	%r10, 24(%rsp)
Packit c4476c
Packit c4476c
#third iteration
Packit c4476c
	mulx	24($inp), $out, %r9
Packit c4476c
	adox	$out, %r12
Packit c4476c
	adcx	%r9, %r13
Packit c4476c
Packit c4476c
	mulx	32($inp), %rax, %rcx
Packit c4476c
	adox	%rax, %r13
Packit c4476c
	adcx	%rcx, %r14
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xc3,0xf6,0x8e,0x28,0x00,0x00,0x00	# mulx	40($inp), $out, %r9
Packit c4476c
	adox	$out, %r14
Packit c4476c
	adcx	%r9, %r15
Packit c4476c
Packit c4476c
	.byte	0xc4,0xe2,0xfb,0xf6,0x8e,0x30,0x00,0x00,0x00	# mulx	48($inp), %rax, %rcx
Packit c4476c
	adox	%rax, %r15
Packit c4476c
	adcx	%rcx, %r8
Packit c4476c
Packit c4476c
	mulx	56($inp), $out, %r9
Packit c4476c
	adox	$out, %r8
Packit c4476c
	adcx	%rbp, %r9
Packit c4476c
	 mulx	%rdx, %rax, $out
Packit c4476c
	adox	%rbp, %r9
Packit c4476c
	 mov	24($inp), %rdx
Packit c4476c
Packit c4476c
	xor	%rcx, %rcx
Packit c4476c
	 adox	%r11, %r11
Packit c4476c
	# rbx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	adcx	%rbx, %rax
Packit c4476c
	adox	%r12, %r12
Packit c4476c
	adcx	%rax, %r11
Packit c4476c
	adox	%rbp, %rcx
Packit c4476c
	adcx	$out, %r12
Packit c4476c
	adcx	%rbp, %rcx
Packit c4476c
Packit c4476c
	mov	%r11, 32(%rsp)
Packit c4476c
	mov	%r12, 40(%rsp)
Packit c4476c
Packit c4476c
#fourth iteration
Packit c4476c
	mulx	32($inp), %rax, %rbx
Packit c4476c
	adox	%rax, %r14
Packit c4476c
	adcx	%rbx, %r15
Packit c4476c
Packit c4476c
	mulx	40($inp), $out, %r10
Packit c4476c
	adox	$out, %r15
Packit c4476c
	adcx	%r10, %r8
Packit c4476c
Packit c4476c
	mulx	48($inp), %rax, %rbx
Packit c4476c
	adox	%rax, %r8
Packit c4476c
	adcx	%rbx, %r9
Packit c4476c
Packit c4476c
	mulx	56($inp), $out, %r10
Packit c4476c
	adox	$out, %r9
Packit c4476c
	adcx	%rbp, %r10
Packit c4476c
	 mulx	%rdx, %rax, $out
Packit c4476c
	adox	%rbp, %r10
Packit c4476c
	 mov	32($inp), %rdx
Packit c4476c
Packit c4476c
	xor	%rbx, %rbx
Packit c4476c
	 adox	%r13, %r13
Packit c4476c
	# rcx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	adcx	%rcx, %rax
Packit c4476c
	adox	%r14, %r14
Packit c4476c
	adcx	%rax, %r13
Packit c4476c
	adox	%rbp, %rbx
Packit c4476c
	adcx	$out, %r14
Packit c4476c
	adcx	%rbp, %rbx
Packit c4476c
Packit c4476c
	mov	%r13, 48(%rsp)
Packit c4476c
	mov	%r14, 56(%rsp)
Packit c4476c
Packit c4476c
#fifth iteration
Packit c4476c
	mulx	40($inp), $out, %r11
Packit c4476c
	adox	$out, %r8
Packit c4476c
	adcx	%r11, %r9
Packit c4476c
Packit c4476c
	mulx	48($inp), %rax, %rcx
Packit c4476c
	adox	%rax, %r9
Packit c4476c
	adcx	%rcx, %r10
Packit c4476c
Packit c4476c
	mulx	56($inp), $out, %r11
Packit c4476c
	adox	$out, %r10
Packit c4476c
	adcx	%rbp, %r11
Packit c4476c
	 mulx	%rdx, %rax, $out
Packit c4476c
	 mov	40($inp), %rdx
Packit c4476c
	adox	%rbp, %r11
Packit c4476c
Packit c4476c
	xor	%rcx, %rcx
Packit c4476c
	 adox	%r15, %r15
Packit c4476c
	# rbx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	adcx	%rbx, %rax
Packit c4476c
	adox	%r8, %r8
Packit c4476c
	adcx	%rax, %r15
Packit c4476c
	adox	%rbp, %rcx
Packit c4476c
	adcx	$out, %r8
Packit c4476c
	adcx	%rbp, %rcx
Packit c4476c
Packit c4476c
	mov	%r15, 64(%rsp)
Packit c4476c
	mov	%r8, 72(%rsp)
Packit c4476c
Packit c4476c
#sixth iteration
Packit c4476c
	.byte	0xc4,0xe2,0xfb,0xf6,0x9e,0x30,0x00,0x00,0x00	# mulx	48($inp), %rax, %rbx
Packit c4476c
	adox	%rax, %r10
Packit c4476c
	adcx	%rbx, %r11
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xc3,0xf6,0xa6,0x38,0x00,0x00,0x00	# mulx	56($inp), $out, %r12
Packit c4476c
	adox	$out, %r11
Packit c4476c
	adcx	%rbp, %r12
Packit c4476c
	 mulx	%rdx, %rax, $out
Packit c4476c
	adox	%rbp, %r12
Packit c4476c
	 mov	48($inp), %rdx
Packit c4476c
Packit c4476c
	xor	%rbx, %rbx
Packit c4476c
	 adox	%r9, %r9
Packit c4476c
	# rcx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	adcx	%rcx, %rax
Packit c4476c
	adox	%r10, %r10
Packit c4476c
	adcx	%rax, %r9
Packit c4476c
	adcx	$out, %r10
Packit c4476c
	adox	%rbp, %rbx
Packit c4476c
	adcx	%rbp, %rbx
Packit c4476c
Packit c4476c
	mov	%r9, 80(%rsp)
Packit c4476c
	mov	%r10, 88(%rsp)
Packit c4476c
Packit c4476c
#seventh iteration
Packit c4476c
	.byte	0xc4,0x62,0xfb,0xf6,0xae,0x38,0x00,0x00,0x00	# mulx	56($inp), %rax, %r13
Packit c4476c
	adox	%rax, %r12
Packit c4476c
	adox	%rbp, %r13
Packit c4476c
Packit c4476c
	mulx	%rdx, %rax, $out
Packit c4476c
	xor	%rcx, %rcx
Packit c4476c
	 mov	56($inp), %rdx
Packit c4476c
	 adox	%r11, %r11
Packit c4476c
	# rbx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	adcx	%rbx, %rax
Packit c4476c
	adox	%r12, %r12
Packit c4476c
	adcx	%rax, %r11
Packit c4476c
	adox	%rbp, %rcx
Packit c4476c
	adcx	$out, %r12
Packit c4476c
	adcx	%rbp, %rcx
Packit c4476c
Packit c4476c
	.byte	0x4c,0x89,0x9c,0x24,0x60,0x00,0x00,0x00		# mov	%r11, 96(%rsp)
Packit c4476c
	.byte	0x4c,0x89,0xa4,0x24,0x68,0x00,0x00,0x00		# mov	%r12, 104(%rsp)
Packit c4476c
Packit c4476c
#eighth iteration
Packit c4476c
	mulx	%rdx, %rax, %rdx
Packit c4476c
	xor	%rbx, %rbx
Packit c4476c
	 adox	%r13, %r13
Packit c4476c
	# rcx <= 2 and rax <= 0xFFFF..F9, so carry must be zero here
Packit c4476c
	adcx	%rcx, %rax
Packit c4476c
	adox	%rbp, %rbx
Packit c4476c
	adcx	%r13, %rax
Packit c4476c
	adcx	%rdx, %rbx
Packit c4476c
Packit c4476c
	movq	%xmm0, $out
Packit c4476c
	movq	%xmm1, %rbp
Packit c4476c
Packit c4476c
	movq	128(%rsp), %rdx		# pull $n0
Packit c4476c
	movq	(%rsp), %r8
Packit c4476c
	movq	8(%rsp), %r9
Packit c4476c
	movq	16(%rsp), %r10
Packit c4476c
	movq	24(%rsp), %r11
Packit c4476c
	movq	32(%rsp), %r12
Packit c4476c
	movq	40(%rsp), %r13
Packit c4476c
	movq	48(%rsp), %r14
Packit c4476c
	movq	56(%rsp), %r15
Packit c4476c
Packit c4476c
	movq	%rax, 112(%rsp)
Packit c4476c
	movq	%rbx, 120(%rsp)
Packit c4476c
Packit c4476c
	call	__rsaz_512_reducex
Packit c4476c
Packit c4476c
	addq	64(%rsp), %r8
Packit c4476c
	adcq	72(%rsp), %r9
Packit c4476c
	adcq	80(%rsp), %r10
Packit c4476c
	adcq	88(%rsp), %r11
Packit c4476c
	adcq	96(%rsp), %r12
Packit c4476c
	adcq	104(%rsp), %r13
Packit c4476c
	adcq	112(%rsp), %r14
Packit c4476c
	adcq	120(%rsp), %r15
Packit c4476c
	sbbq	%rcx, %rcx
Packit c4476c
Packit c4476c
	call	__rsaz_512_subtract
Packit c4476c
Packit c4476c
	movq	%r8, %rdx
Packit c4476c
	movq	%r9, %rax
Packit c4476c
	movl	128+8(%rsp), $times
Packit c4476c
	movq	$out, $inp
Packit c4476c
Packit c4476c
	decl	$times
Packit c4476c
	jnz	.Loop_sqrx
Packit c4476c
Packit c4476c
.Lsqr_tail:
Packit c4476c
___
Packit c4476c
}
Packit c4476c
$code.=<<___;
Packit c4476c
Packit c4476c
	leaq	128+24+48(%rsp), %rax
Packit c4476c
.cfi_def_cfa	%rax,8
Packit c4476c
	movq	-48(%rax), %r15
Packit c4476c
.cfi_restore	%r15
Packit c4476c
	movq	-40(%rax), %r14
Packit c4476c
.cfi_restore	%r14
Packit c4476c
	movq	-32(%rax), %r13
Packit c4476c
.cfi_restore	%r13
Packit c4476c
	movq	-24(%rax), %r12
Packit c4476c
.cfi_restore	%r12
Packit c4476c
	movq	-16(%rax), %rbp
Packit c4476c
.cfi_restore	%rbp
Packit c4476c
	movq	-8(%rax), %rbx
Packit c4476c
.cfi_restore	%rbx
Packit c4476c
	leaq	(%rax), %rsp
Packit c4476c
.cfi_def_cfa_register	%rsp
Packit c4476c
.Lsqr_epilogue:
Packit c4476c
	ret
Packit c4476c
.cfi_endproc
Packit c4476c
.size	rsaz_512_sqr,.-rsaz_512_sqr
Packit c4476c
___
Packit c4476c
}
Packit c4476c
{
Packit c4476c
my ($out,$ap,$bp,$mod,$n0) = ("%rdi","%rsi","%rdx","%rcx","%r8");
Packit c4476c
$code.=<<___;
Packit c4476c
.globl	rsaz_512_mul
Packit c4476c
.type	rsaz_512_mul,\@function,5
Packit c4476c
.align	32
Packit c4476c
rsaz_512_mul:
Packit c4476c
.cfi_startproc
Packit c4476c
	push	%rbx
Packit c4476c
.cfi_push	%rbx
Packit c4476c
	push	%rbp
Packit c4476c
.cfi_push	%rbp
Packit c4476c
	push	%r12
Packit c4476c
.cfi_push	%r12
Packit c4476c
	push	%r13
Packit c4476c
.cfi_push	%r13
Packit c4476c
	push	%r14
Packit c4476c
.cfi_push	%r14
Packit c4476c
	push	%r15
Packit c4476c
.cfi_push	%r15
Packit c4476c
Packit c4476c
	subq	\$128+24, %rsp
Packit c4476c
.cfi_adjust_cfa_offset	128+24
Packit c4476c
.Lmul_body:
Packit c4476c
	movq	$out, %xmm0		# off-load arguments
Packit c4476c
	movq	$mod, %xmm1
Packit c4476c
	movq	$n0, 128(%rsp)
Packit c4476c
___
Packit c4476c
$code.=<<___ if ($addx);
Packit c4476c
	movl	\$0x80100,%r11d
Packit c4476c
	andl	OPENSSL_ia32cap_P+8(%rip),%r11d
Packit c4476c
	cmpl	\$0x80100,%r11d		# check for MULX and ADO/CX
Packit c4476c
	je	.Lmulx
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
	movq	($bp), %rbx		# pass b[0]
Packit c4476c
	movq	$bp, %rbp		# pass argument
Packit c4476c
	call	__rsaz_512_mul
Packit c4476c
Packit c4476c
	movq	%xmm0, $out
Packit c4476c
	movq	%xmm1, %rbp
Packit c4476c
Packit c4476c
	movq	(%rsp), %r8
Packit c4476c
	movq	8(%rsp), %r9
Packit c4476c
	movq	16(%rsp), %r10
Packit c4476c
	movq	24(%rsp), %r11
Packit c4476c
	movq	32(%rsp), %r12
Packit c4476c
	movq	40(%rsp), %r13
Packit c4476c
	movq	48(%rsp), %r14
Packit c4476c
	movq	56(%rsp), %r15
Packit c4476c
Packit c4476c
	call	__rsaz_512_reduce
Packit c4476c
___
Packit c4476c
$code.=<<___ if ($addx);
Packit c4476c
	jmp	.Lmul_tail
Packit c4476c
Packit c4476c
.align	32
Packit c4476c
.Lmulx:
Packit c4476c
	movq	$bp, %rbp		# pass argument
Packit c4476c
	movq	($bp), %rdx		# pass b[0]
Packit c4476c
	call	__rsaz_512_mulx
Packit c4476c
Packit c4476c
	movq	%xmm0, $out
Packit c4476c
	movq	%xmm1, %rbp
Packit c4476c
Packit c4476c
	movq	128(%rsp), %rdx		# pull $n0
Packit c4476c
	movq	(%rsp), %r8
Packit c4476c
	movq	8(%rsp), %r9
Packit c4476c
	movq	16(%rsp), %r10
Packit c4476c
	movq	24(%rsp), %r11
Packit c4476c
	movq	32(%rsp), %r12
Packit c4476c
	movq	40(%rsp), %r13
Packit c4476c
	movq	48(%rsp), %r14
Packit c4476c
	movq	56(%rsp), %r15
Packit c4476c
Packit c4476c
	call	__rsaz_512_reducex
Packit c4476c
.Lmul_tail:
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
	addq	64(%rsp), %r8
Packit c4476c
	adcq	72(%rsp), %r9
Packit c4476c
	adcq	80(%rsp), %r10
Packit c4476c
	adcq	88(%rsp), %r11
Packit c4476c
	adcq	96(%rsp), %r12
Packit c4476c
	adcq	104(%rsp), %r13
Packit c4476c
	adcq	112(%rsp), %r14
Packit c4476c
	adcq	120(%rsp), %r15
Packit c4476c
	sbbq	%rcx, %rcx
Packit c4476c
Packit c4476c
	call	__rsaz_512_subtract
Packit c4476c
Packit c4476c
	leaq	128+24+48(%rsp), %rax
Packit c4476c
.cfi_def_cfa	%rax,8
Packit c4476c
	movq	-48(%rax), %r15
Packit c4476c
.cfi_restore	%r15
Packit c4476c
	movq	-40(%rax), %r14
Packit c4476c
.cfi_restore	%r14
Packit c4476c
	movq	-32(%rax), %r13
Packit c4476c
.cfi_restore	%r13
Packit c4476c
	movq	-24(%rax), %r12
Packit c4476c
.cfi_restore	%r12
Packit c4476c
	movq	-16(%rax), %rbp
Packit c4476c
.cfi_restore	%rbp
Packit c4476c
	movq	-8(%rax), %rbx
Packit c4476c
.cfi_restore	%rbx
Packit c4476c
	leaq	(%rax), %rsp
Packit c4476c
.cfi_def_cfa_register	%rsp
Packit c4476c
.Lmul_epilogue:
Packit c4476c
	ret
Packit c4476c
.cfi_endproc
Packit c4476c
.size	rsaz_512_mul,.-rsaz_512_mul
Packit c4476c
___
Packit c4476c
}
Packit c4476c
{
Packit c4476c
my ($out,$ap,$bp,$mod,$n0,$pwr) = ("%rdi","%rsi","%rdx","%rcx","%r8","%r9d");
Packit c4476c
$code.=<<___;
Packit c4476c
.globl	rsaz_512_mul_gather4
Packit c4476c
.type	rsaz_512_mul_gather4,\@function,6
Packit c4476c
.align	32
Packit c4476c
rsaz_512_mul_gather4:
Packit c4476c
.cfi_startproc
Packit c4476c
	push	%rbx
Packit c4476c
.cfi_push	%rbx
Packit c4476c
	push	%rbp
Packit c4476c
.cfi_push	%rbp
Packit c4476c
	push	%r12
Packit c4476c
.cfi_push	%r12
Packit c4476c
	push	%r13
Packit c4476c
.cfi_push	%r13
Packit c4476c
	push	%r14
Packit c4476c
.cfi_push	%r14
Packit c4476c
	push	%r15
Packit c4476c
.cfi_push	%r15
Packit c4476c
Packit c4476c
	subq	\$`128+24+($win64?0xb0:0)`, %rsp
Packit c4476c
.cfi_adjust_cfa_offset	`128+24+($win64?0xb0:0)`
Packit c4476c
___
Packit c4476c
$code.=<<___	if ($win64);
Packit c4476c
	movaps	%xmm6,0xa0(%rsp)
Packit c4476c
	movaps	%xmm7,0xb0(%rsp)
Packit c4476c
	movaps	%xmm8,0xc0(%rsp)
Packit c4476c
	movaps	%xmm9,0xd0(%rsp)
Packit c4476c
	movaps	%xmm10,0xe0(%rsp)
Packit c4476c
	movaps	%xmm11,0xf0(%rsp)
Packit c4476c
	movaps	%xmm12,0x100(%rsp)
Packit c4476c
	movaps	%xmm13,0x110(%rsp)
Packit c4476c
	movaps	%xmm14,0x120(%rsp)
Packit c4476c
	movaps	%xmm15,0x130(%rsp)
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
.Lmul_gather4_body:
Packit c4476c
	movd	$pwr,%xmm8
Packit c4476c
	movdqa	.Linc+16(%rip),%xmm1	# 00000002000000020000000200000002
Packit c4476c
	movdqa	.Linc(%rip),%xmm0	# 00000001000000010000000000000000
Packit c4476c
Packit c4476c
	pshufd	\$0,%xmm8,%xmm8		# broadcast $power
Packit c4476c
	movdqa	%xmm1,%xmm7
Packit c4476c
	movdqa	%xmm1,%xmm2
Packit c4476c
___
Packit c4476c
########################################################################
Packit c4476c
# calculate mask by comparing 0..15 to $power
Packit c4476c
#
Packit c4476c
for($i=0;$i<4;$i++) {
Packit c4476c
$code.=<<___;
Packit c4476c
	paddd	%xmm`$i`,%xmm`$i+1`
Packit c4476c
	pcmpeqd	%xmm8,%xmm`$i`
Packit c4476c
	movdqa	%xmm7,%xmm`$i+3`
Packit c4476c
___
Packit c4476c
}
Packit c4476c
for(;$i<7;$i++) {
Packit c4476c
$code.=<<___;
Packit c4476c
	paddd	%xmm`$i`,%xmm`$i+1`
Packit c4476c
	pcmpeqd	%xmm8,%xmm`$i`
Packit c4476c
___
Packit c4476c
}
Packit c4476c
$code.=<<___;
Packit c4476c
	pcmpeqd	%xmm8,%xmm7
Packit c4476c
Packit c4476c
	movdqa	16*0($bp),%xmm8
Packit c4476c
	movdqa	16*1($bp),%xmm9
Packit c4476c
	movdqa	16*2($bp),%xmm10
Packit c4476c
	movdqa	16*3($bp),%xmm11
Packit c4476c
	pand	%xmm0,%xmm8
Packit c4476c
	movdqa	16*4($bp),%xmm12
Packit c4476c
	pand	%xmm1,%xmm9
Packit c4476c
	movdqa	16*5($bp),%xmm13
Packit c4476c
	pand	%xmm2,%xmm10
Packit c4476c
	movdqa	16*6($bp),%xmm14
Packit c4476c
	pand	%xmm3,%xmm11
Packit c4476c
	movdqa	16*7($bp),%xmm15
Packit c4476c
	leaq	128($bp), %rbp
Packit c4476c
	pand	%xmm4,%xmm12
Packit c4476c
	pand	%xmm5,%xmm13
Packit c4476c
	pand	%xmm6,%xmm14
Packit c4476c
	pand	%xmm7,%xmm15
Packit c4476c
	por	%xmm10,%xmm8
Packit c4476c
	por	%xmm11,%xmm9
Packit c4476c
	por	%xmm12,%xmm8
Packit c4476c
	por	%xmm13,%xmm9
Packit c4476c
	por	%xmm14,%xmm8
Packit c4476c
	por	%xmm15,%xmm9
Packit c4476c
Packit c4476c
	por	%xmm9,%xmm8
Packit c4476c
	pshufd	\$0x4e,%xmm8,%xmm9
Packit c4476c
	por	%xmm9,%xmm8
Packit c4476c
___
Packit c4476c
$code.=<<___ if ($addx);
Packit c4476c
	movl	\$0x80100,%r11d
Packit c4476c
	andl	OPENSSL_ia32cap_P+8(%rip),%r11d
Packit c4476c
	cmpl	\$0x80100,%r11d		# check for MULX and ADO/CX
Packit c4476c
	je	.Lmulx_gather
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
	movq	%xmm8,%rbx
Packit c4476c
Packit c4476c
	movq	$n0, 128(%rsp)		# off-load arguments
Packit c4476c
	movq	$out, 128+8(%rsp)
Packit c4476c
	movq	$mod, 128+16(%rsp)
Packit c4476c
Packit c4476c
	movq	($ap), %rax
Packit c4476c
	 movq	8($ap), %rcx
Packit c4476c
	mulq	%rbx			# 0 iteration
Packit c4476c
	movq	%rax, (%rsp)
Packit c4476c
	movq	%rcx, %rax
Packit c4476c
	movq	%rdx, %r8
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r8
Packit c4476c
	movq	16($ap), %rax
Packit c4476c
	movq	%rdx, %r9
Packit c4476c
	adcq	\$0, %r9
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r9
Packit c4476c
	movq	24($ap), %rax
Packit c4476c
	movq	%rdx, %r10
Packit c4476c
	adcq	\$0, %r10
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r10
Packit c4476c
	movq	32($ap), %rax
Packit c4476c
	movq	%rdx, %r11
Packit c4476c
	adcq	\$0, %r11
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r11
Packit c4476c
	movq	40($ap), %rax
Packit c4476c
	movq	%rdx, %r12
Packit c4476c
	adcq	\$0, %r12
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r12
Packit c4476c
	movq	48($ap), %rax
Packit c4476c
	movq	%rdx, %r13
Packit c4476c
	adcq	\$0, %r13
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r13
Packit c4476c
	movq	56($ap), %rax
Packit c4476c
	movq	%rdx, %r14
Packit c4476c
	adcq	\$0, %r14
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r14
Packit c4476c
	 movq	($ap), %rax
Packit c4476c
	movq	%rdx, %r15
Packit c4476c
	adcq	\$0, %r15
Packit c4476c
Packit c4476c
	leaq	8(%rsp), %rdi
Packit c4476c
	movl	\$7, %ecx
Packit c4476c
	jmp	.Loop_mul_gather
Packit c4476c
Packit c4476c
.align	32
Packit c4476c
.Loop_mul_gather:
Packit c4476c
	movdqa	16*0(%rbp),%xmm8
Packit c4476c
	movdqa	16*1(%rbp),%xmm9
Packit c4476c
	movdqa	16*2(%rbp),%xmm10
Packit c4476c
	movdqa	16*3(%rbp),%xmm11
Packit c4476c
	pand	%xmm0,%xmm8
Packit c4476c
	movdqa	16*4(%rbp),%xmm12
Packit c4476c
	pand	%xmm1,%xmm9
Packit c4476c
	movdqa	16*5(%rbp),%xmm13
Packit c4476c
	pand	%xmm2,%xmm10
Packit c4476c
	movdqa	16*6(%rbp),%xmm14
Packit c4476c
	pand	%xmm3,%xmm11
Packit c4476c
	movdqa	16*7(%rbp),%xmm15
Packit c4476c
	leaq	128(%rbp), %rbp
Packit c4476c
	pand	%xmm4,%xmm12
Packit c4476c
	pand	%xmm5,%xmm13
Packit c4476c
	pand	%xmm6,%xmm14
Packit c4476c
	pand	%xmm7,%xmm15
Packit c4476c
	por	%xmm10,%xmm8
Packit c4476c
	por	%xmm11,%xmm9
Packit c4476c
	por	%xmm12,%xmm8
Packit c4476c
	por	%xmm13,%xmm9
Packit c4476c
	por	%xmm14,%xmm8
Packit c4476c
	por	%xmm15,%xmm9
Packit c4476c
Packit c4476c
	por	%xmm9,%xmm8
Packit c4476c
	pshufd	\$0x4e,%xmm8,%xmm9
Packit c4476c
	por	%xmm9,%xmm8
Packit c4476c
	movq	%xmm8,%rbx
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r8
Packit c4476c
	movq	8($ap), %rax
Packit c4476c
	movq	%r8, (%rdi)
Packit c4476c
	movq	%rdx, %r8
Packit c4476c
	adcq	\$0, %r8
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r9
Packit c4476c
	movq	16($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r9, %r8
Packit c4476c
	movq	%rdx, %r9
Packit c4476c
	adcq	\$0, %r9
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r10
Packit c4476c
	movq	24($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r10, %r9
Packit c4476c
	movq	%rdx, %r10
Packit c4476c
	adcq	\$0, %r10
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r11
Packit c4476c
	movq	32($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r11, %r10
Packit c4476c
	movq	%rdx, %r11
Packit c4476c
	adcq	\$0, %r11
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r12
Packit c4476c
	movq	40($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r12, %r11
Packit c4476c
	movq	%rdx, %r12
Packit c4476c
	adcq	\$0, %r12
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r13
Packit c4476c
	movq	48($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r13, %r12
Packit c4476c
	movq	%rdx, %r13
Packit c4476c
	adcq	\$0, %r13
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r14
Packit c4476c
	movq	56($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r14, %r13
Packit c4476c
	movq	%rdx, %r14
Packit c4476c
	adcq	\$0, %r14
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r15
Packit c4476c
	 movq	($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r15, %r14
Packit c4476c
	movq	%rdx, %r15
Packit c4476c
	adcq	\$0, %r15
Packit c4476c
Packit c4476c
	leaq	8(%rdi), %rdi
Packit c4476c
Packit c4476c
	decl	%ecx
Packit c4476c
	jnz	.Loop_mul_gather
Packit c4476c
Packit c4476c
	movq	%r8, (%rdi)
Packit c4476c
	movq	%r9, 8(%rdi)
Packit c4476c
	movq	%r10, 16(%rdi)
Packit c4476c
	movq	%r11, 24(%rdi)
Packit c4476c
	movq	%r12, 32(%rdi)
Packit c4476c
	movq	%r13, 40(%rdi)
Packit c4476c
	movq	%r14, 48(%rdi)
Packit c4476c
	movq	%r15, 56(%rdi)
Packit c4476c
Packit c4476c
	movq	128+8(%rsp), $out
Packit c4476c
	movq	128+16(%rsp), %rbp
Packit c4476c
Packit c4476c
	movq	(%rsp), %r8
Packit c4476c
	movq	8(%rsp), %r9
Packit c4476c
	movq	16(%rsp), %r10
Packit c4476c
	movq	24(%rsp), %r11
Packit c4476c
	movq	32(%rsp), %r12
Packit c4476c
	movq	40(%rsp), %r13
Packit c4476c
	movq	48(%rsp), %r14
Packit c4476c
	movq	56(%rsp), %r15
Packit c4476c
Packit c4476c
	call	__rsaz_512_reduce
Packit c4476c
___
Packit c4476c
$code.=<<___ if ($addx);
Packit c4476c
	jmp	.Lmul_gather_tail
Packit c4476c
Packit c4476c
.align	32
Packit c4476c
.Lmulx_gather:
Packit c4476c
	movq	%xmm8,%rdx
Packit c4476c
Packit c4476c
	mov	$n0, 128(%rsp)		# off-load arguments
Packit c4476c
	mov	$out, 128+8(%rsp)
Packit c4476c
	mov	$mod, 128+16(%rsp)
Packit c4476c
Packit c4476c
	mulx	($ap), %rbx, %r8	# 0 iteration
Packit c4476c
	mov	%rbx, (%rsp)
Packit c4476c
	xor	%edi, %edi		# cf=0, of=0
Packit c4476c
Packit c4476c
	mulx	8($ap), %rax, %r9
Packit c4476c
Packit c4476c
	mulx	16($ap), %rbx, %r10
Packit c4476c
	adcx	%rax, %r8
Packit c4476c
Packit c4476c
	mulx	24($ap), %rax, %r11
Packit c4476c
	adcx	%rbx, %r9
Packit c4476c
Packit c4476c
	mulx	32($ap), %rbx, %r12
Packit c4476c
	adcx	%rax, %r10
Packit c4476c
Packit c4476c
	mulx	40($ap), %rax, %r13
Packit c4476c
	adcx	%rbx, %r11
Packit c4476c
Packit c4476c
	mulx	48($ap), %rbx, %r14
Packit c4476c
	adcx	%rax, %r12
Packit c4476c
Packit c4476c
	mulx	56($ap), %rax, %r15
Packit c4476c
	adcx	%rbx, %r13
Packit c4476c
	adcx	%rax, %r14
Packit c4476c
	.byte	0x67
Packit c4476c
	mov	%r8, %rbx
Packit c4476c
	adcx	%rdi, %r15		# %rdi is 0
Packit c4476c
Packit c4476c
	mov	\$-7, %rcx
Packit c4476c
	jmp	.Loop_mulx_gather
Packit c4476c
Packit c4476c
.align	32
Packit c4476c
.Loop_mulx_gather:
Packit c4476c
	movdqa	16*0(%rbp),%xmm8
Packit c4476c
	movdqa	16*1(%rbp),%xmm9
Packit c4476c
	movdqa	16*2(%rbp),%xmm10
Packit c4476c
	movdqa	16*3(%rbp),%xmm11
Packit c4476c
	pand	%xmm0,%xmm8
Packit c4476c
	movdqa	16*4(%rbp),%xmm12
Packit c4476c
	pand	%xmm1,%xmm9
Packit c4476c
	movdqa	16*5(%rbp),%xmm13
Packit c4476c
	pand	%xmm2,%xmm10
Packit c4476c
	movdqa	16*6(%rbp),%xmm14
Packit c4476c
	pand	%xmm3,%xmm11
Packit c4476c
	movdqa	16*7(%rbp),%xmm15
Packit c4476c
	leaq	128(%rbp), %rbp
Packit c4476c
	pand	%xmm4,%xmm12
Packit c4476c
	pand	%xmm5,%xmm13
Packit c4476c
	pand	%xmm6,%xmm14
Packit c4476c
	pand	%xmm7,%xmm15
Packit c4476c
	por	%xmm10,%xmm8
Packit c4476c
	por	%xmm11,%xmm9
Packit c4476c
	por	%xmm12,%xmm8
Packit c4476c
	por	%xmm13,%xmm9
Packit c4476c
	por	%xmm14,%xmm8
Packit c4476c
	por	%xmm15,%xmm9
Packit c4476c
Packit c4476c
	por	%xmm9,%xmm8
Packit c4476c
	pshufd	\$0x4e,%xmm8,%xmm9
Packit c4476c
	por	%xmm9,%xmm8
Packit c4476c
	movq	%xmm8,%rdx
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xfb,0xf6,0x86,0x00,0x00,0x00,0x00	# mulx	($ap), %rax, %r8
Packit c4476c
	adcx	%rax, %rbx
Packit c4476c
	adox	%r9, %r8
Packit c4476c
Packit c4476c
	mulx	8($ap), %rax, %r9
Packit c4476c
	adcx	%rax, %r8
Packit c4476c
	adox	%r10, %r9
Packit c4476c
Packit c4476c
	mulx	16($ap), %rax, %r10
Packit c4476c
	adcx	%rax, %r9
Packit c4476c
	adox	%r11, %r10
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xfb,0xf6,0x9e,0x18,0x00,0x00,0x00	# mulx	24($ap), %rax, %r11
Packit c4476c
	adcx	%rax, %r10
Packit c4476c
	adox	%r12, %r11
Packit c4476c
Packit c4476c
	mulx	32($ap), %rax, %r12
Packit c4476c
	adcx	%rax, %r11
Packit c4476c
	adox	%r13, %r12
Packit c4476c
Packit c4476c
	mulx	40($ap), %rax, %r13
Packit c4476c
	adcx	%rax, %r12
Packit c4476c
	adox	%r14, %r13
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xfb,0xf6,0xb6,0x30,0x00,0x00,0x00	# mulx	48($ap), %rax, %r14
Packit c4476c
	adcx	%rax, %r13
Packit c4476c
	.byte	0x67
Packit c4476c
	adox	%r15, %r14
Packit c4476c
Packit c4476c
	mulx	56($ap), %rax, %r15
Packit c4476c
	 mov	%rbx, 64(%rsp,%rcx,8)
Packit c4476c
	adcx	%rax, %r14
Packit c4476c
	adox	%rdi, %r15
Packit c4476c
	mov	%r8, %rbx
Packit c4476c
	adcx	%rdi, %r15		# cf=0
Packit c4476c
Packit c4476c
	inc	%rcx			# of=0
Packit c4476c
	jnz	.Loop_mulx_gather
Packit c4476c
Packit c4476c
	mov	%r8, 64(%rsp)
Packit c4476c
	mov	%r9, 64+8(%rsp)
Packit c4476c
	mov	%r10, 64+16(%rsp)
Packit c4476c
	mov	%r11, 64+24(%rsp)
Packit c4476c
	mov	%r12, 64+32(%rsp)
Packit c4476c
	mov	%r13, 64+40(%rsp)
Packit c4476c
	mov	%r14, 64+48(%rsp)
Packit c4476c
	mov	%r15, 64+56(%rsp)
Packit c4476c
Packit c4476c
	mov	128(%rsp), %rdx		# pull arguments
Packit c4476c
	mov	128+8(%rsp), $out
Packit c4476c
	mov	128+16(%rsp), %rbp
Packit c4476c
Packit c4476c
	mov	(%rsp), %r8
Packit c4476c
	mov	8(%rsp), %r9
Packit c4476c
	mov	16(%rsp), %r10
Packit c4476c
	mov	24(%rsp), %r11
Packit c4476c
	mov	32(%rsp), %r12
Packit c4476c
	mov	40(%rsp), %r13
Packit c4476c
	mov	48(%rsp), %r14
Packit c4476c
	mov	56(%rsp), %r15
Packit c4476c
Packit c4476c
	call	__rsaz_512_reducex
Packit c4476c
Packit c4476c
.Lmul_gather_tail:
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
	addq	64(%rsp), %r8
Packit c4476c
	adcq	72(%rsp), %r9
Packit c4476c
	adcq	80(%rsp), %r10
Packit c4476c
	adcq	88(%rsp), %r11
Packit c4476c
	adcq	96(%rsp), %r12
Packit c4476c
	adcq	104(%rsp), %r13
Packit c4476c
	adcq	112(%rsp), %r14
Packit c4476c
	adcq	120(%rsp), %r15
Packit c4476c
	sbbq	%rcx, %rcx
Packit c4476c
Packit c4476c
	call	__rsaz_512_subtract
Packit c4476c
Packit c4476c
	leaq	128+24+48(%rsp), %rax
Packit c4476c
___
Packit c4476c
$code.=<<___	if ($win64);
Packit c4476c
	movaps	0xa0-0xc8(%rax),%xmm6
Packit c4476c
	movaps	0xb0-0xc8(%rax),%xmm7
Packit c4476c
	movaps	0xc0-0xc8(%rax),%xmm8
Packit c4476c
	movaps	0xd0-0xc8(%rax),%xmm9
Packit c4476c
	movaps	0xe0-0xc8(%rax),%xmm10
Packit c4476c
	movaps	0xf0-0xc8(%rax),%xmm11
Packit c4476c
	movaps	0x100-0xc8(%rax),%xmm12
Packit c4476c
	movaps	0x110-0xc8(%rax),%xmm13
Packit c4476c
	movaps	0x120-0xc8(%rax),%xmm14
Packit c4476c
	movaps	0x130-0xc8(%rax),%xmm15
Packit c4476c
	lea	0xb0(%rax),%rax
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
.cfi_def_cfa	%rax,8
Packit c4476c
	movq	-48(%rax), %r15
Packit c4476c
.cfi_restore	%r15
Packit c4476c
	movq	-40(%rax), %r14
Packit c4476c
.cfi_restore	%r14
Packit c4476c
	movq	-32(%rax), %r13
Packit c4476c
.cfi_restore	%r13
Packit c4476c
	movq	-24(%rax), %r12
Packit c4476c
.cfi_restore	%r12
Packit c4476c
	movq	-16(%rax), %rbp
Packit c4476c
.cfi_restore	%rbp
Packit c4476c
	movq	-8(%rax), %rbx
Packit c4476c
.cfi_restore	%rbx
Packit c4476c
	leaq	(%rax), %rsp
Packit c4476c
.cfi_def_cfa_register	%rsp
Packit c4476c
.Lmul_gather4_epilogue:
Packit c4476c
	ret
Packit c4476c
.cfi_endproc
Packit c4476c
.size	rsaz_512_mul_gather4,.-rsaz_512_mul_gather4
Packit c4476c
___
Packit c4476c
}
Packit c4476c
{
Packit c4476c
my ($out,$ap,$mod,$n0,$tbl,$pwr) = ("%rdi","%rsi","%rdx","%rcx","%r8","%r9d");
Packit c4476c
$code.=<<___;
Packit c4476c
.globl	rsaz_512_mul_scatter4
Packit c4476c
.type	rsaz_512_mul_scatter4,\@function,6
Packit c4476c
.align	32
Packit c4476c
rsaz_512_mul_scatter4:
Packit c4476c
.cfi_startproc
Packit c4476c
	push	%rbx
Packit c4476c
.cfi_push	%rbx
Packit c4476c
	push	%rbp
Packit c4476c
.cfi_push	%rbp
Packit c4476c
	push	%r12
Packit c4476c
.cfi_push	%r12
Packit c4476c
	push	%r13
Packit c4476c
.cfi_push	%r13
Packit c4476c
	push	%r14
Packit c4476c
.cfi_push	%r14
Packit c4476c
	push	%r15
Packit c4476c
.cfi_push	%r15
Packit c4476c
Packit c4476c
	mov	$pwr, $pwr
Packit c4476c
	subq	\$128+24, %rsp
Packit c4476c
.cfi_adjust_cfa_offset	128+24
Packit c4476c
.Lmul_scatter4_body:
Packit c4476c
	leaq	($tbl,$pwr,8), $tbl
Packit c4476c
	movq	$out, %xmm0		# off-load arguments
Packit c4476c
	movq	$mod, %xmm1
Packit c4476c
	movq	$tbl, %xmm2
Packit c4476c
	movq	$n0, 128(%rsp)
Packit c4476c
Packit c4476c
	movq	$out, %rbp
Packit c4476c
___
Packit c4476c
$code.=<<___ if ($addx);
Packit c4476c
	movl	\$0x80100,%r11d
Packit c4476c
	andl	OPENSSL_ia32cap_P+8(%rip),%r11d
Packit c4476c
	cmpl	\$0x80100,%r11d		# check for MULX and ADO/CX
Packit c4476c
	je	.Lmulx_scatter
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
	movq	($out),%rbx		# pass b[0]
Packit c4476c
	call	__rsaz_512_mul
Packit c4476c
Packit c4476c
	movq	%xmm0, $out
Packit c4476c
	movq	%xmm1, %rbp
Packit c4476c
Packit c4476c
	movq	(%rsp), %r8
Packit c4476c
	movq	8(%rsp), %r9
Packit c4476c
	movq	16(%rsp), %r10
Packit c4476c
	movq	24(%rsp), %r11
Packit c4476c
	movq	32(%rsp), %r12
Packit c4476c
	movq	40(%rsp), %r13
Packit c4476c
	movq	48(%rsp), %r14
Packit c4476c
	movq	56(%rsp), %r15
Packit c4476c
Packit c4476c
	call	__rsaz_512_reduce
Packit c4476c
___
Packit c4476c
$code.=<<___ if ($addx);
Packit c4476c
	jmp	.Lmul_scatter_tail
Packit c4476c
Packit c4476c
.align	32
Packit c4476c
.Lmulx_scatter:
Packit c4476c
	movq	($out), %rdx		# pass b[0]
Packit c4476c
	call	__rsaz_512_mulx
Packit c4476c
Packit c4476c
	movq	%xmm0, $out
Packit c4476c
	movq	%xmm1, %rbp
Packit c4476c
Packit c4476c
	movq	128(%rsp), %rdx		# pull $n0
Packit c4476c
	movq	(%rsp), %r8
Packit c4476c
	movq	8(%rsp), %r9
Packit c4476c
	movq	16(%rsp), %r10
Packit c4476c
	movq	24(%rsp), %r11
Packit c4476c
	movq	32(%rsp), %r12
Packit c4476c
	movq	40(%rsp), %r13
Packit c4476c
	movq	48(%rsp), %r14
Packit c4476c
	movq	56(%rsp), %r15
Packit c4476c
Packit c4476c
	call	__rsaz_512_reducex
Packit c4476c
Packit c4476c
.Lmul_scatter_tail:
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
	addq	64(%rsp), %r8
Packit c4476c
	adcq	72(%rsp), %r9
Packit c4476c
	adcq	80(%rsp), %r10
Packit c4476c
	adcq	88(%rsp), %r11
Packit c4476c
	adcq	96(%rsp), %r12
Packit c4476c
	adcq	104(%rsp), %r13
Packit c4476c
	adcq	112(%rsp), %r14
Packit c4476c
	adcq	120(%rsp), %r15
Packit c4476c
	movq	%xmm2, $inp
Packit c4476c
	sbbq	%rcx, %rcx
Packit c4476c
Packit c4476c
	call	__rsaz_512_subtract
Packit c4476c
Packit c4476c
	movq	%r8, 128*0($inp)	# scatter
Packit c4476c
	movq	%r9, 128*1($inp)
Packit c4476c
	movq	%r10, 128*2($inp)
Packit c4476c
	movq	%r11, 128*3($inp)
Packit c4476c
	movq	%r12, 128*4($inp)
Packit c4476c
	movq	%r13, 128*5($inp)
Packit c4476c
	movq	%r14, 128*6($inp)
Packit c4476c
	movq	%r15, 128*7($inp)
Packit c4476c
Packit c4476c
	leaq	128+24+48(%rsp), %rax
Packit c4476c
.cfi_def_cfa	%rax,8
Packit c4476c
	movq	-48(%rax), %r15
Packit c4476c
.cfi_restore	%r15
Packit c4476c
	movq	-40(%rax), %r14
Packit c4476c
.cfi_restore	%r14
Packit c4476c
	movq	-32(%rax), %r13
Packit c4476c
.cfi_restore	%r13
Packit c4476c
	movq	-24(%rax), %r12
Packit c4476c
.cfi_restore	%r12
Packit c4476c
	movq	-16(%rax), %rbp
Packit c4476c
.cfi_restore	%rbp
Packit c4476c
	movq	-8(%rax), %rbx
Packit c4476c
.cfi_restore	%rbx
Packit c4476c
	leaq	(%rax), %rsp
Packit c4476c
.cfi_def_cfa_register	%rsp
Packit c4476c
.Lmul_scatter4_epilogue:
Packit c4476c
	ret
Packit c4476c
.cfi_endproc
Packit c4476c
.size	rsaz_512_mul_scatter4,.-rsaz_512_mul_scatter4
Packit c4476c
___
Packit c4476c
}
Packit c4476c
{
Packit c4476c
my ($out,$inp,$mod,$n0) = ("%rdi","%rsi","%rdx","%rcx");
Packit c4476c
$code.=<<___;
Packit c4476c
.globl	rsaz_512_mul_by_one
Packit c4476c
.type	rsaz_512_mul_by_one,\@function,4
Packit c4476c
.align	32
Packit c4476c
rsaz_512_mul_by_one:
Packit c4476c
.cfi_startproc
Packit c4476c
	push	%rbx
Packit c4476c
.cfi_push	%rbx
Packit c4476c
	push	%rbp
Packit c4476c
.cfi_push	%rbp
Packit c4476c
	push	%r12
Packit c4476c
.cfi_push	%r12
Packit c4476c
	push	%r13
Packit c4476c
.cfi_push	%r13
Packit c4476c
	push	%r14
Packit c4476c
.cfi_push	%r14
Packit c4476c
	push	%r15
Packit c4476c
.cfi_push	%r15
Packit c4476c
Packit c4476c
	subq	\$128+24, %rsp
Packit c4476c
.cfi_adjust_cfa_offset	128+24
Packit c4476c
.Lmul_by_one_body:
Packit c4476c
___
Packit c4476c
$code.=<<___ if ($addx);
Packit c4476c
	movl	OPENSSL_ia32cap_P+8(%rip),%eax
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
	movq	$mod, %rbp	# reassign argument
Packit c4476c
	movq	$n0, 128(%rsp)
Packit c4476c
Packit c4476c
	movq	($inp), %r8
Packit c4476c
	pxor	%xmm0, %xmm0
Packit c4476c
	movq	8($inp), %r9
Packit c4476c
	movq	16($inp), %r10
Packit c4476c
	movq	24($inp), %r11
Packit c4476c
	movq	32($inp), %r12
Packit c4476c
	movq	40($inp), %r13
Packit c4476c
	movq	48($inp), %r14
Packit c4476c
	movq	56($inp), %r15
Packit c4476c
Packit c4476c
	movdqa	%xmm0, (%rsp)
Packit c4476c
	movdqa	%xmm0, 16(%rsp)
Packit c4476c
	movdqa	%xmm0, 32(%rsp)
Packit c4476c
	movdqa	%xmm0, 48(%rsp)
Packit c4476c
	movdqa	%xmm0, 64(%rsp)
Packit c4476c
	movdqa	%xmm0, 80(%rsp)
Packit c4476c
	movdqa	%xmm0, 96(%rsp)
Packit c4476c
___
Packit c4476c
$code.=<<___ if ($addx);
Packit c4476c
	andl	\$0x80100,%eax
Packit c4476c
	cmpl	\$0x80100,%eax		# check for MULX and ADO/CX
Packit c4476c
	je	.Lby_one_callx
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
	call	__rsaz_512_reduce
Packit c4476c
___
Packit c4476c
$code.=<<___ if ($addx);
Packit c4476c
	jmp	.Lby_one_tail
Packit c4476c
.align	32
Packit c4476c
.Lby_one_callx:
Packit c4476c
	movq	128(%rsp), %rdx		# pull $n0
Packit c4476c
	call	__rsaz_512_reducex
Packit c4476c
.Lby_one_tail:
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
	movq	%r8, ($out)
Packit c4476c
	movq	%r9, 8($out)
Packit c4476c
	movq	%r10, 16($out)
Packit c4476c
	movq	%r11, 24($out)
Packit c4476c
	movq	%r12, 32($out)
Packit c4476c
	movq	%r13, 40($out)
Packit c4476c
	movq	%r14, 48($out)
Packit c4476c
	movq	%r15, 56($out)
Packit c4476c
Packit c4476c
	leaq	128+24+48(%rsp), %rax
Packit c4476c
.cfi_def_cfa	%rax,8
Packit c4476c
	movq	-48(%rax), %r15
Packit c4476c
.cfi_restore	%r15
Packit c4476c
	movq	-40(%rax), %r14
Packit c4476c
.cfi_restore	%r14
Packit c4476c
	movq	-32(%rax), %r13
Packit c4476c
.cfi_restore	%r13
Packit c4476c
	movq	-24(%rax), %r12
Packit c4476c
.cfi_restore	%r12
Packit c4476c
	movq	-16(%rax), %rbp
Packit c4476c
.cfi_restore	%rbp
Packit c4476c
	movq	-8(%rax), %rbx
Packit c4476c
.cfi_restore	%rbx
Packit c4476c
	leaq	(%rax), %rsp
Packit c4476c
.cfi_def_cfa_register	%rsp
Packit c4476c
.Lmul_by_one_epilogue:
Packit c4476c
	ret
Packit c4476c
.cfi_endproc
Packit c4476c
.size	rsaz_512_mul_by_one,.-rsaz_512_mul_by_one
Packit c4476c
___
Packit c4476c
}
Packit c4476c
{	# __rsaz_512_reduce
Packit c4476c
	#
Packit c4476c
	# input:	%r8-%r15, %rbp - mod, 128(%rsp) - n0
Packit c4476c
	# output:	%r8-%r15
Packit c4476c
	# clobbers:	everything except %rbp and %rdi
Packit c4476c
$code.=<<___;
Packit c4476c
.type	__rsaz_512_reduce,\@abi-omnipotent
Packit c4476c
.align	32
Packit c4476c
__rsaz_512_reduce:
Packit c4476c
.cfi_startproc
Packit c4476c
	movq	%r8, %rbx
Packit c4476c
	imulq	128+8(%rsp), %rbx
Packit c4476c
	movq	0(%rbp), %rax
Packit c4476c
	movl	\$8, %ecx
Packit c4476c
	jmp	.Lreduction_loop
Packit c4476c
Packit c4476c
.align	32
Packit c4476c
.Lreduction_loop:
Packit c4476c
	mulq	%rbx
Packit c4476c
	movq	8(%rbp), %rax
Packit c4476c
	negq	%r8
Packit c4476c
	movq	%rdx, %r8
Packit c4476c
	adcq	\$0, %r8
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r9
Packit c4476c
	movq	16(%rbp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r9, %r8
Packit c4476c
	movq	%rdx, %r9
Packit c4476c
	adcq	\$0, %r9
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r10
Packit c4476c
	movq	24(%rbp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r10, %r9
Packit c4476c
	movq	%rdx, %r10
Packit c4476c
	adcq	\$0, %r10
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r11
Packit c4476c
	movq	32(%rbp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r11, %r10
Packit c4476c
	 movq	128+8(%rsp), %rsi
Packit c4476c
	#movq	%rdx, %r11
Packit c4476c
	#adcq	\$0, %r11
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	movq	%rdx, %r11
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r12
Packit c4476c
	movq	40(%rbp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	 imulq	%r8, %rsi
Packit c4476c
	addq	%r12, %r11
Packit c4476c
	movq	%rdx, %r12
Packit c4476c
	adcq	\$0, %r12
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r13
Packit c4476c
	movq	48(%rbp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r13, %r12
Packit c4476c
	movq	%rdx, %r13
Packit c4476c
	adcq	\$0, %r13
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r14
Packit c4476c
	movq	56(%rbp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r14, %r13
Packit c4476c
	movq	%rdx, %r14
Packit c4476c
	adcq	\$0, %r14
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	 movq	%rsi, %rbx
Packit c4476c
	addq	%rax, %r15
Packit c4476c
	 movq	0(%rbp), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r15, %r14
Packit c4476c
	movq	%rdx, %r15
Packit c4476c
	adcq	\$0, %r15
Packit c4476c
Packit c4476c
	decl	%ecx
Packit c4476c
	jne	.Lreduction_loop
Packit c4476c
Packit c4476c
	ret
Packit c4476c
.cfi_endproc
Packit c4476c
.size	__rsaz_512_reduce,.-__rsaz_512_reduce
Packit c4476c
___
Packit c4476c
}
Packit c4476c
if ($addx) {
Packit c4476c
	# __rsaz_512_reducex
Packit c4476c
	#
Packit c4476c
	# input:	%r8-%r15, %rbp - mod, 128(%rsp) - n0
Packit c4476c
	# output:	%r8-%r15
Packit c4476c
	# clobbers:	everything except %rbp and %rdi
Packit c4476c
$code.=<<___;
Packit c4476c
.type	__rsaz_512_reducex,\@abi-omnipotent
Packit c4476c
.align	32
Packit c4476c
__rsaz_512_reducex:
Packit c4476c
.cfi_startproc
Packit c4476c
	#movq	128+8(%rsp), %rdx		# pull $n0
Packit c4476c
	imulq	%r8, %rdx
Packit c4476c
	xorq	%rsi, %rsi			# cf=0,of=0
Packit c4476c
	movl	\$8, %ecx
Packit c4476c
	jmp	.Lreduction_loopx
Packit c4476c
Packit c4476c
.align	32
Packit c4476c
.Lreduction_loopx:
Packit c4476c
	mov	%r8, %rbx
Packit c4476c
	mulx	0(%rbp), %rax, %r8
Packit c4476c
	adcx	%rbx, %rax
Packit c4476c
	adox	%r9, %r8
Packit c4476c
Packit c4476c
	mulx	8(%rbp), %rax, %r9
Packit c4476c
	adcx	%rax, %r8
Packit c4476c
	adox	%r10, %r9
Packit c4476c
Packit c4476c
	mulx	16(%rbp), %rbx, %r10
Packit c4476c
	adcx	%rbx, %r9
Packit c4476c
	adox	%r11, %r10
Packit c4476c
Packit c4476c
	mulx	24(%rbp), %rbx, %r11
Packit c4476c
	adcx	%rbx, %r10
Packit c4476c
	adox	%r12, %r11
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xe3,0xf6,0xa5,0x20,0x00,0x00,0x00	# mulx	32(%rbp), %rbx, %r12
Packit c4476c
	 mov	%rdx, %rax
Packit c4476c
	 mov	%r8, %rdx
Packit c4476c
	adcx	%rbx, %r11
Packit c4476c
	adox	%r13, %r12
Packit c4476c
Packit c4476c
	 mulx	128+8(%rsp), %rbx, %rdx
Packit c4476c
	 mov	%rax, %rdx
Packit c4476c
Packit c4476c
	mulx	40(%rbp), %rax, %r13
Packit c4476c
	adcx	%rax, %r12
Packit c4476c
	adox	%r14, %r13
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xfb,0xf6,0xb5,0x30,0x00,0x00,0x00	# mulx	48(%rbp), %rax, %r14
Packit c4476c
	adcx	%rax, %r13
Packit c4476c
	adox	%r15, %r14
Packit c4476c
Packit c4476c
	mulx	56(%rbp), %rax, %r15
Packit c4476c
	 mov	%rbx, %rdx
Packit c4476c
	adcx	%rax, %r14
Packit c4476c
	adox	%rsi, %r15			# %rsi is 0
Packit c4476c
	adcx	%rsi, %r15			# cf=0
Packit c4476c
Packit c4476c
	decl	%ecx				# of=0
Packit c4476c
	jne	.Lreduction_loopx
Packit c4476c
Packit c4476c
	ret
Packit c4476c
.cfi_endproc
Packit c4476c
.size	__rsaz_512_reducex,.-__rsaz_512_reducex
Packit c4476c
___
Packit c4476c
}
Packit c4476c
{	# __rsaz_512_subtract
Packit c4476c
	# input: %r8-%r15, %rdi - $out, %rbp - $mod, %rcx - mask
Packit c4476c
	# output:
Packit c4476c
	# clobbers: everything but %rdi, %rsi and %rbp
Packit c4476c
$code.=<<___;
Packit c4476c
.type	__rsaz_512_subtract,\@abi-omnipotent
Packit c4476c
.align	32
Packit c4476c
__rsaz_512_subtract:
Packit c4476c
.cfi_startproc
Packit c4476c
	movq	%r8, ($out)
Packit c4476c
	movq	%r9, 8($out)
Packit c4476c
	movq	%r10, 16($out)
Packit c4476c
	movq	%r11, 24($out)
Packit c4476c
	movq	%r12, 32($out)
Packit c4476c
	movq	%r13, 40($out)
Packit c4476c
	movq	%r14, 48($out)
Packit c4476c
	movq	%r15, 56($out)
Packit c4476c
Packit c4476c
	movq	0($mod), %r8
Packit c4476c
	movq	8($mod), %r9
Packit c4476c
	negq	%r8
Packit c4476c
	notq	%r9
Packit c4476c
	andq	%rcx, %r8
Packit c4476c
	movq	16($mod), %r10
Packit c4476c
	andq	%rcx, %r9
Packit c4476c
	notq	%r10
Packit c4476c
	movq	24($mod), %r11
Packit c4476c
	andq	%rcx, %r10
Packit c4476c
	notq	%r11
Packit c4476c
	movq	32($mod), %r12
Packit c4476c
	andq	%rcx, %r11
Packit c4476c
	notq	%r12
Packit c4476c
	movq	40($mod), %r13
Packit c4476c
	andq	%rcx, %r12
Packit c4476c
	notq	%r13
Packit c4476c
	movq	48($mod), %r14
Packit c4476c
	andq	%rcx, %r13
Packit c4476c
	notq	%r14
Packit c4476c
	movq	56($mod), %r15
Packit c4476c
	andq	%rcx, %r14
Packit c4476c
	notq	%r15
Packit c4476c
	andq	%rcx, %r15
Packit c4476c
Packit c4476c
	addq	($out), %r8
Packit c4476c
	adcq	8($out), %r9
Packit c4476c
	adcq	16($out), %r10
Packit c4476c
	adcq	24($out), %r11
Packit c4476c
	adcq	32($out), %r12
Packit c4476c
	adcq	40($out), %r13
Packit c4476c
	adcq	48($out), %r14
Packit c4476c
	adcq	56($out), %r15
Packit c4476c
Packit c4476c
	movq	%r8, ($out)
Packit c4476c
	movq	%r9, 8($out)
Packit c4476c
	movq	%r10, 16($out)
Packit c4476c
	movq	%r11, 24($out)
Packit c4476c
	movq	%r12, 32($out)
Packit c4476c
	movq	%r13, 40($out)
Packit c4476c
	movq	%r14, 48($out)
Packit c4476c
	movq	%r15, 56($out)
Packit c4476c
Packit c4476c
	ret
Packit c4476c
.cfi_endproc
Packit c4476c
.size	__rsaz_512_subtract,.-__rsaz_512_subtract
Packit c4476c
___
Packit c4476c
}
Packit c4476c
{	# __rsaz_512_mul
Packit c4476c
	#
Packit c4476c
	# input: %rsi - ap, %rbp - bp
Packit c4476c
	# output:
Packit c4476c
	# clobbers: everything
Packit c4476c
my ($ap,$bp) = ("%rsi","%rbp");
Packit c4476c
$code.=<<___;
Packit c4476c
.type	__rsaz_512_mul,\@abi-omnipotent
Packit c4476c
.align	32
Packit c4476c
__rsaz_512_mul:
Packit c4476c
.cfi_startproc
Packit c4476c
	leaq	8(%rsp), %rdi
Packit c4476c
Packit c4476c
	movq	($ap), %rax
Packit c4476c
	mulq	%rbx
Packit c4476c
	movq	%rax, (%rdi)
Packit c4476c
	movq	8($ap), %rax
Packit c4476c
	movq	%rdx, %r8
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r8
Packit c4476c
	movq	16($ap), %rax
Packit c4476c
	movq	%rdx, %r9
Packit c4476c
	adcq	\$0, %r9
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r9
Packit c4476c
	movq	24($ap), %rax
Packit c4476c
	movq	%rdx, %r10
Packit c4476c
	adcq	\$0, %r10
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r10
Packit c4476c
	movq	32($ap), %rax
Packit c4476c
	movq	%rdx, %r11
Packit c4476c
	adcq	\$0, %r11
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r11
Packit c4476c
	movq	40($ap), %rax
Packit c4476c
	movq	%rdx, %r12
Packit c4476c
	adcq	\$0, %r12
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r12
Packit c4476c
	movq	48($ap), %rax
Packit c4476c
	movq	%rdx, %r13
Packit c4476c
	adcq	\$0, %r13
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r13
Packit c4476c
	movq	56($ap), %rax
Packit c4476c
	movq	%rdx, %r14
Packit c4476c
	adcq	\$0, %r14
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r14
Packit c4476c
	 movq	($ap), %rax
Packit c4476c
	movq	%rdx, %r15
Packit c4476c
	adcq	\$0, %r15
Packit c4476c
Packit c4476c
	leaq	8($bp), $bp
Packit c4476c
	leaq	8(%rdi), %rdi
Packit c4476c
Packit c4476c
	movl	\$7, %ecx
Packit c4476c
	jmp	.Loop_mul
Packit c4476c
Packit c4476c
.align	32
Packit c4476c
.Loop_mul:
Packit c4476c
	movq	($bp), %rbx
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r8
Packit c4476c
	movq	8($ap), %rax
Packit c4476c
	movq	%r8, (%rdi)
Packit c4476c
	movq	%rdx, %r8
Packit c4476c
	adcq	\$0, %r8
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r9
Packit c4476c
	movq	16($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r9, %r8
Packit c4476c
	movq	%rdx, %r9
Packit c4476c
	adcq	\$0, %r9
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r10
Packit c4476c
	movq	24($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r10, %r9
Packit c4476c
	movq	%rdx, %r10
Packit c4476c
	adcq	\$0, %r10
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r11
Packit c4476c
	movq	32($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r11, %r10
Packit c4476c
	movq	%rdx, %r11
Packit c4476c
	adcq	\$0, %r11
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r12
Packit c4476c
	movq	40($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r12, %r11
Packit c4476c
	movq	%rdx, %r12
Packit c4476c
	adcq	\$0, %r12
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r13
Packit c4476c
	movq	48($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r13, %r12
Packit c4476c
	movq	%rdx, %r13
Packit c4476c
	adcq	\$0, %r13
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r14
Packit c4476c
	movq	56($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r14, %r13
Packit c4476c
	movq	%rdx, %r14
Packit c4476c
	 leaq	8($bp), $bp
Packit c4476c
	adcq	\$0, %r14
Packit c4476c
Packit c4476c
	mulq	%rbx
Packit c4476c
	addq	%rax, %r15
Packit c4476c
	 movq	($ap), %rax
Packit c4476c
	adcq	\$0, %rdx
Packit c4476c
	addq	%r15, %r14
Packit c4476c
	movq	%rdx, %r15
Packit c4476c
	adcq	\$0, %r15
Packit c4476c
Packit c4476c
	leaq	8(%rdi), %rdi
Packit c4476c
Packit c4476c
	decl	%ecx
Packit c4476c
	jnz	.Loop_mul
Packit c4476c
Packit c4476c
	movq	%r8, (%rdi)
Packit c4476c
	movq	%r9, 8(%rdi)
Packit c4476c
	movq	%r10, 16(%rdi)
Packit c4476c
	movq	%r11, 24(%rdi)
Packit c4476c
	movq	%r12, 32(%rdi)
Packit c4476c
	movq	%r13, 40(%rdi)
Packit c4476c
	movq	%r14, 48(%rdi)
Packit c4476c
	movq	%r15, 56(%rdi)
Packit c4476c
Packit c4476c
	ret
Packit c4476c
.cfi_endproc
Packit c4476c
.size	__rsaz_512_mul,.-__rsaz_512_mul
Packit c4476c
___
Packit c4476c
}
Packit c4476c
if ($addx) {
Packit c4476c
	# __rsaz_512_mulx
Packit c4476c
	#
Packit c4476c
	# input: %rsi - ap, %rbp - bp
Packit c4476c
	# output:
Packit c4476c
	# clobbers: everything
Packit c4476c
my ($ap,$bp,$zero) = ("%rsi","%rbp","%rdi");
Packit c4476c
$code.=<<___;
Packit c4476c
.type	__rsaz_512_mulx,\@abi-omnipotent
Packit c4476c
.align	32
Packit c4476c
__rsaz_512_mulx:
Packit c4476c
.cfi_startproc
Packit c4476c
	mulx	($ap), %rbx, %r8	# initial %rdx preloaded by caller
Packit c4476c
	mov	\$-6, %rcx
Packit c4476c
Packit c4476c
	mulx	8($ap), %rax, %r9
Packit c4476c
	movq	%rbx, 8(%rsp)
Packit c4476c
Packit c4476c
	mulx	16($ap), %rbx, %r10
Packit c4476c
	adc	%rax, %r8
Packit c4476c
Packit c4476c
	mulx	24($ap), %rax, %r11
Packit c4476c
	adc	%rbx, %r9
Packit c4476c
Packit c4476c
	mulx	32($ap), %rbx, %r12
Packit c4476c
	adc	%rax, %r10
Packit c4476c
Packit c4476c
	mulx	40($ap), %rax, %r13
Packit c4476c
	adc	%rbx, %r11
Packit c4476c
Packit c4476c
	mulx	48($ap), %rbx, %r14
Packit c4476c
	adc	%rax, %r12
Packit c4476c
Packit c4476c
	mulx	56($ap), %rax, %r15
Packit c4476c
	 mov	8($bp), %rdx
Packit c4476c
	adc	%rbx, %r13
Packit c4476c
	adc	%rax, %r14
Packit c4476c
	adc	\$0, %r15
Packit c4476c
Packit c4476c
	xor	$zero, $zero		# cf=0,of=0
Packit c4476c
	jmp	.Loop_mulx
Packit c4476c
Packit c4476c
.align	32
Packit c4476c
.Loop_mulx:
Packit c4476c
	movq	%r8, %rbx
Packit c4476c
	mulx	($ap), %rax, %r8
Packit c4476c
	adcx	%rax, %rbx
Packit c4476c
	adox	%r9, %r8
Packit c4476c
Packit c4476c
	mulx	8($ap), %rax, %r9
Packit c4476c
	adcx	%rax, %r8
Packit c4476c
	adox	%r10, %r9
Packit c4476c
Packit c4476c
	mulx	16($ap), %rax, %r10
Packit c4476c
	adcx	%rax, %r9
Packit c4476c
	adox	%r11, %r10
Packit c4476c
Packit c4476c
	mulx	24($ap), %rax, %r11
Packit c4476c
	adcx	%rax, %r10
Packit c4476c
	adox	%r12, %r11
Packit c4476c
Packit c4476c
	.byte	0x3e,0xc4,0x62,0xfb,0xf6,0xa6,0x20,0x00,0x00,0x00	# mulx	32($ap), %rax, %r12
Packit c4476c
	adcx	%rax, %r11
Packit c4476c
	adox	%r13, %r12
Packit c4476c
Packit c4476c
	mulx	40($ap), %rax, %r13
Packit c4476c
	adcx	%rax, %r12
Packit c4476c
	adox	%r14, %r13
Packit c4476c
Packit c4476c
	mulx	48($ap), %rax, %r14
Packit c4476c
	adcx	%rax, %r13
Packit c4476c
	adox	%r15, %r14
Packit c4476c
Packit c4476c
	mulx	56($ap), %rax, %r15
Packit c4476c
	 movq	64($bp,%rcx,8), %rdx
Packit c4476c
	 movq	%rbx, 8+64-8(%rsp,%rcx,8)
Packit c4476c
	adcx	%rax, %r14
Packit c4476c
	adox	$zero, %r15
Packit c4476c
	adcx	$zero, %r15		# cf=0
Packit c4476c
Packit c4476c
	inc	%rcx			# of=0
Packit c4476c
	jnz	.Loop_mulx
Packit c4476c
Packit c4476c
	movq	%r8, %rbx
Packit c4476c
	mulx	($ap), %rax, %r8
Packit c4476c
	adcx	%rax, %rbx
Packit c4476c
	adox	%r9, %r8
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xfb,0xf6,0x8e,0x08,0x00,0x00,0x00	# mulx	8($ap), %rax, %r9
Packit c4476c
	adcx	%rax, %r8
Packit c4476c
	adox	%r10, %r9
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xfb,0xf6,0x96,0x10,0x00,0x00,0x00	# mulx	16($ap), %rax, %r10
Packit c4476c
	adcx	%rax, %r9
Packit c4476c
	adox	%r11, %r10
Packit c4476c
Packit c4476c
	mulx	24($ap), %rax, %r11
Packit c4476c
	adcx	%rax, %r10
Packit c4476c
	adox	%r12, %r11
Packit c4476c
Packit c4476c
	mulx	32($ap), %rax, %r12
Packit c4476c
	adcx	%rax, %r11
Packit c4476c
	adox	%r13, %r12
Packit c4476c
Packit c4476c
	mulx	40($ap), %rax, %r13
Packit c4476c
	adcx	%rax, %r12
Packit c4476c
	adox	%r14, %r13
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xfb,0xf6,0xb6,0x30,0x00,0x00,0x00	# mulx	48($ap), %rax, %r14
Packit c4476c
	adcx	%rax, %r13
Packit c4476c
	adox	%r15, %r14
Packit c4476c
Packit c4476c
	.byte	0xc4,0x62,0xfb,0xf6,0xbe,0x38,0x00,0x00,0x00	# mulx	56($ap), %rax, %r15
Packit c4476c
	adcx	%rax, %r14
Packit c4476c
	adox	$zero, %r15
Packit c4476c
	adcx	$zero, %r15
Packit c4476c
Packit c4476c
	mov	%rbx, 8+64-8(%rsp)
Packit c4476c
	mov	%r8, 8+64(%rsp)
Packit c4476c
	mov	%r9, 8+64+8(%rsp)
Packit c4476c
	mov	%r10, 8+64+16(%rsp)
Packit c4476c
	mov	%r11, 8+64+24(%rsp)
Packit c4476c
	mov	%r12, 8+64+32(%rsp)
Packit c4476c
	mov	%r13, 8+64+40(%rsp)
Packit c4476c
	mov	%r14, 8+64+48(%rsp)
Packit c4476c
	mov	%r15, 8+64+56(%rsp)
Packit c4476c
Packit c4476c
	ret
Packit c4476c
.cfi_endproc
Packit c4476c
.size	__rsaz_512_mulx,.-__rsaz_512_mulx
Packit c4476c
___
Packit c4476c
}
Packit c4476c
{
Packit c4476c
my ($out,$inp,$power)= $win64 ? ("%rcx","%rdx","%r8d") : ("%rdi","%rsi","%edx");
Packit c4476c
$code.=<<___;
Packit c4476c
.globl	rsaz_512_scatter4
Packit c4476c
.type	rsaz_512_scatter4,\@abi-omnipotent
Packit c4476c
.align	16
Packit c4476c
rsaz_512_scatter4:
Packit c4476c
.cfi_startproc
Packit c4476c
	leaq	($out,$power,8), $out
Packit c4476c
	movl	\$8, %r9d
Packit c4476c
	jmp	.Loop_scatter
Packit c4476c
.align	16
Packit c4476c
.Loop_scatter:
Packit c4476c
	movq	($inp), %rax
Packit c4476c
	leaq	8($inp), $inp
Packit c4476c
	movq	%rax, ($out)
Packit c4476c
	leaq	128($out), $out
Packit c4476c
	decl	%r9d
Packit c4476c
	jnz	.Loop_scatter
Packit c4476c
	ret
Packit c4476c
.cfi_endproc
Packit c4476c
.size	rsaz_512_scatter4,.-rsaz_512_scatter4
Packit c4476c
Packit c4476c
.globl	rsaz_512_gather4
Packit c4476c
.type	rsaz_512_gather4,\@abi-omnipotent
Packit c4476c
.align	16
Packit c4476c
rsaz_512_gather4:
Packit c4476c
.cfi_startproc
Packit c4476c
___
Packit c4476c
$code.=<<___	if ($win64);
Packit c4476c
.LSEH_begin_rsaz_512_gather4:
Packit c4476c
	.byte	0x48,0x81,0xec,0xa8,0x00,0x00,0x00	# sub    $0xa8,%rsp
Packit c4476c
	.byte	0x0f,0x29,0x34,0x24			# movaps %xmm6,(%rsp)
Packit c4476c
	.byte	0x0f,0x29,0x7c,0x24,0x10		# movaps %xmm7,0x10(%rsp)
Packit c4476c
	.byte	0x44,0x0f,0x29,0x44,0x24,0x20		# movaps %xmm8,0x20(%rsp)
Packit c4476c
	.byte	0x44,0x0f,0x29,0x4c,0x24,0x30		# movaps %xmm9,0x30(%rsp)
Packit c4476c
	.byte	0x44,0x0f,0x29,0x54,0x24,0x40		# movaps %xmm10,0x40(%rsp)
Packit c4476c
	.byte	0x44,0x0f,0x29,0x5c,0x24,0x50		# movaps %xmm11,0x50(%rsp)
Packit c4476c
	.byte	0x44,0x0f,0x29,0x64,0x24,0x60		# movaps %xmm12,0x60(%rsp)
Packit c4476c
	.byte	0x44,0x0f,0x29,0x6c,0x24,0x70		# movaps %xmm13,0x70(%rsp)
Packit c4476c
	.byte	0x44,0x0f,0x29,0xb4,0x24,0x80,0,0,0	# movaps %xmm14,0x80(%rsp)
Packit c4476c
	.byte	0x44,0x0f,0x29,0xbc,0x24,0x90,0,0,0	# movaps %xmm15,0x90(%rsp)
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
	movd	$power,%xmm8
Packit c4476c
	movdqa	.Linc+16(%rip),%xmm1	# 00000002000000020000000200000002
Packit c4476c
	movdqa	.Linc(%rip),%xmm0	# 00000001000000010000000000000000
Packit c4476c
Packit c4476c
	pshufd	\$0,%xmm8,%xmm8		# broadcast $power
Packit c4476c
	movdqa	%xmm1,%xmm7
Packit c4476c
	movdqa	%xmm1,%xmm2
Packit c4476c
___
Packit c4476c
########################################################################
Packit c4476c
# calculate mask by comparing 0..15 to $power
Packit c4476c
#
Packit c4476c
for($i=0;$i<4;$i++) {
Packit c4476c
$code.=<<___;
Packit c4476c
	paddd	%xmm`$i`,%xmm`$i+1`
Packit c4476c
	pcmpeqd	%xmm8,%xmm`$i`
Packit c4476c
	movdqa	%xmm7,%xmm`$i+3`
Packit c4476c
___
Packit c4476c
}
Packit c4476c
for(;$i<7;$i++) {
Packit c4476c
$code.=<<___;
Packit c4476c
	paddd	%xmm`$i`,%xmm`$i+1`
Packit c4476c
	pcmpeqd	%xmm8,%xmm`$i`
Packit c4476c
___
Packit c4476c
}
Packit c4476c
$code.=<<___;
Packit c4476c
	pcmpeqd	%xmm8,%xmm7
Packit c4476c
	movl	\$8, %r9d
Packit c4476c
	jmp	.Loop_gather
Packit c4476c
.align	16
Packit c4476c
.Loop_gather:
Packit c4476c
	movdqa	16*0($inp),%xmm8
Packit c4476c
	movdqa	16*1($inp),%xmm9
Packit c4476c
	movdqa	16*2($inp),%xmm10
Packit c4476c
	movdqa	16*3($inp),%xmm11
Packit c4476c
	pand	%xmm0,%xmm8
Packit c4476c
	movdqa	16*4($inp),%xmm12
Packit c4476c
	pand	%xmm1,%xmm9
Packit c4476c
	movdqa	16*5($inp),%xmm13
Packit c4476c
	pand	%xmm2,%xmm10
Packit c4476c
	movdqa	16*6($inp),%xmm14
Packit c4476c
	pand	%xmm3,%xmm11
Packit c4476c
	movdqa	16*7($inp),%xmm15
Packit c4476c
	leaq	128($inp), $inp
Packit c4476c
	pand	%xmm4,%xmm12
Packit c4476c
	pand	%xmm5,%xmm13
Packit c4476c
	pand	%xmm6,%xmm14
Packit c4476c
	pand	%xmm7,%xmm15
Packit c4476c
	por	%xmm10,%xmm8
Packit c4476c
	por	%xmm11,%xmm9
Packit c4476c
	por	%xmm12,%xmm8
Packit c4476c
	por	%xmm13,%xmm9
Packit c4476c
	por	%xmm14,%xmm8
Packit c4476c
	por	%xmm15,%xmm9
Packit c4476c
Packit c4476c
	por	%xmm9,%xmm8
Packit c4476c
	pshufd	\$0x4e,%xmm8,%xmm9
Packit c4476c
	por	%xmm9,%xmm8
Packit c4476c
	movq	%xmm8,($out)
Packit c4476c
	leaq	8($out), $out
Packit c4476c
	decl	%r9d
Packit c4476c
	jnz	.Loop_gather
Packit c4476c
___
Packit c4476c
$code.=<<___	if ($win64);
Packit c4476c
	movaps	0x00(%rsp),%xmm6
Packit c4476c
	movaps	0x10(%rsp),%xmm7
Packit c4476c
	movaps	0x20(%rsp),%xmm8
Packit c4476c
	movaps	0x30(%rsp),%xmm9
Packit c4476c
	movaps	0x40(%rsp),%xmm10
Packit c4476c
	movaps	0x50(%rsp),%xmm11
Packit c4476c
	movaps	0x60(%rsp),%xmm12
Packit c4476c
	movaps	0x70(%rsp),%xmm13
Packit c4476c
	movaps	0x80(%rsp),%xmm14
Packit c4476c
	movaps	0x90(%rsp),%xmm15
Packit c4476c
	add	\$0xa8,%rsp
Packit c4476c
___
Packit c4476c
$code.=<<___;
Packit c4476c
	ret
Packit c4476c
.LSEH_end_rsaz_512_gather4:
Packit c4476c
.cfi_endproc
Packit c4476c
.size	rsaz_512_gather4,.-rsaz_512_gather4
Packit c4476c
Packit c4476c
.align	64
Packit c4476c
.Linc:
Packit c4476c
	.long	0,0, 1,1
Packit c4476c
	.long	2,2, 2,2
Packit c4476c
___
Packit c4476c
}
Packit c4476c
Packit c4476c
# EXCEPTION_DISPOSITION handler (EXCEPTION_RECORD *rec,ULONG64 frame,
Packit c4476c
#		CONTEXT *context,DISPATCHER_CONTEXT *disp)
Packit c4476c
if ($win64) {
Packit c4476c
$rec="%rcx";
Packit c4476c
$frame="%rdx";
Packit c4476c
$context="%r8";
Packit c4476c
$disp="%r9";
Packit c4476c
Packit c4476c
$code.=<<___;
Packit c4476c
.extern	__imp_RtlVirtualUnwind
Packit c4476c
.type	se_handler,\@abi-omnipotent
Packit c4476c
.align	16
Packit c4476c
se_handler:
Packit c4476c
	push	%rsi
Packit c4476c
	push	%rdi
Packit c4476c
	push	%rbx
Packit c4476c
	push	%rbp
Packit c4476c
	push	%r12
Packit c4476c
	push	%r13
Packit c4476c
	push	%r14
Packit c4476c
	push	%r15
Packit c4476c
	pushfq
Packit c4476c
	sub	\$64,%rsp
Packit c4476c
Packit c4476c
	mov	120($context),%rax	# pull context->Rax
Packit c4476c
	mov	248($context),%rbx	# pull context->Rip
Packit c4476c
Packit c4476c
	mov	8($disp),%rsi		# disp->ImageBase
Packit c4476c
	mov	56($disp),%r11		# disp->HandlerData
Packit c4476c
Packit c4476c
	mov	0(%r11),%r10d		# HandlerData[0]
Packit c4476c
	lea	(%rsi,%r10),%r10	# end of prologue label
Packit c4476c
	cmp	%r10,%rbx		# context->Rip
Packit c4476c
	jb	.Lcommon_seh_tail
Packit c4476c
Packit c4476c
	mov	152($context),%rax	# pull context->Rsp
Packit c4476c
Packit c4476c
	mov	4(%r11),%r10d		# HandlerData[1]
Packit c4476c
	lea	(%rsi,%r10),%r10	# epilogue label
Packit c4476c
	cmp	%r10,%rbx		# context->Rip>=epilogue label
Packit c4476c
	jae	.Lcommon_seh_tail
Packit c4476c
Packit c4476c
	lea	128+24+48(%rax),%rax
Packit c4476c
Packit c4476c
	lea	.Lmul_gather4_epilogue(%rip),%rbx
Packit c4476c
	cmp	%r10,%rbx
Packit c4476c
	jne	.Lse_not_in_mul_gather4
Packit c4476c
Packit c4476c
	lea	0xb0(%rax),%rax
Packit c4476c
Packit c4476c
	lea	-48-0xa8(%rax),%rsi
Packit c4476c
	lea	512($context),%rdi
Packit c4476c
	mov	\$20,%ecx
Packit c4476c
	.long	0xa548f3fc		# cld; rep movsq
Packit c4476c
Packit c4476c
.Lse_not_in_mul_gather4:
Packit c4476c
	mov	-8(%rax),%rbx
Packit c4476c
	mov	-16(%rax),%rbp
Packit c4476c
	mov	-24(%rax),%r12
Packit c4476c
	mov	-32(%rax),%r13
Packit c4476c
	mov	-40(%rax),%r14
Packit c4476c
	mov	-48(%rax),%r15
Packit c4476c
	mov	%rbx,144($context)	# restore context->Rbx
Packit c4476c
	mov	%rbp,160($context)	# restore context->Rbp
Packit c4476c
	mov	%r12,216($context)	# restore context->R12
Packit c4476c
	mov	%r13,224($context)	# restore context->R13
Packit c4476c
	mov	%r14,232($context)	# restore context->R14
Packit c4476c
	mov	%r15,240($context)	# restore context->R15
Packit c4476c
Packit c4476c
.Lcommon_seh_tail:
Packit c4476c
	mov	8(%rax),%rdi
Packit c4476c
	mov	16(%rax),%rsi
Packit c4476c
	mov	%rax,152($context)	# restore context->Rsp
Packit c4476c
	mov	%rsi,168($context)	# restore context->Rsi
Packit c4476c
	mov	%rdi,176($context)	# restore context->Rdi
Packit c4476c
Packit c4476c
	mov	40($disp),%rdi		# disp->ContextRecord
Packit c4476c
	mov	$context,%rsi		# context
Packit c4476c
	mov	\$154,%ecx		# sizeof(CONTEXT)
Packit c4476c
	.long	0xa548f3fc		# cld; rep movsq
Packit c4476c
Packit c4476c
	mov	$disp,%rsi
Packit c4476c
	xor	%rcx,%rcx		# arg1, UNW_FLAG_NHANDLER
Packit c4476c
	mov	8(%rsi),%rdx		# arg2, disp->ImageBase
Packit c4476c
	mov	0(%rsi),%r8		# arg3, disp->ControlPc
Packit c4476c
	mov	16(%rsi),%r9		# arg4, disp->FunctionEntry
Packit c4476c
	mov	40(%rsi),%r10		# disp->ContextRecord
Packit c4476c
	lea	56(%rsi),%r11		# &disp->HandlerData
Packit c4476c
	lea	24(%rsi),%r12		# &disp->EstablisherFrame
Packit c4476c
	mov	%r10,32(%rsp)		# arg5
Packit c4476c
	mov	%r11,40(%rsp)		# arg6
Packit c4476c
	mov	%r12,48(%rsp)		# arg7
Packit c4476c
	mov	%rcx,56(%rsp)		# arg8, (NULL)
Packit c4476c
	call	*__imp_RtlVirtualUnwind(%rip)
Packit c4476c
Packit c4476c
	mov	\$1,%eax		# ExceptionContinueSearch
Packit c4476c
	add	\$64,%rsp
Packit c4476c
	popfq
Packit c4476c
	pop	%r15
Packit c4476c
	pop	%r14
Packit c4476c
	pop	%r13
Packit c4476c
	pop	%r12
Packit c4476c
	pop	%rbp
Packit c4476c
	pop	%rbx
Packit c4476c
	pop	%rdi
Packit c4476c
	pop	%rsi
Packit c4476c
	ret
Packit c4476c
.size	se_handler,.-se_handler
Packit c4476c
Packit c4476c
.section	.pdata
Packit c4476c
.align	4
Packit c4476c
	.rva	.LSEH_begin_rsaz_512_sqr
Packit c4476c
	.rva	.LSEH_end_rsaz_512_sqr
Packit c4476c
	.rva	.LSEH_info_rsaz_512_sqr
Packit c4476c
Packit c4476c
	.rva	.LSEH_begin_rsaz_512_mul
Packit c4476c
	.rva	.LSEH_end_rsaz_512_mul
Packit c4476c
	.rva	.LSEH_info_rsaz_512_mul
Packit c4476c
Packit c4476c
	.rva	.LSEH_begin_rsaz_512_mul_gather4
Packit c4476c
	.rva	.LSEH_end_rsaz_512_mul_gather4
Packit c4476c
	.rva	.LSEH_info_rsaz_512_mul_gather4
Packit c4476c
Packit c4476c
	.rva	.LSEH_begin_rsaz_512_mul_scatter4
Packit c4476c
	.rva	.LSEH_end_rsaz_512_mul_scatter4
Packit c4476c
	.rva	.LSEH_info_rsaz_512_mul_scatter4
Packit c4476c
Packit c4476c
	.rva	.LSEH_begin_rsaz_512_mul_by_one
Packit c4476c
	.rva	.LSEH_end_rsaz_512_mul_by_one
Packit c4476c
	.rva	.LSEH_info_rsaz_512_mul_by_one
Packit c4476c
Packit c4476c
	.rva	.LSEH_begin_rsaz_512_gather4
Packit c4476c
	.rva	.LSEH_end_rsaz_512_gather4
Packit c4476c
	.rva	.LSEH_info_rsaz_512_gather4
Packit c4476c
Packit c4476c
.section	.xdata
Packit c4476c
.align	8
Packit c4476c
.LSEH_info_rsaz_512_sqr:
Packit c4476c
	.byte	9,0,0,0
Packit c4476c
	.rva	se_handler
Packit c4476c
	.rva	.Lsqr_body,.Lsqr_epilogue			# HandlerData[]
Packit c4476c
.LSEH_info_rsaz_512_mul:
Packit c4476c
	.byte	9,0,0,0
Packit c4476c
	.rva	se_handler
Packit c4476c
	.rva	.Lmul_body,.Lmul_epilogue			# HandlerData[]
Packit c4476c
.LSEH_info_rsaz_512_mul_gather4:
Packit c4476c
	.byte	9,0,0,0
Packit c4476c
	.rva	se_handler
Packit c4476c
	.rva	.Lmul_gather4_body,.Lmul_gather4_epilogue	# HandlerData[]
Packit c4476c
.LSEH_info_rsaz_512_mul_scatter4:
Packit c4476c
	.byte	9,0,0,0
Packit c4476c
	.rva	se_handler
Packit c4476c
	.rva	.Lmul_scatter4_body,.Lmul_scatter4_epilogue	# HandlerData[]
Packit c4476c
.LSEH_info_rsaz_512_mul_by_one:
Packit c4476c
	.byte	9,0,0,0
Packit c4476c
	.rva	se_handler
Packit c4476c
	.rva	.Lmul_by_one_body,.Lmul_by_one_epilogue		# HandlerData[]
Packit c4476c
.LSEH_info_rsaz_512_gather4:
Packit c4476c
	.byte	0x01,0x46,0x16,0x00
Packit c4476c
	.byte	0x46,0xf8,0x09,0x00	# vmovaps 0x90(rsp),xmm15
Packit c4476c
	.byte	0x3d,0xe8,0x08,0x00	# vmovaps 0x80(rsp),xmm14
Packit c4476c
	.byte	0x34,0xd8,0x07,0x00	# vmovaps 0x70(rsp),xmm13
Packit c4476c
	.byte	0x2e,0xc8,0x06,0x00	# vmovaps 0x60(rsp),xmm12
Packit c4476c
	.byte	0x28,0xb8,0x05,0x00	# vmovaps 0x50(rsp),xmm11
Packit c4476c
	.byte	0x22,0xa8,0x04,0x00	# vmovaps 0x40(rsp),xmm10
Packit c4476c
	.byte	0x1c,0x98,0x03,0x00	# vmovaps 0x30(rsp),xmm9
Packit c4476c
	.byte	0x16,0x88,0x02,0x00	# vmovaps 0x20(rsp),xmm8
Packit c4476c
	.byte	0x10,0x78,0x01,0x00	# vmovaps 0x10(rsp),xmm7
Packit c4476c
	.byte	0x0b,0x68,0x00,0x00	# vmovaps 0x00(rsp),xmm6
Packit c4476c
	.byte	0x07,0x01,0x15,0x00	# sub     rsp,0xa8
Packit c4476c
___
Packit c4476c
}
Packit c4476c
Packit c4476c
$code =~ s/\`([^\`]*)\`/eval $1/gem;
Packit c4476c
print $code;
Packit c4476c
close STDOUT or die "error closing STDOUT: $!";