6f0c4db90e
The C implementation is still our existing C implementation, but slightly tweaked to fit with upstream's init/block/emits convention. I've tested this by looking at code coverage in kcachegrind and valgrind --tool=callgrind --dump-instr=yes --collect-jumps=yes (NB: valgrind 3.11.0 is needed for AVX2. And even that only does 64-bit AVX2, so we can't get coverage for the 32-bit code yet. But I had to disable that anyway.) This was paired with a hacked up version of poly1305_test that would repeat tests with different ia32cap and armcap values. This isn't checked in, but we badly need a story for testing all the different variants. I'm not happy with upstream's code in either the C/asm boundary or how it dispatches between different versions, but just debugging the code has been a significant time investment. I'd hoped to extract the SIMD parts and do the rest in C, but I think we need to focus on testing first (and use that to guide what modifications would help). For now, this version seems to work at least. The x86 (not x86_64) AVX2 code needs to be disabled because it's broken. It also seems pretty unnecessary. https://rt.openssl.org/Ticket/Display.html?id=4346 Otherwise it seems to work and buys us a decent performance improvement. Notably, my Nexus 6P is finally faster at ChaCha20-Poly1305 than my Nexus 4! bssl speed numbers follow: x86 --- Old: Did 1554000 ChaCha20-Poly1305 (16 bytes) seal operations in 1000536us (1553167.5 ops/sec): 24.9 MB/s Did 136000 ChaCha20-Poly1305 (1350 bytes) seal operations in 1003947us (135465.3 ops/sec): 182.9 MB/s Did 30000 ChaCha20-Poly1305 (8192 bytes) seal operations in 1022990us (29325.8 ops/sec): 240.2 MB/s Did 1888000 ChaCha20-Poly1305-Old (16 bytes) seal operations in 1000206us (1887611.2 ops/sec): 30.2 MB/s Did 173000 ChaCha20-Poly1305-Old (1350 bytes) seal operations in 1003036us (172476.4 ops/sec): 232.8 MB/s Did 30000 ChaCha20-Poly1305-Old (8192 bytes) seal operations in 1027759us (29189.7 ops/sec): 239.1 MB/s New: Did 2030000 ChaCha20-Poly1305 (16 bytes) seal operations in 1000507us (2028971.3 ops/sec): 32.5 MB/s Did 404000 ChaCha20-Poly1305 (1350 bytes) seal operations in 1000287us (403884.1 ops/sec): 545.2 MB/s Did 83000 ChaCha20-Poly1305 (8192 bytes) seal operations in 1001258us (82895.7 ops/sec): 679.1 MB/s Did 2018000 ChaCha20-Poly1305-Old (16 bytes) seal operations in 1000006us (2017987.9 ops/sec): 32.3 MB/s Did 360000 ChaCha20-Poly1305-Old (1350 bytes) seal operations in 1001962us (359295.1 ops/sec): 485.0 MB/s Did 85000 ChaCha20-Poly1305-Old (8192 bytes) seal operations in 1002479us (84789.8 ops/sec): 694.6 MB/s x86_64, no AVX2 --- Old: Did 2023000 ChaCha20-Poly1305 (16 bytes) seal operations in 1000258us (2022478.2 ops/sec): 32.4 MB/s Did 466000 ChaCha20-Poly1305 (1350 bytes) seal operations in 1002619us (464782.7 ops/sec): 627.5 MB/s Did 90000 ChaCha20-Poly1305 (8192 bytes) seal operations in 1001133us (89898.1 ops/sec): 736.4 MB/s Did 2238000 ChaCha20-Poly1305-Old (16 bytes) seal operations in 1000175us (2237608.4 ops/sec): 35.8 MB/s Did 483000 ChaCha20-Poly1305-Old (1350 bytes) seal operations in 1001348us (482349.8 ops/sec): 651.2 MB/s Did 90000 ChaCha20-Poly1305-Old (8192 bytes) seal operations in 1003141us (89718.2 ops/sec): 735.0 MB/s New: Did 2558000 ChaCha20-Poly1305 (16 bytes) seal operations in 1000275us (2557296.7 ops/sec): 40.9 MB/s Did 510000 ChaCha20-Poly1305 (1350 bytes) seal operations in 1001810us (509078.6 ops/sec): 687.3 MB/s Did 115000 ChaCha20-Poly1305 (8192 bytes) seal operations in 1006457us (114262.2 ops/sec): 936.0 MB/s Did 2818000 ChaCha20-Poly1305-Old (16 bytes) seal operations in 1000187us (2817473.1 ops/sec): 45.1 MB/s Did 418000 ChaCha20-Poly1305-Old (1350 bytes) seal operations in 1001140us (417524.0 ops/sec): 563.7 MB/s Did 91000 ChaCha20-Poly1305-Old (8192 bytes) seal operations in 1002539us (90769.5 ops/sec): 743.6 MB/s x86_64, AVX2 --- Old: Did 2516000 ChaCha20-Poly1305 (16 bytes) seal operations in 1000115us (2515710.7 ops/sec): 40.3 MB/s Did 774000 ChaCha20-Poly1305 (1350 bytes) seal operations in 1000300us (773767.9 ops/sec): 1044.6 MB/s Did 171000 ChaCha20-Poly1305 (8192 bytes) seal operations in 1004373us (170255.5 ops/sec): 1394.7 MB/s Did 2580000 ChaCha20-Poly1305-Old (16 bytes) seal operations in 1000144us (2579628.5 ops/sec): 41.3 MB/s Did 769000 ChaCha20-Poly1305-Old (1350 bytes) seal operations in 1000472us (768637.2 ops/sec): 1037.7 MB/s Did 169000 ChaCha20-Poly1305-Old (8192 bytes) seal operations in 1000320us (168945.9 ops/sec): 1384.0 MB/s New: Did 3240000 ChaCha20-Poly1305 (16 bytes) seal operations in 1000114us (3239630.7 ops/sec): 51.8 MB/s Did 932000 ChaCha20-Poly1305 (1350 bytes) seal operations in 1000059us (931945.0 ops/sec): 1258.1 MB/s Did 217000 ChaCha20-Poly1305 (8192 bytes) seal operations in 1003282us (216290.1 ops/sec): 1771.8 MB/s Did 3187000 ChaCha20-Poly1305-Old (16 bytes) seal operations in 1000100us (3186681.3 ops/sec): 51.0 MB/s Did 926000 ChaCha20-Poly1305-Old (1350 bytes) seal operations in 1000071us (925934.3 ops/sec): 1250.0 MB/s Did 215000 ChaCha20-Poly1305-Old (8192 bytes) seal operations in 1000479us (214897.1 ops/sec): 1760.4 MB/s arm, Nexus 4 --- Old: Did 430248 ChaCha20-Poly1305 (16 bytes) seal operations in 1000153us (430182.2 ops/sec): 6.9 MB/s Did 115250 ChaCha20-Poly1305 (1350 bytes) seal operations in 1000549us (115186.8 ops/sec): 155.5 MB/s Did 27000 ChaCha20-Poly1305 (8192 bytes) seal operations in 1030124us (26210.4 ops/sec): 214.7 MB/s Did 451750 ChaCha20-Poly1305-Old (16 bytes) seal operations in 1000549us (451502.1 ops/sec): 7.2 MB/s Did 118000 ChaCha20-Poly1305-Old (1350 bytes) seal operations in 1001557us (117816.6 ops/sec): 159.1 MB/s Did 27000 ChaCha20-Poly1305-Old (8192 bytes) seal operations in 1024263us (26360.4 ops/sec): 215.9 MB/s New: Did 553644 ChaCha20-Poly1305 (16 bytes) seal operations in 1000183us (553542.7 ops/sec): 8.9 MB/s Did 126000 ChaCha20-Poly1305 (1350 bytes) seal operations in 1000396us (125950.1 ops/sec): 170.0 MB/s Did 27000 ChaCha20-Poly1305 (8192 bytes) seal operations in 1000336us (26990.9 ops/sec): 221.1 MB/s Did 559000 ChaCha20-Poly1305-Old (16 bytes) seal operations in 1001465us (558182.3 ops/sec): 8.9 MB/s Did 124000 ChaCha20-Poly1305-Old (1350 bytes) seal operations in 1000824us (123897.9 ops/sec): 167.3 MB/s Did 28000 ChaCha20-Poly1305-Old (8192 bytes) seal operations in 1034854us (27057.0 ops/sec): 221.7 MB/s aarch64, Nexus 6P --- Old: Did 358000 ChaCha20-Poly1305 (16 bytes) seal operations in 1000358us (357871.9 ops/sec): 5.7 MB/s Did 45000 ChaCha20-Poly1305 (1350 bytes) seal operations in 1022386us (44014.7 ops/sec): 59.4 MB/s Did 8657 ChaCha20-Poly1305 (8192 bytes) seal operations in 1063722us (8138.4 ops/sec): 66.7 MB/s Did 350000 ChaCha20-Poly1305-Old (16 bytes) seal operations in 1000074us (349974.1 ops/sec): 5.6 MB/s Did 44000 ChaCha20-Poly1305-Old (1350 bytes) seal operations in 1007907us (43654.8 ops/sec): 58.9 MB/s Did 8525 ChaCha20-Poly1305-Old (8192 bytes) seal operations in 1042644us (8176.3 ops/sec): 67.0 MB/s New: Did 713000 ChaCha20-Poly1305 (16 bytes) seal operations in 1000190us (712864.6 ops/sec): 11.4 MB/s Did 180000 ChaCha20-Poly1305 (1350 bytes) seal operations in 1004249us (179238.4 ops/sec): 242.0 MB/s Did 41000 ChaCha20-Poly1305 (8192 bytes) seal operations in 1005811us (40763.1 ops/sec): 333.9 MB/s Did 775000 ChaCha20-Poly1305-Old (16 bytes) seal operations in 1000719us (774443.2 ops/sec): 12.4 MB/s Did 182000 ChaCha20-Poly1305-Old (1350 bytes) seal operations in 1003529us (181360.0 ops/sec): 244.8 MB/s Did 41000 ChaCha20-Poly1305-Old (8192 bytes) seal operations in 1010576us (40570.9 ops/sec): 332.4 MB/s Change-Id: Iaa4ab86ac1174b79833077963cc3616cfb08e686 Reviewed-on: https://boringssl-review.googlesource.com/7226 Reviewed-by: Adam Langley <agl@google.com>
1217 lines
27 KiB
Prolog
Executable File
1217 lines
27 KiB
Prolog
Executable File
#!/usr/bin/env perl
|
|
#
|
|
# ====================================================================
|
|
# Written by Andy Polyakov <appro@openssl.org> for the OpenSSL
|
|
# project. The module is, however, dual licensed under OpenSSL and
|
|
# CRYPTOGAMS licenses depending on where you obtain it. For further
|
|
# details see http://www.openssl.org/~appro/cryptogams/.
|
|
# ====================================================================
|
|
#
|
|
# IALU(*)/gcc-4.4 NEON
|
|
#
|
|
# ARM11xx(ARMv6) 7.78/+100% -
|
|
# Cortex-A5 6.30/+130% 2.96
|
|
# Cortex-A8 6.25/+115% 2.36
|
|
# Cortex-A9 5.10/+95% 2.55
|
|
# Cortex-A15 3.79/+85% 1.25(**)
|
|
# Snapdragon S4 5.70/+100% 1.48(**)
|
|
#
|
|
# (*) this is for -march=armv6, i.e. with bunch of ldrb loading data;
|
|
# (**) these are trade-off results, they can be improved by ~8% but at
|
|
# the cost of 15/12% regression on Cortex-A5/A7, it's even possible
|
|
# to improve Cortex-A9 result, but then A5/A7 loose more than 20%;
|
|
|
|
$flavour = shift;
|
|
if ($flavour=~/^\w[\w\-]*\.\w+$/) { $output=$flavour; undef $flavour; }
|
|
else { while (($output=shift) && ($output!~/^\w[\w\-]*\.\w+$/)) {} }
|
|
|
|
if ($flavour && $flavour ne "void") {
|
|
$0 =~ m/(.*[\/\\])[^\/\\]+$/; $dir=$1;
|
|
( $xlate="${dir}arm-xlate.pl" and -f $xlate ) or
|
|
( $xlate="${dir}../../perlasm/arm-xlate.pl" and -f $xlate) or
|
|
die "can't locate arm-xlate.pl";
|
|
|
|
open STDOUT,"| \"$^X\" $xlate $flavour $output";
|
|
} else {
|
|
open STDOUT,">$output";
|
|
}
|
|
|
|
($ctx,$inp,$len,$padbit)=map("r$_",(0..3));
|
|
|
|
$code.=<<___;
|
|
#include <openssl/arm_arch.h>
|
|
|
|
.text
|
|
#if defined(__thumb2__)
|
|
.syntax unified
|
|
.thumb
|
|
#else
|
|
.code 32
|
|
#endif
|
|
|
|
.globl poly1305_emit
|
|
.globl poly1305_blocks
|
|
.globl poly1305_init
|
|
.type poly1305_init,%function
|
|
.align 5
|
|
poly1305_init:
|
|
.Lpoly1305_init:
|
|
stmdb sp!,{r4-r11}
|
|
|
|
eor r3,r3,r3
|
|
cmp $inp,#0
|
|
str r3,[$ctx,#0] @ zero hash value
|
|
str r3,[$ctx,#4]
|
|
str r3,[$ctx,#8]
|
|
str r3,[$ctx,#12]
|
|
str r3,[$ctx,#16]
|
|
str r3,[$ctx,#36] @ is_base2_26
|
|
add $ctx,$ctx,#20
|
|
|
|
#ifdef __thumb2__
|
|
it eq
|
|
#endif
|
|
moveq r0,#0
|
|
beq .Lno_key
|
|
|
|
#if __ARM_MAX_ARCH__>=7
|
|
adr r11,.Lpoly1305_init
|
|
ldr r12,.LOPENSSL_armcap
|
|
#endif
|
|
ldrb r4,[$inp,#0]
|
|
mov r10,#0x0fffffff
|
|
ldrb r5,[$inp,#1]
|
|
and r3,r10,#-4 @ 0x0ffffffc
|
|
ldrb r6,[$inp,#2]
|
|
ldrb r7,[$inp,#3]
|
|
orr r4,r4,r5,lsl#8
|
|
ldrb r5,[$inp,#4]
|
|
orr r4,r4,r6,lsl#16
|
|
ldrb r6,[$inp,#5]
|
|
orr r4,r4,r7,lsl#24
|
|
ldrb r7,[$inp,#6]
|
|
and r4,r4,r10
|
|
|
|
#if __ARM_MAX_ARCH__>=7
|
|
ldr r12,[r11,r12] @ OPENSSL_armcap_P
|
|
# ifdef __APPLE__
|
|
ldr r12,[r12]
|
|
# endif
|
|
#endif
|
|
ldrb r8,[$inp,#7]
|
|
orr r5,r5,r6,lsl#8
|
|
ldrb r6,[$inp,#8]
|
|
orr r5,r5,r7,lsl#16
|
|
ldrb r7,[$inp,#9]
|
|
orr r5,r5,r8,lsl#24
|
|
ldrb r8,[$inp,#10]
|
|
and r5,r5,r3
|
|
|
|
#if __ARM_MAX_ARCH__>=7
|
|
tst r12,#ARMV7_NEON @ check for NEON
|
|
# ifdef __APPLE__
|
|
adr r9,poly1305_blocks_neon
|
|
adr r11,poly1305_blocks
|
|
# ifdef __thumb2__
|
|
it ne
|
|
# endif
|
|
movne r11,r9
|
|
adr r12,poly1305_emit
|
|
adr r10,poly1305_emit_neon
|
|
# ifdef __thumb2__
|
|
it ne
|
|
# endif
|
|
movne r12,r10
|
|
# else
|
|
# ifdef __thumb2__
|
|
itete eq
|
|
# endif
|
|
addeq r12,r11,#(poly1305_emit-.Lpoly1305_init)
|
|
addne r12,r11,#(poly1305_emit_neon-.Lpoly1305_init)
|
|
addeq r11,r11,#(poly1305_blocks-.Lpoly1305_init)
|
|
addne r11,r11,#(poly1305_blocks_neon-.Lpoly1305_init)
|
|
# endif
|
|
# ifdef __thumb2__
|
|
orr r12,r12,#1 @ thumb-ify address
|
|
orr r11,r11,#1
|
|
# endif
|
|
#endif
|
|
ldrb r9,[$inp,#11]
|
|
orr r6,r6,r7,lsl#8
|
|
ldrb r7,[$inp,#12]
|
|
orr r6,r6,r8,lsl#16
|
|
ldrb r8,[$inp,#13]
|
|
orr r6,r6,r9,lsl#24
|
|
ldrb r9,[$inp,#14]
|
|
and r6,r6,r3
|
|
|
|
ldrb r10,[$inp,#15]
|
|
orr r7,r7,r8,lsl#8
|
|
str r4,[$ctx,#0]
|
|
orr r7,r7,r9,lsl#16
|
|
str r5,[$ctx,#4]
|
|
orr r7,r7,r10,lsl#24
|
|
str r6,[$ctx,#8]
|
|
and r7,r7,r3
|
|
str r7,[$ctx,#12]
|
|
#if __ARM_MAX_ARCH__>=7
|
|
stmia r2,{r11,r12} @ fill functions table
|
|
mov r0,#1
|
|
#else
|
|
mov r0,#0
|
|
#endif
|
|
.Lno_key:
|
|
ldmia sp!,{r4-r11}
|
|
#if __ARM_ARCH__>=5
|
|
ret @ bx lr
|
|
#else
|
|
tst lr,#1
|
|
moveq pc,lr @ be binary compatible with V4, yet
|
|
bx lr @ interoperable with Thumb ISA:-)
|
|
#endif
|
|
.size poly1305_init,.-poly1305_init
|
|
___
|
|
{
|
|
my ($h0,$h1,$h2,$h3,$h4,$r0,$r1,$r2,$r3)=map("r$_",(4..12));
|
|
my ($s1,$s2,$s3)=($r1,$r2,$r3);
|
|
|
|
$code.=<<___;
|
|
.type poly1305_blocks,%function
|
|
.align 5
|
|
poly1305_blocks:
|
|
stmdb sp!,{r3-r11,lr}
|
|
|
|
ands $len,$len,#-16
|
|
beq .Lno_data
|
|
|
|
cmp $padbit,#0
|
|
add $len,$len,$inp @ end pointer
|
|
sub sp,sp,#32
|
|
|
|
ldmia $ctx,{$h0-$r3} @ load context
|
|
|
|
str $ctx,[sp,#12] @ offload stuff
|
|
mov lr,$inp
|
|
str $len,[sp,#16]
|
|
str $r1,[sp,#20]
|
|
str $r2,[sp,#24]
|
|
str $r3,[sp,#28]
|
|
b .Loop
|
|
|
|
.Loop:
|
|
#if __ARM_ARCH__<7
|
|
ldrb r0,[lr],#16 @ load input
|
|
# ifdef __thumb2__
|
|
it hi
|
|
# endif
|
|
addhi $h4,$h4,#1 @ 1<<128
|
|
ldrb r1,[lr,#-15]
|
|
ldrb r2,[lr,#-14]
|
|
ldrb r3,[lr,#-13]
|
|
orr r1,r0,r1,lsl#8
|
|
ldrb r0,[lr,#-12]
|
|
orr r2,r1,r2,lsl#16
|
|
ldrb r1,[lr,#-11]
|
|
orr r3,r2,r3,lsl#24
|
|
ldrb r2,[lr,#-10]
|
|
adds $h0,$h0,r3 @ accumulate input
|
|
|
|
ldrb r3,[lr,#-9]
|
|
orr r1,r0,r1,lsl#8
|
|
ldrb r0,[lr,#-8]
|
|
orr r2,r1,r2,lsl#16
|
|
ldrb r1,[lr,#-7]
|
|
orr r3,r2,r3,lsl#24
|
|
ldrb r2,[lr,#-6]
|
|
adcs $h1,$h1,r3
|
|
|
|
ldrb r3,[lr,#-5]
|
|
orr r1,r0,r1,lsl#8
|
|
ldrb r0,[lr,#-4]
|
|
orr r2,r1,r2,lsl#16
|
|
ldrb r1,[lr,#-3]
|
|
orr r3,r2,r3,lsl#24
|
|
ldrb r2,[lr,#-2]
|
|
adcs $h2,$h2,r3
|
|
|
|
ldrb r3,[lr,#-1]
|
|
orr r1,r0,r1,lsl#8
|
|
str lr,[sp,#8] @ offload input pointer
|
|
orr r2,r1,r2,lsl#16
|
|
add $s1,$r1,$r1,lsr#2
|
|
orr r3,r2,r3,lsl#24
|
|
#else
|
|
ldr r0,[lr],#16 @ load input
|
|
# ifdef __thumb2__
|
|
it hi
|
|
# endif
|
|
addhi $h4,$h4,#1 @ padbit
|
|
ldr r1,[lr,#-12]
|
|
ldr r2,[lr,#-8]
|
|
ldr r3,[lr,#-4]
|
|
# ifdef __ARMEB__
|
|
rev r0,r0
|
|
rev r1,r1
|
|
rev r2,r2
|
|
rev r3,r3
|
|
# endif
|
|
adds $h0,$h0,r0 @ accumulate input
|
|
str lr,[sp,#8] @ offload input pointer
|
|
adcs $h1,$h1,r1
|
|
add $s1,$r1,$r1,lsr#2
|
|
adcs $h2,$h2,r2
|
|
#endif
|
|
add $s2,$r2,$r2,lsr#2
|
|
adcs $h3,$h3,r3
|
|
add $s3,$r3,$r3,lsr#2
|
|
|
|
umull r2,r3,$h1,$r0
|
|
adc $h4,$h4,#0
|
|
umull r0,r1,$h0,$r0
|
|
umlal r2,r3,$h4,$s1
|
|
umlal r0,r1,$h3,$s1
|
|
ldr $r1,[sp,#20] @ reload $r1
|
|
umlal r2,r3,$h2,$s3
|
|
umlal r0,r1,$h1,$s3
|
|
umlal r2,r3,$h3,$s2
|
|
umlal r0,r1,$h2,$s2
|
|
umlal r2,r3,$h0,$r1
|
|
str r0,[sp,#0] @ future $h0
|
|
mul r0,$s2,$h4
|
|
ldr $r2,[sp,#24] @ reload $r2
|
|
adds r2,r2,r1 @ d1+=d0>>32
|
|
eor r1,r1,r1
|
|
adc lr,r3,#0 @ future $h2
|
|
str r2,[sp,#4] @ future $h1
|
|
|
|
mul r2,$s3,$h4
|
|
eor r3,r3,r3
|
|
umlal r0,r1,$h3,$s3
|
|
ldr $r3,[sp,#28] @ reload $r3
|
|
umlal r2,r3,$h3,$r0
|
|
umlal r0,r1,$h2,$r0
|
|
umlal r2,r3,$h2,$r1
|
|
umlal r0,r1,$h1,$r1
|
|
umlal r2,r3,$h1,$r2
|
|
umlal r0,r1,$h0,$r2
|
|
umlal r2,r3,$h0,$r3
|
|
ldr $h0,[sp,#0]
|
|
mul $h4,$r0,$h4
|
|
ldr $h1,[sp,#4]
|
|
|
|
adds $h2,lr,r0 @ d2+=d1>>32
|
|
ldr lr,[sp,#8] @ reload input pointer
|
|
adc r1,r1,#0
|
|
adds $h3,r2,r1 @ d3+=d2>>32
|
|
ldr r0,[sp,#16] @ reload end pointer
|
|
adc r3,r3,#0
|
|
add $h4,$h4,r3 @ h4+=d3>>32
|
|
|
|
and r1,$h4,#-4
|
|
and $h4,$h4,#3
|
|
add r1,r1,r1,lsr#2 @ *=5
|
|
adds $h0,$h0,r1
|
|
adcs $h1,$h1,#0
|
|
adcs $h2,$h2,#0
|
|
adc $h3,$h3,#0
|
|
|
|
cmp r0,lr @ done yet?
|
|
bhi .Loop
|
|
|
|
ldr $ctx,[sp,#12]
|
|
add sp,sp,#32
|
|
stmia $ctx,{$h0-$h4} @ store the result
|
|
|
|
.Lno_data:
|
|
#if __ARM_ARCH__>=5
|
|
ldmia sp!,{r3-r11,pc}
|
|
#else
|
|
ldmia sp!,{r3-r11,lr}
|
|
tst lr,#1
|
|
moveq pc,lr @ be binary compatible with V4, yet
|
|
bx lr @ interoperable with Thumb ISA:-)
|
|
#endif
|
|
.size poly1305_blocks,.-poly1305_blocks
|
|
___
|
|
}
|
|
{
|
|
my ($ctx,$mac,$nonce)=map("r$_",(0..2));
|
|
my ($h0,$h1,$h2,$h3,$h4,$g0,$g1,$g2,$g3)=map("r$_",(3..11));
|
|
my $g4=$h4;
|
|
|
|
$code.=<<___;
|
|
.type poly1305_emit,%function
|
|
.align 5
|
|
poly1305_emit:
|
|
stmdb sp!,{r4-r11}
|
|
.Lpoly1305_emit_enter:
|
|
|
|
ldmia $ctx,{$h0-$h4}
|
|
adds $g0,$h0,#5 @ compare to modulus
|
|
adcs $g1,$h1,#0
|
|
adcs $g2,$h2,#0
|
|
adcs $g3,$h3,#0
|
|
adc $g4,$h4,#0
|
|
tst $g4,#4 @ did it carry/borrow?
|
|
|
|
#ifdef __thumb2__
|
|
it ne
|
|
#endif
|
|
movne $h0,$g0
|
|
ldr $g0,[$nonce,#0]
|
|
#ifdef __thumb2__
|
|
it ne
|
|
#endif
|
|
movne $h1,$g1
|
|
ldr $g1,[$nonce,#4]
|
|
#ifdef __thumb2__
|
|
it ne
|
|
#endif
|
|
movne $h2,$g2
|
|
ldr $g2,[$nonce,#8]
|
|
#ifdef __thumb2__
|
|
it ne
|
|
#endif
|
|
movne $h3,$g3
|
|
ldr $g3,[$nonce,#12]
|
|
|
|
adds $h0,$h0,$g0
|
|
adcs $h1,$h1,$g1
|
|
adcs $h2,$h2,$g2
|
|
adc $h3,$h3,$g3
|
|
|
|
#if __ARM_ARCH__>=7
|
|
# ifdef __ARMEB__
|
|
rev $h0,$h0
|
|
rev $h1,$h1
|
|
rev $h2,$h2
|
|
rev $h3,$h3
|
|
# endif
|
|
str $h0,[$mac,#0]
|
|
str $h1,[$mac,#4]
|
|
str $h2,[$mac,#8]
|
|
str $h3,[$mac,#12]
|
|
#else
|
|
strb $h0,[$mac,#0]
|
|
mov $h0,$h0,lsr#8
|
|
strb $h1,[$mac,#4]
|
|
mov $h1,$h1,lsr#8
|
|
strb $h2,[$mac,#8]
|
|
mov $h2,$h2,lsr#8
|
|
strb $h3,[$mac,#12]
|
|
mov $h3,$h3,lsr#8
|
|
|
|
strb $h0,[$mac,#1]
|
|
mov $h0,$h0,lsr#8
|
|
strb $h1,[$mac,#5]
|
|
mov $h1,$h1,lsr#8
|
|
strb $h2,[$mac,#9]
|
|
mov $h2,$h2,lsr#8
|
|
strb $h3,[$mac,#13]
|
|
mov $h3,$h3,lsr#8
|
|
|
|
strb $h0,[$mac,#2]
|
|
mov $h0,$h0,lsr#8
|
|
strb $h1,[$mac,#6]
|
|
mov $h1,$h1,lsr#8
|
|
strb $h2,[$mac,#10]
|
|
mov $h2,$h2,lsr#8
|
|
strb $h3,[$mac,#14]
|
|
mov $h3,$h3,lsr#8
|
|
|
|
strb $h0,[$mac,#3]
|
|
strb $h1,[$mac,#7]
|
|
strb $h2,[$mac,#11]
|
|
strb $h3,[$mac,#15]
|
|
#endif
|
|
ldmia sp!,{r4-r11}
|
|
#if __ARM_ARCH__>=5
|
|
ret @ bx lr
|
|
#else
|
|
tst lr,#1
|
|
moveq pc,lr @ be binary compatible with V4, yet
|
|
bx lr @ interoperable with Thumb ISA:-)
|
|
#endif
|
|
.size poly1305_emit,.-poly1305_emit
|
|
___
|
|
{
|
|
my ($R0,$R1,$S1,$R2,$S2,$R3,$S3,$R4,$S4) = map("d$_",(0..9));
|
|
my ($D0,$D1,$D2,$D3,$D4, $H0,$H1,$H2,$H3,$H4) = map("q$_",(5..14));
|
|
my ($T0,$T1,$MASK) = map("q$_",(15,4,0));
|
|
|
|
my ($in2,$zeros,$tbl0,$tbl1) = map("r$_",(4..7));
|
|
|
|
$code.=<<___;
|
|
#if __ARM_MAX_ARCH__>=7
|
|
.fpu neon
|
|
|
|
.type poly1305_init_neon,%function
|
|
.align 5
|
|
poly1305_init_neon:
|
|
ldr r4,[$ctx,#20] @ load key base 2^32
|
|
ldr r5,[$ctx,#24]
|
|
ldr r6,[$ctx,#28]
|
|
ldr r7,[$ctx,#32]
|
|
|
|
and r2,r4,#0x03ffffff @ base 2^32 -> base 2^26
|
|
mov r3,r4,lsr#26
|
|
mov r4,r5,lsr#20
|
|
orr r3,r3,r5,lsl#6
|
|
mov r5,r6,lsr#14
|
|
orr r4,r4,r6,lsl#12
|
|
mov r6,r7,lsr#8
|
|
orr r5,r5,r7,lsl#18
|
|
and r3,r3,#0x03ffffff
|
|
and r4,r4,#0x03ffffff
|
|
and r5,r5,#0x03ffffff
|
|
|
|
vdup.32 $R0,r2 @ r^1 in both lanes
|
|
add r2,r3,r3,lsl#2 @ *5
|
|
vdup.32 $R1,r3
|
|
add r3,r4,r4,lsl#2
|
|
vdup.32 $S1,r2
|
|
vdup.32 $R2,r4
|
|
add r4,r5,r5,lsl#2
|
|
vdup.32 $S2,r3
|
|
vdup.32 $R3,r5
|
|
add r5,r6,r6,lsl#2
|
|
vdup.32 $S3,r4
|
|
vdup.32 $R4,r6
|
|
vdup.32 $S4,r5
|
|
|
|
mov $zeros,#2 @ counter
|
|
|
|
.Lsquare_neon:
|
|
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
@ d0 = h0*r0 + h4*5*r1 + h3*5*r2 + h2*5*r3 + h1*5*r4
|
|
@ d1 = h1*r0 + h0*r1 + h4*5*r2 + h3*5*r3 + h2*5*r4
|
|
@ d2 = h2*r0 + h1*r1 + h0*r2 + h4*5*r3 + h3*5*r4
|
|
@ d3 = h3*r0 + h2*r1 + h1*r2 + h0*r3 + h4*5*r4
|
|
@ d4 = h4*r0 + h3*r1 + h2*r2 + h1*r3 + h0*r4
|
|
|
|
vmull.u32 $D0,$R0,${R0}[1]
|
|
vmull.u32 $D1,$R1,${R0}[1]
|
|
vmull.u32 $D2,$R2,${R0}[1]
|
|
vmull.u32 $D3,$R3,${R0}[1]
|
|
vmull.u32 $D4,$R4,${R0}[1]
|
|
|
|
vmlal.u32 $D0,$R4,${S1}[1]
|
|
vmlal.u32 $D1,$R0,${R1}[1]
|
|
vmlal.u32 $D2,$R1,${R1}[1]
|
|
vmlal.u32 $D3,$R2,${R1}[1]
|
|
vmlal.u32 $D4,$R3,${R1}[1]
|
|
|
|
vmlal.u32 $D0,$R3,${S2}[1]
|
|
vmlal.u32 $D1,$R4,${S2}[1]
|
|
vmlal.u32 $D3,$R1,${R2}[1]
|
|
vmlal.u32 $D2,$R0,${R2}[1]
|
|
vmlal.u32 $D4,$R2,${R2}[1]
|
|
|
|
vmlal.u32 $D0,$R2,${S3}[1]
|
|
vmlal.u32 $D3,$R0,${R3}[1]
|
|
vmlal.u32 $D1,$R3,${S3}[1]
|
|
vmlal.u32 $D2,$R4,${S3}[1]
|
|
vmlal.u32 $D4,$R1,${R3}[1]
|
|
|
|
vmlal.u32 $D3,$R4,${S4}[1]
|
|
vmlal.u32 $D0,$R1,${S4}[1]
|
|
vmlal.u32 $D1,$R2,${S4}[1]
|
|
vmlal.u32 $D2,$R3,${S4}[1]
|
|
vmlal.u32 $D4,$R0,${R4}[1]
|
|
|
|
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
@ lazy reduction as discussed in "NEON crypto" by D.J. Bernstein
|
|
@ and P. Schwabe
|
|
|
|
vshr.u64 $T0,$D3,#26
|
|
vmovn.i64 $D3#lo,$D3
|
|
vshr.u64 $T1,$D0,#26
|
|
vmovn.i64 $D0#lo,$D0
|
|
vadd.i64 $D4,$D4,$T0 @ h3 -> h4
|
|
vbic.i32 $D3#lo,#0xfc000000 @ &=0x03ffffff
|
|
vadd.i64 $D1,$D1,$T1 @ h0 -> h1
|
|
vbic.i32 $D0#lo,#0xfc000000
|
|
|
|
vshrn.u64 $T0#lo,$D4,#26
|
|
vmovn.i64 $D4#lo,$D4
|
|
vshr.u64 $T1,$D1,#26
|
|
vmovn.i64 $D1#lo,$D1
|
|
vadd.i64 $D2,$D2,$T1 @ h1 -> h2
|
|
vbic.i32 $D4#lo,#0xfc000000
|
|
vbic.i32 $D1#lo,#0xfc000000
|
|
|
|
vadd.i32 $D0#lo,$D0#lo,$T0#lo
|
|
vshl.u32 $T0#lo,$T0#lo,#2
|
|
vshrn.u64 $T1#lo,$D2,#26
|
|
vmovn.i64 $D2#lo,$D2
|
|
vadd.i32 $D0#lo,$D0#lo,$T0#lo @ h4 -> h0
|
|
vadd.i32 $D3#lo,$D3#lo,$T1#lo @ h2 -> h3
|
|
vbic.i32 $D2#lo,#0xfc000000
|
|
|
|
vshr.u32 $T0#lo,$D0#lo,#26
|
|
vbic.i32 $D0#lo,#0xfc000000
|
|
vshr.u32 $T1#lo,$D3#lo,#26
|
|
vbic.i32 $D3#lo,#0xfc000000
|
|
vadd.i32 $D1#lo,$D1#lo,$T0#lo @ h0 -> h1
|
|
vadd.i32 $D4#lo,$D4#lo,$T1#lo @ h3 -> h4
|
|
|
|
subs $zeros,$zeros,#1
|
|
beq .Lsquare_break_neon
|
|
|
|
add $tbl0,$ctx,#(48+0*9*4)
|
|
add $tbl1,$ctx,#(48+1*9*4)
|
|
|
|
vtrn.32 $R0,$D0#lo @ r^2:r^1
|
|
vtrn.32 $R2,$D2#lo
|
|
vtrn.32 $R3,$D3#lo
|
|
vtrn.32 $R1,$D1#lo
|
|
vtrn.32 $R4,$D4#lo
|
|
|
|
vshl.u32 $S2,$R2,#2 @ *5
|
|
vshl.u32 $S3,$R3,#2
|
|
vshl.u32 $S1,$R1,#2
|
|
vshl.u32 $S4,$R4,#2
|
|
vadd.i32 $S2,$S2,$R2
|
|
vadd.i32 $S1,$S1,$R1
|
|
vadd.i32 $S3,$S3,$R3
|
|
vadd.i32 $S4,$S4,$R4
|
|
|
|
vst4.32 {${R0}[0],${R1}[0],${S1}[0],${R2}[0]},[$tbl0]!
|
|
vst4.32 {${R0}[1],${R1}[1],${S1}[1],${R2}[1]},[$tbl1]!
|
|
vst4.32 {${S2}[0],${R3}[0],${S3}[0],${R4}[0]},[$tbl0]!
|
|
vst4.32 {${S2}[1],${R3}[1],${S3}[1],${R4}[1]},[$tbl1]!
|
|
vst1.32 {${S4}[0]},[$tbl0,:32]
|
|
vst1.32 {${S4}[1]},[$tbl1,:32]
|
|
|
|
b .Lsquare_neon
|
|
|
|
.align 4
|
|
.Lsquare_break_neon:
|
|
add $tbl0,$ctx,#(48+2*4*9)
|
|
add $tbl1,$ctx,#(48+3*4*9)
|
|
|
|
vmov $R0,$D0#lo @ r^4:r^3
|
|
vshl.u32 $S1,$D1#lo,#2 @ *5
|
|
vmov $R1,$D1#lo
|
|
vshl.u32 $S2,$D2#lo,#2
|
|
vmov $R2,$D2#lo
|
|
vshl.u32 $S3,$D3#lo,#2
|
|
vmov $R3,$D3#lo
|
|
vshl.u32 $S4,$D4#lo,#2
|
|
vmov $R4,$D4#lo
|
|
vadd.i32 $S1,$S1,$D1#lo
|
|
vadd.i32 $S2,$S2,$D2#lo
|
|
vadd.i32 $S3,$S3,$D3#lo
|
|
vadd.i32 $S4,$S4,$D4#lo
|
|
|
|
vst4.32 {${R0}[0],${R1}[0],${S1}[0],${R2}[0]},[$tbl0]!
|
|
vst4.32 {${R0}[1],${R1}[1],${S1}[1],${R2}[1]},[$tbl1]!
|
|
vst4.32 {${S2}[0],${R3}[0],${S3}[0],${R4}[0]},[$tbl0]!
|
|
vst4.32 {${S2}[1],${R3}[1],${S3}[1],${R4}[1]},[$tbl1]!
|
|
vst1.32 {${S4}[0]},[$tbl0]
|
|
vst1.32 {${S4}[1]},[$tbl1]
|
|
|
|
ret @ bx lr
|
|
.size poly1305_init_neon,.-poly1305_init_neon
|
|
|
|
.type poly1305_blocks_neon,%function
|
|
.align 5
|
|
poly1305_blocks_neon:
|
|
ldr ip,[$ctx,#36] @ is_base2_26
|
|
ands $len,$len,#-16
|
|
beq .Lno_data_neon
|
|
|
|
cmp $len,#64
|
|
bhs .Lenter_neon
|
|
tst ip,ip @ is_base2_26?
|
|
beq poly1305_blocks
|
|
|
|
.Lenter_neon:
|
|
stmdb sp!,{r4-r7}
|
|
vstmdb sp!,{d8-d15} @ ABI specification says so
|
|
|
|
tst ip,ip @ is_base2_26?
|
|
bne .Lbase2_26_neon
|
|
|
|
stmdb sp!,{r1-r3,lr}
|
|
bl poly1305_init_neon
|
|
|
|
ldr r4,[$ctx,#0] @ load hash value base 2^32
|
|
ldr r5,[$ctx,#4]
|
|
ldr r6,[$ctx,#8]
|
|
ldr r7,[$ctx,#12]
|
|
ldr ip,[$ctx,#16]
|
|
|
|
and r2,r4,#0x03ffffff @ base 2^32 -> base 2^26
|
|
mov r3,r4,lsr#26
|
|
veor $D0#lo,$D0#lo,$D0#lo
|
|
mov r4,r5,lsr#20
|
|
orr r3,r3,r5,lsl#6
|
|
veor $D1#lo,$D1#lo,$D1#lo
|
|
mov r5,r6,lsr#14
|
|
orr r4,r4,r6,lsl#12
|
|
veor $D2#lo,$D2#lo,$D2#lo
|
|
mov r6,r7,lsr#8
|
|
orr r5,r5,r7,lsl#18
|
|
veor $D3#lo,$D3#lo,$D3#lo
|
|
and r3,r3,#0x03ffffff
|
|
orr r6,r6,ip,lsl#24
|
|
veor $D4#lo,$D4#lo,$D4#lo
|
|
and r4,r4,#0x03ffffff
|
|
mov r1,#1
|
|
and r5,r5,#0x03ffffff
|
|
str r1,[$ctx,#36] @ is_base2_26
|
|
|
|
vmov.32 $D0#lo[0],r2
|
|
vmov.32 $D1#lo[0],r3
|
|
vmov.32 $D2#lo[0],r4
|
|
vmov.32 $D3#lo[0],r5
|
|
vmov.32 $D4#lo[0],r6
|
|
adr $zeros,.Lzeros
|
|
|
|
ldmia sp!,{r1-r3,lr}
|
|
b .Lbase2_32_neon
|
|
|
|
.align 4
|
|
.Lbase2_26_neon:
|
|
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
@ load hash value
|
|
|
|
veor $D0#lo,$D0#lo,$D0#lo
|
|
veor $D1#lo,$D1#lo,$D1#lo
|
|
veor $D2#lo,$D2#lo,$D2#lo
|
|
veor $D3#lo,$D3#lo,$D3#lo
|
|
veor $D4#lo,$D4#lo,$D4#lo
|
|
vld4.32 {$D0#lo[0],$D1#lo[0],$D2#lo[0],$D3#lo[0]},[$ctx]!
|
|
adr $zeros,.Lzeros
|
|
vld1.32 {$D4#lo[0]},[$ctx]
|
|
sub $ctx,$ctx,#16 @ rewind
|
|
|
|
.Lbase2_32_neon:
|
|
add $in2,$inp,#32
|
|
mov $padbit,$padbit,lsl#24
|
|
tst $len,#31
|
|
beq .Leven
|
|
|
|
vld4.32 {$H0#lo[0],$H1#lo[0],$H2#lo[0],$H3#lo[0]},[$inp]!
|
|
vmov.32 $H4#lo[0],$padbit
|
|
sub $len,$len,#16
|
|
add $in2,$inp,#32
|
|
|
|
# ifdef __ARMEB__
|
|
vrev32.8 $H0,$H0
|
|
vrev32.8 $H3,$H3
|
|
vrev32.8 $H1,$H1
|
|
vrev32.8 $H2,$H2
|
|
# endif
|
|
vsri.u32 $H4#lo,$H3#lo,#8 @ base 2^32 -> base 2^26
|
|
vshl.u32 $H3#lo,$H3#lo,#18
|
|
|
|
vsri.u32 $H3#lo,$H2#lo,#14
|
|
vshl.u32 $H2#lo,$H2#lo,#12
|
|
vadd.i32 $H4#hi,$H4#lo,$D4#lo @ add hash value and move to #hi
|
|
|
|
vbic.i32 $H3#lo,#0xfc000000
|
|
vsri.u32 $H2#lo,$H1#lo,#20
|
|
vshl.u32 $H1#lo,$H1#lo,#6
|
|
|
|
vbic.i32 $H2#lo,#0xfc000000
|
|
vsri.u32 $H1#lo,$H0#lo,#26
|
|
vadd.i32 $H3#hi,$H3#lo,$D3#lo
|
|
|
|
vbic.i32 $H0#lo,#0xfc000000
|
|
vbic.i32 $H1#lo,#0xfc000000
|
|
vadd.i32 $H2#hi,$H2#lo,$D2#lo
|
|
|
|
vadd.i32 $H0#hi,$H0#lo,$D0#lo
|
|
vadd.i32 $H1#hi,$H1#lo,$D1#lo
|
|
|
|
mov $tbl1,$zeros
|
|
add $tbl0,$ctx,#48
|
|
|
|
cmp $len,$len
|
|
b .Long_tail
|
|
|
|
.align 4
|
|
.Leven:
|
|
subs $len,$len,#64
|
|
# ifdef __thumb2__
|
|
it lo
|
|
# endif
|
|
movlo $in2,$zeros
|
|
|
|
vmov.i32 $H4,#1<<24 @ padbit, yes, always
|
|
vld4.32 {$H0#lo,$H1#lo,$H2#lo,$H3#lo},[$inp] @ inp[0:1]
|
|
add $inp,$inp,#64
|
|
vld4.32 {$H0#hi,$H1#hi,$H2#hi,$H3#hi},[$in2] @ inp[2:3] (or 0)
|
|
add $in2,$in2,#64
|
|
# ifdef __thumb2__
|
|
itt hi
|
|
# endif
|
|
addhi $tbl1,$ctx,#(48+1*9*4)
|
|
addhi $tbl0,$ctx,#(48+3*9*4)
|
|
|
|
# ifdef __ARMEB__
|
|
vrev32.8 $H0,$H0
|
|
vrev32.8 $H3,$H3
|
|
vrev32.8 $H1,$H1
|
|
vrev32.8 $H2,$H2
|
|
# endif
|
|
vsri.u32 $H4,$H3,#8 @ base 2^32 -> base 2^26
|
|
vshl.u32 $H3,$H3,#18
|
|
|
|
vsri.u32 $H3,$H2,#14
|
|
vshl.u32 $H2,$H2,#12
|
|
|
|
vbic.i32 $H3,#0xfc000000
|
|
vsri.u32 $H2,$H1,#20
|
|
vshl.u32 $H1,$H1,#6
|
|
|
|
vbic.i32 $H2,#0xfc000000
|
|
vsri.u32 $H1,$H0,#26
|
|
|
|
vbic.i32 $H0,#0xfc000000
|
|
vbic.i32 $H1,#0xfc000000
|
|
|
|
bls .Lskip_loop
|
|
|
|
vld4.32 {${R0}[1],${R1}[1],${S1}[1],${R2}[1]},[$tbl1]! @ load r^2
|
|
vld4.32 {${R0}[0],${R1}[0],${S1}[0],${R2}[0]},[$tbl0]! @ load r^4
|
|
vld4.32 {${S2}[1],${R3}[1],${S3}[1],${R4}[1]},[$tbl1]!
|
|
vld4.32 {${S2}[0],${R3}[0],${S3}[0],${R4}[0]},[$tbl0]!
|
|
b .Loop_neon
|
|
|
|
.align 5
|
|
.Loop_neon:
|
|
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
@ ((inp[0]*r^4+inp[2]*r^2+inp[4])*r^4+inp[6]*r^2
|
|
@ ((inp[1]*r^4+inp[3]*r^2+inp[5])*r^3+inp[7]*r
|
|
@ \___________________/
|
|
@ ((inp[0]*r^4+inp[2]*r^2+inp[4])*r^4+inp[6]*r^2+inp[8])*r^2
|
|
@ ((inp[1]*r^4+inp[3]*r^2+inp[5])*r^4+inp[7]*r^2+inp[9])*r
|
|
@ \___________________/ \____________________/
|
|
@
|
|
@ Note that we start with inp[2:3]*r^2. This is because it
|
|
@ doesn't depend on reduction in previous iteration.
|
|
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
@ d4 = h4*r0 + h3*r1 + h2*r2 + h1*r3 + h0*r4
|
|
@ d3 = h3*r0 + h2*r1 + h1*r2 + h0*r3 + h4*5*r4
|
|
@ d2 = h2*r0 + h1*r1 + h0*r2 + h4*5*r3 + h3*5*r4
|
|
@ d1 = h1*r0 + h0*r1 + h4*5*r2 + h3*5*r3 + h2*5*r4
|
|
@ d0 = h0*r0 + h4*5*r1 + h3*5*r2 + h2*5*r3 + h1*5*r4
|
|
|
|
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
@ inp[2:3]*r^2
|
|
|
|
vadd.i32 $H2#lo,$H2#lo,$D2#lo @ accumulate inp[0:1]
|
|
vmull.u32 $D2,$H2#hi,${R0}[1]
|
|
vadd.i32 $H0#lo,$H0#lo,$D0#lo
|
|
vmull.u32 $D0,$H0#hi,${R0}[1]
|
|
vadd.i32 $H3#lo,$H3#lo,$D3#lo
|
|
vmull.u32 $D3,$H3#hi,${R0}[1]
|
|
vmlal.u32 $D2,$H1#hi,${R1}[1]
|
|
vadd.i32 $H1#lo,$H1#lo,$D1#lo
|
|
vmull.u32 $D1,$H1#hi,${R0}[1]
|
|
|
|
vadd.i32 $H4#lo,$H4#lo,$D4#lo
|
|
vmull.u32 $D4,$H4#hi,${R0}[1]
|
|
subs $len,$len,#64
|
|
vmlal.u32 $D0,$H4#hi,${S1}[1]
|
|
# ifdef __thumb2__
|
|
it lo
|
|
# endif
|
|
movlo $in2,$zeros
|
|
vmlal.u32 $D3,$H2#hi,${R1}[1]
|
|
vld1.32 ${S4}[1],[$tbl1,:32]
|
|
vmlal.u32 $D1,$H0#hi,${R1}[1]
|
|
vmlal.u32 $D4,$H3#hi,${R1}[1]
|
|
|
|
vmlal.u32 $D0,$H3#hi,${S2}[1]
|
|
vmlal.u32 $D3,$H1#hi,${R2}[1]
|
|
vmlal.u32 $D4,$H2#hi,${R2}[1]
|
|
vmlal.u32 $D1,$H4#hi,${S2}[1]
|
|
vmlal.u32 $D2,$H0#hi,${R2}[1]
|
|
|
|
vmlal.u32 $D3,$H0#hi,${R3}[1]
|
|
vmlal.u32 $D0,$H2#hi,${S3}[1]
|
|
vmlal.u32 $D4,$H1#hi,${R3}[1]
|
|
vmlal.u32 $D1,$H3#hi,${S3}[1]
|
|
vmlal.u32 $D2,$H4#hi,${S3}[1]
|
|
|
|
vmlal.u32 $D3,$H4#hi,${S4}[1]
|
|
vmlal.u32 $D0,$H1#hi,${S4}[1]
|
|
vmlal.u32 $D4,$H0#hi,${R4}[1]
|
|
vmlal.u32 $D1,$H2#hi,${S4}[1]
|
|
vmlal.u32 $D2,$H3#hi,${S4}[1]
|
|
|
|
vld4.32 {$H0#hi,$H1#hi,$H2#hi,$H3#hi},[$in2] @ inp[2:3] (or 0)
|
|
add $in2,$in2,#64
|
|
|
|
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
@ (hash+inp[0:1])*r^4 and accumulate
|
|
|
|
vmlal.u32 $D3,$H3#lo,${R0}[0]
|
|
vmlal.u32 $D0,$H0#lo,${R0}[0]
|
|
vmlal.u32 $D4,$H4#lo,${R0}[0]
|
|
vmlal.u32 $D1,$H1#lo,${R0}[0]
|
|
vmlal.u32 $D2,$H2#lo,${R0}[0]
|
|
vld1.32 ${S4}[0],[$tbl0,:32]
|
|
|
|
vmlal.u32 $D3,$H2#lo,${R1}[0]
|
|
vmlal.u32 $D0,$H4#lo,${S1}[0]
|
|
vmlal.u32 $D4,$H3#lo,${R1}[0]
|
|
vmlal.u32 $D1,$H0#lo,${R1}[0]
|
|
vmlal.u32 $D2,$H1#lo,${R1}[0]
|
|
|
|
vmlal.u32 $D3,$H1#lo,${R2}[0]
|
|
vmlal.u32 $D0,$H3#lo,${S2}[0]
|
|
vmlal.u32 $D4,$H2#lo,${R2}[0]
|
|
vmlal.u32 $D1,$H4#lo,${S2}[0]
|
|
vmlal.u32 $D2,$H0#lo,${R2}[0]
|
|
|
|
vmlal.u32 $D3,$H0#lo,${R3}[0]
|
|
vmlal.u32 $D0,$H2#lo,${S3}[0]
|
|
vmlal.u32 $D4,$H1#lo,${R3}[0]
|
|
vmlal.u32 $D1,$H3#lo,${S3}[0]
|
|
vmlal.u32 $D3,$H4#lo,${S4}[0]
|
|
|
|
vmlal.u32 $D2,$H4#lo,${S3}[0]
|
|
vmlal.u32 $D0,$H1#lo,${S4}[0]
|
|
vmlal.u32 $D4,$H0#lo,${R4}[0]
|
|
vmov.i32 $H4,#1<<24 @ padbit, yes, always
|
|
vmlal.u32 $D1,$H2#lo,${S4}[0]
|
|
vmlal.u32 $D2,$H3#lo,${S4}[0]
|
|
|
|
vld4.32 {$H0#lo,$H1#lo,$H2#lo,$H3#lo},[$inp] @ inp[0:1]
|
|
add $inp,$inp,#64
|
|
# ifdef __ARMEB__
|
|
vrev32.8 $H0,$H0
|
|
vrev32.8 $H1,$H1
|
|
vrev32.8 $H2,$H2
|
|
vrev32.8 $H3,$H3
|
|
# endif
|
|
|
|
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
@ lazy reduction interleaved with base 2^32 -> base 2^26
|
|
|
|
vshr.u64 $T0,$D3,#26
|
|
vmovn.i64 $D3#lo,$D3
|
|
vshr.u64 $T1,$D0,#26
|
|
vmovn.i64 $D0#lo,$D0
|
|
vadd.i64 $D4,$D4,$T0 @ h3 -> h4
|
|
vbic.i32 $D3#lo,#0xfc000000
|
|
vsri.u32 $H4,$H3,#8 @ base 2^32 -> base 2^26
|
|
vadd.i64 $D1,$D1,$T1 @ h0 -> h1
|
|
vshl.u32 $H3,$H3,#18
|
|
vbic.i32 $D0#lo,#0xfc000000
|
|
|
|
vshrn.u64 $T0#lo,$D4,#26
|
|
vmovn.i64 $D4#lo,$D4
|
|
vshr.u64 $T1,$D1,#26
|
|
vmovn.i64 $D1#lo,$D1
|
|
vadd.i64 $D2,$D2,$T1 @ h1 -> h2
|
|
vsri.u32 $H3,$H2,#14
|
|
vbic.i32 $D4#lo,#0xfc000000
|
|
vshl.u32 $H2,$H2,#12
|
|
vbic.i32 $D1#lo,#0xfc000000
|
|
|
|
vadd.i32 $D0#lo,$D0#lo,$T0#lo
|
|
vshl.u32 $T0#lo,$T0#lo,#2
|
|
vbic.i32 $H3,#0xfc000000
|
|
vshrn.u64 $T1#lo,$D2,#26
|
|
vmovn.i64 $D2#lo,$D2
|
|
vadd.i32 $D0#lo,$D0#lo,$T0#lo @ h4 -> h0
|
|
vsri.u32 $H2,$H1,#20
|
|
vadd.i32 $D3#lo,$D3#lo,$T1#lo @ h2 -> h3
|
|
vshl.u32 $H1,$H1,#6
|
|
vbic.i32 $D2#lo,#0xfc000000
|
|
vbic.i32 $H2,#0xfc000000
|
|
|
|
vshr.u32 $T0#lo,$D0#lo,#26
|
|
vbic.i32 $D0#lo,#0xfc000000
|
|
vsri.u32 $H1,$H0,#26
|
|
vbic.i32 $H0,#0xfc000000
|
|
vshr.u32 $T1#lo,$D3#lo,#26
|
|
vbic.i32 $D3#lo,#0xfc000000
|
|
vadd.i32 $D1#lo,$D1#lo,$T0#lo @ h0 -> h1
|
|
vadd.i32 $D4#lo,$D4#lo,$T1#lo @ h3 -> h4
|
|
vbic.i32 $H1,#0xfc000000
|
|
|
|
bhi .Loop_neon
|
|
|
|
.Lskip_loop:
|
|
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
@ multiply (inp[0:1]+hash) or inp[2:3] by r^2:r^1
|
|
|
|
add $tbl1,$ctx,#(48+0*9*4)
|
|
add $tbl0,$ctx,#(48+1*9*4)
|
|
adds $len,$len,#32
|
|
# ifdef __thumb2__
|
|
it ne
|
|
# endif
|
|
movne $len,#0
|
|
bne .Long_tail
|
|
|
|
vadd.i32 $H2#hi,$H2#lo,$D2#lo @ add hash value and move to #hi
|
|
vadd.i32 $H0#hi,$H0#lo,$D0#lo
|
|
vadd.i32 $H3#hi,$H3#lo,$D3#lo
|
|
vadd.i32 $H1#hi,$H1#lo,$D1#lo
|
|
vadd.i32 $H4#hi,$H4#lo,$D4#lo
|
|
|
|
.Long_tail:
|
|
vld4.32 {${R0}[1],${R1}[1],${S1}[1],${R2}[1]},[$tbl1]! @ load r^1
|
|
vld4.32 {${R0}[0],${R1}[0],${S1}[0],${R2}[0]},[$tbl0]! @ load r^2
|
|
|
|
vadd.i32 $H2#lo,$H2#lo,$D2#lo @ can be redundant
|
|
vmull.u32 $D2,$H2#hi,$R0
|
|
vadd.i32 $H0#lo,$H0#lo,$D0#lo
|
|
vmull.u32 $D0,$H0#hi,$R0
|
|
vadd.i32 $H3#lo,$H3#lo,$D3#lo
|
|
vmull.u32 $D3,$H3#hi,$R0
|
|
vadd.i32 $H1#lo,$H1#lo,$D1#lo
|
|
vmull.u32 $D1,$H1#hi,$R0
|
|
vadd.i32 $H4#lo,$H4#lo,$D4#lo
|
|
vmull.u32 $D4,$H4#hi,$R0
|
|
|
|
vmlal.u32 $D0,$H4#hi,$S1
|
|
vld4.32 {${S2}[1],${R3}[1],${S3}[1],${R4}[1]},[$tbl1]!
|
|
vmlal.u32 $D3,$H2#hi,$R1
|
|
vld4.32 {${S2}[0],${R3}[0],${S3}[0],${R4}[0]},[$tbl0]!
|
|
vmlal.u32 $D1,$H0#hi,$R1
|
|
vmlal.u32 $D4,$H3#hi,$R1
|
|
vmlal.u32 $D2,$H1#hi,$R1
|
|
|
|
vmlal.u32 $D3,$H1#hi,$R2
|
|
vld1.32 ${S4}[1],[$tbl1,:32]
|
|
vmlal.u32 $D0,$H3#hi,$S2
|
|
vld1.32 ${S4}[0],[$tbl0,:32]
|
|
vmlal.u32 $D4,$H2#hi,$R2
|
|
vmlal.u32 $D1,$H4#hi,$S2
|
|
vmlal.u32 $D2,$H0#hi,$R2
|
|
|
|
vmlal.u32 $D3,$H0#hi,$R3
|
|
# ifdef __thumb2__
|
|
it ne
|
|
# endif
|
|
addne $tbl1,$ctx,#(48+2*9*4)
|
|
vmlal.u32 $D0,$H2#hi,$S3
|
|
# ifdef __thumb2__
|
|
it ne
|
|
# endif
|
|
addne $tbl0,$ctx,#(48+3*9*4)
|
|
vmlal.u32 $D4,$H1#hi,$R3
|
|
vmlal.u32 $D1,$H3#hi,$S3
|
|
vmlal.u32 $D2,$H4#hi,$S3
|
|
|
|
vmlal.u32 $D3,$H4#hi,$S4
|
|
vorn $MASK,$MASK,$MASK @ all-ones, can be redundant
|
|
vmlal.u32 $D0,$H1#hi,$S4
|
|
vshr.u64 $MASK,$MASK,#38
|
|
vmlal.u32 $D4,$H0#hi,$R4
|
|
vmlal.u32 $D1,$H2#hi,$S4
|
|
vmlal.u32 $D2,$H3#hi,$S4
|
|
|
|
beq .Lshort_tail
|
|
|
|
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
@ (hash+inp[0:1])*r^4:r^3 and accumulate
|
|
|
|
vld4.32 {${R0}[1],${R1}[1],${S1}[1],${R2}[1]},[$tbl1]! @ load r^3
|
|
vld4.32 {${R0}[0],${R1}[0],${S1}[0],${R2}[0]},[$tbl0]! @ load r^4
|
|
|
|
vmlal.u32 $D2,$H2#lo,$R0
|
|
vmlal.u32 $D0,$H0#lo,$R0
|
|
vmlal.u32 $D3,$H3#lo,$R0
|
|
vmlal.u32 $D1,$H1#lo,$R0
|
|
vmlal.u32 $D4,$H4#lo,$R0
|
|
|
|
vmlal.u32 $D0,$H4#lo,$S1
|
|
vld4.32 {${S2}[1],${R3}[1],${S3}[1],${R4}[1]},[$tbl1]!
|
|
vmlal.u32 $D3,$H2#lo,$R1
|
|
vld4.32 {${S2}[0],${R3}[0],${S3}[0],${R4}[0]},[$tbl0]!
|
|
vmlal.u32 $D1,$H0#lo,$R1
|
|
vmlal.u32 $D4,$H3#lo,$R1
|
|
vmlal.u32 $D2,$H1#lo,$R1
|
|
|
|
vmlal.u32 $D3,$H1#lo,$R2
|
|
vld1.32 ${S4}[1],[$tbl1,:32]
|
|
vmlal.u32 $D0,$H3#lo,$S2
|
|
vld1.32 ${S4}[0],[$tbl0,:32]
|
|
vmlal.u32 $D4,$H2#lo,$R2
|
|
vmlal.u32 $D1,$H4#lo,$S2
|
|
vmlal.u32 $D2,$H0#lo,$R2
|
|
|
|
vmlal.u32 $D3,$H0#lo,$R3
|
|
vmlal.u32 $D0,$H2#lo,$S3
|
|
vmlal.u32 $D4,$H1#lo,$R3
|
|
vmlal.u32 $D1,$H3#lo,$S3
|
|
vmlal.u32 $D2,$H4#lo,$S3
|
|
|
|
vmlal.u32 $D3,$H4#lo,$S4
|
|
vorn $MASK,$MASK,$MASK @ all-ones
|
|
vmlal.u32 $D0,$H1#lo,$S4
|
|
vshr.u64 $MASK,$MASK,#38
|
|
vmlal.u32 $D4,$H0#lo,$R4
|
|
vmlal.u32 $D1,$H2#lo,$S4
|
|
vmlal.u32 $D2,$H3#lo,$S4
|
|
|
|
.Lshort_tail:
|
|
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
@ lazy reduction, but without narrowing
|
|
|
|
vshr.u64 $T0,$D3,#26
|
|
vand.i64 $D3,$D3,$MASK
|
|
vshr.u64 $T1,$D0,#26
|
|
vand.i64 $D0,$D0,$MASK
|
|
vadd.i64 $D4,$D4,$T0 @ h3 -> h4
|
|
vadd.i64 $D1,$D1,$T1 @ h0 -> h1
|
|
|
|
vshr.u64 $T0,$D4,#26
|
|
vand.i64 $D4,$D4,$MASK
|
|
vshr.u64 $T1,$D1,#26
|
|
vand.i64 $D1,$D1,$MASK
|
|
vadd.i64 $D2,$D2,$T1 @ h1 -> h2
|
|
|
|
vadd.i64 $D0,$D0,$T0
|
|
vshl.u64 $T0,$T0,#2
|
|
vshr.u64 $T1,$D2,#26
|
|
vand.i64 $D2,$D2,$MASK
|
|
vadd.i64 $D0,$D0,$T0 @ h4 -> h0
|
|
vadd.i64 $D3,$D3,$T1 @ h2 -> h3
|
|
|
|
vshr.u64 $T0,$D0,#26
|
|
vand.i64 $D0,$D0,$MASK
|
|
vshr.u64 $T1,$D3,#26
|
|
vand.i64 $D3,$D3,$MASK
|
|
vadd.i64 $D1,$D1,$T0 @ h0 -> h1
|
|
vadd.i64 $D4,$D4,$T1 @ h3 -> h4
|
|
|
|
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
@ horizontal addition
|
|
|
|
vadd.i64 $D2#lo,$D2#lo,$D2#hi
|
|
vadd.i64 $D0#lo,$D0#lo,$D0#hi
|
|
vadd.i64 $D3#lo,$D3#lo,$D3#hi
|
|
vadd.i64 $D1#lo,$D1#lo,$D1#hi
|
|
vadd.i64 $D4#lo,$D4#lo,$D4#hi
|
|
|
|
cmp $len,#0
|
|
bne .Leven
|
|
|
|
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
|
|
@ store hash value
|
|
|
|
vst4.32 {$D0#lo[0],$D1#lo[0],$D2#lo[0],$D3#lo[0]},[$ctx]!
|
|
vst1.32 {$D4#lo[0]},[$ctx]
|
|
|
|
vldmia sp!,{d8-d15} @ epilogue
|
|
ldmia sp!,{r4-r7}
|
|
.Lno_data_neon:
|
|
ret @ bx lr
|
|
.size poly1305_blocks_neon,.-poly1305_blocks_neon
|
|
|
|
.type poly1305_emit_neon,%function
|
|
.align 5
|
|
poly1305_emit_neon:
|
|
ldr ip,[$ctx,#36] @ is_base2_26
|
|
|
|
stmdb sp!,{r4-r11}
|
|
|
|
tst ip,ip
|
|
beq .Lpoly1305_emit_enter
|
|
|
|
ldmia $ctx,{$h0-$h4}
|
|
eor $g0,$g0,$g0
|
|
|
|
adds $h0,$h0,$h1,lsl#26 @ base 2^26 -> base 2^32
|
|
mov $h1,$h1,lsr#6
|
|
adcs $h1,$h1,$h2,lsl#20
|
|
mov $h2,$h2,lsr#12
|
|
adcs $h2,$h2,$h3,lsl#14
|
|
mov $h3,$h3,lsr#18
|
|
adcs $h3,$h3,$h4,lsl#8
|
|
adc $h4,$g0,$h4,lsr#24 @ can be partially reduced ...
|
|
|
|
and $g0,$h4,#-4 @ ... so reduce
|
|
and $h4,$h3,#3
|
|
add $g0,$g0,$g0,lsr#2 @ *= 5
|
|
adds $h0,$h0,$g0
|
|
adcs $h1,$h1,#0
|
|
adcs $h2,$h2,#0
|
|
adc $h3,$h3,#0
|
|
|
|
adds $g0,$h0,#5 @ compare to modulus
|
|
adcs $g1,$h1,#0
|
|
adcs $g2,$h2,#0
|
|
adcs $g3,$h3,#0
|
|
adc $g4,$h4,#0
|
|
tst $g4,#4 @ did it carry/borrow?
|
|
|
|
# ifdef __thumb2__
|
|
it ne
|
|
# endif
|
|
movne $h0,$g0
|
|
ldr $g0,[$nonce,#0]
|
|
# ifdef __thumb2__
|
|
it ne
|
|
# endif
|
|
movne $h1,$g1
|
|
ldr $g1,[$nonce,#4]
|
|
# ifdef __thumb2__
|
|
it ne
|
|
# endif
|
|
movne $h2,$g2
|
|
ldr $g2,[$nonce,#8]
|
|
# ifdef __thumb2__
|
|
it ne
|
|
# endif
|
|
movne $h3,$g3
|
|
ldr $g3,[$nonce,#12]
|
|
|
|
adds $h0,$h0,$g0 @ accumulate nonce
|
|
adcs $h1,$h1,$g1
|
|
adcs $h2,$h2,$g2
|
|
adc $h3,$h3,$g3
|
|
|
|
# ifdef __ARMEB__
|
|
rev $h0,$h0
|
|
rev $h1,$h1
|
|
rev $h2,$h2
|
|
rev $h3,$h3
|
|
# endif
|
|
str $h0,[$mac,#0] @ store the result
|
|
str $h1,[$mac,#4]
|
|
str $h2,[$mac,#8]
|
|
str $h3,[$mac,#12]
|
|
|
|
ldmia sp!,{r4-r11}
|
|
ret @ bx lr
|
|
.size poly1305_emit_neon,.-poly1305_emit_neon
|
|
|
|
.align 5
|
|
.Lzeros:
|
|
.long 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
|
|
.LOPENSSL_armcap:
|
|
.word OPENSSL_armcap_P-.Lpoly1305_init
|
|
#endif
|
|
___
|
|
} }
|
|
$code.=<<___;
|
|
.asciz "Poly1305 for ARMv4/NEON, CRYPTOGAMS by <appro\@openssl.org>"
|
|
.align 2
|
|
#if __ARM_MAX_ARCH__>=7
|
|
.comm OPENSSL_armcap_P,4,4
|
|
#endif
|
|
___
|
|
|
|
foreach (split("\n",$code)) {
|
|
s/\`([^\`]*)\`/eval $1/geo;
|
|
|
|
s/\bq([0-9]+)#(lo|hi)/sprintf "d%d",2*$1+($2 eq "hi")/geo or
|
|
s/\bret\b/bx lr/go or
|
|
s/\bbx\s+lr\b/.word\t0xe12fff1e/go; # make it possible to compile with -march=armv4
|
|
|
|
print $_,"\n";
|
|
}
|
|
close STDOUT; # enforce flush
|