rc4-amd64.pl 5.7 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
#!/usr/bin/env perl
#
# ====================================================================
# Written by Andy Polyakov <appro@fy.chalmers.se> for the OpenSSL
# project. Rights for redistribution and usage in source and binary
# forms are granted according to the OpenSSL license.
# ====================================================================
#
# 2.22x RC4 tune-up:-) It should be noted though that my hand [as in
# "hand-coded assembler"] doesn't stand for the whole improvement
# coefficient. It turned out that eliminating RC4_CHAR from config
# line results in ~40% improvement (yes, even for C implementation).
# Presumably it has everything to do with AMD cache architecture and
# RAW or whatever penalties. Once again! The module *requires* config
# line *without* RC4_CHAR! As for coding "secret," I bet on partial
16
# register arithmetics. For example instead of 'inc %r8; and $255,%r8'
17 18 19 20
# I simply 'inc %r8b'. Even though optimization manual discourages
# to operate on partial registers, it turned out to be the best bet.
# At least for AMD... How IA32E would perform remains to be seen...

21 22 23 24 25 26 27
# As was shown by Marc Bevand reordering of couple of load operations
# results in even higher performance gain of 3.3x:-) At least on
# Opteron... For reference, 1x in this case is RC4_CHAR C-code
# compiled with gcc 3.3.2, which performs at ~54MBps per 1GHz clock.
# Latter means that if you want to *estimate* what to expect from
# *your* CPU, then multiply 54 by 3.3 and clock frequency in GHz.

28 29 30 31 32 33 34
# Intel P4 EM64T core was found to run the AMD64 code really slow...
# The only way to achieve comparable performance on P4 is to keep
# RC4_CHAR. Kind of ironic, huh? As it's apparently impossible to
# compose blended code, which would perform even within 30% marginal
# on either AMD and Intel platforms, I implement both cases. See
# rc4_skey.c for further details...

35 36
$output=shift;

37
$win64a=1 if ($output =~ /win64a.[s|asm]/);
38 39 40

open STDOUT,">$output" || die "can't open $output: $!";

41
if (defined($win64a)) {
42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59
    $dat="%rcx";	# arg1
    $len="%rdx";	# arg2
    $inp="%rsi";	# r8, arg3 moves here
    $out="%rdi";	# r9, arg4 moves here
} else {
    $dat="%rdi";	# arg1
    $len="%rsi";	# arg2
    $inp="%rdx";	# arg3
    $out="%rcx";	# arg4
}

$XX="%r10";
$TX="%r8";
$YY="%r11";
$TY="%r9";

sub PTR() {
    my $ret=shift;
60 61 62
    if (defined($win64a)) {
	$ret =~ s/\[([\S]+)\+([\S]+)\]/[$2+$1]/g;   # [%rN+%rM*4]->[%rM*4+%rN]
	$ret =~ s/:([^\[]+)\[([^\]]+)\]/:[$2+$1]/g; # :off[ea]->:[ea+off]
63 64 65 66 67 68 69
    } else {
	$ret =~ s/[\+\*]/,/g;		# [%rN+%rM*4]->[%rN,%rM,4]
	$ret =~ s/\[([^\]]+)\]/($1)/g;	# [%rN]->(%rN)
    }
    $ret;
}

70
$code=<<___ if (!defined($win64a));
71 72 73 74 75 76 77
.text

.globl	RC4
.type	RC4,\@function
.align	16
RC4:	or	$len,$len
	jne	.Lentry
78
	repret
79 80
.Lentry:
___
81
$code=<<___ if (defined($win64a));
82
_TEXT	SEGMENT
83 84
PUBLIC	RC4
ALIGN	16
85
RC4	PROC
86 87
	or	$len,$len
	jne	.Lentry
88
	repret
89
.Lentry:
90 91 92
	push	%rdi
	push	%rsi
	sub	\$40,%rsp
93 94 95 96 97
	mov	%r8,$inp
	mov	%r9,$out
___
$code.=<<___;
	add	\$8,$dat
98 99
	movl	`&PTR("DWORD:-8[$dat]")`,$XX#d
	movl	`&PTR("DWORD:-4[$dat]")`,$YY#d
100 101
	cmpl	\$-1,`&PTR("DWORD:256[$dat]")`
	je	.LRC4_CHAR
102 103 104 105 106
	test	\$-8,$len
	jz	.Lloop1
.align	16
.Lloop8:
	inc	$XX#b
107
	movl	`&PTR("DWORD:[$dat+$XX*4]")`,$TX#d
108
	add	$TX#b,$YY#b
109 110 111
	movl	`&PTR("DWORD:[$dat+$YY*4]")`,$TY#d
	movl	$TX#d,`&PTR("DWORD:[$dat+$YY*4]")`
	movl	$TY#d,`&PTR("DWORD:[$dat+$XX*4]")`
112
	add	$TX#b,$TY#b
113
	inc	$XX#b
114 115
	movl	`&PTR("DWORD:[$dat+$XX*4]")`,$TX#d
	movb	`&PTR("BYTE:[$dat+$TY*4]")`,%al
116 117 118 119
___
for ($i=1;$i<=6;$i++) {
$code.=<<___;
	add	$TX#b,$YY#b
120
	ror	\$8,%rax
121 122 123
	movl	`&PTR("DWORD:[$dat+$YY*4]")`,$TY#d
	movl	$TX#d,`&PTR("DWORD:[$dat+$YY*4]")`
	movl	$TY#d,`&PTR("DWORD:[$dat+$XX*4]")`
124
	add	$TX#b,$TY#b
125
	inc	$XX#b
126 127
	movl	`&PTR("DWORD:[$dat+$XX*4]")`,$TX#d
	movb	`&PTR("BYTE:[$dat+$TY*4]")`,%al
128 129 130 131
___
}
$code.=<<___;
	add	$TX#b,$YY#b
132
	ror	\$8,%rax
133 134 135
	movl	`&PTR("DWORD:[$dat+$YY*4]")`,$TY#d
	movl	$TX#d,`&PTR("DWORD:[$dat+$YY*4]")`
	movl	$TY#d,`&PTR("DWORD:[$dat+$XX*4]")`
136 137
	sub	\$8,$len
	add	$TY#b,$TX#b
138 139
	movb	`&PTR("BYTE:[$dat+$TX*4]")`,%al
	ror	\$8,%rax
140
	add	\$8,$inp
141
	add	\$8,$out
142

143
	xor	`&PTR("QWORD:-8[$inp]")`,%rax
144
	mov	%rax,`&PTR("QWORD:-8[$out]")`
145 146 147 148 149 150

	test	\$-8,$len
	jnz	.Lloop8
	cmp	\$0,$len
	jne	.Lloop1
.Lexit:
151 152
	movl	$XX#d,`&PTR("DWORD:-8[$dat]")`
	movl	$YY#d,`&PTR("DWORD:-4[$dat]")`
153
___
154 155 156 157
$code.=<<___ if (defined($win64a));
	add	\$40,%rsp
	pop	%rsi
	pop	%rdi
158 159
___
$code.=<<___;
160
	repret
161 162
.align	16
.Lloop1:
163
	movzb	`&PTR("BYTE:[$inp]")`,%eax
164
	inc	$XX#b
165
	movl	`&PTR("DWORD:[$dat+$XX*4]")`,$TX#d
166
	add	$TX#b,$YY#b
167 168 169
	movl	`&PTR("DWORD:[$dat+$YY*4]")`,$TY#d
	movl	$TX#d,`&PTR("DWORD:[$dat+$YY*4]")`
	movl	$TY#d,`&PTR("DWORD:[$dat+$XX*4]")`
170
	add	$TY#b,$TX#b
171
	movl	`&PTR("DWORD:[$dat+$TX*4]")`,$TY#d
172 173
	xor	$TY,%rax
	inc	$inp
174
	movb	%al,`&PTR("BYTE:[$out]")`
175 176 177 178
	inc	$out
	dec	$len
	jnz	.Lloop1
	jmp	.Lexit
179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196

.align	16
.LRC4_CHAR:
	inc	$XX#b
	movzb	`&PTR("BYTE:[$dat+$XX]")`,$TX#d
	add	$TX#b,$YY#b
	movzb	`&PTR("BYTE:[$dat+$YY]")`,$TY#d
	movb	$TX#b,`&PTR("BYTE:[$dat+$YY]")`
	movb	$TY#b,`&PTR("BYTE:[$dat+$XX]")`
	add	$TX#b,$TY#b
	movzb	`&PTR("BYTE:[$dat+$TY]")`,$TY#d
	xorb	`&PTR("BYTE:[$inp]")`,$TY#b
	movb	$TY#b,`&PTR("BYTE:[$out]")`
	inc	$inp
	inc	$out
	dec	$len
	jnz	.LRC4_CHAR
	jmp	.Lexit
197
___
198 199
$code.=<<___ if (defined($win64a));
RC4	ENDP
200
_TEXT	ENDS
201 202 203 204 205
END
___
$code.=<<___ if (!defined($win64a));
.size	RC4,.-RC4
___
206 207 208 209

$code =~ s/#([bwd])/$1/gm;
$code =~ s/\`([^\`]*)\`/eval $1/gem;

210
if (defined($win64a)) {
211 212 213 214
    $code =~ s/\.align/ALIGN/gm;
    $code =~ s/[\$%]//gm;
    $code =~ s/\.L/\$L/gm;
    $code =~ s/([\w]+)([\s]+)([\S]+),([\S]+)/$1$2$4,$3/gm;
215 216 217 218
    $code =~ s/([QD]*WORD|BYTE):/$1 PTR/gm;
    $code =~ s/mov[bwlq]/mov/gm;
    $code =~ s/movzb/movzx/gm;
    $code =~ s/repret/DB\t0F3h,0C3h/gm;
219 220
    $code =~ s/cmpl/cmp/gm;
    $code =~ s/xorb/xor/gm;
221
} else {
222 223
    $code =~ s/([QD]*WORD|BYTE)://gm;
    $code =~ s/repret/.byte\t0xF3,0xC3/gm;
224 225
}
print $code;