diff options
Diffstat (limited to 'arch/arm/crypto')
-rw-r--r-- | arch/arm/crypto/Makefile | 14 | ||||
-rw-r--r-- | arch/arm/crypto/aes-armv4.S | 6 | ||||
-rw-r--r-- | arch/arm/crypto/aes_glue.c | 22 | ||||
-rw-r--r-- | arch/arm/crypto/aes_glue.h | 19 | ||||
-rw-r--r-- | arch/arm/crypto/aesbs-core.S_shipped | 2544 | ||||
-rw-r--r-- | arch/arm/crypto/aesbs-glue.c | 434 | ||||
-rw-r--r-- | arch/arm/crypto/bsaes-armv7.pl | 2467 |
7 files changed, 5485 insertions, 21 deletions
diff --git a/arch/arm/crypto/Makefile b/arch/arm/crypto/Makefile index a2c83851bc90..81cda39860c5 100644 --- a/arch/arm/crypto/Makefile +++ b/arch/arm/crypto/Makefile | |||
@@ -3,7 +3,17 @@ | |||
3 | # | 3 | # |
4 | 4 | ||
5 | obj-$(CONFIG_CRYPTO_AES_ARM) += aes-arm.o | 5 | obj-$(CONFIG_CRYPTO_AES_ARM) += aes-arm.o |
6 | obj-$(CONFIG_CRYPTO_AES_ARM_BS) += aes-arm-bs.o | ||
6 | obj-$(CONFIG_CRYPTO_SHA1_ARM) += sha1-arm.o | 7 | obj-$(CONFIG_CRYPTO_SHA1_ARM) += sha1-arm.o |
7 | 8 | ||
8 | aes-arm-y := aes-armv4.o aes_glue.o | 9 | aes-arm-y := aes-armv4.o aes_glue.o |
9 | sha1-arm-y := sha1-armv4-large.o sha1_glue.o | 10 | aes-arm-bs-y := aesbs-core.o aesbs-glue.o |
11 | sha1-arm-y := sha1-armv4-large.o sha1_glue.o | ||
12 | |||
13 | quiet_cmd_perl = PERL $@ | ||
14 | cmd_perl = $(PERL) $(<) > $(@) | ||
15 | |||
16 | $(src)/aesbs-core.S_shipped: $(src)/bsaes-armv7.pl | ||
17 | $(call cmd,perl) | ||
18 | |||
19 | .PRECIOUS: $(obj)/aesbs-core.S | ||
diff --git a/arch/arm/crypto/aes-armv4.S b/arch/arm/crypto/aes-armv4.S index 19d6cd6f29f9..3a14ea8fe97e 100644 --- a/arch/arm/crypto/aes-armv4.S +++ b/arch/arm/crypto/aes-armv4.S | |||
@@ -148,7 +148,7 @@ AES_Te: | |||
148 | @ const AES_KEY *key) { | 148 | @ const AES_KEY *key) { |
149 | .align 5 | 149 | .align 5 |
150 | ENTRY(AES_encrypt) | 150 | ENTRY(AES_encrypt) |
151 | sub r3,pc,#8 @ AES_encrypt | 151 | adr r3,AES_encrypt |
152 | stmdb sp!,{r1,r4-r12,lr} | 152 | stmdb sp!,{r1,r4-r12,lr} |
153 | mov r12,r0 @ inp | 153 | mov r12,r0 @ inp |
154 | mov r11,r2 | 154 | mov r11,r2 |
@@ -381,7 +381,7 @@ _armv4_AES_encrypt: | |||
381 | .align 5 | 381 | .align 5 |
382 | ENTRY(private_AES_set_encrypt_key) | 382 | ENTRY(private_AES_set_encrypt_key) |
383 | _armv4_AES_set_encrypt_key: | 383 | _armv4_AES_set_encrypt_key: |
384 | sub r3,pc,#8 @ AES_set_encrypt_key | 384 | adr r3,_armv4_AES_set_encrypt_key |
385 | teq r0,#0 | 385 | teq r0,#0 |
386 | moveq r0,#-1 | 386 | moveq r0,#-1 |
387 | beq .Labrt | 387 | beq .Labrt |
@@ -843,7 +843,7 @@ AES_Td: | |||
843 | @ const AES_KEY *key) { | 843 | @ const AES_KEY *key) { |
844 | .align 5 | 844 | .align 5 |
845 | ENTRY(AES_decrypt) | 845 | ENTRY(AES_decrypt) |
846 | sub r3,pc,#8 @ AES_decrypt | 846 | adr r3,AES_decrypt |
847 | stmdb sp!,{r1,r4-r12,lr} | 847 | stmdb sp!,{r1,r4-r12,lr} |
848 | mov r12,r0 @ inp | 848 | mov r12,r0 @ inp |
849 | mov r11,r2 | 849 | mov r11,r2 |
diff --git a/arch/arm/crypto/aes_glue.c b/arch/arm/crypto/aes_glue.c index 59f7877ead6a..3003fa1f6fb4 100644 --- a/arch/arm/crypto/aes_glue.c +++ b/arch/arm/crypto/aes_glue.c | |||
@@ -6,22 +6,12 @@ | |||
6 | #include <linux/crypto.h> | 6 | #include <linux/crypto.h> |
7 | #include <crypto/aes.h> | 7 | #include <crypto/aes.h> |
8 | 8 | ||
9 | #define AES_MAXNR 14 | 9 | #include "aes_glue.h" |
10 | 10 | ||
11 | typedef struct { | 11 | EXPORT_SYMBOL(AES_encrypt); |
12 | unsigned int rd_key[4 *(AES_MAXNR + 1)]; | 12 | EXPORT_SYMBOL(AES_decrypt); |
13 | int rounds; | 13 | EXPORT_SYMBOL(private_AES_set_encrypt_key); |
14 | } AES_KEY; | 14 | EXPORT_SYMBOL(private_AES_set_decrypt_key); |
15 | |||
16 | struct AES_CTX { | ||
17 | AES_KEY enc_key; | ||
18 | AES_KEY dec_key; | ||
19 | }; | ||
20 | |||
21 | asmlinkage void AES_encrypt(const u8 *in, u8 *out, AES_KEY *ctx); | ||
22 | asmlinkage void AES_decrypt(const u8 *in, u8 *out, AES_KEY *ctx); | ||
23 | asmlinkage int private_AES_set_decrypt_key(const unsigned char *userKey, const int bits, AES_KEY *key); | ||
24 | asmlinkage int private_AES_set_encrypt_key(const unsigned char *userKey, const int bits, AES_KEY *key); | ||
25 | 15 | ||
26 | static void aes_encrypt(struct crypto_tfm *tfm, u8 *dst, const u8 *src) | 16 | static void aes_encrypt(struct crypto_tfm *tfm, u8 *dst, const u8 *src) |
27 | { | 17 | { |
@@ -81,7 +71,7 @@ static struct crypto_alg aes_alg = { | |||
81 | .cipher = { | 71 | .cipher = { |
82 | .cia_min_keysize = AES_MIN_KEY_SIZE, | 72 | .cia_min_keysize = AES_MIN_KEY_SIZE, |
83 | .cia_max_keysize = AES_MAX_KEY_SIZE, | 73 | .cia_max_keysize = AES_MAX_KEY_SIZE, |
84 | .cia_setkey = aes_set_key, | 74 | .cia_setkey = aes_set_key, |
85 | .cia_encrypt = aes_encrypt, | 75 | .cia_encrypt = aes_encrypt, |
86 | .cia_decrypt = aes_decrypt | 76 | .cia_decrypt = aes_decrypt |
87 | } | 77 | } |
diff --git a/arch/arm/crypto/aes_glue.h b/arch/arm/crypto/aes_glue.h new file mode 100644 index 000000000000..cca3e51eb606 --- /dev/null +++ b/arch/arm/crypto/aes_glue.h | |||
@@ -0,0 +1,19 @@ | |||
1 | |||
2 | #define AES_MAXNR 14 | ||
3 | |||
4 | struct AES_KEY { | ||
5 | unsigned int rd_key[4 * (AES_MAXNR + 1)]; | ||
6 | int rounds; | ||
7 | }; | ||
8 | |||
9 | struct AES_CTX { | ||
10 | struct AES_KEY enc_key; | ||
11 | struct AES_KEY dec_key; | ||
12 | }; | ||
13 | |||
14 | asmlinkage void AES_encrypt(const u8 *in, u8 *out, struct AES_KEY *ctx); | ||
15 | asmlinkage void AES_decrypt(const u8 *in, u8 *out, struct AES_KEY *ctx); | ||
16 | asmlinkage int private_AES_set_decrypt_key(const unsigned char *userKey, | ||
17 | const int bits, struct AES_KEY *key); | ||
18 | asmlinkage int private_AES_set_encrypt_key(const unsigned char *userKey, | ||
19 | const int bits, struct AES_KEY *key); | ||
diff --git a/arch/arm/crypto/aesbs-core.S_shipped b/arch/arm/crypto/aesbs-core.S_shipped new file mode 100644 index 000000000000..64205d453260 --- /dev/null +++ b/arch/arm/crypto/aesbs-core.S_shipped | |||
@@ -0,0 +1,2544 @@ | |||
1 | |||
2 | @ ==================================================================== | ||
3 | @ Written by Andy Polyakov <appro@openssl.org> for the OpenSSL | ||
4 | @ project. The module is, however, dual licensed under OpenSSL and | ||
5 | @ CRYPTOGAMS licenses depending on where you obtain it. For further | ||
6 | @ details see http://www.openssl.org/~appro/cryptogams/. | ||
7 | @ | ||
8 | @ Specific modes and adaptation for Linux kernel by Ard Biesheuvel | ||
9 | @ <ard.biesheuvel@linaro.org>. Permission to use under GPL terms is | ||
10 | @ granted. | ||
11 | @ ==================================================================== | ||
12 | |||
13 | @ Bit-sliced AES for ARM NEON | ||
14 | @ | ||
15 | @ February 2012. | ||
16 | @ | ||
17 | @ This implementation is direct adaptation of bsaes-x86_64 module for | ||
18 | @ ARM NEON. Except that this module is endian-neutral [in sense that | ||
19 | @ it can be compiled for either endianness] by courtesy of vld1.8's | ||
20 | @ neutrality. Initial version doesn't implement interface to OpenSSL, | ||
21 | @ only low-level primitives and unsupported entry points, just enough | ||
22 | @ to collect performance results, which for Cortex-A8 core are: | ||
23 | @ | ||
24 | @ encrypt 19.5 cycles per byte processed with 128-bit key | ||
25 | @ decrypt 22.1 cycles per byte processed with 128-bit key | ||
26 | @ key conv. 440 cycles per 128-bit key/0.18 of 8x block | ||
27 | @ | ||
28 | @ Snapdragon S4 encrypts byte in 17.6 cycles and decrypts in 19.7, | ||
29 | @ which is [much] worse than anticipated (for further details see | ||
30 | @ http://www.openssl.org/~appro/Snapdragon-S4.html). | ||
31 | @ | ||
32 | @ Cortex-A15 manages in 14.2/16.1 cycles [when integer-only code | ||
33 | @ manages in 20.0 cycles]. | ||
34 | @ | ||
35 | @ When comparing to x86_64 results keep in mind that NEON unit is | ||
36 | @ [mostly] single-issue and thus can't [fully] benefit from | ||
37 | @ instruction-level parallelism. And when comparing to aes-armv4 | ||
38 | @ results keep in mind key schedule conversion overhead (see | ||
39 | @ bsaes-x86_64.pl for further details)... | ||
40 | @ | ||
41 | @ <appro@openssl.org> | ||
42 | |||
43 | @ April-August 2013 | ||
44 | @ | ||
45 | @ Add CBC, CTR and XTS subroutines, adapt for kernel use. | ||
46 | @ | ||
47 | @ <ard.biesheuvel@linaro.org> | ||
48 | |||
49 | #ifndef __KERNEL__ | ||
50 | # include "arm_arch.h" | ||
51 | |||
52 | # define VFP_ABI_PUSH vstmdb sp!,{d8-d15} | ||
53 | # define VFP_ABI_POP vldmia sp!,{d8-d15} | ||
54 | # define VFP_ABI_FRAME 0x40 | ||
55 | #else | ||
56 | # define VFP_ABI_PUSH | ||
57 | # define VFP_ABI_POP | ||
58 | # define VFP_ABI_FRAME 0 | ||
59 | # define BSAES_ASM_EXTENDED_KEY | ||
60 | # define XTS_CHAIN_TWEAK | ||
61 | # define __ARM_ARCH__ __LINUX_ARM_ARCH__ | ||
62 | #endif | ||
63 | |||
64 | #ifdef __thumb__ | ||
65 | # define adrl adr | ||
66 | #endif | ||
67 | |||
68 | #if __ARM_ARCH__>=7 | ||
69 | .text | ||
70 | .syntax unified @ ARMv7-capable assembler is expected to handle this | ||
71 | #ifdef __thumb2__ | ||
72 | .thumb | ||
73 | #else | ||
74 | .code 32 | ||
75 | #endif | ||
76 | |||
77 | .fpu neon | ||
78 | |||
79 | .type _bsaes_decrypt8,%function | ||
80 | .align 4 | ||
81 | _bsaes_decrypt8: | ||
82 | adr r6,_bsaes_decrypt8 | ||
83 | vldmia r4!, {q9} @ round 0 key | ||
84 | add r6,r6,#.LM0ISR-_bsaes_decrypt8 | ||
85 | |||
86 | vldmia r6!, {q8} @ .LM0ISR | ||
87 | veor q10, q0, q9 @ xor with round0 key | ||
88 | veor q11, q1, q9 | ||
89 | vtbl.8 d0, {q10}, d16 | ||
90 | vtbl.8 d1, {q10}, d17 | ||
91 | veor q12, q2, q9 | ||
92 | vtbl.8 d2, {q11}, d16 | ||
93 | vtbl.8 d3, {q11}, d17 | ||
94 | veor q13, q3, q9 | ||
95 | vtbl.8 d4, {q12}, d16 | ||
96 | vtbl.8 d5, {q12}, d17 | ||
97 | veor q14, q4, q9 | ||
98 | vtbl.8 d6, {q13}, d16 | ||
99 | vtbl.8 d7, {q13}, d17 | ||
100 | veor q15, q5, q9 | ||
101 | vtbl.8 d8, {q14}, d16 | ||
102 | vtbl.8 d9, {q14}, d17 | ||
103 | veor q10, q6, q9 | ||
104 | vtbl.8 d10, {q15}, d16 | ||
105 | vtbl.8 d11, {q15}, d17 | ||
106 | veor q11, q7, q9 | ||
107 | vtbl.8 d12, {q10}, d16 | ||
108 | vtbl.8 d13, {q10}, d17 | ||
109 | vtbl.8 d14, {q11}, d16 | ||
110 | vtbl.8 d15, {q11}, d17 | ||
111 | vmov.i8 q8,#0x55 @ compose .LBS0 | ||
112 | vmov.i8 q9,#0x33 @ compose .LBS1 | ||
113 | vshr.u64 q10, q6, #1 | ||
114 | vshr.u64 q11, q4, #1 | ||
115 | veor q10, q10, q7 | ||
116 | veor q11, q11, q5 | ||
117 | vand q10, q10, q8 | ||
118 | vand q11, q11, q8 | ||
119 | veor q7, q7, q10 | ||
120 | vshl.u64 q10, q10, #1 | ||
121 | veor q5, q5, q11 | ||
122 | vshl.u64 q11, q11, #1 | ||
123 | veor q6, q6, q10 | ||
124 | veor q4, q4, q11 | ||
125 | vshr.u64 q10, q2, #1 | ||
126 | vshr.u64 q11, q0, #1 | ||
127 | veor q10, q10, q3 | ||
128 | veor q11, q11, q1 | ||
129 | vand q10, q10, q8 | ||
130 | vand q11, q11, q8 | ||
131 | veor q3, q3, q10 | ||
132 | vshl.u64 q10, q10, #1 | ||
133 | veor q1, q1, q11 | ||
134 | vshl.u64 q11, q11, #1 | ||
135 | veor q2, q2, q10 | ||
136 | veor q0, q0, q11 | ||
137 | vmov.i8 q8,#0x0f @ compose .LBS2 | ||
138 | vshr.u64 q10, q5, #2 | ||
139 | vshr.u64 q11, q4, #2 | ||
140 | veor q10, q10, q7 | ||
141 | veor q11, q11, q6 | ||
142 | vand q10, q10, q9 | ||
143 | vand q11, q11, q9 | ||
144 | veor q7, q7, q10 | ||
145 | vshl.u64 q10, q10, #2 | ||
146 | veor q6, q6, q11 | ||
147 | vshl.u64 q11, q11, #2 | ||
148 | veor q5, q5, q10 | ||
149 | veor q4, q4, q11 | ||
150 | vshr.u64 q10, q1, #2 | ||
151 | vshr.u64 q11, q0, #2 | ||
152 | veor q10, q10, q3 | ||
153 | veor q11, q11, q2 | ||
154 | vand q10, q10, q9 | ||
155 | vand q11, q11, q9 | ||
156 | veor q3, q3, q10 | ||
157 | vshl.u64 q10, q10, #2 | ||
158 | veor q2, q2, q11 | ||
159 | vshl.u64 q11, q11, #2 | ||
160 | veor q1, q1, q10 | ||
161 | veor q0, q0, q11 | ||
162 | vshr.u64 q10, q3, #4 | ||
163 | vshr.u64 q11, q2, #4 | ||
164 | veor q10, q10, q7 | ||
165 | veor q11, q11, q6 | ||
166 | vand q10, q10, q8 | ||
167 | vand q11, q11, q8 | ||
168 | veor q7, q7, q10 | ||
169 | vshl.u64 q10, q10, #4 | ||
170 | veor q6, q6, q11 | ||
171 | vshl.u64 q11, q11, #4 | ||
172 | veor q3, q3, q10 | ||
173 | veor q2, q2, q11 | ||
174 | vshr.u64 q10, q1, #4 | ||
175 | vshr.u64 q11, q0, #4 | ||
176 | veor q10, q10, q5 | ||
177 | veor q11, q11, q4 | ||
178 | vand q10, q10, q8 | ||
179 | vand q11, q11, q8 | ||
180 | veor q5, q5, q10 | ||
181 | vshl.u64 q10, q10, #4 | ||
182 | veor q4, q4, q11 | ||
183 | vshl.u64 q11, q11, #4 | ||
184 | veor q1, q1, q10 | ||
185 | veor q0, q0, q11 | ||
186 | sub r5,r5,#1 | ||
187 | b .Ldec_sbox | ||
188 | .align 4 | ||
189 | .Ldec_loop: | ||
190 | vldmia r4!, {q8-q11} | ||
191 | veor q8, q8, q0 | ||
192 | veor q9, q9, q1 | ||
193 | vtbl.8 d0, {q8}, d24 | ||
194 | vtbl.8 d1, {q8}, d25 | ||
195 | vldmia r4!, {q8} | ||
196 | veor q10, q10, q2 | ||
197 | vtbl.8 d2, {q9}, d24 | ||
198 | vtbl.8 d3, {q9}, d25 | ||
199 | vldmia r4!, {q9} | ||
200 | veor q11, q11, q3 | ||
201 | vtbl.8 d4, {q10}, d24 | ||
202 | vtbl.8 d5, {q10}, d25 | ||
203 | vldmia r4!, {q10} | ||
204 | vtbl.8 d6, {q11}, d24 | ||
205 | vtbl.8 d7, {q11}, d25 | ||
206 | vldmia r4!, {q11} | ||
207 | veor q8, q8, q4 | ||
208 | veor q9, q9, q5 | ||
209 | vtbl.8 d8, {q8}, d24 | ||
210 | vtbl.8 d9, {q8}, d25 | ||
211 | veor q10, q10, q6 | ||
212 | vtbl.8 d10, {q9}, d24 | ||
213 | vtbl.8 d11, {q9}, d25 | ||
214 | veor q11, q11, q7 | ||
215 | vtbl.8 d12, {q10}, d24 | ||
216 | vtbl.8 d13, {q10}, d25 | ||
217 | vtbl.8 d14, {q11}, d24 | ||
218 | vtbl.8 d15, {q11}, d25 | ||
219 | .Ldec_sbox: | ||
220 | veor q1, q1, q4 | ||
221 | veor q3, q3, q4 | ||
222 | |||
223 | veor q4, q4, q7 | ||
224 | veor q1, q1, q6 | ||
225 | veor q2, q2, q7 | ||
226 | veor q6, q6, q4 | ||
227 | |||
228 | veor q0, q0, q1 | ||
229 | veor q2, q2, q5 | ||
230 | veor q7, q7, q6 | ||
231 | veor q3, q3, q0 | ||
232 | veor q5, q5, q0 | ||
233 | veor q1, q1, q3 | ||
234 | veor q11, q3, q0 | ||
235 | veor q10, q7, q4 | ||
236 | veor q9, q1, q6 | ||
237 | veor q13, q4, q0 | ||
238 | vmov q8, q10 | ||
239 | veor q12, q5, q2 | ||
240 | |||
241 | vorr q10, q10, q9 | ||
242 | veor q15, q11, q8 | ||
243 | vand q14, q11, q12 | ||
244 | vorr q11, q11, q12 | ||
245 | veor q12, q12, q9 | ||
246 | vand q8, q8, q9 | ||
247 | veor q9, q6, q2 | ||
248 | vand q15, q15, q12 | ||
249 | vand q13, q13, q9 | ||
250 | veor q9, q3, q7 | ||
251 | veor q12, q1, q5 | ||
252 | veor q11, q11, q13 | ||
253 | veor q10, q10, q13 | ||
254 | vand q13, q9, q12 | ||
255 | vorr q9, q9, q12 | ||
256 | veor q11, q11, q15 | ||
257 | veor q8, q8, q13 | ||
258 | veor q10, q10, q14 | ||
259 | veor q9, q9, q15 | ||
260 | veor q8, q8, q14 | ||
261 | vand q12, q4, q6 | ||
262 | veor q9, q9, q14 | ||
263 | vand q13, q0, q2 | ||
264 | vand q14, q7, q1 | ||
265 | vorr q15, q3, q5 | ||
266 | veor q11, q11, q12 | ||
267 | veor q9, q9, q14 | ||
268 | veor q8, q8, q15 | ||
269 | veor q10, q10, q13 | ||
270 | |||
271 | @ Inv_GF16 0, 1, 2, 3, s0, s1, s2, s3 | ||
272 | |||
273 | @ new smaller inversion | ||
274 | |||
275 | vand q14, q11, q9 | ||
276 | vmov q12, q8 | ||
277 | |||
278 | veor q13, q10, q14 | ||
279 | veor q15, q8, q14 | ||
280 | veor q14, q8, q14 @ q14=q15 | ||
281 | |||
282 | vbsl q13, q9, q8 | ||
283 | vbsl q15, q11, q10 | ||
284 | veor q11, q11, q10 | ||
285 | |||
286 | vbsl q12, q13, q14 | ||
287 | vbsl q8, q14, q13 | ||
288 | |||
289 | vand q14, q12, q15 | ||
290 | veor q9, q9, q8 | ||
291 | |||
292 | veor q14, q14, q11 | ||
293 | veor q12, q5, q2 | ||
294 | veor q8, q1, q6 | ||
295 | veor q10, q15, q14 | ||
296 | vand q10, q10, q5 | ||
297 | veor q5, q5, q1 | ||
298 | vand q11, q1, q15 | ||
299 | vand q5, q5, q14 | ||
300 | veor q1, q11, q10 | ||
301 | veor q5, q5, q11 | ||
302 | veor q15, q15, q13 | ||
303 | veor q14, q14, q9 | ||
304 | veor q11, q15, q14 | ||
305 | veor q10, q13, q9 | ||
306 | vand q11, q11, q12 | ||
307 | vand q10, q10, q2 | ||
308 | veor q12, q12, q8 | ||
309 | veor q2, q2, q6 | ||
310 | vand q8, q8, q15 | ||
311 | vand q6, q6, q13 | ||
312 | vand q12, q12, q14 | ||
313 | vand q2, q2, q9 | ||
314 | veor q8, q8, q12 | ||
315 | veor q2, q2, q6 | ||
316 | veor q12, q12, q11 | ||
317 | veor q6, q6, q10 | ||
318 | veor q5, q5, q12 | ||
319 | veor q2, q2, q12 | ||
320 | veor q1, q1, q8 | ||
321 | veor q6, q6, q8 | ||
322 | |||
323 | veor q12, q3, q0 | ||
324 | veor q8, q7, q4 | ||
325 | veor q11, q15, q14 | ||
326 | veor q10, q13, q9 | ||
327 | vand q11, q11, q12 | ||
328 | vand q10, q10, q0 | ||
329 | veor q12, q12, q8 | ||
330 | veor q0, q0, q4 | ||
331 | vand q8, q8, q15 | ||
332 | vand q4, q4, q13 | ||
333 | vand q12, q12, q14 | ||
334 | vand q0, q0, q9 | ||
335 | veor q8, q8, q12 | ||
336 | veor q0, q0, q4 | ||
337 | veor q12, q12, q11 | ||
338 | veor q4, q4, q10 | ||
339 | veor q15, q15, q13 | ||
340 | veor q14, q14, q9 | ||
341 | veor q10, q15, q14 | ||
342 | vand q10, q10, q3 | ||
343 | veor q3, q3, q7 | ||
344 | vand q11, q7, q15 | ||
345 | vand q3, q3, q14 | ||
346 | veor q7, q11, q10 | ||
347 | veor q3, q3, q11 | ||
348 | veor q3, q3, q12 | ||
349 | veor q0, q0, q12 | ||
350 | veor q7, q7, q8 | ||
351 | veor q4, q4, q8 | ||
352 | veor q1, q1, q7 | ||
353 | veor q6, q6, q5 | ||
354 | |||
355 | veor q4, q4, q1 | ||
356 | veor q2, q2, q7 | ||
357 | veor q5, q5, q7 | ||
358 | veor q4, q4, q2 | ||
359 | veor q7, q7, q0 | ||
360 | veor q4, q4, q5 | ||
361 | veor q3, q3, q6 | ||
362 | veor q6, q6, q1 | ||
363 | veor q3, q3, q4 | ||
364 | |||
365 | veor q4, q4, q0 | ||
366 | veor q7, q7, q3 | ||
367 | subs r5,r5,#1 | ||
368 | bcc .Ldec_done | ||
369 | @ multiplication by 0x05-0x00-0x04-0x00 | ||
370 | vext.8 q8, q0, q0, #8 | ||
371 | vext.8 q14, q3, q3, #8 | ||
372 | vext.8 q15, q5, q5, #8 | ||
373 | veor q8, q8, q0 | ||
374 | vext.8 q9, q1, q1, #8 | ||
375 | veor q14, q14, q3 | ||
376 | vext.8 q10, q6, q6, #8 | ||
377 | veor q15, q15, q5 | ||
378 | vext.8 q11, q4, q4, #8 | ||
379 | veor q9, q9, q1 | ||
380 | vext.8 q12, q2, q2, #8 | ||
381 | veor q10, q10, q6 | ||
382 | vext.8 q13, q7, q7, #8 | ||
383 | veor q11, q11, q4 | ||
384 | veor q12, q12, q2 | ||
385 | veor q13, q13, q7 | ||
386 | |||
387 | veor q0, q0, q14 | ||
388 | veor q1, q1, q14 | ||
389 | veor q6, q6, q8 | ||
390 | veor q2, q2, q10 | ||
391 | veor q4, q4, q9 | ||
392 | veor q1, q1, q15 | ||
393 | veor q6, q6, q15 | ||
394 | veor q2, q2, q14 | ||
395 | veor q7, q7, q11 | ||
396 | veor q4, q4, q14 | ||
397 | veor q3, q3, q12 | ||
398 | veor q2, q2, q15 | ||
399 | veor q7, q7, q15 | ||
400 | veor q5, q5, q13 | ||
401 | vext.8 q8, q0, q0, #12 @ x0 <<< 32 | ||
402 | vext.8 q9, q1, q1, #12 | ||
403 | veor q0, q0, q8 @ x0 ^ (x0 <<< 32) | ||
404 | vext.8 q10, q6, q6, #12 | ||
405 | veor q1, q1, q9 | ||
406 | vext.8 q11, q4, q4, #12 | ||
407 | veor q6, q6, q10 | ||
408 | vext.8 q12, q2, q2, #12 | ||
409 | veor q4, q4, q11 | ||
410 | vext.8 q13, q7, q7, #12 | ||
411 | veor q2, q2, q12 | ||
412 | vext.8 q14, q3, q3, #12 | ||
413 | veor q7, q7, q13 | ||
414 | vext.8 q15, q5, q5, #12 | ||
415 | veor q3, q3, q14 | ||
416 | |||
417 | veor q9, q9, q0 | ||
418 | veor q5, q5, q15 | ||
419 | vext.8 q0, q0, q0, #8 @ (x0 ^ (x0 <<< 32)) <<< 64) | ||
420 | veor q10, q10, q1 | ||
421 | veor q8, q8, q5 | ||
422 | veor q9, q9, q5 | ||
423 | vext.8 q1, q1, q1, #8 | ||
424 | veor q13, q13, q2 | ||
425 | veor q0, q0, q8 | ||
426 | veor q14, q14, q7 | ||
427 | veor q1, q1, q9 | ||
428 | vext.8 q8, q2, q2, #8 | ||
429 | veor q12, q12, q4 | ||
430 | vext.8 q9, q7, q7, #8 | ||
431 | veor q15, q15, q3 | ||
432 | vext.8 q2, q4, q4, #8 | ||
433 | veor q11, q11, q6 | ||
434 | vext.8 q7, q5, q5, #8 | ||
435 | veor q12, q12, q5 | ||
436 | vext.8 q4, q3, q3, #8 | ||
437 | veor q11, q11, q5 | ||
438 | vext.8 q3, q6, q6, #8 | ||
439 | veor q5, q9, q13 | ||
440 | veor q11, q11, q2 | ||
441 | veor q7, q7, q15 | ||
442 | veor q6, q4, q14 | ||
443 | veor q4, q8, q12 | ||
444 | veor q2, q3, q10 | ||
445 | vmov q3, q11 | ||
446 | @ vmov q5, q9 | ||
447 | vldmia r6, {q12} @ .LISR | ||
448 | ite eq @ Thumb2 thing, sanity check in ARM | ||
449 | addeq r6,r6,#0x10 | ||
450 | bne .Ldec_loop | ||
451 | vldmia r6, {q12} @ .LISRM0 | ||
452 | b .Ldec_loop | ||
453 | .align 4 | ||
454 | .Ldec_done: | ||
455 | vmov.i8 q8,#0x55 @ compose .LBS0 | ||
456 | vmov.i8 q9,#0x33 @ compose .LBS1 | ||
457 | vshr.u64 q10, q3, #1 | ||
458 | vshr.u64 q11, q2, #1 | ||
459 | veor q10, q10, q5 | ||
460 | veor q11, q11, q7 | ||
461 | vand q10, q10, q8 | ||
462 | vand q11, q11, q8 | ||
463 | veor q5, q5, q10 | ||
464 | vshl.u64 q10, q10, #1 | ||
465 | veor q7, q7, q11 | ||
466 | vshl.u64 q11, q11, #1 | ||
467 | veor q3, q3, q10 | ||
468 | veor q2, q2, q11 | ||
469 | vshr.u64 q10, q6, #1 | ||
470 | vshr.u64 q11, q0, #1 | ||
471 | veor q10, q10, q4 | ||
472 | veor q11, q11, q1 | ||
473 | vand q10, q10, q8 | ||
474 | vand q11, q11, q8 | ||
475 | veor q4, q4, q10 | ||
476 | vshl.u64 q10, q10, #1 | ||
477 | veor q1, q1, q11 | ||
478 | vshl.u64 q11, q11, #1 | ||
479 | veor q6, q6, q10 | ||
480 | veor q0, q0, q11 | ||
481 | vmov.i8 q8,#0x0f @ compose .LBS2 | ||
482 | vshr.u64 q10, q7, #2 | ||
483 | vshr.u64 q11, q2, #2 | ||
484 | veor q10, q10, q5 | ||
485 | veor q11, q11, q3 | ||
486 | vand q10, q10, q9 | ||
487 | vand q11, q11, q9 | ||
488 | veor q5, q5, q10 | ||
489 | vshl.u64 q10, q10, #2 | ||
490 | veor q3, q3, q11 | ||
491 | vshl.u64 q11, q11, #2 | ||
492 | veor q7, q7, q10 | ||
493 | veor q2, q2, q11 | ||
494 | vshr.u64 q10, q1, #2 | ||
495 | vshr.u64 q11, q0, #2 | ||
496 | veor q10, q10, q4 | ||
497 | veor q11, q11, q6 | ||
498 | vand q10, q10, q9 | ||
499 | vand q11, q11, q9 | ||
500 | veor q4, q4, q10 | ||
501 | vshl.u64 q10, q10, #2 | ||
502 | veor q6, q6, q11 | ||
503 | vshl.u64 q11, q11, #2 | ||
504 | veor q1, q1, q10 | ||
505 | veor q0, q0, q11 | ||
506 | vshr.u64 q10, q4, #4 | ||
507 | vshr.u64 q11, q6, #4 | ||
508 | veor q10, q10, q5 | ||
509 | veor q11, q11, q3 | ||
510 | vand q10, q10, q8 | ||
511 | vand q11, q11, q8 | ||
512 | veor q5, q5, q10 | ||
513 | vshl.u64 q10, q10, #4 | ||
514 | veor q3, q3, q11 | ||
515 | vshl.u64 q11, q11, #4 | ||
516 | veor q4, q4, q10 | ||
517 | veor q6, q6, q11 | ||
518 | vshr.u64 q10, q1, #4 | ||
519 | vshr.u64 q11, q0, #4 | ||
520 | veor q10, q10, q7 | ||
521 | veor q11, q11, q2 | ||
522 | vand q10, q10, q8 | ||
523 | vand q11, q11, q8 | ||
524 | veor q7, q7, q10 | ||
525 | vshl.u64 q10, q10, #4 | ||
526 | veor q2, q2, q11 | ||
527 | vshl.u64 q11, q11, #4 | ||
528 | veor q1, q1, q10 | ||
529 | veor q0, q0, q11 | ||
530 | vldmia r4, {q8} @ last round key | ||
531 | veor q6, q6, q8 | ||
532 | veor q4, q4, q8 | ||
533 | veor q2, q2, q8 | ||
534 | veor q7, q7, q8 | ||
535 | veor q3, q3, q8 | ||
536 | veor q5, q5, q8 | ||
537 | veor q0, q0, q8 | ||
538 | veor q1, q1, q8 | ||
539 | bx lr | ||
540 | .size _bsaes_decrypt8,.-_bsaes_decrypt8 | ||
541 | |||
542 | .type _bsaes_const,%object | ||
543 | .align 6 | ||
544 | _bsaes_const: | ||
545 | .LM0ISR: @ InvShiftRows constants | ||
546 | .quad 0x0a0e0206070b0f03, 0x0004080c0d010509 | ||
547 | .LISR: | ||
548 | .quad 0x0504070602010003, 0x0f0e0d0c080b0a09 | ||
549 | .LISRM0: | ||
550 | .quad 0x01040b0e0205080f, 0x0306090c00070a0d | ||
551 | .LM0SR: @ ShiftRows constants | ||
552 | .quad 0x0a0e02060f03070b, 0x0004080c05090d01 | ||
553 | .LSR: | ||
554 | .quad 0x0504070600030201, 0x0f0e0d0c0a09080b | ||
555 | .LSRM0: | ||
556 | .quad 0x0304090e00050a0f, 0x01060b0c0207080d | ||
557 | .LM0: | ||
558 | .quad 0x02060a0e03070b0f, 0x0004080c0105090d | ||
559 | .LREVM0SR: | ||
560 | .quad 0x090d01050c000408, 0x03070b0f060a0e02 | ||
561 | .asciz "Bit-sliced AES for NEON, CRYPTOGAMS by <appro@openssl.org>" | ||
562 | .align 6 | ||
563 | .size _bsaes_const,.-_bsaes_const | ||
564 | |||
565 | .type _bsaes_encrypt8,%function | ||
566 | .align 4 | ||
567 | _bsaes_encrypt8: | ||
568 | adr r6,_bsaes_encrypt8 | ||
569 | vldmia r4!, {q9} @ round 0 key | ||
570 | sub r6,r6,#_bsaes_encrypt8-.LM0SR | ||
571 | |||
572 | vldmia r6!, {q8} @ .LM0SR | ||
573 | _bsaes_encrypt8_alt: | ||
574 | veor q10, q0, q9 @ xor with round0 key | ||
575 | veor q11, q1, q9 | ||
576 | vtbl.8 d0, {q10}, d16 | ||
577 | vtbl.8 d1, {q10}, d17 | ||
578 | veor q12, q2, q9 | ||
579 | vtbl.8 d2, {q11}, d16 | ||
580 | vtbl.8 d3, {q11}, d17 | ||
581 | veor q13, q3, q9 | ||
582 | vtbl.8 d4, {q12}, d16 | ||
583 | vtbl.8 d5, {q12}, d17 | ||
584 | veor q14, q4, q9 | ||
585 | vtbl.8 d6, {q13}, d16 | ||
586 | vtbl.8 d7, {q13}, d17 | ||
587 | veor q15, q5, q9 | ||
588 | vtbl.8 d8, {q14}, d16 | ||
589 | vtbl.8 d9, {q14}, d17 | ||
590 | veor q10, q6, q9 | ||
591 | vtbl.8 d10, {q15}, d16 | ||
592 | vtbl.8 d11, {q15}, d17 | ||
593 | veor q11, q7, q9 | ||
594 | vtbl.8 d12, {q10}, d16 | ||
595 | vtbl.8 d13, {q10}, d17 | ||
596 | vtbl.8 d14, {q11}, d16 | ||
597 | vtbl.8 d15, {q11}, d17 | ||
598 | _bsaes_encrypt8_bitslice: | ||
599 | vmov.i8 q8,#0x55 @ compose .LBS0 | ||
600 | vmov.i8 q9,#0x33 @ compose .LBS1 | ||
601 | vshr.u64 q10, q6, #1 | ||
602 | vshr.u64 q11, q4, #1 | ||
603 | veor q10, q10, q7 | ||
604 | veor q11, q11, q5 | ||
605 | vand q10, q10, q8 | ||
606 | vand q11, q11, q8 | ||
607 | veor q7, q7, q10 | ||
608 | vshl.u64 q10, q10, #1 | ||
609 | veor q5, q5, q11 | ||
610 | vshl.u64 q11, q11, #1 | ||
611 | veor q6, q6, q10 | ||
612 | veor q4, q4, q11 | ||
613 | vshr.u64 q10, q2, #1 | ||
614 | vshr.u64 q11, q0, #1 | ||
615 | veor q10, q10, q3 | ||
616 | veor q11, q11, q1 | ||
617 | vand q10, q10, q8 | ||
618 | vand q11, q11, q8 | ||
619 | veor q3, q3, q10 | ||
620 | vshl.u64 q10, q10, #1 | ||
621 | veor q1, q1, q11 | ||
622 | vshl.u64 q11, q11, #1 | ||
623 | veor q2, q2, q10 | ||
624 | veor q0, q0, q11 | ||
625 | vmov.i8 q8,#0x0f @ compose .LBS2 | ||
626 | vshr.u64 q10, q5, #2 | ||
627 | vshr.u64 q11, q4, #2 | ||
628 | veor q10, q10, q7 | ||
629 | veor q11, q11, q6 | ||
630 | vand q10, q10, q9 | ||
631 | vand q11, q11, q9 | ||
632 | veor q7, q7, q10 | ||
633 | vshl.u64 q10, q10, #2 | ||
634 | veor q6, q6, q11 | ||
635 | vshl.u64 q11, q11, #2 | ||
636 | veor q5, q5, q10 | ||
637 | veor q4, q4, q11 | ||
638 | vshr.u64 q10, q1, #2 | ||
639 | vshr.u64 q11, q0, #2 | ||
640 | veor q10, q10, q3 | ||
641 | veor q11, q11, q2 | ||
642 | vand q10, q10, q9 | ||
643 | vand q11, q11, q9 | ||
644 | veor q3, q3, q10 | ||
645 | vshl.u64 q10, q10, #2 | ||
646 | veor q2, q2, q11 | ||
647 | vshl.u64 q11, q11, #2 | ||
648 | veor q1, q1, q10 | ||
649 | veor q0, q0, q11 | ||
650 | vshr.u64 q10, q3, #4 | ||
651 | vshr.u64 q11, q2, #4 | ||
652 | veor q10, q10, q7 | ||
653 | veor q11, q11, q6 | ||
654 | vand q10, q10, q8 | ||
655 | vand q11, q11, q8 | ||
656 | veor q7, q7, q10 | ||
657 | vshl.u64 q10, q10, #4 | ||
658 | veor q6, q6, q11 | ||
659 | vshl.u64 q11, q11, #4 | ||
660 | veor q3, q3, q10 | ||
661 | veor q2, q2, q11 | ||
662 | vshr.u64 q10, q1, #4 | ||
663 | vshr.u64 q11, q0, #4 | ||
664 | veor q10, q10, q5 | ||
665 | veor q11, q11, q4 | ||
666 | vand q10, q10, q8 | ||
667 | vand q11, q11, q8 | ||
668 | veor q5, q5, q10 | ||
669 | vshl.u64 q10, q10, #4 | ||
670 | veor q4, q4, q11 | ||
671 | vshl.u64 q11, q11, #4 | ||
672 | veor q1, q1, q10 | ||
673 | veor q0, q0, q11 | ||
674 | sub r5,r5,#1 | ||
675 | b .Lenc_sbox | ||
676 | .align 4 | ||
677 | .Lenc_loop: | ||
678 | vldmia r4!, {q8-q11} | ||
679 | veor q8, q8, q0 | ||
680 | veor q9, q9, q1 | ||
681 | vtbl.8 d0, {q8}, d24 | ||
682 | vtbl.8 d1, {q8}, d25 | ||
683 | vldmia r4!, {q8} | ||
684 | veor q10, q10, q2 | ||
685 | vtbl.8 d2, {q9}, d24 | ||
686 | vtbl.8 d3, {q9}, d25 | ||
687 | vldmia r4!, {q9} | ||
688 | veor q11, q11, q3 | ||
689 | vtbl.8 d4, {q10}, d24 | ||
690 | vtbl.8 d5, {q10}, d25 | ||
691 | vldmia r4!, {q10} | ||
692 | vtbl.8 d6, {q11}, d24 | ||
693 | vtbl.8 d7, {q11}, d25 | ||
694 | vldmia r4!, {q11} | ||
695 | veor q8, q8, q4 | ||
696 | veor q9, q9, q5 | ||
697 | vtbl.8 d8, {q8}, d24 | ||
698 | vtbl.8 d9, {q8}, d25 | ||
699 | veor q10, q10, q6 | ||
700 | vtbl.8 d10, {q9}, d24 | ||
701 | vtbl.8 d11, {q9}, d25 | ||
702 | veor q11, q11, q7 | ||
703 | vtbl.8 d12, {q10}, d24 | ||
704 | vtbl.8 d13, {q10}, d25 | ||
705 | vtbl.8 d14, {q11}, d24 | ||
706 | vtbl.8 d15, {q11}, d25 | ||
707 | .Lenc_sbox: | ||
708 | veor q2, q2, q1 | ||
709 | veor q5, q5, q6 | ||
710 | veor q3, q3, q0 | ||
711 | veor q6, q6, q2 | ||
712 | veor q5, q5, q0 | ||
713 | |||
714 | veor q6, q6, q3 | ||
715 | veor q3, q3, q7 | ||
716 | veor q7, q7, q5 | ||
717 | veor q3, q3, q4 | ||
718 | veor q4, q4, q5 | ||
719 | |||
720 | veor q2, q2, q7 | ||
721 | veor q3, q3, q1 | ||
722 | veor q1, q1, q5 | ||
723 | veor q11, q7, q4 | ||
724 | veor q10, q1, q2 | ||
725 | veor q9, q5, q3 | ||
726 | veor q13, q2, q4 | ||
727 | vmov q8, q10 | ||
728 | veor q12, q6, q0 | ||
729 | |||
730 | vorr q10, q10, q9 | ||
731 | veor q15, q11, q8 | ||
732 | vand q14, q11, q12 | ||
733 | vorr q11, q11, q12 | ||
734 | veor q12, q12, q9 | ||
735 | vand q8, q8, q9 | ||
736 | veor q9, q3, q0 | ||
737 | vand q15, q15, q12 | ||
738 | vand q13, q13, q9 | ||
739 | veor q9, q7, q1 | ||
740 | veor q12, q5, q6 | ||
741 | veor q11, q11, q13 | ||
742 | veor q10, q10, q13 | ||
743 | vand q13, q9, q12 | ||
744 | vorr q9, q9, q12 | ||
745 | veor q11, q11, q15 | ||
746 | veor q8, q8, q13 | ||
747 | veor q10, q10, q14 | ||
748 | veor q9, q9, q15 | ||
749 | veor q8, q8, q14 | ||
750 | vand q12, q2, q3 | ||
751 | veor q9, q9, q14 | ||
752 | vand q13, q4, q0 | ||
753 | vand q14, q1, q5 | ||
754 | vorr q15, q7, q6 | ||
755 | veor q11, q11, q12 | ||
756 | veor q9, q9, q14 | ||
757 | veor q8, q8, q15 | ||
758 | veor q10, q10, q13 | ||
759 | |||
760 | @ Inv_GF16 0, 1, 2, 3, s0, s1, s2, s3 | ||
761 | |||
762 | @ new smaller inversion | ||
763 | |||
764 | vand q14, q11, q9 | ||
765 | vmov q12, q8 | ||
766 | |||
767 | veor q13, q10, q14 | ||
768 | veor q15, q8, q14 | ||
769 | veor q14, q8, q14 @ q14=q15 | ||
770 | |||
771 | vbsl q13, q9, q8 | ||
772 | vbsl q15, q11, q10 | ||
773 | veor q11, q11, q10 | ||
774 | |||
775 | vbsl q12, q13, q14 | ||
776 | vbsl q8, q14, q13 | ||
777 | |||
778 | vand q14, q12, q15 | ||
779 | veor q9, q9, q8 | ||
780 | |||
781 | veor q14, q14, q11 | ||
782 | veor q12, q6, q0 | ||
783 | veor q8, q5, q3 | ||
784 | veor q10, q15, q14 | ||
785 | vand q10, q10, q6 | ||
786 | veor q6, q6, q5 | ||
787 | vand q11, q5, q15 | ||
788 | vand q6, q6, q14 | ||
789 | veor q5, q11, q10 | ||
790 | veor q6, q6, q11 | ||
791 | veor q15, q15, q13 | ||
792 | veor q14, q14, q9 | ||
793 | veor q11, q15, q14 | ||
794 | veor q10, q13, q9 | ||
795 | vand q11, q11, q12 | ||
796 | vand q10, q10, q0 | ||
797 | veor q12, q12, q8 | ||
798 | veor q0, q0, q3 | ||
799 | vand q8, q8, q15 | ||
800 | vand q3, q3, q13 | ||
801 | vand q12, q12, q14 | ||
802 | vand q0, q0, q9 | ||
803 | veor q8, q8, q12 | ||
804 | veor q0, q0, q3 | ||
805 | veor q12, q12, q11 | ||
806 | veor q3, q3, q10 | ||
807 | veor q6, q6, q12 | ||
808 | veor q0, q0, q12 | ||
809 | veor q5, q5, q8 | ||
810 | veor q3, q3, q8 | ||
811 | |||
812 | veor q12, q7, q4 | ||
813 | veor q8, q1, q2 | ||
814 | veor q11, q15, q14 | ||
815 | veor q10, q13, q9 | ||
816 | vand q11, q11, q12 | ||
817 | vand q10, q10, q4 | ||
818 | veor q12, q12, q8 | ||
819 | veor q4, q4, q2 | ||
820 | vand q8, q8, q15 | ||
821 | vand q2, q2, q13 | ||
822 | vand q12, q12, q14 | ||
823 | vand q4, q4, q9 | ||
824 | veor q8, q8, q12 | ||
825 | veor q4, q4, q2 | ||
826 | veor q12, q12, q11 | ||
827 | veor q2, q2, q10 | ||
828 | veor q15, q15, q13 | ||
829 | veor q14, q14, q9 | ||
830 | veor q10, q15, q14 | ||
831 | vand q10, q10, q7 | ||
832 | veor q7, q7, q1 | ||
833 | vand q11, q1, q15 | ||
834 | vand q7, q7, q14 | ||
835 | veor q1, q11, q10 | ||
836 | veor q7, q7, q11 | ||
837 | veor q7, q7, q12 | ||
838 | veor q4, q4, q12 | ||
839 | veor q1, q1, q8 | ||
840 | veor q2, q2, q8 | ||
841 | veor q7, q7, q0 | ||
842 | veor q1, q1, q6 | ||
843 | veor q6, q6, q0 | ||
844 | veor q4, q4, q7 | ||
845 | veor q0, q0, q1 | ||
846 | |||
847 | veor q1, q1, q5 | ||
848 | veor q5, q5, q2 | ||
849 | veor q2, q2, q3 | ||
850 | veor q3, q3, q5 | ||
851 | veor q4, q4, q5 | ||
852 | |||
853 | veor q6, q6, q3 | ||
854 | subs r5,r5,#1 | ||
855 | bcc .Lenc_done | ||
856 | vext.8 q8, q0, q0, #12 @ x0 <<< 32 | ||
857 | vext.8 q9, q1, q1, #12 | ||
858 | veor q0, q0, q8 @ x0 ^ (x0 <<< 32) | ||
859 | vext.8 q10, q4, q4, #12 | ||
860 | veor q1, q1, q9 | ||
861 | vext.8 q11, q6, q6, #12 | ||
862 | veor q4, q4, q10 | ||
863 | vext.8 q12, q3, q3, #12 | ||
864 | veor q6, q6, q11 | ||
865 | vext.8 q13, q7, q7, #12 | ||
866 | veor q3, q3, q12 | ||
867 | vext.8 q14, q2, q2, #12 | ||
868 | veor q7, q7, q13 | ||
869 | vext.8 q15, q5, q5, #12 | ||
870 | veor q2, q2, q14 | ||
871 | |||
872 | veor q9, q9, q0 | ||
873 | veor q5, q5, q15 | ||
874 | vext.8 q0, q0, q0, #8 @ (x0 ^ (x0 <<< 32)) <<< 64) | ||
875 | veor q10, q10, q1 | ||
876 | veor q8, q8, q5 | ||
877 | veor q9, q9, q5 | ||
878 | vext.8 q1, q1, q1, #8 | ||
879 | veor q13, q13, q3 | ||
880 | veor q0, q0, q8 | ||
881 | veor q14, q14, q7 | ||
882 | veor q1, q1, q9 | ||
883 | vext.8 q8, q3, q3, #8 | ||
884 | veor q12, q12, q6 | ||
885 | vext.8 q9, q7, q7, #8 | ||
886 | veor q15, q15, q2 | ||
887 | vext.8 q3, q6, q6, #8 | ||
888 | veor q11, q11, q4 | ||
889 | vext.8 q7, q5, q5, #8 | ||
890 | veor q12, q12, q5 | ||
891 | vext.8 q6, q2, q2, #8 | ||
892 | veor q11, q11, q5 | ||
893 | vext.8 q2, q4, q4, #8 | ||
894 | veor q5, q9, q13 | ||
895 | veor q4, q8, q12 | ||
896 | veor q3, q3, q11 | ||
897 | veor q7, q7, q15 | ||
898 | veor q6, q6, q14 | ||
899 | @ vmov q4, q8 | ||
900 | veor q2, q2, q10 | ||
901 | @ vmov q5, q9 | ||
902 | vldmia r6, {q12} @ .LSR | ||
903 | ite eq @ Thumb2 thing, samity check in ARM | ||
904 | addeq r6,r6,#0x10 | ||
905 | bne .Lenc_loop | ||
906 | vldmia r6, {q12} @ .LSRM0 | ||
907 | b .Lenc_loop | ||
908 | .align 4 | ||
909 | .Lenc_done: | ||
910 | vmov.i8 q8,#0x55 @ compose .LBS0 | ||
911 | vmov.i8 q9,#0x33 @ compose .LBS1 | ||
912 | vshr.u64 q10, q2, #1 | ||
913 | vshr.u64 q11, q3, #1 | ||
914 | veor q10, q10, q5 | ||
915 | veor q11, q11, q7 | ||
916 | vand q10, q10, q8 | ||
917 | vand q11, q11, q8 | ||
918 | veor q5, q5, q10 | ||
919 | vshl.u64 q10, q10, #1 | ||
920 | veor q7, q7, q11 | ||
921 | vshl.u64 q11, q11, #1 | ||
922 | veor q2, q2, q10 | ||
923 | veor q3, q3, q11 | ||
924 | vshr.u64 q10, q4, #1 | ||
925 | vshr.u64 q11, q0, #1 | ||
926 | veor q10, q10, q6 | ||
927 | veor q11, q11, q1 | ||
928 | vand q10, q10, q8 | ||
929 | vand q11, q11, q8 | ||
930 | veor q6, q6, q10 | ||
931 | vshl.u64 q10, q10, #1 | ||
932 | veor q1, q1, q11 | ||
933 | vshl.u64 q11, q11, #1 | ||
934 | veor q4, q4, q10 | ||
935 | veor q0, q0, q11 | ||
936 | vmov.i8 q8,#0x0f @ compose .LBS2 | ||
937 | vshr.u64 q10, q7, #2 | ||
938 | vshr.u64 q11, q3, #2 | ||
939 | veor q10, q10, q5 | ||
940 | veor q11, q11, q2 | ||
941 | vand q10, q10, q9 | ||
942 | vand q11, q11, q9 | ||
943 | veor q5, q5, q10 | ||
944 | vshl.u64 q10, q10, #2 | ||
945 | veor q2, q2, q11 | ||
946 | vshl.u64 q11, q11, #2 | ||
947 | veor q7, q7, q10 | ||
948 | veor q3, q3, q11 | ||
949 | vshr.u64 q10, q1, #2 | ||
950 | vshr.u64 q11, q0, #2 | ||
951 | veor q10, q10, q6 | ||
952 | veor q11, q11, q4 | ||
953 | vand q10, q10, q9 | ||
954 | vand q11, q11, q9 | ||
955 | veor q6, q6, q10 | ||
956 | vshl.u64 q10, q10, #2 | ||
957 | veor q4, q4, q11 | ||
958 | vshl.u64 q11, q11, #2 | ||
959 | veor q1, q1, q10 | ||
960 | veor q0, q0, q11 | ||
961 | vshr.u64 q10, q6, #4 | ||
962 | vshr.u64 q11, q4, #4 | ||
963 | veor q10, q10, q5 | ||
964 | veor q11, q11, q2 | ||
965 | vand q10, q10, q8 | ||
966 | vand q11, q11, q8 | ||
967 | veor q5, q5, q10 | ||
968 | vshl.u64 q10, q10, #4 | ||
969 | veor q2, q2, q11 | ||
970 | vshl.u64 q11, q11, #4 | ||
971 | veor q6, q6, q10 | ||
972 | veor q4, q4, q11 | ||
973 | vshr.u64 q10, q1, #4 | ||
974 | vshr.u64 q11, q0, #4 | ||
975 | veor q10, q10, q7 | ||
976 | veor q11, q11, q3 | ||
977 | vand q10, q10, q8 | ||
978 | vand q11, q11, q8 | ||
979 | veor q7, q7, q10 | ||
980 | vshl.u64 q10, q10, #4 | ||
981 | veor q3, q3, q11 | ||
982 | vshl.u64 q11, q11, #4 | ||
983 | veor q1, q1, q10 | ||
984 | veor q0, q0, q11 | ||
985 | vldmia r4, {q8} @ last round key | ||
986 | veor q4, q4, q8 | ||
987 | veor q6, q6, q8 | ||
988 | veor q3, q3, q8 | ||
989 | veor q7, q7, q8 | ||
990 | veor q2, q2, q8 | ||
991 | veor q5, q5, q8 | ||
992 | veor q0, q0, q8 | ||
993 | veor q1, q1, q8 | ||
994 | bx lr | ||
995 | .size _bsaes_encrypt8,.-_bsaes_encrypt8 | ||
996 | .type _bsaes_key_convert,%function | ||
997 | .align 4 | ||
998 | _bsaes_key_convert: | ||
999 | adr r6,_bsaes_key_convert | ||
1000 | vld1.8 {q7}, [r4]! @ load round 0 key | ||
1001 | sub r6,r6,#_bsaes_key_convert-.LM0 | ||
1002 | vld1.8 {q15}, [r4]! @ load round 1 key | ||
1003 | |||
1004 | vmov.i8 q8, #0x01 @ bit masks | ||
1005 | vmov.i8 q9, #0x02 | ||
1006 | vmov.i8 q10, #0x04 | ||
1007 | vmov.i8 q11, #0x08 | ||
1008 | vmov.i8 q12, #0x10 | ||
1009 | vmov.i8 q13, #0x20 | ||
1010 | vldmia r6, {q14} @ .LM0 | ||
1011 | |||
1012 | #ifdef __ARMEL__ | ||
1013 | vrev32.8 q7, q7 | ||
1014 | vrev32.8 q15, q15 | ||
1015 | #endif | ||
1016 | sub r5,r5,#1 | ||
1017 | vstmia r12!, {q7} @ save round 0 key | ||
1018 | b .Lkey_loop | ||
1019 | |||
1020 | .align 4 | ||
1021 | .Lkey_loop: | ||
1022 | vtbl.8 d14,{q15},d28 | ||
1023 | vtbl.8 d15,{q15},d29 | ||
1024 | vmov.i8 q6, #0x40 | ||
1025 | vmov.i8 q15, #0x80 | ||
1026 | |||
1027 | vtst.8 q0, q7, q8 | ||
1028 | vtst.8 q1, q7, q9 | ||
1029 | vtst.8 q2, q7, q10 | ||
1030 | vtst.8 q3, q7, q11 | ||
1031 | vtst.8 q4, q7, q12 | ||
1032 | vtst.8 q5, q7, q13 | ||
1033 | vtst.8 q6, q7, q6 | ||
1034 | vtst.8 q7, q7, q15 | ||
1035 | vld1.8 {q15}, [r4]! @ load next round key | ||
1036 | vmvn q0, q0 @ "pnot" | ||
1037 | vmvn q1, q1 | ||
1038 | vmvn q5, q5 | ||
1039 | vmvn q6, q6 | ||
1040 | #ifdef __ARMEL__ | ||
1041 | vrev32.8 q15, q15 | ||
1042 | #endif | ||
1043 | subs r5,r5,#1 | ||
1044 | vstmia r12!,{q0-q7} @ write bit-sliced round key | ||
1045 | bne .Lkey_loop | ||
1046 | |||
1047 | vmov.i8 q7,#0x63 @ compose .L63 | ||
1048 | @ don't save last round key | ||
1049 | bx lr | ||
1050 | .size _bsaes_key_convert,.-_bsaes_key_convert | ||
1051 | .extern AES_cbc_encrypt | ||
1052 | .extern AES_decrypt | ||
1053 | |||
1054 | .global bsaes_cbc_encrypt | ||
1055 | .type bsaes_cbc_encrypt,%function | ||
1056 | .align 5 | ||
1057 | bsaes_cbc_encrypt: | ||
1058 | #ifndef __KERNEL__ | ||
1059 | cmp r2, #128 | ||
1060 | #ifndef __thumb__ | ||
1061 | blo AES_cbc_encrypt | ||
1062 | #else | ||
1063 | bhs 1f | ||
1064 | b AES_cbc_encrypt | ||
1065 | 1: | ||
1066 | #endif | ||
1067 | #endif | ||
1068 | |||
1069 | @ it is up to the caller to make sure we are called with enc == 0 | ||
1070 | |||
1071 | mov ip, sp | ||
1072 | stmdb sp!, {r4-r10, lr} | ||
1073 | VFP_ABI_PUSH | ||
1074 | ldr r8, [ip] @ IV is 1st arg on the stack | ||
1075 | mov r2, r2, lsr#4 @ len in 16 byte blocks | ||
1076 | sub sp, #0x10 @ scratch space to carry over the IV | ||
1077 | mov r9, sp @ save sp | ||
1078 | |||
1079 | ldr r10, [r3, #240] @ get # of rounds | ||
1080 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1081 | @ allocate the key schedule on the stack | ||
1082 | sub r12, sp, r10, lsl#7 @ 128 bytes per inner round key | ||
1083 | add r12, #96 @ sifze of bit-slices key schedule | ||
1084 | |||
1085 | @ populate the key schedule | ||
1086 | mov r4, r3 @ pass key | ||
1087 | mov r5, r10 @ pass # of rounds | ||
1088 | mov sp, r12 @ sp is sp | ||
1089 | bl _bsaes_key_convert | ||
1090 | vldmia sp, {q6} | ||
1091 | vstmia r12, {q15} @ save last round key | ||
1092 | veor q7, q7, q6 @ fix up round 0 key | ||
1093 | vstmia sp, {q7} | ||
1094 | #else | ||
1095 | ldr r12, [r3, #244] | ||
1096 | eors r12, #1 | ||
1097 | beq 0f | ||
1098 | |||
1099 | @ populate the key schedule | ||
1100 | str r12, [r3, #244] | ||
1101 | mov r4, r3 @ pass key | ||
1102 | mov r5, r10 @ pass # of rounds | ||
1103 | add r12, r3, #248 @ pass key schedule | ||
1104 | bl _bsaes_key_convert | ||
1105 | add r4, r3, #248 | ||
1106 | vldmia r4, {q6} | ||
1107 | vstmia r12, {q15} @ save last round key | ||
1108 | veor q7, q7, q6 @ fix up round 0 key | ||
1109 | vstmia r4, {q7} | ||
1110 | |||
1111 | .align 2 | ||
1112 | 0: | ||
1113 | #endif | ||
1114 | |||
1115 | vld1.8 {q15}, [r8] @ load IV | ||
1116 | b .Lcbc_dec_loop | ||
1117 | |||
1118 | .align 4 | ||
1119 | .Lcbc_dec_loop: | ||
1120 | subs r2, r2, #0x8 | ||
1121 | bmi .Lcbc_dec_loop_finish | ||
1122 | |||
1123 | vld1.8 {q0-q1}, [r0]! @ load input | ||
1124 | vld1.8 {q2-q3}, [r0]! | ||
1125 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1126 | mov r4, sp @ pass the key | ||
1127 | #else | ||
1128 | add r4, r3, #248 | ||
1129 | #endif | ||
1130 | vld1.8 {q4-q5}, [r0]! | ||
1131 | mov r5, r10 | ||
1132 | vld1.8 {q6-q7}, [r0] | ||
1133 | sub r0, r0, #0x60 | ||
1134 | vstmia r9, {q15} @ put aside IV | ||
1135 | |||
1136 | bl _bsaes_decrypt8 | ||
1137 | |||
1138 | vldmia r9, {q14} @ reload IV | ||
1139 | vld1.8 {q8-q9}, [r0]! @ reload input | ||
1140 | veor q0, q0, q14 @ ^= IV | ||
1141 | vld1.8 {q10-q11}, [r0]! | ||
1142 | veor q1, q1, q8 | ||
1143 | veor q6, q6, q9 | ||
1144 | vld1.8 {q12-q13}, [r0]! | ||
1145 | veor q4, q4, q10 | ||
1146 | veor q2, q2, q11 | ||
1147 | vld1.8 {q14-q15}, [r0]! | ||
1148 | veor q7, q7, q12 | ||
1149 | vst1.8 {q0-q1}, [r1]! @ write output | ||
1150 | veor q3, q3, q13 | ||
1151 | vst1.8 {q6}, [r1]! | ||
1152 | veor q5, q5, q14 | ||
1153 | vst1.8 {q4}, [r1]! | ||
1154 | vst1.8 {q2}, [r1]! | ||
1155 | vst1.8 {q7}, [r1]! | ||
1156 | vst1.8 {q3}, [r1]! | ||
1157 | vst1.8 {q5}, [r1]! | ||
1158 | |||
1159 | b .Lcbc_dec_loop | ||
1160 | |||
1161 | .Lcbc_dec_loop_finish: | ||
1162 | adds r2, r2, #8 | ||
1163 | beq .Lcbc_dec_done | ||
1164 | |||
1165 | vld1.8 {q0}, [r0]! @ load input | ||
1166 | cmp r2, #2 | ||
1167 | blo .Lcbc_dec_one | ||
1168 | vld1.8 {q1}, [r0]! | ||
1169 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1170 | mov r4, sp @ pass the key | ||
1171 | #else | ||
1172 | add r4, r3, #248 | ||
1173 | #endif | ||
1174 | mov r5, r10 | ||
1175 | vstmia r9, {q15} @ put aside IV | ||
1176 | beq .Lcbc_dec_two | ||
1177 | vld1.8 {q2}, [r0]! | ||
1178 | cmp r2, #4 | ||
1179 | blo .Lcbc_dec_three | ||
1180 | vld1.8 {q3}, [r0]! | ||
1181 | beq .Lcbc_dec_four | ||
1182 | vld1.8 {q4}, [r0]! | ||
1183 | cmp r2, #6 | ||
1184 | blo .Lcbc_dec_five | ||
1185 | vld1.8 {q5}, [r0]! | ||
1186 | beq .Lcbc_dec_six | ||
1187 | vld1.8 {q6}, [r0]! | ||
1188 | sub r0, r0, #0x70 | ||
1189 | |||
1190 | bl _bsaes_decrypt8 | ||
1191 | |||
1192 | vldmia r9, {q14} @ reload IV | ||
1193 | vld1.8 {q8-q9}, [r0]! @ reload input | ||
1194 | veor q0, q0, q14 @ ^= IV | ||
1195 | vld1.8 {q10-q11}, [r0]! | ||
1196 | veor q1, q1, q8 | ||
1197 | veor q6, q6, q9 | ||
1198 | vld1.8 {q12-q13}, [r0]! | ||
1199 | veor q4, q4, q10 | ||
1200 | veor q2, q2, q11 | ||
1201 | vld1.8 {q15}, [r0]! | ||
1202 | veor q7, q7, q12 | ||
1203 | vst1.8 {q0-q1}, [r1]! @ write output | ||
1204 | veor q3, q3, q13 | ||
1205 | vst1.8 {q6}, [r1]! | ||
1206 | vst1.8 {q4}, [r1]! | ||
1207 | vst1.8 {q2}, [r1]! | ||
1208 | vst1.8 {q7}, [r1]! | ||
1209 | vst1.8 {q3}, [r1]! | ||
1210 | b .Lcbc_dec_done | ||
1211 | .align 4 | ||
1212 | .Lcbc_dec_six: | ||
1213 | sub r0, r0, #0x60 | ||
1214 | bl _bsaes_decrypt8 | ||
1215 | vldmia r9,{q14} @ reload IV | ||
1216 | vld1.8 {q8-q9}, [r0]! @ reload input | ||
1217 | veor q0, q0, q14 @ ^= IV | ||
1218 | vld1.8 {q10-q11}, [r0]! | ||
1219 | veor q1, q1, q8 | ||
1220 | veor q6, q6, q9 | ||
1221 | vld1.8 {q12}, [r0]! | ||
1222 | veor q4, q4, q10 | ||
1223 | veor q2, q2, q11 | ||
1224 | vld1.8 {q15}, [r0]! | ||
1225 | veor q7, q7, q12 | ||
1226 | vst1.8 {q0-q1}, [r1]! @ write output | ||
1227 | vst1.8 {q6}, [r1]! | ||
1228 | vst1.8 {q4}, [r1]! | ||
1229 | vst1.8 {q2}, [r1]! | ||
1230 | vst1.8 {q7}, [r1]! | ||
1231 | b .Lcbc_dec_done | ||
1232 | .align 4 | ||
1233 | .Lcbc_dec_five: | ||
1234 | sub r0, r0, #0x50 | ||
1235 | bl _bsaes_decrypt8 | ||
1236 | vldmia r9, {q14} @ reload IV | ||
1237 | vld1.8 {q8-q9}, [r0]! @ reload input | ||
1238 | veor q0, q0, q14 @ ^= IV | ||
1239 | vld1.8 {q10-q11}, [r0]! | ||
1240 | veor q1, q1, q8 | ||
1241 | veor q6, q6, q9 | ||
1242 | vld1.8 {q15}, [r0]! | ||
1243 | veor q4, q4, q10 | ||
1244 | vst1.8 {q0-q1}, [r1]! @ write output | ||
1245 | veor q2, q2, q11 | ||
1246 | vst1.8 {q6}, [r1]! | ||
1247 | vst1.8 {q4}, [r1]! | ||
1248 | vst1.8 {q2}, [r1]! | ||
1249 | b .Lcbc_dec_done | ||
1250 | .align 4 | ||
1251 | .Lcbc_dec_four: | ||
1252 | sub r0, r0, #0x40 | ||
1253 | bl _bsaes_decrypt8 | ||
1254 | vldmia r9, {q14} @ reload IV | ||
1255 | vld1.8 {q8-q9}, [r0]! @ reload input | ||
1256 | veor q0, q0, q14 @ ^= IV | ||
1257 | vld1.8 {q10}, [r0]! | ||
1258 | veor q1, q1, q8 | ||
1259 | veor q6, q6, q9 | ||
1260 | vld1.8 {q15}, [r0]! | ||
1261 | veor q4, q4, q10 | ||
1262 | vst1.8 {q0-q1}, [r1]! @ write output | ||
1263 | vst1.8 {q6}, [r1]! | ||
1264 | vst1.8 {q4}, [r1]! | ||
1265 | b .Lcbc_dec_done | ||
1266 | .align 4 | ||
1267 | .Lcbc_dec_three: | ||
1268 | sub r0, r0, #0x30 | ||
1269 | bl _bsaes_decrypt8 | ||
1270 | vldmia r9, {q14} @ reload IV | ||
1271 | vld1.8 {q8-q9}, [r0]! @ reload input | ||
1272 | veor q0, q0, q14 @ ^= IV | ||
1273 | vld1.8 {q15}, [r0]! | ||
1274 | veor q1, q1, q8 | ||
1275 | veor q6, q6, q9 | ||
1276 | vst1.8 {q0-q1}, [r1]! @ write output | ||
1277 | vst1.8 {q6}, [r1]! | ||
1278 | b .Lcbc_dec_done | ||
1279 | .align 4 | ||
1280 | .Lcbc_dec_two: | ||
1281 | sub r0, r0, #0x20 | ||
1282 | bl _bsaes_decrypt8 | ||
1283 | vldmia r9, {q14} @ reload IV | ||
1284 | vld1.8 {q8}, [r0]! @ reload input | ||
1285 | veor q0, q0, q14 @ ^= IV | ||
1286 | vld1.8 {q15}, [r0]! @ reload input | ||
1287 | veor q1, q1, q8 | ||
1288 | vst1.8 {q0-q1}, [r1]! @ write output | ||
1289 | b .Lcbc_dec_done | ||
1290 | .align 4 | ||
1291 | .Lcbc_dec_one: | ||
1292 | sub r0, r0, #0x10 | ||
1293 | mov r10, r1 @ save original out pointer | ||
1294 | mov r1, r9 @ use the iv scratch space as out buffer | ||
1295 | mov r2, r3 | ||
1296 | vmov q4,q15 @ just in case ensure that IV | ||
1297 | vmov q5,q0 @ and input are preserved | ||
1298 | bl AES_decrypt | ||
1299 | vld1.8 {q0}, [r9,:64] @ load result | ||
1300 | veor q0, q0, q4 @ ^= IV | ||
1301 | vmov q15, q5 @ q5 holds input | ||
1302 | vst1.8 {q0}, [r10] @ write output | ||
1303 | |||
1304 | .Lcbc_dec_done: | ||
1305 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1306 | vmov.i32 q0, #0 | ||
1307 | vmov.i32 q1, #0 | ||
1308 | .Lcbc_dec_bzero: @ wipe key schedule [if any] | ||
1309 | vstmia sp!, {q0-q1} | ||
1310 | cmp sp, r9 | ||
1311 | bne .Lcbc_dec_bzero | ||
1312 | #endif | ||
1313 | |||
1314 | mov sp, r9 | ||
1315 | add sp, #0x10 @ add sp,r9,#0x10 is no good for thumb | ||
1316 | vst1.8 {q15}, [r8] @ return IV | ||
1317 | VFP_ABI_POP | ||
1318 | ldmia sp!, {r4-r10, pc} | ||
1319 | .size bsaes_cbc_encrypt,.-bsaes_cbc_encrypt | ||
1320 | .extern AES_encrypt | ||
1321 | .global bsaes_ctr32_encrypt_blocks | ||
1322 | .type bsaes_ctr32_encrypt_blocks,%function | ||
1323 | .align 5 | ||
1324 | bsaes_ctr32_encrypt_blocks: | ||
1325 | cmp r2, #8 @ use plain AES for | ||
1326 | blo .Lctr_enc_short @ small sizes | ||
1327 | |||
1328 | mov ip, sp | ||
1329 | stmdb sp!, {r4-r10, lr} | ||
1330 | VFP_ABI_PUSH | ||
1331 | ldr r8, [ip] @ ctr is 1st arg on the stack | ||
1332 | sub sp, sp, #0x10 @ scratch space to carry over the ctr | ||
1333 | mov r9, sp @ save sp | ||
1334 | |||
1335 | ldr r10, [r3, #240] @ get # of rounds | ||
1336 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1337 | @ allocate the key schedule on the stack | ||
1338 | sub r12, sp, r10, lsl#7 @ 128 bytes per inner round key | ||
1339 | add r12, #96 @ size of bit-sliced key schedule | ||
1340 | |||
1341 | @ populate the key schedule | ||
1342 | mov r4, r3 @ pass key | ||
1343 | mov r5, r10 @ pass # of rounds | ||
1344 | mov sp, r12 @ sp is sp | ||
1345 | bl _bsaes_key_convert | ||
1346 | veor q7,q7,q15 @ fix up last round key | ||
1347 | vstmia r12, {q7} @ save last round key | ||
1348 | |||
1349 | vld1.8 {q0}, [r8] @ load counter | ||
1350 | add r8, r6, #.LREVM0SR-.LM0 @ borrow r8 | ||
1351 | vldmia sp, {q4} @ load round0 key | ||
1352 | #else | ||
1353 | ldr r12, [r3, #244] | ||
1354 | eors r12, #1 | ||
1355 | beq 0f | ||
1356 | |||
1357 | @ populate the key schedule | ||
1358 | str r12, [r3, #244] | ||
1359 | mov r4, r3 @ pass key | ||
1360 | mov r5, r10 @ pass # of rounds | ||
1361 | add r12, r3, #248 @ pass key schedule | ||
1362 | bl _bsaes_key_convert | ||
1363 | veor q7,q7,q15 @ fix up last round key | ||
1364 | vstmia r12, {q7} @ save last round key | ||
1365 | |||
1366 | .align 2 | ||
1367 | 0: add r12, r3, #248 | ||
1368 | vld1.8 {q0}, [r8] @ load counter | ||
1369 | adrl r8, .LREVM0SR @ borrow r8 | ||
1370 | vldmia r12, {q4} @ load round0 key | ||
1371 | sub sp, #0x10 @ place for adjusted round0 key | ||
1372 | #endif | ||
1373 | |||
1374 | vmov.i32 q8,#1 @ compose 1<<96 | ||
1375 | veor q9,q9,q9 | ||
1376 | vrev32.8 q0,q0 | ||
1377 | vext.8 q8,q9,q8,#4 | ||
1378 | vrev32.8 q4,q4 | ||
1379 | vadd.u32 q9,q8,q8 @ compose 2<<96 | ||
1380 | vstmia sp, {q4} @ save adjusted round0 key | ||
1381 | b .Lctr_enc_loop | ||
1382 | |||
1383 | .align 4 | ||
1384 | .Lctr_enc_loop: | ||
1385 | vadd.u32 q10, q8, q9 @ compose 3<<96 | ||
1386 | vadd.u32 q1, q0, q8 @ +1 | ||
1387 | vadd.u32 q2, q0, q9 @ +2 | ||
1388 | vadd.u32 q3, q0, q10 @ +3 | ||
1389 | vadd.u32 q4, q1, q10 | ||
1390 | vadd.u32 q5, q2, q10 | ||
1391 | vadd.u32 q6, q3, q10 | ||
1392 | vadd.u32 q7, q4, q10 | ||
1393 | vadd.u32 q10, q5, q10 @ next counter | ||
1394 | |||
1395 | @ Borrow prologue from _bsaes_encrypt8 to use the opportunity | ||
1396 | @ to flip byte order in 32-bit counter | ||
1397 | |||
1398 | vldmia sp, {q9} @ load round0 key | ||
1399 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1400 | add r4, sp, #0x10 @ pass next round key | ||
1401 | #else | ||
1402 | add r4, r3, #264 | ||
1403 | #endif | ||
1404 | vldmia r8, {q8} @ .LREVM0SR | ||
1405 | mov r5, r10 @ pass rounds | ||
1406 | vstmia r9, {q10} @ save next counter | ||
1407 | sub r6, r8, #.LREVM0SR-.LSR @ pass constants | ||
1408 | |||
1409 | bl _bsaes_encrypt8_alt | ||
1410 | |||
1411 | subs r2, r2, #8 | ||
1412 | blo .Lctr_enc_loop_done | ||
1413 | |||
1414 | vld1.8 {q8-q9}, [r0]! @ load input | ||
1415 | vld1.8 {q10-q11}, [r0]! | ||
1416 | veor q0, q8 | ||
1417 | veor q1, q9 | ||
1418 | vld1.8 {q12-q13}, [r0]! | ||
1419 | veor q4, q10 | ||
1420 | veor q6, q11 | ||
1421 | vld1.8 {q14-q15}, [r0]! | ||
1422 | veor q3, q12 | ||
1423 | vst1.8 {q0-q1}, [r1]! @ write output | ||
1424 | veor q7, q13 | ||
1425 | veor q2, q14 | ||
1426 | vst1.8 {q4}, [r1]! | ||
1427 | veor q5, q15 | ||
1428 | vst1.8 {q6}, [r1]! | ||
1429 | vmov.i32 q8, #1 @ compose 1<<96 | ||
1430 | vst1.8 {q3}, [r1]! | ||
1431 | veor q9, q9, q9 | ||
1432 | vst1.8 {q7}, [r1]! | ||
1433 | vext.8 q8, q9, q8, #4 | ||
1434 | vst1.8 {q2}, [r1]! | ||
1435 | vadd.u32 q9,q8,q8 @ compose 2<<96 | ||
1436 | vst1.8 {q5}, [r1]! | ||
1437 | vldmia r9, {q0} @ load counter | ||
1438 | |||
1439 | bne .Lctr_enc_loop | ||
1440 | b .Lctr_enc_done | ||
1441 | |||
1442 | .align 4 | ||
1443 | .Lctr_enc_loop_done: | ||
1444 | add r2, r2, #8 | ||
1445 | vld1.8 {q8}, [r0]! @ load input | ||
1446 | veor q0, q8 | ||
1447 | vst1.8 {q0}, [r1]! @ write output | ||
1448 | cmp r2, #2 | ||
1449 | blo .Lctr_enc_done | ||
1450 | vld1.8 {q9}, [r0]! | ||
1451 | veor q1, q9 | ||
1452 | vst1.8 {q1}, [r1]! | ||
1453 | beq .Lctr_enc_done | ||
1454 | vld1.8 {q10}, [r0]! | ||
1455 | veor q4, q10 | ||
1456 | vst1.8 {q4}, [r1]! | ||
1457 | cmp r2, #4 | ||
1458 | blo .Lctr_enc_done | ||
1459 | vld1.8 {q11}, [r0]! | ||
1460 | veor q6, q11 | ||
1461 | vst1.8 {q6}, [r1]! | ||
1462 | beq .Lctr_enc_done | ||
1463 | vld1.8 {q12}, [r0]! | ||
1464 | veor q3, q12 | ||
1465 | vst1.8 {q3}, [r1]! | ||
1466 | cmp r2, #6 | ||
1467 | blo .Lctr_enc_done | ||
1468 | vld1.8 {q13}, [r0]! | ||
1469 | veor q7, q13 | ||
1470 | vst1.8 {q7}, [r1]! | ||
1471 | beq .Lctr_enc_done | ||
1472 | vld1.8 {q14}, [r0] | ||
1473 | veor q2, q14 | ||
1474 | vst1.8 {q2}, [r1]! | ||
1475 | |||
1476 | .Lctr_enc_done: | ||
1477 | vmov.i32 q0, #0 | ||
1478 | vmov.i32 q1, #0 | ||
1479 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1480 | .Lctr_enc_bzero: @ wipe key schedule [if any] | ||
1481 | vstmia sp!, {q0-q1} | ||
1482 | cmp sp, r9 | ||
1483 | bne .Lctr_enc_bzero | ||
1484 | #else | ||
1485 | vstmia sp, {q0-q1} | ||
1486 | #endif | ||
1487 | |||
1488 | mov sp, r9 | ||
1489 | add sp, #0x10 @ add sp,r9,#0x10 is no good for thumb | ||
1490 | VFP_ABI_POP | ||
1491 | ldmia sp!, {r4-r10, pc} @ return | ||
1492 | |||
1493 | .align 4 | ||
1494 | .Lctr_enc_short: | ||
1495 | ldr ip, [sp] @ ctr pointer is passed on stack | ||
1496 | stmdb sp!, {r4-r8, lr} | ||
1497 | |||
1498 | mov r4, r0 @ copy arguments | ||
1499 | mov r5, r1 | ||
1500 | mov r6, r2 | ||
1501 | mov r7, r3 | ||
1502 | ldr r8, [ip, #12] @ load counter LSW | ||
1503 | vld1.8 {q1}, [ip] @ load whole counter value | ||
1504 | #ifdef __ARMEL__ | ||
1505 | rev r8, r8 | ||
1506 | #endif | ||
1507 | sub sp, sp, #0x10 | ||
1508 | vst1.8 {q1}, [sp,:64] @ copy counter value | ||
1509 | sub sp, sp, #0x10 | ||
1510 | |||
1511 | .Lctr_enc_short_loop: | ||
1512 | add r0, sp, #0x10 @ input counter value | ||
1513 | mov r1, sp @ output on the stack | ||
1514 | mov r2, r7 @ key | ||
1515 | |||
1516 | bl AES_encrypt | ||
1517 | |||
1518 | vld1.8 {q0}, [r4]! @ load input | ||
1519 | vld1.8 {q1}, [sp,:64] @ load encrypted counter | ||
1520 | add r8, r8, #1 | ||
1521 | #ifdef __ARMEL__ | ||
1522 | rev r0, r8 | ||
1523 | str r0, [sp, #0x1c] @ next counter value | ||
1524 | #else | ||
1525 | str r8, [sp, #0x1c] @ next counter value | ||
1526 | #endif | ||
1527 | veor q0,q0,q1 | ||
1528 | vst1.8 {q0}, [r5]! @ store output | ||
1529 | subs r6, r6, #1 | ||
1530 | bne .Lctr_enc_short_loop | ||
1531 | |||
1532 | vmov.i32 q0, #0 | ||
1533 | vmov.i32 q1, #0 | ||
1534 | vstmia sp!, {q0-q1} | ||
1535 | |||
1536 | ldmia sp!, {r4-r8, pc} | ||
1537 | .size bsaes_ctr32_encrypt_blocks,.-bsaes_ctr32_encrypt_blocks | ||
1538 | .globl bsaes_xts_encrypt | ||
1539 | .type bsaes_xts_encrypt,%function | ||
1540 | .align 4 | ||
1541 | bsaes_xts_encrypt: | ||
1542 | mov ip, sp | ||
1543 | stmdb sp!, {r4-r10, lr} @ 0x20 | ||
1544 | VFP_ABI_PUSH | ||
1545 | mov r6, sp @ future r3 | ||
1546 | |||
1547 | mov r7, r0 | ||
1548 | mov r8, r1 | ||
1549 | mov r9, r2 | ||
1550 | mov r10, r3 | ||
1551 | |||
1552 | sub r0, sp, #0x10 @ 0x10 | ||
1553 | bic r0, #0xf @ align at 16 bytes | ||
1554 | mov sp, r0 | ||
1555 | |||
1556 | #ifdef XTS_CHAIN_TWEAK | ||
1557 | ldr r0, [ip] @ pointer to input tweak | ||
1558 | #else | ||
1559 | @ generate initial tweak | ||
1560 | ldr r0, [ip, #4] @ iv[] | ||
1561 | mov r1, sp | ||
1562 | ldr r2, [ip, #0] @ key2 | ||
1563 | bl AES_encrypt | ||
1564 | mov r0,sp @ pointer to initial tweak | ||
1565 | #endif | ||
1566 | |||
1567 | ldr r1, [r10, #240] @ get # of rounds | ||
1568 | mov r3, r6 | ||
1569 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1570 | @ allocate the key schedule on the stack | ||
1571 | sub r12, sp, r1, lsl#7 @ 128 bytes per inner round key | ||
1572 | @ add r12, #96 @ size of bit-sliced key schedule | ||
1573 | sub r12, #48 @ place for tweak[9] | ||
1574 | |||
1575 | @ populate the key schedule | ||
1576 | mov r4, r10 @ pass key | ||
1577 | mov r5, r1 @ pass # of rounds | ||
1578 | mov sp, r12 | ||
1579 | add r12, #0x90 @ pass key schedule | ||
1580 | bl _bsaes_key_convert | ||
1581 | veor q7, q7, q15 @ fix up last round key | ||
1582 | vstmia r12, {q7} @ save last round key | ||
1583 | #else | ||
1584 | ldr r12, [r10, #244] | ||
1585 | eors r12, #1 | ||
1586 | beq 0f | ||
1587 | |||
1588 | str r12, [r10, #244] | ||
1589 | mov r4, r10 @ pass key | ||
1590 | mov r5, r1 @ pass # of rounds | ||
1591 | add r12, r10, #248 @ pass key schedule | ||
1592 | bl _bsaes_key_convert | ||
1593 | veor q7, q7, q15 @ fix up last round key | ||
1594 | vstmia r12, {q7} | ||
1595 | |||
1596 | .align 2 | ||
1597 | 0: sub sp, #0x90 @ place for tweak[9] | ||
1598 | #endif | ||
1599 | |||
1600 | vld1.8 {q8}, [r0] @ initial tweak | ||
1601 | adr r2, .Lxts_magic | ||
1602 | |||
1603 | subs r9, #0x80 | ||
1604 | blo .Lxts_enc_short | ||
1605 | b .Lxts_enc_loop | ||
1606 | |||
1607 | .align 4 | ||
1608 | .Lxts_enc_loop: | ||
1609 | vldmia r2, {q5} @ load XTS magic | ||
1610 | vshr.s64 q6, q8, #63 | ||
1611 | mov r0, sp | ||
1612 | vand q6, q6, q5 | ||
1613 | vadd.u64 q9, q8, q8 | ||
1614 | vst1.64 {q8}, [r0,:128]! | ||
1615 | vswp d13,d12 | ||
1616 | vshr.s64 q7, q9, #63 | ||
1617 | veor q9, q9, q6 | ||
1618 | vand q7, q7, q5 | ||
1619 | vadd.u64 q10, q9, q9 | ||
1620 | vst1.64 {q9}, [r0,:128]! | ||
1621 | vswp d15,d14 | ||
1622 | vshr.s64 q6, q10, #63 | ||
1623 | veor q10, q10, q7 | ||
1624 | vand q6, q6, q5 | ||
1625 | vld1.8 {q0}, [r7]! | ||
1626 | vadd.u64 q11, q10, q10 | ||
1627 | vst1.64 {q10}, [r0,:128]! | ||
1628 | vswp d13,d12 | ||
1629 | vshr.s64 q7, q11, #63 | ||
1630 | veor q11, q11, q6 | ||
1631 | vand q7, q7, q5 | ||
1632 | vld1.8 {q1}, [r7]! | ||
1633 | veor q0, q0, q8 | ||
1634 | vadd.u64 q12, q11, q11 | ||
1635 | vst1.64 {q11}, [r0,:128]! | ||
1636 | vswp d15,d14 | ||
1637 | vshr.s64 q6, q12, #63 | ||
1638 | veor q12, q12, q7 | ||
1639 | vand q6, q6, q5 | ||
1640 | vld1.8 {q2}, [r7]! | ||
1641 | veor q1, q1, q9 | ||
1642 | vadd.u64 q13, q12, q12 | ||
1643 | vst1.64 {q12}, [r0,:128]! | ||
1644 | vswp d13,d12 | ||
1645 | vshr.s64 q7, q13, #63 | ||
1646 | veor q13, q13, q6 | ||
1647 | vand q7, q7, q5 | ||
1648 | vld1.8 {q3}, [r7]! | ||
1649 | veor q2, q2, q10 | ||
1650 | vadd.u64 q14, q13, q13 | ||
1651 | vst1.64 {q13}, [r0,:128]! | ||
1652 | vswp d15,d14 | ||
1653 | vshr.s64 q6, q14, #63 | ||
1654 | veor q14, q14, q7 | ||
1655 | vand q6, q6, q5 | ||
1656 | vld1.8 {q4}, [r7]! | ||
1657 | veor q3, q3, q11 | ||
1658 | vadd.u64 q15, q14, q14 | ||
1659 | vst1.64 {q14}, [r0,:128]! | ||
1660 | vswp d13,d12 | ||
1661 | vshr.s64 q7, q15, #63 | ||
1662 | veor q15, q15, q6 | ||
1663 | vand q7, q7, q5 | ||
1664 | vld1.8 {q5}, [r7]! | ||
1665 | veor q4, q4, q12 | ||
1666 | vadd.u64 q8, q15, q15 | ||
1667 | vst1.64 {q15}, [r0,:128]! | ||
1668 | vswp d15,d14 | ||
1669 | veor q8, q8, q7 | ||
1670 | vst1.64 {q8}, [r0,:128] @ next round tweak | ||
1671 | |||
1672 | vld1.8 {q6-q7}, [r7]! | ||
1673 | veor q5, q5, q13 | ||
1674 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1675 | add r4, sp, #0x90 @ pass key schedule | ||
1676 | #else | ||
1677 | add r4, r10, #248 @ pass key schedule | ||
1678 | #endif | ||
1679 | veor q6, q6, q14 | ||
1680 | mov r5, r1 @ pass rounds | ||
1681 | veor q7, q7, q15 | ||
1682 | mov r0, sp | ||
1683 | |||
1684 | bl _bsaes_encrypt8 | ||
1685 | |||
1686 | vld1.64 {q8-q9}, [r0,:128]! | ||
1687 | vld1.64 {q10-q11}, [r0,:128]! | ||
1688 | veor q0, q0, q8 | ||
1689 | vld1.64 {q12-q13}, [r0,:128]! | ||
1690 | veor q1, q1, q9 | ||
1691 | veor q8, q4, q10 | ||
1692 | vst1.8 {q0-q1}, [r8]! | ||
1693 | veor q9, q6, q11 | ||
1694 | vld1.64 {q14-q15}, [r0,:128]! | ||
1695 | veor q10, q3, q12 | ||
1696 | vst1.8 {q8-q9}, [r8]! | ||
1697 | veor q11, q7, q13 | ||
1698 | veor q12, q2, q14 | ||
1699 | vst1.8 {q10-q11}, [r8]! | ||
1700 | veor q13, q5, q15 | ||
1701 | vst1.8 {q12-q13}, [r8]! | ||
1702 | |||
1703 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
1704 | |||
1705 | subs r9, #0x80 | ||
1706 | bpl .Lxts_enc_loop | ||
1707 | |||
1708 | .Lxts_enc_short: | ||
1709 | adds r9, #0x70 | ||
1710 | bmi .Lxts_enc_done | ||
1711 | |||
1712 | vldmia r2, {q5} @ load XTS magic | ||
1713 | vshr.s64 q7, q8, #63 | ||
1714 | mov r0, sp | ||
1715 | vand q7, q7, q5 | ||
1716 | vadd.u64 q9, q8, q8 | ||
1717 | vst1.64 {q8}, [r0,:128]! | ||
1718 | vswp d15,d14 | ||
1719 | vshr.s64 q6, q9, #63 | ||
1720 | veor q9, q9, q7 | ||
1721 | vand q6, q6, q5 | ||
1722 | vadd.u64 q10, q9, q9 | ||
1723 | vst1.64 {q9}, [r0,:128]! | ||
1724 | vswp d13,d12 | ||
1725 | vshr.s64 q7, q10, #63 | ||
1726 | veor q10, q10, q6 | ||
1727 | vand q7, q7, q5 | ||
1728 | vld1.8 {q0}, [r7]! | ||
1729 | subs r9, #0x10 | ||
1730 | bmi .Lxts_enc_1 | ||
1731 | vadd.u64 q11, q10, q10 | ||
1732 | vst1.64 {q10}, [r0,:128]! | ||
1733 | vswp d15,d14 | ||
1734 | vshr.s64 q6, q11, #63 | ||
1735 | veor q11, q11, q7 | ||
1736 | vand q6, q6, q5 | ||
1737 | vld1.8 {q1}, [r7]! | ||
1738 | subs r9, #0x10 | ||
1739 | bmi .Lxts_enc_2 | ||
1740 | veor q0, q0, q8 | ||
1741 | vadd.u64 q12, q11, q11 | ||
1742 | vst1.64 {q11}, [r0,:128]! | ||
1743 | vswp d13,d12 | ||
1744 | vshr.s64 q7, q12, #63 | ||
1745 | veor q12, q12, q6 | ||
1746 | vand q7, q7, q5 | ||
1747 | vld1.8 {q2}, [r7]! | ||
1748 | subs r9, #0x10 | ||
1749 | bmi .Lxts_enc_3 | ||
1750 | veor q1, q1, q9 | ||
1751 | vadd.u64 q13, q12, q12 | ||
1752 | vst1.64 {q12}, [r0,:128]! | ||
1753 | vswp d15,d14 | ||
1754 | vshr.s64 q6, q13, #63 | ||
1755 | veor q13, q13, q7 | ||
1756 | vand q6, q6, q5 | ||
1757 | vld1.8 {q3}, [r7]! | ||
1758 | subs r9, #0x10 | ||
1759 | bmi .Lxts_enc_4 | ||
1760 | veor q2, q2, q10 | ||
1761 | vadd.u64 q14, q13, q13 | ||
1762 | vst1.64 {q13}, [r0,:128]! | ||
1763 | vswp d13,d12 | ||
1764 | vshr.s64 q7, q14, #63 | ||
1765 | veor q14, q14, q6 | ||
1766 | vand q7, q7, q5 | ||
1767 | vld1.8 {q4}, [r7]! | ||
1768 | subs r9, #0x10 | ||
1769 | bmi .Lxts_enc_5 | ||
1770 | veor q3, q3, q11 | ||
1771 | vadd.u64 q15, q14, q14 | ||
1772 | vst1.64 {q14}, [r0,:128]! | ||
1773 | vswp d15,d14 | ||
1774 | vshr.s64 q6, q15, #63 | ||
1775 | veor q15, q15, q7 | ||
1776 | vand q6, q6, q5 | ||
1777 | vld1.8 {q5}, [r7]! | ||
1778 | subs r9, #0x10 | ||
1779 | bmi .Lxts_enc_6 | ||
1780 | veor q4, q4, q12 | ||
1781 | sub r9, #0x10 | ||
1782 | vst1.64 {q15}, [r0,:128] @ next round tweak | ||
1783 | |||
1784 | vld1.8 {q6}, [r7]! | ||
1785 | veor q5, q5, q13 | ||
1786 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1787 | add r4, sp, #0x90 @ pass key schedule | ||
1788 | #else | ||
1789 | add r4, r10, #248 @ pass key schedule | ||
1790 | #endif | ||
1791 | veor q6, q6, q14 | ||
1792 | mov r5, r1 @ pass rounds | ||
1793 | mov r0, sp | ||
1794 | |||
1795 | bl _bsaes_encrypt8 | ||
1796 | |||
1797 | vld1.64 {q8-q9}, [r0,:128]! | ||
1798 | vld1.64 {q10-q11}, [r0,:128]! | ||
1799 | veor q0, q0, q8 | ||
1800 | vld1.64 {q12-q13}, [r0,:128]! | ||
1801 | veor q1, q1, q9 | ||
1802 | veor q8, q4, q10 | ||
1803 | vst1.8 {q0-q1}, [r8]! | ||
1804 | veor q9, q6, q11 | ||
1805 | vld1.64 {q14}, [r0,:128]! | ||
1806 | veor q10, q3, q12 | ||
1807 | vst1.8 {q8-q9}, [r8]! | ||
1808 | veor q11, q7, q13 | ||
1809 | veor q12, q2, q14 | ||
1810 | vst1.8 {q10-q11}, [r8]! | ||
1811 | vst1.8 {q12}, [r8]! | ||
1812 | |||
1813 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
1814 | b .Lxts_enc_done | ||
1815 | .align 4 | ||
1816 | .Lxts_enc_6: | ||
1817 | vst1.64 {q14}, [r0,:128] @ next round tweak | ||
1818 | |||
1819 | veor q4, q4, q12 | ||
1820 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1821 | add r4, sp, #0x90 @ pass key schedule | ||
1822 | #else | ||
1823 | add r4, r10, #248 @ pass key schedule | ||
1824 | #endif | ||
1825 | veor q5, q5, q13 | ||
1826 | mov r5, r1 @ pass rounds | ||
1827 | mov r0, sp | ||
1828 | |||
1829 | bl _bsaes_encrypt8 | ||
1830 | |||
1831 | vld1.64 {q8-q9}, [r0,:128]! | ||
1832 | vld1.64 {q10-q11}, [r0,:128]! | ||
1833 | veor q0, q0, q8 | ||
1834 | vld1.64 {q12-q13}, [r0,:128]! | ||
1835 | veor q1, q1, q9 | ||
1836 | veor q8, q4, q10 | ||
1837 | vst1.8 {q0-q1}, [r8]! | ||
1838 | veor q9, q6, q11 | ||
1839 | veor q10, q3, q12 | ||
1840 | vst1.8 {q8-q9}, [r8]! | ||
1841 | veor q11, q7, q13 | ||
1842 | vst1.8 {q10-q11}, [r8]! | ||
1843 | |||
1844 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
1845 | b .Lxts_enc_done | ||
1846 | |||
1847 | @ put this in range for both ARM and Thumb mode adr instructions | ||
1848 | .align 5 | ||
1849 | .Lxts_magic: | ||
1850 | .quad 1, 0x87 | ||
1851 | |||
1852 | .align 5 | ||
1853 | .Lxts_enc_5: | ||
1854 | vst1.64 {q13}, [r0,:128] @ next round tweak | ||
1855 | |||
1856 | veor q3, q3, q11 | ||
1857 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1858 | add r4, sp, #0x90 @ pass key schedule | ||
1859 | #else | ||
1860 | add r4, r10, #248 @ pass key schedule | ||
1861 | #endif | ||
1862 | veor q4, q4, q12 | ||
1863 | mov r5, r1 @ pass rounds | ||
1864 | mov r0, sp | ||
1865 | |||
1866 | bl _bsaes_encrypt8 | ||
1867 | |||
1868 | vld1.64 {q8-q9}, [r0,:128]! | ||
1869 | vld1.64 {q10-q11}, [r0,:128]! | ||
1870 | veor q0, q0, q8 | ||
1871 | vld1.64 {q12}, [r0,:128]! | ||
1872 | veor q1, q1, q9 | ||
1873 | veor q8, q4, q10 | ||
1874 | vst1.8 {q0-q1}, [r8]! | ||
1875 | veor q9, q6, q11 | ||
1876 | veor q10, q3, q12 | ||
1877 | vst1.8 {q8-q9}, [r8]! | ||
1878 | vst1.8 {q10}, [r8]! | ||
1879 | |||
1880 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
1881 | b .Lxts_enc_done | ||
1882 | .align 4 | ||
1883 | .Lxts_enc_4: | ||
1884 | vst1.64 {q12}, [r0,:128] @ next round tweak | ||
1885 | |||
1886 | veor q2, q2, q10 | ||
1887 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1888 | add r4, sp, #0x90 @ pass key schedule | ||
1889 | #else | ||
1890 | add r4, r10, #248 @ pass key schedule | ||
1891 | #endif | ||
1892 | veor q3, q3, q11 | ||
1893 | mov r5, r1 @ pass rounds | ||
1894 | mov r0, sp | ||
1895 | |||
1896 | bl _bsaes_encrypt8 | ||
1897 | |||
1898 | vld1.64 {q8-q9}, [r0,:128]! | ||
1899 | vld1.64 {q10-q11}, [r0,:128]! | ||
1900 | veor q0, q0, q8 | ||
1901 | veor q1, q1, q9 | ||
1902 | veor q8, q4, q10 | ||
1903 | vst1.8 {q0-q1}, [r8]! | ||
1904 | veor q9, q6, q11 | ||
1905 | vst1.8 {q8-q9}, [r8]! | ||
1906 | |||
1907 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
1908 | b .Lxts_enc_done | ||
1909 | .align 4 | ||
1910 | .Lxts_enc_3: | ||
1911 | vst1.64 {q11}, [r0,:128] @ next round tweak | ||
1912 | |||
1913 | veor q1, q1, q9 | ||
1914 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1915 | add r4, sp, #0x90 @ pass key schedule | ||
1916 | #else | ||
1917 | add r4, r10, #248 @ pass key schedule | ||
1918 | #endif | ||
1919 | veor q2, q2, q10 | ||
1920 | mov r5, r1 @ pass rounds | ||
1921 | mov r0, sp | ||
1922 | |||
1923 | bl _bsaes_encrypt8 | ||
1924 | |||
1925 | vld1.64 {q8-q9}, [r0,:128]! | ||
1926 | vld1.64 {q10}, [r0,:128]! | ||
1927 | veor q0, q0, q8 | ||
1928 | veor q1, q1, q9 | ||
1929 | veor q8, q4, q10 | ||
1930 | vst1.8 {q0-q1}, [r8]! | ||
1931 | vst1.8 {q8}, [r8]! | ||
1932 | |||
1933 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
1934 | b .Lxts_enc_done | ||
1935 | .align 4 | ||
1936 | .Lxts_enc_2: | ||
1937 | vst1.64 {q10}, [r0,:128] @ next round tweak | ||
1938 | |||
1939 | veor q0, q0, q8 | ||
1940 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1941 | add r4, sp, #0x90 @ pass key schedule | ||
1942 | #else | ||
1943 | add r4, r10, #248 @ pass key schedule | ||
1944 | #endif | ||
1945 | veor q1, q1, q9 | ||
1946 | mov r5, r1 @ pass rounds | ||
1947 | mov r0, sp | ||
1948 | |||
1949 | bl _bsaes_encrypt8 | ||
1950 | |||
1951 | vld1.64 {q8-q9}, [r0,:128]! | ||
1952 | veor q0, q0, q8 | ||
1953 | veor q1, q1, q9 | ||
1954 | vst1.8 {q0-q1}, [r8]! | ||
1955 | |||
1956 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
1957 | b .Lxts_enc_done | ||
1958 | .align 4 | ||
1959 | .Lxts_enc_1: | ||
1960 | mov r0, sp | ||
1961 | veor q0, q8 | ||
1962 | mov r1, sp | ||
1963 | vst1.8 {q0}, [sp,:128] | ||
1964 | mov r2, r10 | ||
1965 | mov r4, r3 @ preserve fp | ||
1966 | |||
1967 | bl AES_encrypt | ||
1968 | |||
1969 | vld1.8 {q0}, [sp,:128] | ||
1970 | veor q0, q0, q8 | ||
1971 | vst1.8 {q0}, [r8]! | ||
1972 | mov r3, r4 | ||
1973 | |||
1974 | vmov q8, q9 @ next round tweak | ||
1975 | |||
1976 | .Lxts_enc_done: | ||
1977 | #ifndef XTS_CHAIN_TWEAK | ||
1978 | adds r9, #0x10 | ||
1979 | beq .Lxts_enc_ret | ||
1980 | sub r6, r8, #0x10 | ||
1981 | |||
1982 | .Lxts_enc_steal: | ||
1983 | ldrb r0, [r7], #1 | ||
1984 | ldrb r1, [r8, #-0x10] | ||
1985 | strb r0, [r8, #-0x10] | ||
1986 | strb r1, [r8], #1 | ||
1987 | |||
1988 | subs r9, #1 | ||
1989 | bhi .Lxts_enc_steal | ||
1990 | |||
1991 | vld1.8 {q0}, [r6] | ||
1992 | mov r0, sp | ||
1993 | veor q0, q0, q8 | ||
1994 | mov r1, sp | ||
1995 | vst1.8 {q0}, [sp,:128] | ||
1996 | mov r2, r10 | ||
1997 | mov r4, r3 @ preserve fp | ||
1998 | |||
1999 | bl AES_encrypt | ||
2000 | |||
2001 | vld1.8 {q0}, [sp,:128] | ||
2002 | veor q0, q0, q8 | ||
2003 | vst1.8 {q0}, [r6] | ||
2004 | mov r3, r4 | ||
2005 | #endif | ||
2006 | |||
2007 | .Lxts_enc_ret: | ||
2008 | bic r0, r3, #0xf | ||
2009 | vmov.i32 q0, #0 | ||
2010 | vmov.i32 q1, #0 | ||
2011 | #ifdef XTS_CHAIN_TWEAK | ||
2012 | ldr r1, [r3, #0x20+VFP_ABI_FRAME] @ chain tweak | ||
2013 | #endif | ||
2014 | .Lxts_enc_bzero: @ wipe key schedule [if any] | ||
2015 | vstmia sp!, {q0-q1} | ||
2016 | cmp sp, r0 | ||
2017 | bne .Lxts_enc_bzero | ||
2018 | |||
2019 | mov sp, r3 | ||
2020 | #ifdef XTS_CHAIN_TWEAK | ||
2021 | vst1.8 {q8}, [r1] | ||
2022 | #endif | ||
2023 | VFP_ABI_POP | ||
2024 | ldmia sp!, {r4-r10, pc} @ return | ||
2025 | |||
2026 | .size bsaes_xts_encrypt,.-bsaes_xts_encrypt | ||
2027 | |||
2028 | .globl bsaes_xts_decrypt | ||
2029 | .type bsaes_xts_decrypt,%function | ||
2030 | .align 4 | ||
2031 | bsaes_xts_decrypt: | ||
2032 | mov ip, sp | ||
2033 | stmdb sp!, {r4-r10, lr} @ 0x20 | ||
2034 | VFP_ABI_PUSH | ||
2035 | mov r6, sp @ future r3 | ||
2036 | |||
2037 | mov r7, r0 | ||
2038 | mov r8, r1 | ||
2039 | mov r9, r2 | ||
2040 | mov r10, r3 | ||
2041 | |||
2042 | sub r0, sp, #0x10 @ 0x10 | ||
2043 | bic r0, #0xf @ align at 16 bytes | ||
2044 | mov sp, r0 | ||
2045 | |||
2046 | #ifdef XTS_CHAIN_TWEAK | ||
2047 | ldr r0, [ip] @ pointer to input tweak | ||
2048 | #else | ||
2049 | @ generate initial tweak | ||
2050 | ldr r0, [ip, #4] @ iv[] | ||
2051 | mov r1, sp | ||
2052 | ldr r2, [ip, #0] @ key2 | ||
2053 | bl AES_encrypt | ||
2054 | mov r0, sp @ pointer to initial tweak | ||
2055 | #endif | ||
2056 | |||
2057 | ldr r1, [r10, #240] @ get # of rounds | ||
2058 | mov r3, r6 | ||
2059 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2060 | @ allocate the key schedule on the stack | ||
2061 | sub r12, sp, r1, lsl#7 @ 128 bytes per inner round key | ||
2062 | @ add r12, #96 @ size of bit-sliced key schedule | ||
2063 | sub r12, #48 @ place for tweak[9] | ||
2064 | |||
2065 | @ populate the key schedule | ||
2066 | mov r4, r10 @ pass key | ||
2067 | mov r5, r1 @ pass # of rounds | ||
2068 | mov sp, r12 | ||
2069 | add r12, #0x90 @ pass key schedule | ||
2070 | bl _bsaes_key_convert | ||
2071 | add r4, sp, #0x90 | ||
2072 | vldmia r4, {q6} | ||
2073 | vstmia r12, {q15} @ save last round key | ||
2074 | veor q7, q7, q6 @ fix up round 0 key | ||
2075 | vstmia r4, {q7} | ||
2076 | #else | ||
2077 | ldr r12, [r10, #244] | ||
2078 | eors r12, #1 | ||
2079 | beq 0f | ||
2080 | |||
2081 | str r12, [r10, #244] | ||
2082 | mov r4, r10 @ pass key | ||
2083 | mov r5, r1 @ pass # of rounds | ||
2084 | add r12, r10, #248 @ pass key schedule | ||
2085 | bl _bsaes_key_convert | ||
2086 | add r4, r10, #248 | ||
2087 | vldmia r4, {q6} | ||
2088 | vstmia r12, {q15} @ save last round key | ||
2089 | veor q7, q7, q6 @ fix up round 0 key | ||
2090 | vstmia r4, {q7} | ||
2091 | |||
2092 | .align 2 | ||
2093 | 0: sub sp, #0x90 @ place for tweak[9] | ||
2094 | #endif | ||
2095 | vld1.8 {q8}, [r0] @ initial tweak | ||
2096 | adr r2, .Lxts_magic | ||
2097 | |||
2098 | tst r9, #0xf @ if not multiple of 16 | ||
2099 | it ne @ Thumb2 thing, sanity check in ARM | ||
2100 | subne r9, #0x10 @ subtract another 16 bytes | ||
2101 | subs r9, #0x80 | ||
2102 | |||
2103 | blo .Lxts_dec_short | ||
2104 | b .Lxts_dec_loop | ||
2105 | |||
2106 | .align 4 | ||
2107 | .Lxts_dec_loop: | ||
2108 | vldmia r2, {q5} @ load XTS magic | ||
2109 | vshr.s64 q6, q8, #63 | ||
2110 | mov r0, sp | ||
2111 | vand q6, q6, q5 | ||
2112 | vadd.u64 q9, q8, q8 | ||
2113 | vst1.64 {q8}, [r0,:128]! | ||
2114 | vswp d13,d12 | ||
2115 | vshr.s64 q7, q9, #63 | ||
2116 | veor q9, q9, q6 | ||
2117 | vand q7, q7, q5 | ||
2118 | vadd.u64 q10, q9, q9 | ||
2119 | vst1.64 {q9}, [r0,:128]! | ||
2120 | vswp d15,d14 | ||
2121 | vshr.s64 q6, q10, #63 | ||
2122 | veor q10, q10, q7 | ||
2123 | vand q6, q6, q5 | ||
2124 | vld1.8 {q0}, [r7]! | ||
2125 | vadd.u64 q11, q10, q10 | ||
2126 | vst1.64 {q10}, [r0,:128]! | ||
2127 | vswp d13,d12 | ||
2128 | vshr.s64 q7, q11, #63 | ||
2129 | veor q11, q11, q6 | ||
2130 | vand q7, q7, q5 | ||
2131 | vld1.8 {q1}, [r7]! | ||
2132 | veor q0, q0, q8 | ||
2133 | vadd.u64 q12, q11, q11 | ||
2134 | vst1.64 {q11}, [r0,:128]! | ||
2135 | vswp d15,d14 | ||
2136 | vshr.s64 q6, q12, #63 | ||
2137 | veor q12, q12, q7 | ||
2138 | vand q6, q6, q5 | ||
2139 | vld1.8 {q2}, [r7]! | ||
2140 | veor q1, q1, q9 | ||
2141 | vadd.u64 q13, q12, q12 | ||
2142 | vst1.64 {q12}, [r0,:128]! | ||
2143 | vswp d13,d12 | ||
2144 | vshr.s64 q7, q13, #63 | ||
2145 | veor q13, q13, q6 | ||
2146 | vand q7, q7, q5 | ||
2147 | vld1.8 {q3}, [r7]! | ||
2148 | veor q2, q2, q10 | ||
2149 | vadd.u64 q14, q13, q13 | ||
2150 | vst1.64 {q13}, [r0,:128]! | ||
2151 | vswp d15,d14 | ||
2152 | vshr.s64 q6, q14, #63 | ||
2153 | veor q14, q14, q7 | ||
2154 | vand q6, q6, q5 | ||
2155 | vld1.8 {q4}, [r7]! | ||
2156 | veor q3, q3, q11 | ||
2157 | vadd.u64 q15, q14, q14 | ||
2158 | vst1.64 {q14}, [r0,:128]! | ||
2159 | vswp d13,d12 | ||
2160 | vshr.s64 q7, q15, #63 | ||
2161 | veor q15, q15, q6 | ||
2162 | vand q7, q7, q5 | ||
2163 | vld1.8 {q5}, [r7]! | ||
2164 | veor q4, q4, q12 | ||
2165 | vadd.u64 q8, q15, q15 | ||
2166 | vst1.64 {q15}, [r0,:128]! | ||
2167 | vswp d15,d14 | ||
2168 | veor q8, q8, q7 | ||
2169 | vst1.64 {q8}, [r0,:128] @ next round tweak | ||
2170 | |||
2171 | vld1.8 {q6-q7}, [r7]! | ||
2172 | veor q5, q5, q13 | ||
2173 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2174 | add r4, sp, #0x90 @ pass key schedule | ||
2175 | #else | ||
2176 | add r4, r10, #248 @ pass key schedule | ||
2177 | #endif | ||
2178 | veor q6, q6, q14 | ||
2179 | mov r5, r1 @ pass rounds | ||
2180 | veor q7, q7, q15 | ||
2181 | mov r0, sp | ||
2182 | |||
2183 | bl _bsaes_decrypt8 | ||
2184 | |||
2185 | vld1.64 {q8-q9}, [r0,:128]! | ||
2186 | vld1.64 {q10-q11}, [r0,:128]! | ||
2187 | veor q0, q0, q8 | ||
2188 | vld1.64 {q12-q13}, [r0,:128]! | ||
2189 | veor q1, q1, q9 | ||
2190 | veor q8, q6, q10 | ||
2191 | vst1.8 {q0-q1}, [r8]! | ||
2192 | veor q9, q4, q11 | ||
2193 | vld1.64 {q14-q15}, [r0,:128]! | ||
2194 | veor q10, q2, q12 | ||
2195 | vst1.8 {q8-q9}, [r8]! | ||
2196 | veor q11, q7, q13 | ||
2197 | veor q12, q3, q14 | ||
2198 | vst1.8 {q10-q11}, [r8]! | ||
2199 | veor q13, q5, q15 | ||
2200 | vst1.8 {q12-q13}, [r8]! | ||
2201 | |||
2202 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
2203 | |||
2204 | subs r9, #0x80 | ||
2205 | bpl .Lxts_dec_loop | ||
2206 | |||
2207 | .Lxts_dec_short: | ||
2208 | adds r9, #0x70 | ||
2209 | bmi .Lxts_dec_done | ||
2210 | |||
2211 | vldmia r2, {q5} @ load XTS magic | ||
2212 | vshr.s64 q7, q8, #63 | ||
2213 | mov r0, sp | ||
2214 | vand q7, q7, q5 | ||
2215 | vadd.u64 q9, q8, q8 | ||
2216 | vst1.64 {q8}, [r0,:128]! | ||
2217 | vswp d15,d14 | ||
2218 | vshr.s64 q6, q9, #63 | ||
2219 | veor q9, q9, q7 | ||
2220 | vand q6, q6, q5 | ||
2221 | vadd.u64 q10, q9, q9 | ||
2222 | vst1.64 {q9}, [r0,:128]! | ||
2223 | vswp d13,d12 | ||
2224 | vshr.s64 q7, q10, #63 | ||
2225 | veor q10, q10, q6 | ||
2226 | vand q7, q7, q5 | ||
2227 | vld1.8 {q0}, [r7]! | ||
2228 | subs r9, #0x10 | ||
2229 | bmi .Lxts_dec_1 | ||
2230 | vadd.u64 q11, q10, q10 | ||
2231 | vst1.64 {q10}, [r0,:128]! | ||
2232 | vswp d15,d14 | ||
2233 | vshr.s64 q6, q11, #63 | ||
2234 | veor q11, q11, q7 | ||
2235 | vand q6, q6, q5 | ||
2236 | vld1.8 {q1}, [r7]! | ||
2237 | subs r9, #0x10 | ||
2238 | bmi .Lxts_dec_2 | ||
2239 | veor q0, q0, q8 | ||
2240 | vadd.u64 q12, q11, q11 | ||
2241 | vst1.64 {q11}, [r0,:128]! | ||
2242 | vswp d13,d12 | ||
2243 | vshr.s64 q7, q12, #63 | ||
2244 | veor q12, q12, q6 | ||
2245 | vand q7, q7, q5 | ||
2246 | vld1.8 {q2}, [r7]! | ||
2247 | subs r9, #0x10 | ||
2248 | bmi .Lxts_dec_3 | ||
2249 | veor q1, q1, q9 | ||
2250 | vadd.u64 q13, q12, q12 | ||
2251 | vst1.64 {q12}, [r0,:128]! | ||
2252 | vswp d15,d14 | ||
2253 | vshr.s64 q6, q13, #63 | ||
2254 | veor q13, q13, q7 | ||
2255 | vand q6, q6, q5 | ||
2256 | vld1.8 {q3}, [r7]! | ||
2257 | subs r9, #0x10 | ||
2258 | bmi .Lxts_dec_4 | ||
2259 | veor q2, q2, q10 | ||
2260 | vadd.u64 q14, q13, q13 | ||
2261 | vst1.64 {q13}, [r0,:128]! | ||
2262 | vswp d13,d12 | ||
2263 | vshr.s64 q7, q14, #63 | ||
2264 | veor q14, q14, q6 | ||
2265 | vand q7, q7, q5 | ||
2266 | vld1.8 {q4}, [r7]! | ||
2267 | subs r9, #0x10 | ||
2268 | bmi .Lxts_dec_5 | ||
2269 | veor q3, q3, q11 | ||
2270 | vadd.u64 q15, q14, q14 | ||
2271 | vst1.64 {q14}, [r0,:128]! | ||
2272 | vswp d15,d14 | ||
2273 | vshr.s64 q6, q15, #63 | ||
2274 | veor q15, q15, q7 | ||
2275 | vand q6, q6, q5 | ||
2276 | vld1.8 {q5}, [r7]! | ||
2277 | subs r9, #0x10 | ||
2278 | bmi .Lxts_dec_6 | ||
2279 | veor q4, q4, q12 | ||
2280 | sub r9, #0x10 | ||
2281 | vst1.64 {q15}, [r0,:128] @ next round tweak | ||
2282 | |||
2283 | vld1.8 {q6}, [r7]! | ||
2284 | veor q5, q5, q13 | ||
2285 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2286 | add r4, sp, #0x90 @ pass key schedule | ||
2287 | #else | ||
2288 | add r4, r10, #248 @ pass key schedule | ||
2289 | #endif | ||
2290 | veor q6, q6, q14 | ||
2291 | mov r5, r1 @ pass rounds | ||
2292 | mov r0, sp | ||
2293 | |||
2294 | bl _bsaes_decrypt8 | ||
2295 | |||
2296 | vld1.64 {q8-q9}, [r0,:128]! | ||
2297 | vld1.64 {q10-q11}, [r0,:128]! | ||
2298 | veor q0, q0, q8 | ||
2299 | vld1.64 {q12-q13}, [r0,:128]! | ||
2300 | veor q1, q1, q9 | ||
2301 | veor q8, q6, q10 | ||
2302 | vst1.8 {q0-q1}, [r8]! | ||
2303 | veor q9, q4, q11 | ||
2304 | vld1.64 {q14}, [r0,:128]! | ||
2305 | veor q10, q2, q12 | ||
2306 | vst1.8 {q8-q9}, [r8]! | ||
2307 | veor q11, q7, q13 | ||
2308 | veor q12, q3, q14 | ||
2309 | vst1.8 {q10-q11}, [r8]! | ||
2310 | vst1.8 {q12}, [r8]! | ||
2311 | |||
2312 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
2313 | b .Lxts_dec_done | ||
2314 | .align 4 | ||
2315 | .Lxts_dec_6: | ||
2316 | vst1.64 {q14}, [r0,:128] @ next round tweak | ||
2317 | |||
2318 | veor q4, q4, q12 | ||
2319 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2320 | add r4, sp, #0x90 @ pass key schedule | ||
2321 | #else | ||
2322 | add r4, r10, #248 @ pass key schedule | ||
2323 | #endif | ||
2324 | veor q5, q5, q13 | ||
2325 | mov r5, r1 @ pass rounds | ||
2326 | mov r0, sp | ||
2327 | |||
2328 | bl _bsaes_decrypt8 | ||
2329 | |||
2330 | vld1.64 {q8-q9}, [r0,:128]! | ||
2331 | vld1.64 {q10-q11}, [r0,:128]! | ||
2332 | veor q0, q0, q8 | ||
2333 | vld1.64 {q12-q13}, [r0,:128]! | ||
2334 | veor q1, q1, q9 | ||
2335 | veor q8, q6, q10 | ||
2336 | vst1.8 {q0-q1}, [r8]! | ||
2337 | veor q9, q4, q11 | ||
2338 | veor q10, q2, q12 | ||
2339 | vst1.8 {q8-q9}, [r8]! | ||
2340 | veor q11, q7, q13 | ||
2341 | vst1.8 {q10-q11}, [r8]! | ||
2342 | |||
2343 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
2344 | b .Lxts_dec_done | ||
2345 | .align 4 | ||
2346 | .Lxts_dec_5: | ||
2347 | vst1.64 {q13}, [r0,:128] @ next round tweak | ||
2348 | |||
2349 | veor q3, q3, q11 | ||
2350 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2351 | add r4, sp, #0x90 @ pass key schedule | ||
2352 | #else | ||
2353 | add r4, r10, #248 @ pass key schedule | ||
2354 | #endif | ||
2355 | veor q4, q4, q12 | ||
2356 | mov r5, r1 @ pass rounds | ||
2357 | mov r0, sp | ||
2358 | |||
2359 | bl _bsaes_decrypt8 | ||
2360 | |||
2361 | vld1.64 {q8-q9}, [r0,:128]! | ||
2362 | vld1.64 {q10-q11}, [r0,:128]! | ||
2363 | veor q0, q0, q8 | ||
2364 | vld1.64 {q12}, [r0,:128]! | ||
2365 | veor q1, q1, q9 | ||
2366 | veor q8, q6, q10 | ||
2367 | vst1.8 {q0-q1}, [r8]! | ||
2368 | veor q9, q4, q11 | ||
2369 | veor q10, q2, q12 | ||
2370 | vst1.8 {q8-q9}, [r8]! | ||
2371 | vst1.8 {q10}, [r8]! | ||
2372 | |||
2373 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
2374 | b .Lxts_dec_done | ||
2375 | .align 4 | ||
2376 | .Lxts_dec_4: | ||
2377 | vst1.64 {q12}, [r0,:128] @ next round tweak | ||
2378 | |||
2379 | veor q2, q2, q10 | ||
2380 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2381 | add r4, sp, #0x90 @ pass key schedule | ||
2382 | #else | ||
2383 | add r4, r10, #248 @ pass key schedule | ||
2384 | #endif | ||
2385 | veor q3, q3, q11 | ||
2386 | mov r5, r1 @ pass rounds | ||
2387 | mov r0, sp | ||
2388 | |||
2389 | bl _bsaes_decrypt8 | ||
2390 | |||
2391 | vld1.64 {q8-q9}, [r0,:128]! | ||
2392 | vld1.64 {q10-q11}, [r0,:128]! | ||
2393 | veor q0, q0, q8 | ||
2394 | veor q1, q1, q9 | ||
2395 | veor q8, q6, q10 | ||
2396 | vst1.8 {q0-q1}, [r8]! | ||
2397 | veor q9, q4, q11 | ||
2398 | vst1.8 {q8-q9}, [r8]! | ||
2399 | |||
2400 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
2401 | b .Lxts_dec_done | ||
2402 | .align 4 | ||
2403 | .Lxts_dec_3: | ||
2404 | vst1.64 {q11}, [r0,:128] @ next round tweak | ||
2405 | |||
2406 | veor q1, q1, q9 | ||
2407 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2408 | add r4, sp, #0x90 @ pass key schedule | ||
2409 | #else | ||
2410 | add r4, r10, #248 @ pass key schedule | ||
2411 | #endif | ||
2412 | veor q2, q2, q10 | ||
2413 | mov r5, r1 @ pass rounds | ||
2414 | mov r0, sp | ||
2415 | |||
2416 | bl _bsaes_decrypt8 | ||
2417 | |||
2418 | vld1.64 {q8-q9}, [r0,:128]! | ||
2419 | vld1.64 {q10}, [r0,:128]! | ||
2420 | veor q0, q0, q8 | ||
2421 | veor q1, q1, q9 | ||
2422 | veor q8, q6, q10 | ||
2423 | vst1.8 {q0-q1}, [r8]! | ||
2424 | vst1.8 {q8}, [r8]! | ||
2425 | |||
2426 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
2427 | b .Lxts_dec_done | ||
2428 | .align 4 | ||
2429 | .Lxts_dec_2: | ||
2430 | vst1.64 {q10}, [r0,:128] @ next round tweak | ||
2431 | |||
2432 | veor q0, q0, q8 | ||
2433 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2434 | add r4, sp, #0x90 @ pass key schedule | ||
2435 | #else | ||
2436 | add r4, r10, #248 @ pass key schedule | ||
2437 | #endif | ||
2438 | veor q1, q1, q9 | ||
2439 | mov r5, r1 @ pass rounds | ||
2440 | mov r0, sp | ||
2441 | |||
2442 | bl _bsaes_decrypt8 | ||
2443 | |||
2444 | vld1.64 {q8-q9}, [r0,:128]! | ||
2445 | veor q0, q0, q8 | ||
2446 | veor q1, q1, q9 | ||
2447 | vst1.8 {q0-q1}, [r8]! | ||
2448 | |||
2449 | vld1.64 {q8}, [r0,:128] @ next round tweak | ||
2450 | b .Lxts_dec_done | ||
2451 | .align 4 | ||
2452 | .Lxts_dec_1: | ||
2453 | mov r0, sp | ||
2454 | veor q0, q8 | ||
2455 | mov r1, sp | ||
2456 | vst1.8 {q0}, [sp,:128] | ||
2457 | mov r2, r10 | ||
2458 | mov r4, r3 @ preserve fp | ||
2459 | mov r5, r2 @ preserve magic | ||
2460 | |||
2461 | bl AES_decrypt | ||
2462 | |||
2463 | vld1.8 {q0}, [sp,:128] | ||
2464 | veor q0, q0, q8 | ||
2465 | vst1.8 {q0}, [r8]! | ||
2466 | mov r3, r4 | ||
2467 | mov r2, r5 | ||
2468 | |||
2469 | vmov q8, q9 @ next round tweak | ||
2470 | |||
2471 | .Lxts_dec_done: | ||
2472 | #ifndef XTS_CHAIN_TWEAK | ||
2473 | adds r9, #0x10 | ||
2474 | beq .Lxts_dec_ret | ||
2475 | |||
2476 | @ calculate one round of extra tweak for the stolen ciphertext | ||
2477 | vldmia r2, {q5} | ||
2478 | vshr.s64 q6, q8, #63 | ||
2479 | vand q6, q6, q5 | ||
2480 | vadd.u64 q9, q8, q8 | ||
2481 | vswp d13,d12 | ||
2482 | veor q9, q9, q6 | ||
2483 | |||
2484 | @ perform the final decryption with the last tweak value | ||
2485 | vld1.8 {q0}, [r7]! | ||
2486 | mov r0, sp | ||
2487 | veor q0, q0, q9 | ||
2488 | mov r1, sp | ||
2489 | vst1.8 {q0}, [sp,:128] | ||
2490 | mov r2, r10 | ||
2491 | mov r4, r3 @ preserve fp | ||
2492 | |||
2493 | bl AES_decrypt | ||
2494 | |||
2495 | vld1.8 {q0}, [sp,:128] | ||
2496 | veor q0, q0, q9 | ||
2497 | vst1.8 {q0}, [r8] | ||
2498 | |||
2499 | mov r6, r8 | ||
2500 | .Lxts_dec_steal: | ||
2501 | ldrb r1, [r8] | ||
2502 | ldrb r0, [r7], #1 | ||
2503 | strb r1, [r8, #0x10] | ||
2504 | strb r0, [r8], #1 | ||
2505 | |||
2506 | subs r9, #1 | ||
2507 | bhi .Lxts_dec_steal | ||
2508 | |||
2509 | vld1.8 {q0}, [r6] | ||
2510 | mov r0, sp | ||
2511 | veor q0, q8 | ||
2512 | mov r1, sp | ||
2513 | vst1.8 {q0}, [sp,:128] | ||
2514 | mov r2, r10 | ||
2515 | |||
2516 | bl AES_decrypt | ||
2517 | |||
2518 | vld1.8 {q0}, [sp,:128] | ||
2519 | veor q0, q0, q8 | ||
2520 | vst1.8 {q0}, [r6] | ||
2521 | mov r3, r4 | ||
2522 | #endif | ||
2523 | |||
2524 | .Lxts_dec_ret: | ||
2525 | bic r0, r3, #0xf | ||
2526 | vmov.i32 q0, #0 | ||
2527 | vmov.i32 q1, #0 | ||
2528 | #ifdef XTS_CHAIN_TWEAK | ||
2529 | ldr r1, [r3, #0x20+VFP_ABI_FRAME] @ chain tweak | ||
2530 | #endif | ||
2531 | .Lxts_dec_bzero: @ wipe key schedule [if any] | ||
2532 | vstmia sp!, {q0-q1} | ||
2533 | cmp sp, r0 | ||
2534 | bne .Lxts_dec_bzero | ||
2535 | |||
2536 | mov sp, r3 | ||
2537 | #ifdef XTS_CHAIN_TWEAK | ||
2538 | vst1.8 {q8}, [r1] | ||
2539 | #endif | ||
2540 | VFP_ABI_POP | ||
2541 | ldmia sp!, {r4-r10, pc} @ return | ||
2542 | |||
2543 | .size bsaes_xts_decrypt,.-bsaes_xts_decrypt | ||
2544 | #endif | ||
diff --git a/arch/arm/crypto/aesbs-glue.c b/arch/arm/crypto/aesbs-glue.c new file mode 100644 index 000000000000..4522366da759 --- /dev/null +++ b/arch/arm/crypto/aesbs-glue.c | |||
@@ -0,0 +1,434 @@ | |||
1 | /* | ||
2 | * linux/arch/arm/crypto/aesbs-glue.c - glue code for NEON bit sliced AES | ||
3 | * | ||
4 | * Copyright (C) 2013 Linaro Ltd <ard.biesheuvel@linaro.org> | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License version 2 as | ||
8 | * published by the Free Software Foundation. | ||
9 | */ | ||
10 | |||
11 | #include <asm/neon.h> | ||
12 | #include <crypto/aes.h> | ||
13 | #include <crypto/ablk_helper.h> | ||
14 | #include <crypto/algapi.h> | ||
15 | #include <linux/module.h> | ||
16 | |||
17 | #include "aes_glue.h" | ||
18 | |||
19 | #define BIT_SLICED_KEY_MAXSIZE (128 * (AES_MAXNR - 1) + 2 * AES_BLOCK_SIZE) | ||
20 | |||
21 | struct BS_KEY { | ||
22 | struct AES_KEY rk; | ||
23 | int converted; | ||
24 | u8 __aligned(8) bs[BIT_SLICED_KEY_MAXSIZE]; | ||
25 | } __aligned(8); | ||
26 | |||
27 | asmlinkage void bsaes_enc_key_convert(u8 out[], struct AES_KEY const *in); | ||
28 | asmlinkage void bsaes_dec_key_convert(u8 out[], struct AES_KEY const *in); | ||
29 | |||
30 | asmlinkage void bsaes_cbc_encrypt(u8 const in[], u8 out[], u32 bytes, | ||
31 | struct BS_KEY *key, u8 iv[]); | ||
32 | |||
33 | asmlinkage void bsaes_ctr32_encrypt_blocks(u8 const in[], u8 out[], u32 blocks, | ||
34 | struct BS_KEY *key, u8 const iv[]); | ||
35 | |||
36 | asmlinkage void bsaes_xts_encrypt(u8 const in[], u8 out[], u32 bytes, | ||
37 | struct BS_KEY *key, u8 tweak[]); | ||
38 | |||
39 | asmlinkage void bsaes_xts_decrypt(u8 const in[], u8 out[], u32 bytes, | ||
40 | struct BS_KEY *key, u8 tweak[]); | ||
41 | |||
42 | struct aesbs_cbc_ctx { | ||
43 | struct AES_KEY enc; | ||
44 | struct BS_KEY dec; | ||
45 | }; | ||
46 | |||
47 | struct aesbs_ctr_ctx { | ||
48 | struct BS_KEY enc; | ||
49 | }; | ||
50 | |||
51 | struct aesbs_xts_ctx { | ||
52 | struct BS_KEY enc; | ||
53 | struct BS_KEY dec; | ||
54 | struct AES_KEY twkey; | ||
55 | }; | ||
56 | |||
57 | static int aesbs_cbc_set_key(struct crypto_tfm *tfm, const u8 *in_key, | ||
58 | unsigned int key_len) | ||
59 | { | ||
60 | struct aesbs_cbc_ctx *ctx = crypto_tfm_ctx(tfm); | ||
61 | int bits = key_len * 8; | ||
62 | |||
63 | if (private_AES_set_encrypt_key(in_key, bits, &ctx->enc)) { | ||
64 | tfm->crt_flags |= CRYPTO_TFM_RES_BAD_KEY_LEN; | ||
65 | return -EINVAL; | ||
66 | } | ||
67 | ctx->dec.rk = ctx->enc; | ||
68 | private_AES_set_decrypt_key(in_key, bits, &ctx->dec.rk); | ||
69 | ctx->dec.converted = 0; | ||
70 | return 0; | ||
71 | } | ||
72 | |||
73 | static int aesbs_ctr_set_key(struct crypto_tfm *tfm, const u8 *in_key, | ||
74 | unsigned int key_len) | ||
75 | { | ||
76 | struct aesbs_ctr_ctx *ctx = crypto_tfm_ctx(tfm); | ||
77 | int bits = key_len * 8; | ||
78 | |||
79 | if (private_AES_set_encrypt_key(in_key, bits, &ctx->enc.rk)) { | ||
80 | tfm->crt_flags |= CRYPTO_TFM_RES_BAD_KEY_LEN; | ||
81 | return -EINVAL; | ||
82 | } | ||
83 | ctx->enc.converted = 0; | ||
84 | return 0; | ||
85 | } | ||
86 | |||
87 | static int aesbs_xts_set_key(struct crypto_tfm *tfm, const u8 *in_key, | ||
88 | unsigned int key_len) | ||
89 | { | ||
90 | struct aesbs_xts_ctx *ctx = crypto_tfm_ctx(tfm); | ||
91 | int bits = key_len * 4; | ||
92 | |||
93 | if (private_AES_set_encrypt_key(in_key, bits, &ctx->enc.rk)) { | ||
94 | tfm->crt_flags |= CRYPTO_TFM_RES_BAD_KEY_LEN; | ||
95 | return -EINVAL; | ||
96 | } | ||
97 | ctx->dec.rk = ctx->enc.rk; | ||
98 | private_AES_set_decrypt_key(in_key, bits, &ctx->dec.rk); | ||
99 | private_AES_set_encrypt_key(in_key + key_len / 2, bits, &ctx->twkey); | ||
100 | ctx->enc.converted = ctx->dec.converted = 0; | ||
101 | return 0; | ||
102 | } | ||
103 | |||
104 | static int aesbs_cbc_encrypt(struct blkcipher_desc *desc, | ||
105 | struct scatterlist *dst, | ||
106 | struct scatterlist *src, unsigned int nbytes) | ||
107 | { | ||
108 | struct aesbs_cbc_ctx *ctx = crypto_blkcipher_ctx(desc->tfm); | ||
109 | struct blkcipher_walk walk; | ||
110 | int err; | ||
111 | |||
112 | blkcipher_walk_init(&walk, dst, src, nbytes); | ||
113 | err = blkcipher_walk_virt(desc, &walk); | ||
114 | |||
115 | while (walk.nbytes) { | ||
116 | u32 blocks = walk.nbytes / AES_BLOCK_SIZE; | ||
117 | u8 *src = walk.src.virt.addr; | ||
118 | |||
119 | if (walk.dst.virt.addr == walk.src.virt.addr) { | ||
120 | u8 *iv = walk.iv; | ||
121 | |||
122 | do { | ||
123 | crypto_xor(src, iv, AES_BLOCK_SIZE); | ||
124 | AES_encrypt(src, src, &ctx->enc); | ||
125 | iv = src; | ||
126 | src += AES_BLOCK_SIZE; | ||
127 | } while (--blocks); | ||
128 | memcpy(walk.iv, iv, AES_BLOCK_SIZE); | ||
129 | } else { | ||
130 | u8 *dst = walk.dst.virt.addr; | ||
131 | |||
132 | do { | ||
133 | crypto_xor(walk.iv, src, AES_BLOCK_SIZE); | ||
134 | AES_encrypt(walk.iv, dst, &ctx->enc); | ||
135 | memcpy(walk.iv, dst, AES_BLOCK_SIZE); | ||
136 | src += AES_BLOCK_SIZE; | ||
137 | dst += AES_BLOCK_SIZE; | ||
138 | } while (--blocks); | ||
139 | } | ||
140 | err = blkcipher_walk_done(desc, &walk, 0); | ||
141 | } | ||
142 | return err; | ||
143 | } | ||
144 | |||
145 | static int aesbs_cbc_decrypt(struct blkcipher_desc *desc, | ||
146 | struct scatterlist *dst, | ||
147 | struct scatterlist *src, unsigned int nbytes) | ||
148 | { | ||
149 | struct aesbs_cbc_ctx *ctx = crypto_blkcipher_ctx(desc->tfm); | ||
150 | struct blkcipher_walk walk; | ||
151 | int err; | ||
152 | |||
153 | blkcipher_walk_init(&walk, dst, src, nbytes); | ||
154 | err = blkcipher_walk_virt_block(desc, &walk, 8 * AES_BLOCK_SIZE); | ||
155 | |||
156 | while ((walk.nbytes / AES_BLOCK_SIZE) >= 8) { | ||
157 | kernel_neon_begin(); | ||
158 | bsaes_cbc_encrypt(walk.src.virt.addr, walk.dst.virt.addr, | ||
159 | walk.nbytes, &ctx->dec, walk.iv); | ||
160 | kernel_neon_end(); | ||
161 | err = blkcipher_walk_done(desc, &walk, 0); | ||
162 | } | ||
163 | while (walk.nbytes) { | ||
164 | u32 blocks = walk.nbytes / AES_BLOCK_SIZE; | ||
165 | u8 *dst = walk.dst.virt.addr; | ||
166 | u8 *src = walk.src.virt.addr; | ||
167 | u8 bk[2][AES_BLOCK_SIZE]; | ||
168 | u8 *iv = walk.iv; | ||
169 | |||
170 | do { | ||
171 | if (walk.dst.virt.addr == walk.src.virt.addr) | ||
172 | memcpy(bk[blocks & 1], src, AES_BLOCK_SIZE); | ||
173 | |||
174 | AES_decrypt(src, dst, &ctx->dec.rk); | ||
175 | crypto_xor(dst, iv, AES_BLOCK_SIZE); | ||
176 | |||
177 | if (walk.dst.virt.addr == walk.src.virt.addr) | ||
178 | iv = bk[blocks & 1]; | ||
179 | else | ||
180 | iv = src; | ||
181 | |||
182 | dst += AES_BLOCK_SIZE; | ||
183 | src += AES_BLOCK_SIZE; | ||
184 | } while (--blocks); | ||
185 | err = blkcipher_walk_done(desc, &walk, 0); | ||
186 | } | ||
187 | return err; | ||
188 | } | ||
189 | |||
190 | static void inc_be128_ctr(__be32 ctr[], u32 addend) | ||
191 | { | ||
192 | int i; | ||
193 | |||
194 | for (i = 3; i >= 0; i--, addend = 1) { | ||
195 | u32 n = be32_to_cpu(ctr[i]) + addend; | ||
196 | |||
197 | ctr[i] = cpu_to_be32(n); | ||
198 | if (n >= addend) | ||
199 | break; | ||
200 | } | ||
201 | } | ||
202 | |||
203 | static int aesbs_ctr_encrypt(struct blkcipher_desc *desc, | ||
204 | struct scatterlist *dst, struct scatterlist *src, | ||
205 | unsigned int nbytes) | ||
206 | { | ||
207 | struct aesbs_ctr_ctx *ctx = crypto_blkcipher_ctx(desc->tfm); | ||
208 | struct blkcipher_walk walk; | ||
209 | u32 blocks; | ||
210 | int err; | ||
211 | |||
212 | blkcipher_walk_init(&walk, dst, src, nbytes); | ||
213 | err = blkcipher_walk_virt_block(desc, &walk, 8 * AES_BLOCK_SIZE); | ||
214 | |||
215 | while ((blocks = walk.nbytes / AES_BLOCK_SIZE)) { | ||
216 | u32 tail = walk.nbytes % AES_BLOCK_SIZE; | ||
217 | __be32 *ctr = (__be32 *)walk.iv; | ||
218 | u32 headroom = UINT_MAX - be32_to_cpu(ctr[3]); | ||
219 | |||
220 | /* avoid 32 bit counter overflow in the NEON code */ | ||
221 | if (unlikely(headroom < blocks)) { | ||
222 | blocks = headroom + 1; | ||
223 | tail = walk.nbytes - blocks * AES_BLOCK_SIZE; | ||
224 | } | ||
225 | kernel_neon_begin(); | ||
226 | bsaes_ctr32_encrypt_blocks(walk.src.virt.addr, | ||
227 | walk.dst.virt.addr, blocks, | ||
228 | &ctx->enc, walk.iv); | ||
229 | kernel_neon_end(); | ||
230 | inc_be128_ctr(ctr, blocks); | ||
231 | |||
232 | nbytes -= blocks * AES_BLOCK_SIZE; | ||
233 | if (nbytes && nbytes == tail && nbytes <= AES_BLOCK_SIZE) | ||
234 | break; | ||
235 | |||
236 | err = blkcipher_walk_done(desc, &walk, tail); | ||
237 | } | ||
238 | if (walk.nbytes) { | ||
239 | u8 *tdst = walk.dst.virt.addr + blocks * AES_BLOCK_SIZE; | ||
240 | u8 *tsrc = walk.src.virt.addr + blocks * AES_BLOCK_SIZE; | ||
241 | u8 ks[AES_BLOCK_SIZE]; | ||
242 | |||
243 | AES_encrypt(walk.iv, ks, &ctx->enc.rk); | ||
244 | if (tdst != tsrc) | ||
245 | memcpy(tdst, tsrc, nbytes); | ||
246 | crypto_xor(tdst, ks, nbytes); | ||
247 | err = blkcipher_walk_done(desc, &walk, 0); | ||
248 | } | ||
249 | return err; | ||
250 | } | ||
251 | |||
252 | static int aesbs_xts_encrypt(struct blkcipher_desc *desc, | ||
253 | struct scatterlist *dst, | ||
254 | struct scatterlist *src, unsigned int nbytes) | ||
255 | { | ||
256 | struct aesbs_xts_ctx *ctx = crypto_blkcipher_ctx(desc->tfm); | ||
257 | struct blkcipher_walk walk; | ||
258 | int err; | ||
259 | |||
260 | blkcipher_walk_init(&walk, dst, src, nbytes); | ||
261 | err = blkcipher_walk_virt_block(desc, &walk, 8 * AES_BLOCK_SIZE); | ||
262 | |||
263 | /* generate the initial tweak */ | ||
264 | AES_encrypt(walk.iv, walk.iv, &ctx->twkey); | ||
265 | |||
266 | while (walk.nbytes) { | ||
267 | kernel_neon_begin(); | ||
268 | bsaes_xts_encrypt(walk.src.virt.addr, walk.dst.virt.addr, | ||
269 | walk.nbytes, &ctx->enc, walk.iv); | ||
270 | kernel_neon_end(); | ||
271 | err = blkcipher_walk_done(desc, &walk, 0); | ||
272 | } | ||
273 | return err; | ||
274 | } | ||
275 | |||
276 | static int aesbs_xts_decrypt(struct blkcipher_desc *desc, | ||
277 | struct scatterlist *dst, | ||
278 | struct scatterlist *src, unsigned int nbytes) | ||
279 | { | ||
280 | struct aesbs_xts_ctx *ctx = crypto_blkcipher_ctx(desc->tfm); | ||
281 | struct blkcipher_walk walk; | ||
282 | int err; | ||
283 | |||
284 | blkcipher_walk_init(&walk, dst, src, nbytes); | ||
285 | err = blkcipher_walk_virt_block(desc, &walk, 8 * AES_BLOCK_SIZE); | ||
286 | |||
287 | /* generate the initial tweak */ | ||
288 | AES_encrypt(walk.iv, walk.iv, &ctx->twkey); | ||
289 | |||
290 | while (walk.nbytes) { | ||
291 | kernel_neon_begin(); | ||
292 | bsaes_xts_decrypt(walk.src.virt.addr, walk.dst.virt.addr, | ||
293 | walk.nbytes, &ctx->dec, walk.iv); | ||
294 | kernel_neon_end(); | ||
295 | err = blkcipher_walk_done(desc, &walk, 0); | ||
296 | } | ||
297 | return err; | ||
298 | } | ||
299 | |||
300 | static struct crypto_alg aesbs_algs[] = { { | ||
301 | .cra_name = "__cbc-aes-neonbs", | ||
302 | .cra_driver_name = "__driver-cbc-aes-neonbs", | ||
303 | .cra_priority = 0, | ||
304 | .cra_flags = CRYPTO_ALG_TYPE_BLKCIPHER, | ||
305 | .cra_blocksize = AES_BLOCK_SIZE, | ||
306 | .cra_ctxsize = sizeof(struct aesbs_cbc_ctx), | ||
307 | .cra_alignmask = 7, | ||
308 | .cra_type = &crypto_blkcipher_type, | ||
309 | .cra_module = THIS_MODULE, | ||
310 | .cra_blkcipher = { | ||
311 | .min_keysize = AES_MIN_KEY_SIZE, | ||
312 | .max_keysize = AES_MAX_KEY_SIZE, | ||
313 | .ivsize = AES_BLOCK_SIZE, | ||
314 | .setkey = aesbs_cbc_set_key, | ||
315 | .encrypt = aesbs_cbc_encrypt, | ||
316 | .decrypt = aesbs_cbc_decrypt, | ||
317 | }, | ||
318 | }, { | ||
319 | .cra_name = "__ctr-aes-neonbs", | ||
320 | .cra_driver_name = "__driver-ctr-aes-neonbs", | ||
321 | .cra_priority = 0, | ||
322 | .cra_flags = CRYPTO_ALG_TYPE_BLKCIPHER, | ||
323 | .cra_blocksize = 1, | ||
324 | .cra_ctxsize = sizeof(struct aesbs_ctr_ctx), | ||
325 | .cra_alignmask = 7, | ||
326 | .cra_type = &crypto_blkcipher_type, | ||
327 | .cra_module = THIS_MODULE, | ||
328 | .cra_blkcipher = { | ||
329 | .min_keysize = AES_MIN_KEY_SIZE, | ||
330 | .max_keysize = AES_MAX_KEY_SIZE, | ||
331 | .ivsize = AES_BLOCK_SIZE, | ||
332 | .setkey = aesbs_ctr_set_key, | ||
333 | .encrypt = aesbs_ctr_encrypt, | ||
334 | .decrypt = aesbs_ctr_encrypt, | ||
335 | }, | ||
336 | }, { | ||
337 | .cra_name = "__xts-aes-neonbs", | ||
338 | .cra_driver_name = "__driver-xts-aes-neonbs", | ||
339 | .cra_priority = 0, | ||
340 | .cra_flags = CRYPTO_ALG_TYPE_BLKCIPHER, | ||
341 | .cra_blocksize = AES_BLOCK_SIZE, | ||
342 | .cra_ctxsize = sizeof(struct aesbs_xts_ctx), | ||
343 | .cra_alignmask = 7, | ||
344 | .cra_type = &crypto_blkcipher_type, | ||
345 | .cra_module = THIS_MODULE, | ||
346 | .cra_blkcipher = { | ||
347 | .min_keysize = 2 * AES_MIN_KEY_SIZE, | ||
348 | .max_keysize = 2 * AES_MAX_KEY_SIZE, | ||
349 | .ivsize = AES_BLOCK_SIZE, | ||
350 | .setkey = aesbs_xts_set_key, | ||
351 | .encrypt = aesbs_xts_encrypt, | ||
352 | .decrypt = aesbs_xts_decrypt, | ||
353 | }, | ||
354 | }, { | ||
355 | .cra_name = "cbc(aes)", | ||
356 | .cra_driver_name = "cbc-aes-neonbs", | ||
357 | .cra_priority = 300, | ||
358 | .cra_flags = CRYPTO_ALG_TYPE_ABLKCIPHER|CRYPTO_ALG_ASYNC, | ||
359 | .cra_blocksize = AES_BLOCK_SIZE, | ||
360 | .cra_ctxsize = sizeof(struct async_helper_ctx), | ||
361 | .cra_alignmask = 7, | ||
362 | .cra_type = &crypto_ablkcipher_type, | ||
363 | .cra_module = THIS_MODULE, | ||
364 | .cra_init = ablk_init, | ||
365 | .cra_exit = ablk_exit, | ||
366 | .cra_ablkcipher = { | ||
367 | .min_keysize = AES_MIN_KEY_SIZE, | ||
368 | .max_keysize = AES_MAX_KEY_SIZE, | ||
369 | .ivsize = AES_BLOCK_SIZE, | ||
370 | .setkey = ablk_set_key, | ||
371 | .encrypt = __ablk_encrypt, | ||
372 | .decrypt = ablk_decrypt, | ||
373 | } | ||
374 | }, { | ||
375 | .cra_name = "ctr(aes)", | ||
376 | .cra_driver_name = "ctr-aes-neonbs", | ||
377 | .cra_priority = 300, | ||
378 | .cra_flags = CRYPTO_ALG_TYPE_ABLKCIPHER|CRYPTO_ALG_ASYNC, | ||
379 | .cra_blocksize = 1, | ||
380 | .cra_ctxsize = sizeof(struct async_helper_ctx), | ||
381 | .cra_alignmask = 7, | ||
382 | .cra_type = &crypto_ablkcipher_type, | ||
383 | .cra_module = THIS_MODULE, | ||
384 | .cra_init = ablk_init, | ||
385 | .cra_exit = ablk_exit, | ||
386 | .cra_ablkcipher = { | ||
387 | .min_keysize = AES_MIN_KEY_SIZE, | ||
388 | .max_keysize = AES_MAX_KEY_SIZE, | ||
389 | .ivsize = AES_BLOCK_SIZE, | ||
390 | .setkey = ablk_set_key, | ||
391 | .encrypt = ablk_encrypt, | ||
392 | .decrypt = ablk_decrypt, | ||
393 | } | ||
394 | }, { | ||
395 | .cra_name = "xts(aes)", | ||
396 | .cra_driver_name = "xts-aes-neonbs", | ||
397 | .cra_priority = 300, | ||
398 | .cra_flags = CRYPTO_ALG_TYPE_ABLKCIPHER|CRYPTO_ALG_ASYNC, | ||
399 | .cra_blocksize = AES_BLOCK_SIZE, | ||
400 | .cra_ctxsize = sizeof(struct async_helper_ctx), | ||
401 | .cra_alignmask = 7, | ||
402 | .cra_type = &crypto_ablkcipher_type, | ||
403 | .cra_module = THIS_MODULE, | ||
404 | .cra_init = ablk_init, | ||
405 | .cra_exit = ablk_exit, | ||
406 | .cra_ablkcipher = { | ||
407 | .min_keysize = 2 * AES_MIN_KEY_SIZE, | ||
408 | .max_keysize = 2 * AES_MAX_KEY_SIZE, | ||
409 | .ivsize = AES_BLOCK_SIZE, | ||
410 | .setkey = ablk_set_key, | ||
411 | .encrypt = ablk_encrypt, | ||
412 | .decrypt = ablk_decrypt, | ||
413 | } | ||
414 | } }; | ||
415 | |||
416 | static int __init aesbs_mod_init(void) | ||
417 | { | ||
418 | if (!cpu_has_neon()) | ||
419 | return -ENODEV; | ||
420 | |||
421 | return crypto_register_algs(aesbs_algs, ARRAY_SIZE(aesbs_algs)); | ||
422 | } | ||
423 | |||
424 | static void __exit aesbs_mod_exit(void) | ||
425 | { | ||
426 | crypto_unregister_algs(aesbs_algs, ARRAY_SIZE(aesbs_algs)); | ||
427 | } | ||
428 | |||
429 | module_init(aesbs_mod_init); | ||
430 | module_exit(aesbs_mod_exit); | ||
431 | |||
432 | MODULE_DESCRIPTION("Bit sliced AES in CBC/CTR/XTS modes using NEON"); | ||
433 | MODULE_AUTHOR("Ard Biesheuvel <ard.biesheuvel@linaro.org>"); | ||
434 | MODULE_LICENSE("GPL"); | ||
diff --git a/arch/arm/crypto/bsaes-armv7.pl b/arch/arm/crypto/bsaes-armv7.pl new file mode 100644 index 000000000000..f3d96d932573 --- /dev/null +++ b/arch/arm/crypto/bsaes-armv7.pl | |||
@@ -0,0 +1,2467 @@ | |||
1 | #!/usr/bin/env perl | ||
2 | |||
3 | # ==================================================================== | ||
4 | # Written by Andy Polyakov <appro@openssl.org> for the OpenSSL | ||
5 | # project. The module is, however, dual licensed under OpenSSL and | ||
6 | # CRYPTOGAMS licenses depending on where you obtain it. For further | ||
7 | # details see http://www.openssl.org/~appro/cryptogams/. | ||
8 | # | ||
9 | # Specific modes and adaptation for Linux kernel by Ard Biesheuvel | ||
10 | # <ard.biesheuvel@linaro.org>. Permission to use under GPL terms is | ||
11 | # granted. | ||
12 | # ==================================================================== | ||
13 | |||
14 | # Bit-sliced AES for ARM NEON | ||
15 | # | ||
16 | # February 2012. | ||
17 | # | ||
18 | # This implementation is direct adaptation of bsaes-x86_64 module for | ||
19 | # ARM NEON. Except that this module is endian-neutral [in sense that | ||
20 | # it can be compiled for either endianness] by courtesy of vld1.8's | ||
21 | # neutrality. Initial version doesn't implement interface to OpenSSL, | ||
22 | # only low-level primitives and unsupported entry points, just enough | ||
23 | # to collect performance results, which for Cortex-A8 core are: | ||
24 | # | ||
25 | # encrypt 19.5 cycles per byte processed with 128-bit key | ||
26 | # decrypt 22.1 cycles per byte processed with 128-bit key | ||
27 | # key conv. 440 cycles per 128-bit key/0.18 of 8x block | ||
28 | # | ||
29 | # Snapdragon S4 encrypts byte in 17.6 cycles and decrypts in 19.7, | ||
30 | # which is [much] worse than anticipated (for further details see | ||
31 | # http://www.openssl.org/~appro/Snapdragon-S4.html). | ||
32 | # | ||
33 | # Cortex-A15 manages in 14.2/16.1 cycles [when integer-only code | ||
34 | # manages in 20.0 cycles]. | ||
35 | # | ||
36 | # When comparing to x86_64 results keep in mind that NEON unit is | ||
37 | # [mostly] single-issue and thus can't [fully] benefit from | ||
38 | # instruction-level parallelism. And when comparing to aes-armv4 | ||
39 | # results keep in mind key schedule conversion overhead (see | ||
40 | # bsaes-x86_64.pl for further details)... | ||
41 | # | ||
42 | # <appro@openssl.org> | ||
43 | |||
44 | # April-August 2013 | ||
45 | # | ||
46 | # Add CBC, CTR and XTS subroutines, adapt for kernel use. | ||
47 | # | ||
48 | # <ard.biesheuvel@linaro.org> | ||
49 | |||
50 | while (($output=shift) && ($output!~/^\w[\w\-]*\.\w+$/)) {} | ||
51 | open STDOUT,">$output"; | ||
52 | |||
53 | my ($inp,$out,$len,$key)=("r0","r1","r2","r3"); | ||
54 | my @XMM=map("q$_",(0..15)); | ||
55 | |||
56 | { | ||
57 | my ($key,$rounds,$const)=("r4","r5","r6"); | ||
58 | |||
59 | sub Dlo() { shift=~m|q([1]?[0-9])|?"d".($1*2):""; } | ||
60 | sub Dhi() { shift=~m|q([1]?[0-9])|?"d".($1*2+1):""; } | ||
61 | |||
62 | sub Sbox { | ||
63 | # input in lsb > [b0, b1, b2, b3, b4, b5, b6, b7] < msb | ||
64 | # output in lsb > [b0, b1, b4, b6, b3, b7, b2, b5] < msb | ||
65 | my @b=@_[0..7]; | ||
66 | my @t=@_[8..11]; | ||
67 | my @s=@_[12..15]; | ||
68 | &InBasisChange (@b); | ||
69 | &Inv_GF256 (@b[6,5,0,3,7,1,4,2],@t,@s); | ||
70 | &OutBasisChange (@b[7,1,4,2,6,5,0,3]); | ||
71 | } | ||
72 | |||
73 | sub InBasisChange { | ||
74 | # input in lsb > [b0, b1, b2, b3, b4, b5, b6, b7] < msb | ||
75 | # output in lsb > [b6, b5, b0, b3, b7, b1, b4, b2] < msb | ||
76 | my @b=@_[0..7]; | ||
77 | $code.=<<___; | ||
78 | veor @b[2], @b[2], @b[1] | ||
79 | veor @b[5], @b[5], @b[6] | ||
80 | veor @b[3], @b[3], @b[0] | ||
81 | veor @b[6], @b[6], @b[2] | ||
82 | veor @b[5], @b[5], @b[0] | ||
83 | |||
84 | veor @b[6], @b[6], @b[3] | ||
85 | veor @b[3], @b[3], @b[7] | ||
86 | veor @b[7], @b[7], @b[5] | ||
87 | veor @b[3], @b[3], @b[4] | ||
88 | veor @b[4], @b[4], @b[5] | ||
89 | |||
90 | veor @b[2], @b[2], @b[7] | ||
91 | veor @b[3], @b[3], @b[1] | ||
92 | veor @b[1], @b[1], @b[5] | ||
93 | ___ | ||
94 | } | ||
95 | |||
96 | sub OutBasisChange { | ||
97 | # input in lsb > [b0, b1, b2, b3, b4, b5, b6, b7] < msb | ||
98 | # output in lsb > [b6, b1, b2, b4, b7, b0, b3, b5] < msb | ||
99 | my @b=@_[0..7]; | ||
100 | $code.=<<___; | ||
101 | veor @b[0], @b[0], @b[6] | ||
102 | veor @b[1], @b[1], @b[4] | ||
103 | veor @b[4], @b[4], @b[6] | ||
104 | veor @b[2], @b[2], @b[0] | ||
105 | veor @b[6], @b[6], @b[1] | ||
106 | |||
107 | veor @b[1], @b[1], @b[5] | ||
108 | veor @b[5], @b[5], @b[3] | ||
109 | veor @b[3], @b[3], @b[7] | ||
110 | veor @b[7], @b[7], @b[5] | ||
111 | veor @b[2], @b[2], @b[5] | ||
112 | |||
113 | veor @b[4], @b[4], @b[7] | ||
114 | ___ | ||
115 | } | ||
116 | |||
117 | sub InvSbox { | ||
118 | # input in lsb > [b0, b1, b2, b3, b4, b5, b6, b7] < msb | ||
119 | # output in lsb > [b0, b1, b6, b4, b2, b7, b3, b5] < msb | ||
120 | my @b=@_[0..7]; | ||
121 | my @t=@_[8..11]; | ||
122 | my @s=@_[12..15]; | ||
123 | &InvInBasisChange (@b); | ||
124 | &Inv_GF256 (@b[5,1,2,6,3,7,0,4],@t,@s); | ||
125 | &InvOutBasisChange (@b[3,7,0,4,5,1,2,6]); | ||
126 | } | ||
127 | |||
128 | sub InvInBasisChange { # OutBasisChange in reverse (with twist) | ||
129 | my @b=@_[5,1,2,6,3,7,0,4]; | ||
130 | $code.=<<___ | ||
131 | veor @b[1], @b[1], @b[7] | ||
132 | veor @b[4], @b[4], @b[7] | ||
133 | |||
134 | veor @b[7], @b[7], @b[5] | ||
135 | veor @b[1], @b[1], @b[3] | ||
136 | veor @b[2], @b[2], @b[5] | ||
137 | veor @b[3], @b[3], @b[7] | ||
138 | |||
139 | veor @b[6], @b[6], @b[1] | ||
140 | veor @b[2], @b[2], @b[0] | ||
141 | veor @b[5], @b[5], @b[3] | ||
142 | veor @b[4], @b[4], @b[6] | ||
143 | veor @b[0], @b[0], @b[6] | ||
144 | veor @b[1], @b[1], @b[4] | ||
145 | ___ | ||
146 | } | ||
147 | |||
148 | sub InvOutBasisChange { # InBasisChange in reverse | ||
149 | my @b=@_[2,5,7,3,6,1,0,4]; | ||
150 | $code.=<<___; | ||
151 | veor @b[1], @b[1], @b[5] | ||
152 | veor @b[2], @b[2], @b[7] | ||
153 | |||
154 | veor @b[3], @b[3], @b[1] | ||
155 | veor @b[4], @b[4], @b[5] | ||
156 | veor @b[7], @b[7], @b[5] | ||
157 | veor @b[3], @b[3], @b[4] | ||
158 | veor @b[5], @b[5], @b[0] | ||
159 | veor @b[3], @b[3], @b[7] | ||
160 | veor @b[6], @b[6], @b[2] | ||
161 | veor @b[2], @b[2], @b[1] | ||
162 | veor @b[6], @b[6], @b[3] | ||
163 | |||
164 | veor @b[3], @b[3], @b[0] | ||
165 | veor @b[5], @b[5], @b[6] | ||
166 | ___ | ||
167 | } | ||
168 | |||
169 | sub Mul_GF4 { | ||
170 | #;************************************************************* | ||
171 | #;* Mul_GF4: Input x0-x1,y0-y1 Output x0-x1 Temp t0 (8) * | ||
172 | #;************************************************************* | ||
173 | my ($x0,$x1,$y0,$y1,$t0,$t1)=@_; | ||
174 | $code.=<<___; | ||
175 | veor $t0, $y0, $y1 | ||
176 | vand $t0, $t0, $x0 | ||
177 | veor $x0, $x0, $x1 | ||
178 | vand $t1, $x1, $y0 | ||
179 | vand $x0, $x0, $y1 | ||
180 | veor $x1, $t1, $t0 | ||
181 | veor $x0, $x0, $t1 | ||
182 | ___ | ||
183 | } | ||
184 | |||
185 | sub Mul_GF4_N { # not used, see next subroutine | ||
186 | # multiply and scale by N | ||
187 | my ($x0,$x1,$y0,$y1,$t0)=@_; | ||
188 | $code.=<<___; | ||
189 | veor $t0, $y0, $y1 | ||
190 | vand $t0, $t0, $x0 | ||
191 | veor $x0, $x0, $x1 | ||
192 | vand $x1, $x1, $y0 | ||
193 | vand $x0, $x0, $y1 | ||
194 | veor $x1, $x1, $x0 | ||
195 | veor $x0, $x0, $t0 | ||
196 | ___ | ||
197 | } | ||
198 | |||
199 | sub Mul_GF4_N_GF4 { | ||
200 | # interleaved Mul_GF4_N and Mul_GF4 | ||
201 | my ($x0,$x1,$y0,$y1,$t0, | ||
202 | $x2,$x3,$y2,$y3,$t1)=@_; | ||
203 | $code.=<<___; | ||
204 | veor $t0, $y0, $y1 | ||
205 | veor $t1, $y2, $y3 | ||
206 | vand $t0, $t0, $x0 | ||
207 | vand $t1, $t1, $x2 | ||
208 | veor $x0, $x0, $x1 | ||
209 | veor $x2, $x2, $x3 | ||
210 | vand $x1, $x1, $y0 | ||
211 | vand $x3, $x3, $y2 | ||
212 | vand $x0, $x0, $y1 | ||
213 | vand $x2, $x2, $y3 | ||
214 | veor $x1, $x1, $x0 | ||
215 | veor $x2, $x2, $x3 | ||
216 | veor $x0, $x0, $t0 | ||
217 | veor $x3, $x3, $t1 | ||
218 | ___ | ||
219 | } | ||
220 | sub Mul_GF16_2 { | ||
221 | my @x=@_[0..7]; | ||
222 | my @y=@_[8..11]; | ||
223 | my @t=@_[12..15]; | ||
224 | $code.=<<___; | ||
225 | veor @t[0], @x[0], @x[2] | ||
226 | veor @t[1], @x[1], @x[3] | ||
227 | ___ | ||
228 | &Mul_GF4 (@x[0], @x[1], @y[0], @y[1], @t[2..3]); | ||
229 | $code.=<<___; | ||
230 | veor @y[0], @y[0], @y[2] | ||
231 | veor @y[1], @y[1], @y[3] | ||
232 | ___ | ||
233 | Mul_GF4_N_GF4 (@t[0], @t[1], @y[0], @y[1], @t[3], | ||
234 | @x[2], @x[3], @y[2], @y[3], @t[2]); | ||
235 | $code.=<<___; | ||
236 | veor @x[0], @x[0], @t[0] | ||
237 | veor @x[2], @x[2], @t[0] | ||
238 | veor @x[1], @x[1], @t[1] | ||
239 | veor @x[3], @x[3], @t[1] | ||
240 | |||
241 | veor @t[0], @x[4], @x[6] | ||
242 | veor @t[1], @x[5], @x[7] | ||
243 | ___ | ||
244 | &Mul_GF4_N_GF4 (@t[0], @t[1], @y[0], @y[1], @t[3], | ||
245 | @x[6], @x[7], @y[2], @y[3], @t[2]); | ||
246 | $code.=<<___; | ||
247 | veor @y[0], @y[0], @y[2] | ||
248 | veor @y[1], @y[1], @y[3] | ||
249 | ___ | ||
250 | &Mul_GF4 (@x[4], @x[5], @y[0], @y[1], @t[2..3]); | ||
251 | $code.=<<___; | ||
252 | veor @x[4], @x[4], @t[0] | ||
253 | veor @x[6], @x[6], @t[0] | ||
254 | veor @x[5], @x[5], @t[1] | ||
255 | veor @x[7], @x[7], @t[1] | ||
256 | ___ | ||
257 | } | ||
258 | sub Inv_GF256 { | ||
259 | #;******************************************************************** | ||
260 | #;* Inv_GF256: Input x0-x7 Output x0-x7 Temp t0-t3,s0-s3 (144) * | ||
261 | #;******************************************************************** | ||
262 | my @x=@_[0..7]; | ||
263 | my @t=@_[8..11]; | ||
264 | my @s=@_[12..15]; | ||
265 | # direct optimizations from hardware | ||
266 | $code.=<<___; | ||
267 | veor @t[3], @x[4], @x[6] | ||
268 | veor @t[2], @x[5], @x[7] | ||
269 | veor @t[1], @x[1], @x[3] | ||
270 | veor @s[1], @x[7], @x[6] | ||
271 | vmov @t[0], @t[2] | ||
272 | veor @s[0], @x[0], @x[2] | ||
273 | |||
274 | vorr @t[2], @t[2], @t[1] | ||
275 | veor @s[3], @t[3], @t[0] | ||
276 | vand @s[2], @t[3], @s[0] | ||
277 | vorr @t[3], @t[3], @s[0] | ||
278 | veor @s[0], @s[0], @t[1] | ||
279 | vand @t[0], @t[0], @t[1] | ||
280 | veor @t[1], @x[3], @x[2] | ||
281 | vand @s[3], @s[3], @s[0] | ||
282 | vand @s[1], @s[1], @t[1] | ||
283 | veor @t[1], @x[4], @x[5] | ||
284 | veor @s[0], @x[1], @x[0] | ||
285 | veor @t[3], @t[3], @s[1] | ||
286 | veor @t[2], @t[2], @s[1] | ||
287 | vand @s[1], @t[1], @s[0] | ||
288 | vorr @t[1], @t[1], @s[0] | ||
289 | veor @t[3], @t[3], @s[3] | ||
290 | veor @t[0], @t[0], @s[1] | ||
291 | veor @t[2], @t[2], @s[2] | ||
292 | veor @t[1], @t[1], @s[3] | ||
293 | veor @t[0], @t[0], @s[2] | ||
294 | vand @s[0], @x[7], @x[3] | ||
295 | veor @t[1], @t[1], @s[2] | ||
296 | vand @s[1], @x[6], @x[2] | ||
297 | vand @s[2], @x[5], @x[1] | ||
298 | vorr @s[3], @x[4], @x[0] | ||
299 | veor @t[3], @t[3], @s[0] | ||
300 | veor @t[1], @t[1], @s[2] | ||
301 | veor @t[0], @t[0], @s[3] | ||
302 | veor @t[2], @t[2], @s[1] | ||
303 | |||
304 | @ Inv_GF16 \t0, \t1, \t2, \t3, \s0, \s1, \s2, \s3 | ||
305 | |||
306 | @ new smaller inversion | ||
307 | |||
308 | vand @s[2], @t[3], @t[1] | ||
309 | vmov @s[0], @t[0] | ||
310 | |||
311 | veor @s[1], @t[2], @s[2] | ||
312 | veor @s[3], @t[0], @s[2] | ||
313 | veor @s[2], @t[0], @s[2] @ @s[2]=@s[3] | ||
314 | |||
315 | vbsl @s[1], @t[1], @t[0] | ||
316 | vbsl @s[3], @t[3], @t[2] | ||
317 | veor @t[3], @t[3], @t[2] | ||
318 | |||
319 | vbsl @s[0], @s[1], @s[2] | ||
320 | vbsl @t[0], @s[2], @s[1] | ||
321 | |||
322 | vand @s[2], @s[0], @s[3] | ||
323 | veor @t[1], @t[1], @t[0] | ||
324 | |||
325 | veor @s[2], @s[2], @t[3] | ||
326 | ___ | ||
327 | # output in s3, s2, s1, t1 | ||
328 | |||
329 | # Mul_GF16_2 \x0, \x1, \x2, \x3, \x4, \x5, \x6, \x7, \t2, \t3, \t0, \t1, \s0, \s1, \s2, \s3 | ||
330 | |||
331 | # Mul_GF16_2 \x0, \x1, \x2, \x3, \x4, \x5, \x6, \x7, \s3, \s2, \s1, \t1, \s0, \t0, \t2, \t3 | ||
332 | &Mul_GF16_2(@x,@s[3,2,1],@t[1],@s[0],@t[0,2,3]); | ||
333 | |||
334 | ### output msb > [x3,x2,x1,x0,x7,x6,x5,x4] < lsb | ||
335 | } | ||
336 | |||
337 | # AES linear components | ||
338 | |||
339 | sub ShiftRows { | ||
340 | my @x=@_[0..7]; | ||
341 | my @t=@_[8..11]; | ||
342 | my $mask=pop; | ||
343 | $code.=<<___; | ||
344 | vldmia $key!, {@t[0]-@t[3]} | ||
345 | veor @t[0], @t[0], @x[0] | ||
346 | veor @t[1], @t[1], @x[1] | ||
347 | vtbl.8 `&Dlo(@x[0])`, {@t[0]}, `&Dlo($mask)` | ||
348 | vtbl.8 `&Dhi(@x[0])`, {@t[0]}, `&Dhi($mask)` | ||
349 | vldmia $key!, {@t[0]} | ||
350 | veor @t[2], @t[2], @x[2] | ||
351 | vtbl.8 `&Dlo(@x[1])`, {@t[1]}, `&Dlo($mask)` | ||
352 | vtbl.8 `&Dhi(@x[1])`, {@t[1]}, `&Dhi($mask)` | ||
353 | vldmia $key!, {@t[1]} | ||
354 | veor @t[3], @t[3], @x[3] | ||
355 | vtbl.8 `&Dlo(@x[2])`, {@t[2]}, `&Dlo($mask)` | ||
356 | vtbl.8 `&Dhi(@x[2])`, {@t[2]}, `&Dhi($mask)` | ||
357 | vldmia $key!, {@t[2]} | ||
358 | vtbl.8 `&Dlo(@x[3])`, {@t[3]}, `&Dlo($mask)` | ||
359 | vtbl.8 `&Dhi(@x[3])`, {@t[3]}, `&Dhi($mask)` | ||
360 | vldmia $key!, {@t[3]} | ||
361 | veor @t[0], @t[0], @x[4] | ||
362 | veor @t[1], @t[1], @x[5] | ||
363 | vtbl.8 `&Dlo(@x[4])`, {@t[0]}, `&Dlo($mask)` | ||
364 | vtbl.8 `&Dhi(@x[4])`, {@t[0]}, `&Dhi($mask)` | ||
365 | veor @t[2], @t[2], @x[6] | ||
366 | vtbl.8 `&Dlo(@x[5])`, {@t[1]}, `&Dlo($mask)` | ||
367 | vtbl.8 `&Dhi(@x[5])`, {@t[1]}, `&Dhi($mask)` | ||
368 | veor @t[3], @t[3], @x[7] | ||
369 | vtbl.8 `&Dlo(@x[6])`, {@t[2]}, `&Dlo($mask)` | ||
370 | vtbl.8 `&Dhi(@x[6])`, {@t[2]}, `&Dhi($mask)` | ||
371 | vtbl.8 `&Dlo(@x[7])`, {@t[3]}, `&Dlo($mask)` | ||
372 | vtbl.8 `&Dhi(@x[7])`, {@t[3]}, `&Dhi($mask)` | ||
373 | ___ | ||
374 | } | ||
375 | |||
376 | sub MixColumns { | ||
377 | # modified to emit output in order suitable for feeding back to aesenc[last] | ||
378 | my @x=@_[0..7]; | ||
379 | my @t=@_[8..15]; | ||
380 | my $inv=@_[16]; # optional | ||
381 | $code.=<<___; | ||
382 | vext.8 @t[0], @x[0], @x[0], #12 @ x0 <<< 32 | ||
383 | vext.8 @t[1], @x[1], @x[1], #12 | ||
384 | veor @x[0], @x[0], @t[0] @ x0 ^ (x0 <<< 32) | ||
385 | vext.8 @t[2], @x[2], @x[2], #12 | ||
386 | veor @x[1], @x[1], @t[1] | ||
387 | vext.8 @t[3], @x[3], @x[3], #12 | ||
388 | veor @x[2], @x[2], @t[2] | ||
389 | vext.8 @t[4], @x[4], @x[4], #12 | ||
390 | veor @x[3], @x[3], @t[3] | ||
391 | vext.8 @t[5], @x[5], @x[5], #12 | ||
392 | veor @x[4], @x[4], @t[4] | ||
393 | vext.8 @t[6], @x[6], @x[6], #12 | ||
394 | veor @x[5], @x[5], @t[5] | ||
395 | vext.8 @t[7], @x[7], @x[7], #12 | ||
396 | veor @x[6], @x[6], @t[6] | ||
397 | |||
398 | veor @t[1], @t[1], @x[0] | ||
399 | veor @x[7], @x[7], @t[7] | ||
400 | vext.8 @x[0], @x[0], @x[0], #8 @ (x0 ^ (x0 <<< 32)) <<< 64) | ||
401 | veor @t[2], @t[2], @x[1] | ||
402 | veor @t[0], @t[0], @x[7] | ||
403 | veor @t[1], @t[1], @x[7] | ||
404 | vext.8 @x[1], @x[1], @x[1], #8 | ||
405 | veor @t[5], @t[5], @x[4] | ||
406 | veor @x[0], @x[0], @t[0] | ||
407 | veor @t[6], @t[6], @x[5] | ||
408 | veor @x[1], @x[1], @t[1] | ||
409 | vext.8 @t[0], @x[4], @x[4], #8 | ||
410 | veor @t[4], @t[4], @x[3] | ||
411 | vext.8 @t[1], @x[5], @x[5], #8 | ||
412 | veor @t[7], @t[7], @x[6] | ||
413 | vext.8 @x[4], @x[3], @x[3], #8 | ||
414 | veor @t[3], @t[3], @x[2] | ||
415 | vext.8 @x[5], @x[7], @x[7], #8 | ||
416 | veor @t[4], @t[4], @x[7] | ||
417 | vext.8 @x[3], @x[6], @x[6], #8 | ||
418 | veor @t[3], @t[3], @x[7] | ||
419 | vext.8 @x[6], @x[2], @x[2], #8 | ||
420 | veor @x[7], @t[1], @t[5] | ||
421 | ___ | ||
422 | $code.=<<___ if (!$inv); | ||
423 | veor @x[2], @t[0], @t[4] | ||
424 | veor @x[4], @x[4], @t[3] | ||
425 | veor @x[5], @x[5], @t[7] | ||
426 | veor @x[3], @x[3], @t[6] | ||
427 | @ vmov @x[2], @t[0] | ||
428 | veor @x[6], @x[6], @t[2] | ||
429 | @ vmov @x[7], @t[1] | ||
430 | ___ | ||
431 | $code.=<<___ if ($inv); | ||
432 | veor @t[3], @t[3], @x[4] | ||
433 | veor @x[5], @x[5], @t[7] | ||
434 | veor @x[2], @x[3], @t[6] | ||
435 | veor @x[3], @t[0], @t[4] | ||
436 | veor @x[4], @x[6], @t[2] | ||
437 | vmov @x[6], @t[3] | ||
438 | @ vmov @x[7], @t[1] | ||
439 | ___ | ||
440 | } | ||
441 | |||
442 | sub InvMixColumns_orig { | ||
443 | my @x=@_[0..7]; | ||
444 | my @t=@_[8..15]; | ||
445 | |||
446 | $code.=<<___; | ||
447 | @ multiplication by 0x0e | ||
448 | vext.8 @t[7], @x[7], @x[7], #12 | ||
449 | vmov @t[2], @x[2] | ||
450 | veor @x[2], @x[2], @x[5] @ 2 5 | ||
451 | veor @x[7], @x[7], @x[5] @ 7 5 | ||
452 | vext.8 @t[0], @x[0], @x[0], #12 | ||
453 | vmov @t[5], @x[5] | ||
454 | veor @x[5], @x[5], @x[0] @ 5 0 [1] | ||
455 | veor @x[0], @x[0], @x[1] @ 0 1 | ||
456 | vext.8 @t[1], @x[1], @x[1], #12 | ||
457 | veor @x[1], @x[1], @x[2] @ 1 25 | ||
458 | veor @x[0], @x[0], @x[6] @ 01 6 [2] | ||
459 | vext.8 @t[3], @x[3], @x[3], #12 | ||
460 | veor @x[1], @x[1], @x[3] @ 125 3 [4] | ||
461 | veor @x[2], @x[2], @x[0] @ 25 016 [3] | ||
462 | veor @x[3], @x[3], @x[7] @ 3 75 | ||
463 | veor @x[7], @x[7], @x[6] @ 75 6 [0] | ||
464 | vext.8 @t[6], @x[6], @x[6], #12 | ||
465 | vmov @t[4], @x[4] | ||
466 | veor @x[6], @x[6], @x[4] @ 6 4 | ||
467 | veor @x[4], @x[4], @x[3] @ 4 375 [6] | ||
468 | veor @x[3], @x[3], @x[7] @ 375 756=36 | ||
469 | veor @x[6], @x[6], @t[5] @ 64 5 [7] | ||
470 | veor @x[3], @x[3], @t[2] @ 36 2 | ||
471 | vext.8 @t[5], @t[5], @t[5], #12 | ||
472 | veor @x[3], @x[3], @t[4] @ 362 4 [5] | ||
473 | ___ | ||
474 | my @y = @x[7,5,0,2,1,3,4,6]; | ||
475 | $code.=<<___; | ||
476 | @ multiplication by 0x0b | ||
477 | veor @y[1], @y[1], @y[0] | ||
478 | veor @y[0], @y[0], @t[0] | ||
479 | vext.8 @t[2], @t[2], @t[2], #12 | ||
480 | veor @y[1], @y[1], @t[1] | ||
481 | veor @y[0], @y[0], @t[5] | ||
482 | vext.8 @t[4], @t[4], @t[4], #12 | ||
483 | veor @y[1], @y[1], @t[6] | ||
484 | veor @y[0], @y[0], @t[7] | ||
485 | veor @t[7], @t[7], @t[6] @ clobber t[7] | ||
486 | |||
487 | veor @y[3], @y[3], @t[0] | ||
488 | veor @y[1], @y[1], @y[0] | ||
489 | vext.8 @t[0], @t[0], @t[0], #12 | ||
490 | veor @y[2], @y[2], @t[1] | ||
491 | veor @y[4], @y[4], @t[1] | ||
492 | vext.8 @t[1], @t[1], @t[1], #12 | ||
493 | veor @y[2], @y[2], @t[2] | ||
494 | veor @y[3], @y[3], @t[2] | ||
495 | veor @y[5], @y[5], @t[2] | ||
496 | veor @y[2], @y[2], @t[7] | ||
497 | vext.8 @t[2], @t[2], @t[2], #12 | ||
498 | veor @y[3], @y[3], @t[3] | ||
499 | veor @y[6], @y[6], @t[3] | ||
500 | veor @y[4], @y[4], @t[3] | ||
501 | veor @y[7], @y[7], @t[4] | ||
502 | vext.8 @t[3], @t[3], @t[3], #12 | ||
503 | veor @y[5], @y[5], @t[4] | ||
504 | veor @y[7], @y[7], @t[7] | ||
505 | veor @t[7], @t[7], @t[5] @ clobber t[7] even more | ||
506 | veor @y[3], @y[3], @t[5] | ||
507 | veor @y[4], @y[4], @t[4] | ||
508 | |||
509 | veor @y[5], @y[5], @t[7] | ||
510 | vext.8 @t[4], @t[4], @t[4], #12 | ||
511 | veor @y[6], @y[6], @t[7] | ||
512 | veor @y[4], @y[4], @t[7] | ||
513 | |||
514 | veor @t[7], @t[7], @t[5] | ||
515 | vext.8 @t[5], @t[5], @t[5], #12 | ||
516 | |||
517 | @ multiplication by 0x0d | ||
518 | veor @y[4], @y[4], @y[7] | ||
519 | veor @t[7], @t[7], @t[6] @ restore t[7] | ||
520 | veor @y[7], @y[7], @t[4] | ||
521 | vext.8 @t[6], @t[6], @t[6], #12 | ||
522 | veor @y[2], @y[2], @t[0] | ||
523 | veor @y[7], @y[7], @t[5] | ||
524 | vext.8 @t[7], @t[7], @t[7], #12 | ||
525 | veor @y[2], @y[2], @t[2] | ||
526 | |||
527 | veor @y[3], @y[3], @y[1] | ||
528 | veor @y[1], @y[1], @t[1] | ||
529 | veor @y[0], @y[0], @t[0] | ||
530 | veor @y[3], @y[3], @t[0] | ||
531 | veor @y[1], @y[1], @t[5] | ||
532 | veor @y[0], @y[0], @t[5] | ||
533 | vext.8 @t[0], @t[0], @t[0], #12 | ||
534 | veor @y[1], @y[1], @t[7] | ||
535 | veor @y[0], @y[0], @t[6] | ||
536 | veor @y[3], @y[3], @y[1] | ||
537 | veor @y[4], @y[4], @t[1] | ||
538 | vext.8 @t[1], @t[1], @t[1], #12 | ||
539 | |||
540 | veor @y[7], @y[7], @t[7] | ||
541 | veor @y[4], @y[4], @t[2] | ||
542 | veor @y[5], @y[5], @t[2] | ||
543 | veor @y[2], @y[2], @t[6] | ||
544 | veor @t[6], @t[6], @t[3] @ clobber t[6] | ||
545 | vext.8 @t[2], @t[2], @t[2], #12 | ||
546 | veor @y[4], @y[4], @y[7] | ||
547 | veor @y[3], @y[3], @t[6] | ||
548 | |||
549 | veor @y[6], @y[6], @t[6] | ||
550 | veor @y[5], @y[5], @t[5] | ||
551 | vext.8 @t[5], @t[5], @t[5], #12 | ||
552 | veor @y[6], @y[6], @t[4] | ||
553 | vext.8 @t[4], @t[4], @t[4], #12 | ||
554 | veor @y[5], @y[5], @t[6] | ||
555 | veor @y[6], @y[6], @t[7] | ||
556 | vext.8 @t[7], @t[7], @t[7], #12 | ||
557 | veor @t[6], @t[6], @t[3] @ restore t[6] | ||
558 | vext.8 @t[3], @t[3], @t[3], #12 | ||
559 | |||
560 | @ multiplication by 0x09 | ||
561 | veor @y[4], @y[4], @y[1] | ||
562 | veor @t[1], @t[1], @y[1] @ t[1]=y[1] | ||
563 | veor @t[0], @t[0], @t[5] @ clobber t[0] | ||
564 | vext.8 @t[6], @t[6], @t[6], #12 | ||
565 | veor @t[1], @t[1], @t[5] | ||
566 | veor @y[3], @y[3], @t[0] | ||
567 | veor @t[0], @t[0], @y[0] @ t[0]=y[0] | ||
568 | veor @t[1], @t[1], @t[6] | ||
569 | veor @t[6], @t[6], @t[7] @ clobber t[6] | ||
570 | veor @y[4], @y[4], @t[1] | ||
571 | veor @y[7], @y[7], @t[4] | ||
572 | veor @y[6], @y[6], @t[3] | ||
573 | veor @y[5], @y[5], @t[2] | ||
574 | veor @t[4], @t[4], @y[4] @ t[4]=y[4] | ||
575 | veor @t[3], @t[3], @y[3] @ t[3]=y[3] | ||
576 | veor @t[5], @t[5], @y[5] @ t[5]=y[5] | ||
577 | veor @t[2], @t[2], @y[2] @ t[2]=y[2] | ||
578 | veor @t[3], @t[3], @t[7] | ||
579 | veor @XMM[5], @t[5], @t[6] | ||
580 | veor @XMM[6], @t[6], @y[6] @ t[6]=y[6] | ||
581 | veor @XMM[2], @t[2], @t[6] | ||
582 | veor @XMM[7], @t[7], @y[7] @ t[7]=y[7] | ||
583 | |||
584 | vmov @XMM[0], @t[0] | ||
585 | vmov @XMM[1], @t[1] | ||
586 | @ vmov @XMM[2], @t[2] | ||
587 | vmov @XMM[3], @t[3] | ||
588 | vmov @XMM[4], @t[4] | ||
589 | @ vmov @XMM[5], @t[5] | ||
590 | @ vmov @XMM[6], @t[6] | ||
591 | @ vmov @XMM[7], @t[7] | ||
592 | ___ | ||
593 | } | ||
594 | |||
595 | sub InvMixColumns { | ||
596 | my @x=@_[0..7]; | ||
597 | my @t=@_[8..15]; | ||
598 | |||
599 | # Thanks to Jussi Kivilinna for providing pointer to | ||
600 | # | ||
601 | # | 0e 0b 0d 09 | | 02 03 01 01 | | 05 00 04 00 | | ||
602 | # | 09 0e 0b 0d | = | 01 02 03 01 | x | 00 05 00 04 | | ||
603 | # | 0d 09 0e 0b | | 01 01 02 03 | | 04 00 05 00 | | ||
604 | # | 0b 0d 09 0e | | 03 01 01 02 | | 00 04 00 05 | | ||
605 | |||
606 | $code.=<<___; | ||
607 | @ multiplication by 0x05-0x00-0x04-0x00 | ||
608 | vext.8 @t[0], @x[0], @x[0], #8 | ||
609 | vext.8 @t[6], @x[6], @x[6], #8 | ||
610 | vext.8 @t[7], @x[7], @x[7], #8 | ||
611 | veor @t[0], @t[0], @x[0] | ||
612 | vext.8 @t[1], @x[1], @x[1], #8 | ||
613 | veor @t[6], @t[6], @x[6] | ||
614 | vext.8 @t[2], @x[2], @x[2], #8 | ||
615 | veor @t[7], @t[7], @x[7] | ||
616 | vext.8 @t[3], @x[3], @x[3], #8 | ||
617 | veor @t[1], @t[1], @x[1] | ||
618 | vext.8 @t[4], @x[4], @x[4], #8 | ||
619 | veor @t[2], @t[2], @x[2] | ||
620 | vext.8 @t[5], @x[5], @x[5], #8 | ||
621 | veor @t[3], @t[3], @x[3] | ||
622 | veor @t[4], @t[4], @x[4] | ||
623 | veor @t[5], @t[5], @x[5] | ||
624 | |||
625 | veor @x[0], @x[0], @t[6] | ||
626 | veor @x[1], @x[1], @t[6] | ||
627 | veor @x[2], @x[2], @t[0] | ||
628 | veor @x[4], @x[4], @t[2] | ||
629 | veor @x[3], @x[3], @t[1] | ||
630 | veor @x[1], @x[1], @t[7] | ||
631 | veor @x[2], @x[2], @t[7] | ||
632 | veor @x[4], @x[4], @t[6] | ||
633 | veor @x[5], @x[5], @t[3] | ||
634 | veor @x[3], @x[3], @t[6] | ||
635 | veor @x[6], @x[6], @t[4] | ||
636 | veor @x[4], @x[4], @t[7] | ||
637 | veor @x[5], @x[5], @t[7] | ||
638 | veor @x[7], @x[7], @t[5] | ||
639 | ___ | ||
640 | &MixColumns (@x,@t,1); # flipped 2<->3 and 4<->6 | ||
641 | } | ||
642 | |||
643 | sub swapmove { | ||
644 | my ($a,$b,$n,$mask,$t)=@_; | ||
645 | $code.=<<___; | ||
646 | vshr.u64 $t, $b, #$n | ||
647 | veor $t, $t, $a | ||
648 | vand $t, $t, $mask | ||
649 | veor $a, $a, $t | ||
650 | vshl.u64 $t, $t, #$n | ||
651 | veor $b, $b, $t | ||
652 | ___ | ||
653 | } | ||
654 | sub swapmove2x { | ||
655 | my ($a0,$b0,$a1,$b1,$n,$mask,$t0,$t1)=@_; | ||
656 | $code.=<<___; | ||
657 | vshr.u64 $t0, $b0, #$n | ||
658 | vshr.u64 $t1, $b1, #$n | ||
659 | veor $t0, $t0, $a0 | ||
660 | veor $t1, $t1, $a1 | ||
661 | vand $t0, $t0, $mask | ||
662 | vand $t1, $t1, $mask | ||
663 | veor $a0, $a0, $t0 | ||
664 | vshl.u64 $t0, $t0, #$n | ||
665 | veor $a1, $a1, $t1 | ||
666 | vshl.u64 $t1, $t1, #$n | ||
667 | veor $b0, $b0, $t0 | ||
668 | veor $b1, $b1, $t1 | ||
669 | ___ | ||
670 | } | ||
671 | |||
672 | sub bitslice { | ||
673 | my @x=reverse(@_[0..7]); | ||
674 | my ($t0,$t1,$t2,$t3)=@_[8..11]; | ||
675 | $code.=<<___; | ||
676 | vmov.i8 $t0,#0x55 @ compose .LBS0 | ||
677 | vmov.i8 $t1,#0x33 @ compose .LBS1 | ||
678 | ___ | ||
679 | &swapmove2x(@x[0,1,2,3],1,$t0,$t2,$t3); | ||
680 | &swapmove2x(@x[4,5,6,7],1,$t0,$t2,$t3); | ||
681 | $code.=<<___; | ||
682 | vmov.i8 $t0,#0x0f @ compose .LBS2 | ||
683 | ___ | ||
684 | &swapmove2x(@x[0,2,1,3],2,$t1,$t2,$t3); | ||
685 | &swapmove2x(@x[4,6,5,7],2,$t1,$t2,$t3); | ||
686 | |||
687 | &swapmove2x(@x[0,4,1,5],4,$t0,$t2,$t3); | ||
688 | &swapmove2x(@x[2,6,3,7],4,$t0,$t2,$t3); | ||
689 | } | ||
690 | |||
691 | $code.=<<___; | ||
692 | #ifndef __KERNEL__ | ||
693 | # include "arm_arch.h" | ||
694 | |||
695 | # define VFP_ABI_PUSH vstmdb sp!,{d8-d15} | ||
696 | # define VFP_ABI_POP vldmia sp!,{d8-d15} | ||
697 | # define VFP_ABI_FRAME 0x40 | ||
698 | #else | ||
699 | # define VFP_ABI_PUSH | ||
700 | # define VFP_ABI_POP | ||
701 | # define VFP_ABI_FRAME 0 | ||
702 | # define BSAES_ASM_EXTENDED_KEY | ||
703 | # define XTS_CHAIN_TWEAK | ||
704 | # define __ARM_ARCH__ __LINUX_ARM_ARCH__ | ||
705 | #endif | ||
706 | |||
707 | #ifdef __thumb__ | ||
708 | # define adrl adr | ||
709 | #endif | ||
710 | |||
711 | #if __ARM_ARCH__>=7 | ||
712 | .text | ||
713 | .syntax unified @ ARMv7-capable assembler is expected to handle this | ||
714 | #ifdef __thumb2__ | ||
715 | .thumb | ||
716 | #else | ||
717 | .code 32 | ||
718 | #endif | ||
719 | |||
720 | .fpu neon | ||
721 | |||
722 | .type _bsaes_decrypt8,%function | ||
723 | .align 4 | ||
724 | _bsaes_decrypt8: | ||
725 | adr $const,_bsaes_decrypt8 | ||
726 | vldmia $key!, {@XMM[9]} @ round 0 key | ||
727 | add $const,$const,#.LM0ISR-_bsaes_decrypt8 | ||
728 | |||
729 | vldmia $const!, {@XMM[8]} @ .LM0ISR | ||
730 | veor @XMM[10], @XMM[0], @XMM[9] @ xor with round0 key | ||
731 | veor @XMM[11], @XMM[1], @XMM[9] | ||
732 | vtbl.8 `&Dlo(@XMM[0])`, {@XMM[10]}, `&Dlo(@XMM[8])` | ||
733 | vtbl.8 `&Dhi(@XMM[0])`, {@XMM[10]}, `&Dhi(@XMM[8])` | ||
734 | veor @XMM[12], @XMM[2], @XMM[9] | ||
735 | vtbl.8 `&Dlo(@XMM[1])`, {@XMM[11]}, `&Dlo(@XMM[8])` | ||
736 | vtbl.8 `&Dhi(@XMM[1])`, {@XMM[11]}, `&Dhi(@XMM[8])` | ||
737 | veor @XMM[13], @XMM[3], @XMM[9] | ||
738 | vtbl.8 `&Dlo(@XMM[2])`, {@XMM[12]}, `&Dlo(@XMM[8])` | ||
739 | vtbl.8 `&Dhi(@XMM[2])`, {@XMM[12]}, `&Dhi(@XMM[8])` | ||
740 | veor @XMM[14], @XMM[4], @XMM[9] | ||
741 | vtbl.8 `&Dlo(@XMM[3])`, {@XMM[13]}, `&Dlo(@XMM[8])` | ||
742 | vtbl.8 `&Dhi(@XMM[3])`, {@XMM[13]}, `&Dhi(@XMM[8])` | ||
743 | veor @XMM[15], @XMM[5], @XMM[9] | ||
744 | vtbl.8 `&Dlo(@XMM[4])`, {@XMM[14]}, `&Dlo(@XMM[8])` | ||
745 | vtbl.8 `&Dhi(@XMM[4])`, {@XMM[14]}, `&Dhi(@XMM[8])` | ||
746 | veor @XMM[10], @XMM[6], @XMM[9] | ||
747 | vtbl.8 `&Dlo(@XMM[5])`, {@XMM[15]}, `&Dlo(@XMM[8])` | ||
748 | vtbl.8 `&Dhi(@XMM[5])`, {@XMM[15]}, `&Dhi(@XMM[8])` | ||
749 | veor @XMM[11], @XMM[7], @XMM[9] | ||
750 | vtbl.8 `&Dlo(@XMM[6])`, {@XMM[10]}, `&Dlo(@XMM[8])` | ||
751 | vtbl.8 `&Dhi(@XMM[6])`, {@XMM[10]}, `&Dhi(@XMM[8])` | ||
752 | vtbl.8 `&Dlo(@XMM[7])`, {@XMM[11]}, `&Dlo(@XMM[8])` | ||
753 | vtbl.8 `&Dhi(@XMM[7])`, {@XMM[11]}, `&Dhi(@XMM[8])` | ||
754 | ___ | ||
755 | &bitslice (@XMM[0..7, 8..11]); | ||
756 | $code.=<<___; | ||
757 | sub $rounds,$rounds,#1 | ||
758 | b .Ldec_sbox | ||
759 | .align 4 | ||
760 | .Ldec_loop: | ||
761 | ___ | ||
762 | &ShiftRows (@XMM[0..7, 8..12]); | ||
763 | $code.=".Ldec_sbox:\n"; | ||
764 | &InvSbox (@XMM[0..7, 8..15]); | ||
765 | $code.=<<___; | ||
766 | subs $rounds,$rounds,#1 | ||
767 | bcc .Ldec_done | ||
768 | ___ | ||
769 | &InvMixColumns (@XMM[0,1,6,4,2,7,3,5, 8..15]); | ||
770 | $code.=<<___; | ||
771 | vldmia $const, {@XMM[12]} @ .LISR | ||
772 | ite eq @ Thumb2 thing, sanity check in ARM | ||
773 | addeq $const,$const,#0x10 | ||
774 | bne .Ldec_loop | ||
775 | vldmia $const, {@XMM[12]} @ .LISRM0 | ||
776 | b .Ldec_loop | ||
777 | .align 4 | ||
778 | .Ldec_done: | ||
779 | ___ | ||
780 | &bitslice (@XMM[0,1,6,4,2,7,3,5, 8..11]); | ||
781 | $code.=<<___; | ||
782 | vldmia $key, {@XMM[8]} @ last round key | ||
783 | veor @XMM[6], @XMM[6], @XMM[8] | ||
784 | veor @XMM[4], @XMM[4], @XMM[8] | ||
785 | veor @XMM[2], @XMM[2], @XMM[8] | ||
786 | veor @XMM[7], @XMM[7], @XMM[8] | ||
787 | veor @XMM[3], @XMM[3], @XMM[8] | ||
788 | veor @XMM[5], @XMM[5], @XMM[8] | ||
789 | veor @XMM[0], @XMM[0], @XMM[8] | ||
790 | veor @XMM[1], @XMM[1], @XMM[8] | ||
791 | bx lr | ||
792 | .size _bsaes_decrypt8,.-_bsaes_decrypt8 | ||
793 | |||
794 | .type _bsaes_const,%object | ||
795 | .align 6 | ||
796 | _bsaes_const: | ||
797 | .LM0ISR: @ InvShiftRows constants | ||
798 | .quad 0x0a0e0206070b0f03, 0x0004080c0d010509 | ||
799 | .LISR: | ||
800 | .quad 0x0504070602010003, 0x0f0e0d0c080b0a09 | ||
801 | .LISRM0: | ||
802 | .quad 0x01040b0e0205080f, 0x0306090c00070a0d | ||
803 | .LM0SR: @ ShiftRows constants | ||
804 | .quad 0x0a0e02060f03070b, 0x0004080c05090d01 | ||
805 | .LSR: | ||
806 | .quad 0x0504070600030201, 0x0f0e0d0c0a09080b | ||
807 | .LSRM0: | ||
808 | .quad 0x0304090e00050a0f, 0x01060b0c0207080d | ||
809 | .LM0: | ||
810 | .quad 0x02060a0e03070b0f, 0x0004080c0105090d | ||
811 | .LREVM0SR: | ||
812 | .quad 0x090d01050c000408, 0x03070b0f060a0e02 | ||
813 | .asciz "Bit-sliced AES for NEON, CRYPTOGAMS by <appro\@openssl.org>" | ||
814 | .align 6 | ||
815 | .size _bsaes_const,.-_bsaes_const | ||
816 | |||
817 | .type _bsaes_encrypt8,%function | ||
818 | .align 4 | ||
819 | _bsaes_encrypt8: | ||
820 | adr $const,_bsaes_encrypt8 | ||
821 | vldmia $key!, {@XMM[9]} @ round 0 key | ||
822 | sub $const,$const,#_bsaes_encrypt8-.LM0SR | ||
823 | |||
824 | vldmia $const!, {@XMM[8]} @ .LM0SR | ||
825 | _bsaes_encrypt8_alt: | ||
826 | veor @XMM[10], @XMM[0], @XMM[9] @ xor with round0 key | ||
827 | veor @XMM[11], @XMM[1], @XMM[9] | ||
828 | vtbl.8 `&Dlo(@XMM[0])`, {@XMM[10]}, `&Dlo(@XMM[8])` | ||
829 | vtbl.8 `&Dhi(@XMM[0])`, {@XMM[10]}, `&Dhi(@XMM[8])` | ||
830 | veor @XMM[12], @XMM[2], @XMM[9] | ||
831 | vtbl.8 `&Dlo(@XMM[1])`, {@XMM[11]}, `&Dlo(@XMM[8])` | ||
832 | vtbl.8 `&Dhi(@XMM[1])`, {@XMM[11]}, `&Dhi(@XMM[8])` | ||
833 | veor @XMM[13], @XMM[3], @XMM[9] | ||
834 | vtbl.8 `&Dlo(@XMM[2])`, {@XMM[12]}, `&Dlo(@XMM[8])` | ||
835 | vtbl.8 `&Dhi(@XMM[2])`, {@XMM[12]}, `&Dhi(@XMM[8])` | ||
836 | veor @XMM[14], @XMM[4], @XMM[9] | ||
837 | vtbl.8 `&Dlo(@XMM[3])`, {@XMM[13]}, `&Dlo(@XMM[8])` | ||
838 | vtbl.8 `&Dhi(@XMM[3])`, {@XMM[13]}, `&Dhi(@XMM[8])` | ||
839 | veor @XMM[15], @XMM[5], @XMM[9] | ||
840 | vtbl.8 `&Dlo(@XMM[4])`, {@XMM[14]}, `&Dlo(@XMM[8])` | ||
841 | vtbl.8 `&Dhi(@XMM[4])`, {@XMM[14]}, `&Dhi(@XMM[8])` | ||
842 | veor @XMM[10], @XMM[6], @XMM[9] | ||
843 | vtbl.8 `&Dlo(@XMM[5])`, {@XMM[15]}, `&Dlo(@XMM[8])` | ||
844 | vtbl.8 `&Dhi(@XMM[5])`, {@XMM[15]}, `&Dhi(@XMM[8])` | ||
845 | veor @XMM[11], @XMM[7], @XMM[9] | ||
846 | vtbl.8 `&Dlo(@XMM[6])`, {@XMM[10]}, `&Dlo(@XMM[8])` | ||
847 | vtbl.8 `&Dhi(@XMM[6])`, {@XMM[10]}, `&Dhi(@XMM[8])` | ||
848 | vtbl.8 `&Dlo(@XMM[7])`, {@XMM[11]}, `&Dlo(@XMM[8])` | ||
849 | vtbl.8 `&Dhi(@XMM[7])`, {@XMM[11]}, `&Dhi(@XMM[8])` | ||
850 | _bsaes_encrypt8_bitslice: | ||
851 | ___ | ||
852 | &bitslice (@XMM[0..7, 8..11]); | ||
853 | $code.=<<___; | ||
854 | sub $rounds,$rounds,#1 | ||
855 | b .Lenc_sbox | ||
856 | .align 4 | ||
857 | .Lenc_loop: | ||
858 | ___ | ||
859 | &ShiftRows (@XMM[0..7, 8..12]); | ||
860 | $code.=".Lenc_sbox:\n"; | ||
861 | &Sbox (@XMM[0..7, 8..15]); | ||
862 | $code.=<<___; | ||
863 | subs $rounds,$rounds,#1 | ||
864 | bcc .Lenc_done | ||
865 | ___ | ||
866 | &MixColumns (@XMM[0,1,4,6,3,7,2,5, 8..15]); | ||
867 | $code.=<<___; | ||
868 | vldmia $const, {@XMM[12]} @ .LSR | ||
869 | ite eq @ Thumb2 thing, samity check in ARM | ||
870 | addeq $const,$const,#0x10 | ||
871 | bne .Lenc_loop | ||
872 | vldmia $const, {@XMM[12]} @ .LSRM0 | ||
873 | b .Lenc_loop | ||
874 | .align 4 | ||
875 | .Lenc_done: | ||
876 | ___ | ||
877 | # output in lsb > [t0, t1, t4, t6, t3, t7, t2, t5] < msb | ||
878 | &bitslice (@XMM[0,1,4,6,3,7,2,5, 8..11]); | ||
879 | $code.=<<___; | ||
880 | vldmia $key, {@XMM[8]} @ last round key | ||
881 | veor @XMM[4], @XMM[4], @XMM[8] | ||
882 | veor @XMM[6], @XMM[6], @XMM[8] | ||
883 | veor @XMM[3], @XMM[3], @XMM[8] | ||
884 | veor @XMM[7], @XMM[7], @XMM[8] | ||
885 | veor @XMM[2], @XMM[2], @XMM[8] | ||
886 | veor @XMM[5], @XMM[5], @XMM[8] | ||
887 | veor @XMM[0], @XMM[0], @XMM[8] | ||
888 | veor @XMM[1], @XMM[1], @XMM[8] | ||
889 | bx lr | ||
890 | .size _bsaes_encrypt8,.-_bsaes_encrypt8 | ||
891 | ___ | ||
892 | } | ||
893 | { | ||
894 | my ($out,$inp,$rounds,$const)=("r12","r4","r5","r6"); | ||
895 | |||
896 | sub bitslice_key { | ||
897 | my @x=reverse(@_[0..7]); | ||
898 | my ($bs0,$bs1,$bs2,$t2,$t3)=@_[8..12]; | ||
899 | |||
900 | &swapmove (@x[0,1],1,$bs0,$t2,$t3); | ||
901 | $code.=<<___; | ||
902 | @ &swapmove(@x[2,3],1,$t0,$t2,$t3); | ||
903 | vmov @x[2], @x[0] | ||
904 | vmov @x[3], @x[1] | ||
905 | ___ | ||
906 | #&swapmove2x(@x[4,5,6,7],1,$t0,$t2,$t3); | ||
907 | |||
908 | &swapmove2x (@x[0,2,1,3],2,$bs1,$t2,$t3); | ||
909 | $code.=<<___; | ||
910 | @ &swapmove2x(@x[4,6,5,7],2,$t1,$t2,$t3); | ||
911 | vmov @x[4], @x[0] | ||
912 | vmov @x[6], @x[2] | ||
913 | vmov @x[5], @x[1] | ||
914 | vmov @x[7], @x[3] | ||
915 | ___ | ||
916 | &swapmove2x (@x[0,4,1,5],4,$bs2,$t2,$t3); | ||
917 | &swapmove2x (@x[2,6,3,7],4,$bs2,$t2,$t3); | ||
918 | } | ||
919 | |||
920 | $code.=<<___; | ||
921 | .type _bsaes_key_convert,%function | ||
922 | .align 4 | ||
923 | _bsaes_key_convert: | ||
924 | adr $const,_bsaes_key_convert | ||
925 | vld1.8 {@XMM[7]}, [$inp]! @ load round 0 key | ||
926 | sub $const,$const,#_bsaes_key_convert-.LM0 | ||
927 | vld1.8 {@XMM[15]}, [$inp]! @ load round 1 key | ||
928 | |||
929 | vmov.i8 @XMM[8], #0x01 @ bit masks | ||
930 | vmov.i8 @XMM[9], #0x02 | ||
931 | vmov.i8 @XMM[10], #0x04 | ||
932 | vmov.i8 @XMM[11], #0x08 | ||
933 | vmov.i8 @XMM[12], #0x10 | ||
934 | vmov.i8 @XMM[13], #0x20 | ||
935 | vldmia $const, {@XMM[14]} @ .LM0 | ||
936 | |||
937 | #ifdef __ARMEL__ | ||
938 | vrev32.8 @XMM[7], @XMM[7] | ||
939 | vrev32.8 @XMM[15], @XMM[15] | ||
940 | #endif | ||
941 | sub $rounds,$rounds,#1 | ||
942 | vstmia $out!, {@XMM[7]} @ save round 0 key | ||
943 | b .Lkey_loop | ||
944 | |||
945 | .align 4 | ||
946 | .Lkey_loop: | ||
947 | vtbl.8 `&Dlo(@XMM[7])`,{@XMM[15]},`&Dlo(@XMM[14])` | ||
948 | vtbl.8 `&Dhi(@XMM[7])`,{@XMM[15]},`&Dhi(@XMM[14])` | ||
949 | vmov.i8 @XMM[6], #0x40 | ||
950 | vmov.i8 @XMM[15], #0x80 | ||
951 | |||
952 | vtst.8 @XMM[0], @XMM[7], @XMM[8] | ||
953 | vtst.8 @XMM[1], @XMM[7], @XMM[9] | ||
954 | vtst.8 @XMM[2], @XMM[7], @XMM[10] | ||
955 | vtst.8 @XMM[3], @XMM[7], @XMM[11] | ||
956 | vtst.8 @XMM[4], @XMM[7], @XMM[12] | ||
957 | vtst.8 @XMM[5], @XMM[7], @XMM[13] | ||
958 | vtst.8 @XMM[6], @XMM[7], @XMM[6] | ||
959 | vtst.8 @XMM[7], @XMM[7], @XMM[15] | ||
960 | vld1.8 {@XMM[15]}, [$inp]! @ load next round key | ||
961 | vmvn @XMM[0], @XMM[0] @ "pnot" | ||
962 | vmvn @XMM[1], @XMM[1] | ||
963 | vmvn @XMM[5], @XMM[5] | ||
964 | vmvn @XMM[6], @XMM[6] | ||
965 | #ifdef __ARMEL__ | ||
966 | vrev32.8 @XMM[15], @XMM[15] | ||
967 | #endif | ||
968 | subs $rounds,$rounds,#1 | ||
969 | vstmia $out!,{@XMM[0]-@XMM[7]} @ write bit-sliced round key | ||
970 | bne .Lkey_loop | ||
971 | |||
972 | vmov.i8 @XMM[7],#0x63 @ compose .L63 | ||
973 | @ don't save last round key | ||
974 | bx lr | ||
975 | .size _bsaes_key_convert,.-_bsaes_key_convert | ||
976 | ___ | ||
977 | } | ||
978 | |||
979 | if (0) { # following four functions are unsupported interface | ||
980 | # used for benchmarking... | ||
981 | $code.=<<___; | ||
982 | .globl bsaes_enc_key_convert | ||
983 | .type bsaes_enc_key_convert,%function | ||
984 | .align 4 | ||
985 | bsaes_enc_key_convert: | ||
986 | stmdb sp!,{r4-r6,lr} | ||
987 | vstmdb sp!,{d8-d15} @ ABI specification says so | ||
988 | |||
989 | ldr r5,[$inp,#240] @ pass rounds | ||
990 | mov r4,$inp @ pass key | ||
991 | mov r12,$out @ pass key schedule | ||
992 | bl _bsaes_key_convert | ||
993 | veor @XMM[7],@XMM[7],@XMM[15] @ fix up last round key | ||
994 | vstmia r12, {@XMM[7]} @ save last round key | ||
995 | |||
996 | vldmia sp!,{d8-d15} | ||
997 | ldmia sp!,{r4-r6,pc} | ||
998 | .size bsaes_enc_key_convert,.-bsaes_enc_key_convert | ||
999 | |||
1000 | .globl bsaes_encrypt_128 | ||
1001 | .type bsaes_encrypt_128,%function | ||
1002 | .align 4 | ||
1003 | bsaes_encrypt_128: | ||
1004 | stmdb sp!,{r4-r6,lr} | ||
1005 | vstmdb sp!,{d8-d15} @ ABI specification says so | ||
1006 | .Lenc128_loop: | ||
1007 | vld1.8 {@XMM[0]-@XMM[1]}, [$inp]! @ load input | ||
1008 | vld1.8 {@XMM[2]-@XMM[3]}, [$inp]! | ||
1009 | mov r4,$key @ pass the key | ||
1010 | vld1.8 {@XMM[4]-@XMM[5]}, [$inp]! | ||
1011 | mov r5,#10 @ pass rounds | ||
1012 | vld1.8 {@XMM[6]-@XMM[7]}, [$inp]! | ||
1013 | |||
1014 | bl _bsaes_encrypt8 | ||
1015 | |||
1016 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! @ write output | ||
1017 | vst1.8 {@XMM[4]}, [$out]! | ||
1018 | vst1.8 {@XMM[6]}, [$out]! | ||
1019 | vst1.8 {@XMM[3]}, [$out]! | ||
1020 | vst1.8 {@XMM[7]}, [$out]! | ||
1021 | vst1.8 {@XMM[2]}, [$out]! | ||
1022 | subs $len,$len,#0x80 | ||
1023 | vst1.8 {@XMM[5]}, [$out]! | ||
1024 | bhi .Lenc128_loop | ||
1025 | |||
1026 | vldmia sp!,{d8-d15} | ||
1027 | ldmia sp!,{r4-r6,pc} | ||
1028 | .size bsaes_encrypt_128,.-bsaes_encrypt_128 | ||
1029 | |||
1030 | .globl bsaes_dec_key_convert | ||
1031 | .type bsaes_dec_key_convert,%function | ||
1032 | .align 4 | ||
1033 | bsaes_dec_key_convert: | ||
1034 | stmdb sp!,{r4-r6,lr} | ||
1035 | vstmdb sp!,{d8-d15} @ ABI specification says so | ||
1036 | |||
1037 | ldr r5,[$inp,#240] @ pass rounds | ||
1038 | mov r4,$inp @ pass key | ||
1039 | mov r12,$out @ pass key schedule | ||
1040 | bl _bsaes_key_convert | ||
1041 | vldmia $out, {@XMM[6]} | ||
1042 | vstmia r12, {@XMM[15]} @ save last round key | ||
1043 | veor @XMM[7], @XMM[7], @XMM[6] @ fix up round 0 key | ||
1044 | vstmia $out, {@XMM[7]} | ||
1045 | |||
1046 | vldmia sp!,{d8-d15} | ||
1047 | ldmia sp!,{r4-r6,pc} | ||
1048 | .size bsaes_dec_key_convert,.-bsaes_dec_key_convert | ||
1049 | |||
1050 | .globl bsaes_decrypt_128 | ||
1051 | .type bsaes_decrypt_128,%function | ||
1052 | .align 4 | ||
1053 | bsaes_decrypt_128: | ||
1054 | stmdb sp!,{r4-r6,lr} | ||
1055 | vstmdb sp!,{d8-d15} @ ABI specification says so | ||
1056 | .Ldec128_loop: | ||
1057 | vld1.8 {@XMM[0]-@XMM[1]}, [$inp]! @ load input | ||
1058 | vld1.8 {@XMM[2]-@XMM[3]}, [$inp]! | ||
1059 | mov r4,$key @ pass the key | ||
1060 | vld1.8 {@XMM[4]-@XMM[5]}, [$inp]! | ||
1061 | mov r5,#10 @ pass rounds | ||
1062 | vld1.8 {@XMM[6]-@XMM[7]}, [$inp]! | ||
1063 | |||
1064 | bl _bsaes_decrypt8 | ||
1065 | |||
1066 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! @ write output | ||
1067 | vst1.8 {@XMM[6]}, [$out]! | ||
1068 | vst1.8 {@XMM[4]}, [$out]! | ||
1069 | vst1.8 {@XMM[2]}, [$out]! | ||
1070 | vst1.8 {@XMM[7]}, [$out]! | ||
1071 | vst1.8 {@XMM[3]}, [$out]! | ||
1072 | subs $len,$len,#0x80 | ||
1073 | vst1.8 {@XMM[5]}, [$out]! | ||
1074 | bhi .Ldec128_loop | ||
1075 | |||
1076 | vldmia sp!,{d8-d15} | ||
1077 | ldmia sp!,{r4-r6,pc} | ||
1078 | .size bsaes_decrypt_128,.-bsaes_decrypt_128 | ||
1079 | ___ | ||
1080 | } | ||
1081 | { | ||
1082 | my ($inp,$out,$len,$key, $ivp,$fp,$rounds)=map("r$_",(0..3,8..10)); | ||
1083 | my ($keysched)=("sp"); | ||
1084 | |||
1085 | $code.=<<___; | ||
1086 | .extern AES_cbc_encrypt | ||
1087 | .extern AES_decrypt | ||
1088 | |||
1089 | .global bsaes_cbc_encrypt | ||
1090 | .type bsaes_cbc_encrypt,%function | ||
1091 | .align 5 | ||
1092 | bsaes_cbc_encrypt: | ||
1093 | #ifndef __KERNEL__ | ||
1094 | cmp $len, #128 | ||
1095 | #ifndef __thumb__ | ||
1096 | blo AES_cbc_encrypt | ||
1097 | #else | ||
1098 | bhs 1f | ||
1099 | b AES_cbc_encrypt | ||
1100 | 1: | ||
1101 | #endif | ||
1102 | #endif | ||
1103 | |||
1104 | @ it is up to the caller to make sure we are called with enc == 0 | ||
1105 | |||
1106 | mov ip, sp | ||
1107 | stmdb sp!, {r4-r10, lr} | ||
1108 | VFP_ABI_PUSH | ||
1109 | ldr $ivp, [ip] @ IV is 1st arg on the stack | ||
1110 | mov $len, $len, lsr#4 @ len in 16 byte blocks | ||
1111 | sub sp, #0x10 @ scratch space to carry over the IV | ||
1112 | mov $fp, sp @ save sp | ||
1113 | |||
1114 | ldr $rounds, [$key, #240] @ get # of rounds | ||
1115 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1116 | @ allocate the key schedule on the stack | ||
1117 | sub r12, sp, $rounds, lsl#7 @ 128 bytes per inner round key | ||
1118 | add r12, #`128-32` @ sifze of bit-slices key schedule | ||
1119 | |||
1120 | @ populate the key schedule | ||
1121 | mov r4, $key @ pass key | ||
1122 | mov r5, $rounds @ pass # of rounds | ||
1123 | mov sp, r12 @ sp is $keysched | ||
1124 | bl _bsaes_key_convert | ||
1125 | vldmia $keysched, {@XMM[6]} | ||
1126 | vstmia r12, {@XMM[15]} @ save last round key | ||
1127 | veor @XMM[7], @XMM[7], @XMM[6] @ fix up round 0 key | ||
1128 | vstmia $keysched, {@XMM[7]} | ||
1129 | #else | ||
1130 | ldr r12, [$key, #244] | ||
1131 | eors r12, #1 | ||
1132 | beq 0f | ||
1133 | |||
1134 | @ populate the key schedule | ||
1135 | str r12, [$key, #244] | ||
1136 | mov r4, $key @ pass key | ||
1137 | mov r5, $rounds @ pass # of rounds | ||
1138 | add r12, $key, #248 @ pass key schedule | ||
1139 | bl _bsaes_key_convert | ||
1140 | add r4, $key, #248 | ||
1141 | vldmia r4, {@XMM[6]} | ||
1142 | vstmia r12, {@XMM[15]} @ save last round key | ||
1143 | veor @XMM[7], @XMM[7], @XMM[6] @ fix up round 0 key | ||
1144 | vstmia r4, {@XMM[7]} | ||
1145 | |||
1146 | .align 2 | ||
1147 | 0: | ||
1148 | #endif | ||
1149 | |||
1150 | vld1.8 {@XMM[15]}, [$ivp] @ load IV | ||
1151 | b .Lcbc_dec_loop | ||
1152 | |||
1153 | .align 4 | ||
1154 | .Lcbc_dec_loop: | ||
1155 | subs $len, $len, #0x8 | ||
1156 | bmi .Lcbc_dec_loop_finish | ||
1157 | |||
1158 | vld1.8 {@XMM[0]-@XMM[1]}, [$inp]! @ load input | ||
1159 | vld1.8 {@XMM[2]-@XMM[3]}, [$inp]! | ||
1160 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1161 | mov r4, $keysched @ pass the key | ||
1162 | #else | ||
1163 | add r4, $key, #248 | ||
1164 | #endif | ||
1165 | vld1.8 {@XMM[4]-@XMM[5]}, [$inp]! | ||
1166 | mov r5, $rounds | ||
1167 | vld1.8 {@XMM[6]-@XMM[7]}, [$inp] | ||
1168 | sub $inp, $inp, #0x60 | ||
1169 | vstmia $fp, {@XMM[15]} @ put aside IV | ||
1170 | |||
1171 | bl _bsaes_decrypt8 | ||
1172 | |||
1173 | vldmia $fp, {@XMM[14]} @ reload IV | ||
1174 | vld1.8 {@XMM[8]-@XMM[9]}, [$inp]! @ reload input | ||
1175 | veor @XMM[0], @XMM[0], @XMM[14] @ ^= IV | ||
1176 | vld1.8 {@XMM[10]-@XMM[11]}, [$inp]! | ||
1177 | veor @XMM[1], @XMM[1], @XMM[8] | ||
1178 | veor @XMM[6], @XMM[6], @XMM[9] | ||
1179 | vld1.8 {@XMM[12]-@XMM[13]}, [$inp]! | ||
1180 | veor @XMM[4], @XMM[4], @XMM[10] | ||
1181 | veor @XMM[2], @XMM[2], @XMM[11] | ||
1182 | vld1.8 {@XMM[14]-@XMM[15]}, [$inp]! | ||
1183 | veor @XMM[7], @XMM[7], @XMM[12] | ||
1184 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! @ write output | ||
1185 | veor @XMM[3], @XMM[3], @XMM[13] | ||
1186 | vst1.8 {@XMM[6]}, [$out]! | ||
1187 | veor @XMM[5], @XMM[5], @XMM[14] | ||
1188 | vst1.8 {@XMM[4]}, [$out]! | ||
1189 | vst1.8 {@XMM[2]}, [$out]! | ||
1190 | vst1.8 {@XMM[7]}, [$out]! | ||
1191 | vst1.8 {@XMM[3]}, [$out]! | ||
1192 | vst1.8 {@XMM[5]}, [$out]! | ||
1193 | |||
1194 | b .Lcbc_dec_loop | ||
1195 | |||
1196 | .Lcbc_dec_loop_finish: | ||
1197 | adds $len, $len, #8 | ||
1198 | beq .Lcbc_dec_done | ||
1199 | |||
1200 | vld1.8 {@XMM[0]}, [$inp]! @ load input | ||
1201 | cmp $len, #2 | ||
1202 | blo .Lcbc_dec_one | ||
1203 | vld1.8 {@XMM[1]}, [$inp]! | ||
1204 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1205 | mov r4, $keysched @ pass the key | ||
1206 | #else | ||
1207 | add r4, $key, #248 | ||
1208 | #endif | ||
1209 | mov r5, $rounds | ||
1210 | vstmia $fp, {@XMM[15]} @ put aside IV | ||
1211 | beq .Lcbc_dec_two | ||
1212 | vld1.8 {@XMM[2]}, [$inp]! | ||
1213 | cmp $len, #4 | ||
1214 | blo .Lcbc_dec_three | ||
1215 | vld1.8 {@XMM[3]}, [$inp]! | ||
1216 | beq .Lcbc_dec_four | ||
1217 | vld1.8 {@XMM[4]}, [$inp]! | ||
1218 | cmp $len, #6 | ||
1219 | blo .Lcbc_dec_five | ||
1220 | vld1.8 {@XMM[5]}, [$inp]! | ||
1221 | beq .Lcbc_dec_six | ||
1222 | vld1.8 {@XMM[6]}, [$inp]! | ||
1223 | sub $inp, $inp, #0x70 | ||
1224 | |||
1225 | bl _bsaes_decrypt8 | ||
1226 | |||
1227 | vldmia $fp, {@XMM[14]} @ reload IV | ||
1228 | vld1.8 {@XMM[8]-@XMM[9]}, [$inp]! @ reload input | ||
1229 | veor @XMM[0], @XMM[0], @XMM[14] @ ^= IV | ||
1230 | vld1.8 {@XMM[10]-@XMM[11]}, [$inp]! | ||
1231 | veor @XMM[1], @XMM[1], @XMM[8] | ||
1232 | veor @XMM[6], @XMM[6], @XMM[9] | ||
1233 | vld1.8 {@XMM[12]-@XMM[13]}, [$inp]! | ||
1234 | veor @XMM[4], @XMM[4], @XMM[10] | ||
1235 | veor @XMM[2], @XMM[2], @XMM[11] | ||
1236 | vld1.8 {@XMM[15]}, [$inp]! | ||
1237 | veor @XMM[7], @XMM[7], @XMM[12] | ||
1238 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! @ write output | ||
1239 | veor @XMM[3], @XMM[3], @XMM[13] | ||
1240 | vst1.8 {@XMM[6]}, [$out]! | ||
1241 | vst1.8 {@XMM[4]}, [$out]! | ||
1242 | vst1.8 {@XMM[2]}, [$out]! | ||
1243 | vst1.8 {@XMM[7]}, [$out]! | ||
1244 | vst1.8 {@XMM[3]}, [$out]! | ||
1245 | b .Lcbc_dec_done | ||
1246 | .align 4 | ||
1247 | .Lcbc_dec_six: | ||
1248 | sub $inp, $inp, #0x60 | ||
1249 | bl _bsaes_decrypt8 | ||
1250 | vldmia $fp,{@XMM[14]} @ reload IV | ||
1251 | vld1.8 {@XMM[8]-@XMM[9]}, [$inp]! @ reload input | ||
1252 | veor @XMM[0], @XMM[0], @XMM[14] @ ^= IV | ||
1253 | vld1.8 {@XMM[10]-@XMM[11]}, [$inp]! | ||
1254 | veor @XMM[1], @XMM[1], @XMM[8] | ||
1255 | veor @XMM[6], @XMM[6], @XMM[9] | ||
1256 | vld1.8 {@XMM[12]}, [$inp]! | ||
1257 | veor @XMM[4], @XMM[4], @XMM[10] | ||
1258 | veor @XMM[2], @XMM[2], @XMM[11] | ||
1259 | vld1.8 {@XMM[15]}, [$inp]! | ||
1260 | veor @XMM[7], @XMM[7], @XMM[12] | ||
1261 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! @ write output | ||
1262 | vst1.8 {@XMM[6]}, [$out]! | ||
1263 | vst1.8 {@XMM[4]}, [$out]! | ||
1264 | vst1.8 {@XMM[2]}, [$out]! | ||
1265 | vst1.8 {@XMM[7]}, [$out]! | ||
1266 | b .Lcbc_dec_done | ||
1267 | .align 4 | ||
1268 | .Lcbc_dec_five: | ||
1269 | sub $inp, $inp, #0x50 | ||
1270 | bl _bsaes_decrypt8 | ||
1271 | vldmia $fp, {@XMM[14]} @ reload IV | ||
1272 | vld1.8 {@XMM[8]-@XMM[9]}, [$inp]! @ reload input | ||
1273 | veor @XMM[0], @XMM[0], @XMM[14] @ ^= IV | ||
1274 | vld1.8 {@XMM[10]-@XMM[11]}, [$inp]! | ||
1275 | veor @XMM[1], @XMM[1], @XMM[8] | ||
1276 | veor @XMM[6], @XMM[6], @XMM[9] | ||
1277 | vld1.8 {@XMM[15]}, [$inp]! | ||
1278 | veor @XMM[4], @XMM[4], @XMM[10] | ||
1279 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! @ write output | ||
1280 | veor @XMM[2], @XMM[2], @XMM[11] | ||
1281 | vst1.8 {@XMM[6]}, [$out]! | ||
1282 | vst1.8 {@XMM[4]}, [$out]! | ||
1283 | vst1.8 {@XMM[2]}, [$out]! | ||
1284 | b .Lcbc_dec_done | ||
1285 | .align 4 | ||
1286 | .Lcbc_dec_four: | ||
1287 | sub $inp, $inp, #0x40 | ||
1288 | bl _bsaes_decrypt8 | ||
1289 | vldmia $fp, {@XMM[14]} @ reload IV | ||
1290 | vld1.8 {@XMM[8]-@XMM[9]}, [$inp]! @ reload input | ||
1291 | veor @XMM[0], @XMM[0], @XMM[14] @ ^= IV | ||
1292 | vld1.8 {@XMM[10]}, [$inp]! | ||
1293 | veor @XMM[1], @XMM[1], @XMM[8] | ||
1294 | veor @XMM[6], @XMM[6], @XMM[9] | ||
1295 | vld1.8 {@XMM[15]}, [$inp]! | ||
1296 | veor @XMM[4], @XMM[4], @XMM[10] | ||
1297 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! @ write output | ||
1298 | vst1.8 {@XMM[6]}, [$out]! | ||
1299 | vst1.8 {@XMM[4]}, [$out]! | ||
1300 | b .Lcbc_dec_done | ||
1301 | .align 4 | ||
1302 | .Lcbc_dec_three: | ||
1303 | sub $inp, $inp, #0x30 | ||
1304 | bl _bsaes_decrypt8 | ||
1305 | vldmia $fp, {@XMM[14]} @ reload IV | ||
1306 | vld1.8 {@XMM[8]-@XMM[9]}, [$inp]! @ reload input | ||
1307 | veor @XMM[0], @XMM[0], @XMM[14] @ ^= IV | ||
1308 | vld1.8 {@XMM[15]}, [$inp]! | ||
1309 | veor @XMM[1], @XMM[1], @XMM[8] | ||
1310 | veor @XMM[6], @XMM[6], @XMM[9] | ||
1311 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! @ write output | ||
1312 | vst1.8 {@XMM[6]}, [$out]! | ||
1313 | b .Lcbc_dec_done | ||
1314 | .align 4 | ||
1315 | .Lcbc_dec_two: | ||
1316 | sub $inp, $inp, #0x20 | ||
1317 | bl _bsaes_decrypt8 | ||
1318 | vldmia $fp, {@XMM[14]} @ reload IV | ||
1319 | vld1.8 {@XMM[8]}, [$inp]! @ reload input | ||
1320 | veor @XMM[0], @XMM[0], @XMM[14] @ ^= IV | ||
1321 | vld1.8 {@XMM[15]}, [$inp]! @ reload input | ||
1322 | veor @XMM[1], @XMM[1], @XMM[8] | ||
1323 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! @ write output | ||
1324 | b .Lcbc_dec_done | ||
1325 | .align 4 | ||
1326 | .Lcbc_dec_one: | ||
1327 | sub $inp, $inp, #0x10 | ||
1328 | mov $rounds, $out @ save original out pointer | ||
1329 | mov $out, $fp @ use the iv scratch space as out buffer | ||
1330 | mov r2, $key | ||
1331 | vmov @XMM[4],@XMM[15] @ just in case ensure that IV | ||
1332 | vmov @XMM[5],@XMM[0] @ and input are preserved | ||
1333 | bl AES_decrypt | ||
1334 | vld1.8 {@XMM[0]}, [$fp,:64] @ load result | ||
1335 | veor @XMM[0], @XMM[0], @XMM[4] @ ^= IV | ||
1336 | vmov @XMM[15], @XMM[5] @ @XMM[5] holds input | ||
1337 | vst1.8 {@XMM[0]}, [$rounds] @ write output | ||
1338 | |||
1339 | .Lcbc_dec_done: | ||
1340 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1341 | vmov.i32 q0, #0 | ||
1342 | vmov.i32 q1, #0 | ||
1343 | .Lcbc_dec_bzero: @ wipe key schedule [if any] | ||
1344 | vstmia $keysched!, {q0-q1} | ||
1345 | cmp $keysched, $fp | ||
1346 | bne .Lcbc_dec_bzero | ||
1347 | #endif | ||
1348 | |||
1349 | mov sp, $fp | ||
1350 | add sp, #0x10 @ add sp,$fp,#0x10 is no good for thumb | ||
1351 | vst1.8 {@XMM[15]}, [$ivp] @ return IV | ||
1352 | VFP_ABI_POP | ||
1353 | ldmia sp!, {r4-r10, pc} | ||
1354 | .size bsaes_cbc_encrypt,.-bsaes_cbc_encrypt | ||
1355 | ___ | ||
1356 | } | ||
1357 | { | ||
1358 | my ($inp,$out,$len,$key, $ctr,$fp,$rounds)=(map("r$_",(0..3,8..10))); | ||
1359 | my $const = "r6"; # shared with _bsaes_encrypt8_alt | ||
1360 | my $keysched = "sp"; | ||
1361 | |||
1362 | $code.=<<___; | ||
1363 | .extern AES_encrypt | ||
1364 | .global bsaes_ctr32_encrypt_blocks | ||
1365 | .type bsaes_ctr32_encrypt_blocks,%function | ||
1366 | .align 5 | ||
1367 | bsaes_ctr32_encrypt_blocks: | ||
1368 | cmp $len, #8 @ use plain AES for | ||
1369 | blo .Lctr_enc_short @ small sizes | ||
1370 | |||
1371 | mov ip, sp | ||
1372 | stmdb sp!, {r4-r10, lr} | ||
1373 | VFP_ABI_PUSH | ||
1374 | ldr $ctr, [ip] @ ctr is 1st arg on the stack | ||
1375 | sub sp, sp, #0x10 @ scratch space to carry over the ctr | ||
1376 | mov $fp, sp @ save sp | ||
1377 | |||
1378 | ldr $rounds, [$key, #240] @ get # of rounds | ||
1379 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1380 | @ allocate the key schedule on the stack | ||
1381 | sub r12, sp, $rounds, lsl#7 @ 128 bytes per inner round key | ||
1382 | add r12, #`128-32` @ size of bit-sliced key schedule | ||
1383 | |||
1384 | @ populate the key schedule | ||
1385 | mov r4, $key @ pass key | ||
1386 | mov r5, $rounds @ pass # of rounds | ||
1387 | mov sp, r12 @ sp is $keysched | ||
1388 | bl _bsaes_key_convert | ||
1389 | veor @XMM[7],@XMM[7],@XMM[15] @ fix up last round key | ||
1390 | vstmia r12, {@XMM[7]} @ save last round key | ||
1391 | |||
1392 | vld1.8 {@XMM[0]}, [$ctr] @ load counter | ||
1393 | add $ctr, $const, #.LREVM0SR-.LM0 @ borrow $ctr | ||
1394 | vldmia $keysched, {@XMM[4]} @ load round0 key | ||
1395 | #else | ||
1396 | ldr r12, [$key, #244] | ||
1397 | eors r12, #1 | ||
1398 | beq 0f | ||
1399 | |||
1400 | @ populate the key schedule | ||
1401 | str r12, [$key, #244] | ||
1402 | mov r4, $key @ pass key | ||
1403 | mov r5, $rounds @ pass # of rounds | ||
1404 | add r12, $key, #248 @ pass key schedule | ||
1405 | bl _bsaes_key_convert | ||
1406 | veor @XMM[7],@XMM[7],@XMM[15] @ fix up last round key | ||
1407 | vstmia r12, {@XMM[7]} @ save last round key | ||
1408 | |||
1409 | .align 2 | ||
1410 | 0: add r12, $key, #248 | ||
1411 | vld1.8 {@XMM[0]}, [$ctr] @ load counter | ||
1412 | adrl $ctr, .LREVM0SR @ borrow $ctr | ||
1413 | vldmia r12, {@XMM[4]} @ load round0 key | ||
1414 | sub sp, #0x10 @ place for adjusted round0 key | ||
1415 | #endif | ||
1416 | |||
1417 | vmov.i32 @XMM[8],#1 @ compose 1<<96 | ||
1418 | veor @XMM[9],@XMM[9],@XMM[9] | ||
1419 | vrev32.8 @XMM[0],@XMM[0] | ||
1420 | vext.8 @XMM[8],@XMM[9],@XMM[8],#4 | ||
1421 | vrev32.8 @XMM[4],@XMM[4] | ||
1422 | vadd.u32 @XMM[9],@XMM[8],@XMM[8] @ compose 2<<96 | ||
1423 | vstmia $keysched, {@XMM[4]} @ save adjusted round0 key | ||
1424 | b .Lctr_enc_loop | ||
1425 | |||
1426 | .align 4 | ||
1427 | .Lctr_enc_loop: | ||
1428 | vadd.u32 @XMM[10], @XMM[8], @XMM[9] @ compose 3<<96 | ||
1429 | vadd.u32 @XMM[1], @XMM[0], @XMM[8] @ +1 | ||
1430 | vadd.u32 @XMM[2], @XMM[0], @XMM[9] @ +2 | ||
1431 | vadd.u32 @XMM[3], @XMM[0], @XMM[10] @ +3 | ||
1432 | vadd.u32 @XMM[4], @XMM[1], @XMM[10] | ||
1433 | vadd.u32 @XMM[5], @XMM[2], @XMM[10] | ||
1434 | vadd.u32 @XMM[6], @XMM[3], @XMM[10] | ||
1435 | vadd.u32 @XMM[7], @XMM[4], @XMM[10] | ||
1436 | vadd.u32 @XMM[10], @XMM[5], @XMM[10] @ next counter | ||
1437 | |||
1438 | @ Borrow prologue from _bsaes_encrypt8 to use the opportunity | ||
1439 | @ to flip byte order in 32-bit counter | ||
1440 | |||
1441 | vldmia $keysched, {@XMM[9]} @ load round0 key | ||
1442 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1443 | add r4, $keysched, #0x10 @ pass next round key | ||
1444 | #else | ||
1445 | add r4, $key, #`248+16` | ||
1446 | #endif | ||
1447 | vldmia $ctr, {@XMM[8]} @ .LREVM0SR | ||
1448 | mov r5, $rounds @ pass rounds | ||
1449 | vstmia $fp, {@XMM[10]} @ save next counter | ||
1450 | sub $const, $ctr, #.LREVM0SR-.LSR @ pass constants | ||
1451 | |||
1452 | bl _bsaes_encrypt8_alt | ||
1453 | |||
1454 | subs $len, $len, #8 | ||
1455 | blo .Lctr_enc_loop_done | ||
1456 | |||
1457 | vld1.8 {@XMM[8]-@XMM[9]}, [$inp]! @ load input | ||
1458 | vld1.8 {@XMM[10]-@XMM[11]}, [$inp]! | ||
1459 | veor @XMM[0], @XMM[8] | ||
1460 | veor @XMM[1], @XMM[9] | ||
1461 | vld1.8 {@XMM[12]-@XMM[13]}, [$inp]! | ||
1462 | veor @XMM[4], @XMM[10] | ||
1463 | veor @XMM[6], @XMM[11] | ||
1464 | vld1.8 {@XMM[14]-@XMM[15]}, [$inp]! | ||
1465 | veor @XMM[3], @XMM[12] | ||
1466 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! @ write output | ||
1467 | veor @XMM[7], @XMM[13] | ||
1468 | veor @XMM[2], @XMM[14] | ||
1469 | vst1.8 {@XMM[4]}, [$out]! | ||
1470 | veor @XMM[5], @XMM[15] | ||
1471 | vst1.8 {@XMM[6]}, [$out]! | ||
1472 | vmov.i32 @XMM[8], #1 @ compose 1<<96 | ||
1473 | vst1.8 {@XMM[3]}, [$out]! | ||
1474 | veor @XMM[9], @XMM[9], @XMM[9] | ||
1475 | vst1.8 {@XMM[7]}, [$out]! | ||
1476 | vext.8 @XMM[8], @XMM[9], @XMM[8], #4 | ||
1477 | vst1.8 {@XMM[2]}, [$out]! | ||
1478 | vadd.u32 @XMM[9],@XMM[8],@XMM[8] @ compose 2<<96 | ||
1479 | vst1.8 {@XMM[5]}, [$out]! | ||
1480 | vldmia $fp, {@XMM[0]} @ load counter | ||
1481 | |||
1482 | bne .Lctr_enc_loop | ||
1483 | b .Lctr_enc_done | ||
1484 | |||
1485 | .align 4 | ||
1486 | .Lctr_enc_loop_done: | ||
1487 | add $len, $len, #8 | ||
1488 | vld1.8 {@XMM[8]}, [$inp]! @ load input | ||
1489 | veor @XMM[0], @XMM[8] | ||
1490 | vst1.8 {@XMM[0]}, [$out]! @ write output | ||
1491 | cmp $len, #2 | ||
1492 | blo .Lctr_enc_done | ||
1493 | vld1.8 {@XMM[9]}, [$inp]! | ||
1494 | veor @XMM[1], @XMM[9] | ||
1495 | vst1.8 {@XMM[1]}, [$out]! | ||
1496 | beq .Lctr_enc_done | ||
1497 | vld1.8 {@XMM[10]}, [$inp]! | ||
1498 | veor @XMM[4], @XMM[10] | ||
1499 | vst1.8 {@XMM[4]}, [$out]! | ||
1500 | cmp $len, #4 | ||
1501 | blo .Lctr_enc_done | ||
1502 | vld1.8 {@XMM[11]}, [$inp]! | ||
1503 | veor @XMM[6], @XMM[11] | ||
1504 | vst1.8 {@XMM[6]}, [$out]! | ||
1505 | beq .Lctr_enc_done | ||
1506 | vld1.8 {@XMM[12]}, [$inp]! | ||
1507 | veor @XMM[3], @XMM[12] | ||
1508 | vst1.8 {@XMM[3]}, [$out]! | ||
1509 | cmp $len, #6 | ||
1510 | blo .Lctr_enc_done | ||
1511 | vld1.8 {@XMM[13]}, [$inp]! | ||
1512 | veor @XMM[7], @XMM[13] | ||
1513 | vst1.8 {@XMM[7]}, [$out]! | ||
1514 | beq .Lctr_enc_done | ||
1515 | vld1.8 {@XMM[14]}, [$inp] | ||
1516 | veor @XMM[2], @XMM[14] | ||
1517 | vst1.8 {@XMM[2]}, [$out]! | ||
1518 | |||
1519 | .Lctr_enc_done: | ||
1520 | vmov.i32 q0, #0 | ||
1521 | vmov.i32 q1, #0 | ||
1522 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1523 | .Lctr_enc_bzero: @ wipe key schedule [if any] | ||
1524 | vstmia $keysched!, {q0-q1} | ||
1525 | cmp $keysched, $fp | ||
1526 | bne .Lctr_enc_bzero | ||
1527 | #else | ||
1528 | vstmia $keysched, {q0-q1} | ||
1529 | #endif | ||
1530 | |||
1531 | mov sp, $fp | ||
1532 | add sp, #0x10 @ add sp,$fp,#0x10 is no good for thumb | ||
1533 | VFP_ABI_POP | ||
1534 | ldmia sp!, {r4-r10, pc} @ return | ||
1535 | |||
1536 | .align 4 | ||
1537 | .Lctr_enc_short: | ||
1538 | ldr ip, [sp] @ ctr pointer is passed on stack | ||
1539 | stmdb sp!, {r4-r8, lr} | ||
1540 | |||
1541 | mov r4, $inp @ copy arguments | ||
1542 | mov r5, $out | ||
1543 | mov r6, $len | ||
1544 | mov r7, $key | ||
1545 | ldr r8, [ip, #12] @ load counter LSW | ||
1546 | vld1.8 {@XMM[1]}, [ip] @ load whole counter value | ||
1547 | #ifdef __ARMEL__ | ||
1548 | rev r8, r8 | ||
1549 | #endif | ||
1550 | sub sp, sp, #0x10 | ||
1551 | vst1.8 {@XMM[1]}, [sp,:64] @ copy counter value | ||
1552 | sub sp, sp, #0x10 | ||
1553 | |||
1554 | .Lctr_enc_short_loop: | ||
1555 | add r0, sp, #0x10 @ input counter value | ||
1556 | mov r1, sp @ output on the stack | ||
1557 | mov r2, r7 @ key | ||
1558 | |||
1559 | bl AES_encrypt | ||
1560 | |||
1561 | vld1.8 {@XMM[0]}, [r4]! @ load input | ||
1562 | vld1.8 {@XMM[1]}, [sp,:64] @ load encrypted counter | ||
1563 | add r8, r8, #1 | ||
1564 | #ifdef __ARMEL__ | ||
1565 | rev r0, r8 | ||
1566 | str r0, [sp, #0x1c] @ next counter value | ||
1567 | #else | ||
1568 | str r8, [sp, #0x1c] @ next counter value | ||
1569 | #endif | ||
1570 | veor @XMM[0],@XMM[0],@XMM[1] | ||
1571 | vst1.8 {@XMM[0]}, [r5]! @ store output | ||
1572 | subs r6, r6, #1 | ||
1573 | bne .Lctr_enc_short_loop | ||
1574 | |||
1575 | vmov.i32 q0, #0 | ||
1576 | vmov.i32 q1, #0 | ||
1577 | vstmia sp!, {q0-q1} | ||
1578 | |||
1579 | ldmia sp!, {r4-r8, pc} | ||
1580 | .size bsaes_ctr32_encrypt_blocks,.-bsaes_ctr32_encrypt_blocks | ||
1581 | ___ | ||
1582 | } | ||
1583 | { | ||
1584 | ###################################################################### | ||
1585 | # void bsaes_xts_[en|de]crypt(const char *inp,char *out,size_t len, | ||
1586 | # const AES_KEY *key1, const AES_KEY *key2, | ||
1587 | # const unsigned char iv[16]); | ||
1588 | # | ||
1589 | my ($inp,$out,$len,$key,$rounds,$magic,$fp)=(map("r$_",(7..10,1..3))); | ||
1590 | my $const="r6"; # returned by _bsaes_key_convert | ||
1591 | my $twmask=@XMM[5]; | ||
1592 | my @T=@XMM[6..7]; | ||
1593 | |||
1594 | $code.=<<___; | ||
1595 | .globl bsaes_xts_encrypt | ||
1596 | .type bsaes_xts_encrypt,%function | ||
1597 | .align 4 | ||
1598 | bsaes_xts_encrypt: | ||
1599 | mov ip, sp | ||
1600 | stmdb sp!, {r4-r10, lr} @ 0x20 | ||
1601 | VFP_ABI_PUSH | ||
1602 | mov r6, sp @ future $fp | ||
1603 | |||
1604 | mov $inp, r0 | ||
1605 | mov $out, r1 | ||
1606 | mov $len, r2 | ||
1607 | mov $key, r3 | ||
1608 | |||
1609 | sub r0, sp, #0x10 @ 0x10 | ||
1610 | bic r0, #0xf @ align at 16 bytes | ||
1611 | mov sp, r0 | ||
1612 | |||
1613 | #ifdef XTS_CHAIN_TWEAK | ||
1614 | ldr r0, [ip] @ pointer to input tweak | ||
1615 | #else | ||
1616 | @ generate initial tweak | ||
1617 | ldr r0, [ip, #4] @ iv[] | ||
1618 | mov r1, sp | ||
1619 | ldr r2, [ip, #0] @ key2 | ||
1620 | bl AES_encrypt | ||
1621 | mov r0,sp @ pointer to initial tweak | ||
1622 | #endif | ||
1623 | |||
1624 | ldr $rounds, [$key, #240] @ get # of rounds | ||
1625 | mov $fp, r6 | ||
1626 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1627 | @ allocate the key schedule on the stack | ||
1628 | sub r12, sp, $rounds, lsl#7 @ 128 bytes per inner round key | ||
1629 | @ add r12, #`128-32` @ size of bit-sliced key schedule | ||
1630 | sub r12, #`32+16` @ place for tweak[9] | ||
1631 | |||
1632 | @ populate the key schedule | ||
1633 | mov r4, $key @ pass key | ||
1634 | mov r5, $rounds @ pass # of rounds | ||
1635 | mov sp, r12 | ||
1636 | add r12, #0x90 @ pass key schedule | ||
1637 | bl _bsaes_key_convert | ||
1638 | veor @XMM[7], @XMM[7], @XMM[15] @ fix up last round key | ||
1639 | vstmia r12, {@XMM[7]} @ save last round key | ||
1640 | #else | ||
1641 | ldr r12, [$key, #244] | ||
1642 | eors r12, #1 | ||
1643 | beq 0f | ||
1644 | |||
1645 | str r12, [$key, #244] | ||
1646 | mov r4, $key @ pass key | ||
1647 | mov r5, $rounds @ pass # of rounds | ||
1648 | add r12, $key, #248 @ pass key schedule | ||
1649 | bl _bsaes_key_convert | ||
1650 | veor @XMM[7], @XMM[7], @XMM[15] @ fix up last round key | ||
1651 | vstmia r12, {@XMM[7]} | ||
1652 | |||
1653 | .align 2 | ||
1654 | 0: sub sp, #0x90 @ place for tweak[9] | ||
1655 | #endif | ||
1656 | |||
1657 | vld1.8 {@XMM[8]}, [r0] @ initial tweak | ||
1658 | adr $magic, .Lxts_magic | ||
1659 | |||
1660 | subs $len, #0x80 | ||
1661 | blo .Lxts_enc_short | ||
1662 | b .Lxts_enc_loop | ||
1663 | |||
1664 | .align 4 | ||
1665 | .Lxts_enc_loop: | ||
1666 | vldmia $magic, {$twmask} @ load XTS magic | ||
1667 | vshr.s64 @T[0], @XMM[8], #63 | ||
1668 | mov r0, sp | ||
1669 | vand @T[0], @T[0], $twmask | ||
1670 | ___ | ||
1671 | for($i=9;$i<16;$i++) { | ||
1672 | $code.=<<___; | ||
1673 | vadd.u64 @XMM[$i], @XMM[$i-1], @XMM[$i-1] | ||
1674 | vst1.64 {@XMM[$i-1]}, [r0,:128]! | ||
1675 | vswp `&Dhi("@T[0]")`,`&Dlo("@T[0]")` | ||
1676 | vshr.s64 @T[1], @XMM[$i], #63 | ||
1677 | veor @XMM[$i], @XMM[$i], @T[0] | ||
1678 | vand @T[1], @T[1], $twmask | ||
1679 | ___ | ||
1680 | @T=reverse(@T); | ||
1681 | |||
1682 | $code.=<<___ if ($i>=10); | ||
1683 | vld1.8 {@XMM[$i-10]}, [$inp]! | ||
1684 | ___ | ||
1685 | $code.=<<___ if ($i>=11); | ||
1686 | veor @XMM[$i-11], @XMM[$i-11], @XMM[$i-3] | ||
1687 | ___ | ||
1688 | } | ||
1689 | $code.=<<___; | ||
1690 | vadd.u64 @XMM[8], @XMM[15], @XMM[15] | ||
1691 | vst1.64 {@XMM[15]}, [r0,:128]! | ||
1692 | vswp `&Dhi("@T[0]")`,`&Dlo("@T[0]")` | ||
1693 | veor @XMM[8], @XMM[8], @T[0] | ||
1694 | vst1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
1695 | |||
1696 | vld1.8 {@XMM[6]-@XMM[7]}, [$inp]! | ||
1697 | veor @XMM[5], @XMM[5], @XMM[13] | ||
1698 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1699 | add r4, sp, #0x90 @ pass key schedule | ||
1700 | #else | ||
1701 | add r4, $key, #248 @ pass key schedule | ||
1702 | #endif | ||
1703 | veor @XMM[6], @XMM[6], @XMM[14] | ||
1704 | mov r5, $rounds @ pass rounds | ||
1705 | veor @XMM[7], @XMM[7], @XMM[15] | ||
1706 | mov r0, sp | ||
1707 | |||
1708 | bl _bsaes_encrypt8 | ||
1709 | |||
1710 | vld1.64 {@XMM[ 8]-@XMM[ 9]}, [r0,:128]! | ||
1711 | vld1.64 {@XMM[10]-@XMM[11]}, [r0,:128]! | ||
1712 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
1713 | vld1.64 {@XMM[12]-@XMM[13]}, [r0,:128]! | ||
1714 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
1715 | veor @XMM[8], @XMM[4], @XMM[10] | ||
1716 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
1717 | veor @XMM[9], @XMM[6], @XMM[11] | ||
1718 | vld1.64 {@XMM[14]-@XMM[15]}, [r0,:128]! | ||
1719 | veor @XMM[10], @XMM[3], @XMM[12] | ||
1720 | vst1.8 {@XMM[8]-@XMM[9]}, [$out]! | ||
1721 | veor @XMM[11], @XMM[7], @XMM[13] | ||
1722 | veor @XMM[12], @XMM[2], @XMM[14] | ||
1723 | vst1.8 {@XMM[10]-@XMM[11]}, [$out]! | ||
1724 | veor @XMM[13], @XMM[5], @XMM[15] | ||
1725 | vst1.8 {@XMM[12]-@XMM[13]}, [$out]! | ||
1726 | |||
1727 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
1728 | |||
1729 | subs $len, #0x80 | ||
1730 | bpl .Lxts_enc_loop | ||
1731 | |||
1732 | .Lxts_enc_short: | ||
1733 | adds $len, #0x70 | ||
1734 | bmi .Lxts_enc_done | ||
1735 | |||
1736 | vldmia $magic, {$twmask} @ load XTS magic | ||
1737 | vshr.s64 @T[0], @XMM[8], #63 | ||
1738 | mov r0, sp | ||
1739 | vand @T[0], @T[0], $twmask | ||
1740 | ___ | ||
1741 | for($i=9;$i<16;$i++) { | ||
1742 | $code.=<<___; | ||
1743 | vadd.u64 @XMM[$i], @XMM[$i-1], @XMM[$i-1] | ||
1744 | vst1.64 {@XMM[$i-1]}, [r0,:128]! | ||
1745 | vswp `&Dhi("@T[0]")`,`&Dlo("@T[0]")` | ||
1746 | vshr.s64 @T[1], @XMM[$i], #63 | ||
1747 | veor @XMM[$i], @XMM[$i], @T[0] | ||
1748 | vand @T[1], @T[1], $twmask | ||
1749 | ___ | ||
1750 | @T=reverse(@T); | ||
1751 | |||
1752 | $code.=<<___ if ($i>=10); | ||
1753 | vld1.8 {@XMM[$i-10]}, [$inp]! | ||
1754 | subs $len, #0x10 | ||
1755 | bmi .Lxts_enc_`$i-9` | ||
1756 | ___ | ||
1757 | $code.=<<___ if ($i>=11); | ||
1758 | veor @XMM[$i-11], @XMM[$i-11], @XMM[$i-3] | ||
1759 | ___ | ||
1760 | } | ||
1761 | $code.=<<___; | ||
1762 | sub $len, #0x10 | ||
1763 | vst1.64 {@XMM[15]}, [r0,:128] @ next round tweak | ||
1764 | |||
1765 | vld1.8 {@XMM[6]}, [$inp]! | ||
1766 | veor @XMM[5], @XMM[5], @XMM[13] | ||
1767 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1768 | add r4, sp, #0x90 @ pass key schedule | ||
1769 | #else | ||
1770 | add r4, $key, #248 @ pass key schedule | ||
1771 | #endif | ||
1772 | veor @XMM[6], @XMM[6], @XMM[14] | ||
1773 | mov r5, $rounds @ pass rounds | ||
1774 | mov r0, sp | ||
1775 | |||
1776 | bl _bsaes_encrypt8 | ||
1777 | |||
1778 | vld1.64 {@XMM[ 8]-@XMM[ 9]}, [r0,:128]! | ||
1779 | vld1.64 {@XMM[10]-@XMM[11]}, [r0,:128]! | ||
1780 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
1781 | vld1.64 {@XMM[12]-@XMM[13]}, [r0,:128]! | ||
1782 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
1783 | veor @XMM[8], @XMM[4], @XMM[10] | ||
1784 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
1785 | veor @XMM[9], @XMM[6], @XMM[11] | ||
1786 | vld1.64 {@XMM[14]}, [r0,:128]! | ||
1787 | veor @XMM[10], @XMM[3], @XMM[12] | ||
1788 | vst1.8 {@XMM[8]-@XMM[9]}, [$out]! | ||
1789 | veor @XMM[11], @XMM[7], @XMM[13] | ||
1790 | veor @XMM[12], @XMM[2], @XMM[14] | ||
1791 | vst1.8 {@XMM[10]-@XMM[11]}, [$out]! | ||
1792 | vst1.8 {@XMM[12]}, [$out]! | ||
1793 | |||
1794 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
1795 | b .Lxts_enc_done | ||
1796 | .align 4 | ||
1797 | .Lxts_enc_6: | ||
1798 | vst1.64 {@XMM[14]}, [r0,:128] @ next round tweak | ||
1799 | |||
1800 | veor @XMM[4], @XMM[4], @XMM[12] | ||
1801 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1802 | add r4, sp, #0x90 @ pass key schedule | ||
1803 | #else | ||
1804 | add r4, $key, #248 @ pass key schedule | ||
1805 | #endif | ||
1806 | veor @XMM[5], @XMM[5], @XMM[13] | ||
1807 | mov r5, $rounds @ pass rounds | ||
1808 | mov r0, sp | ||
1809 | |||
1810 | bl _bsaes_encrypt8 | ||
1811 | |||
1812 | vld1.64 {@XMM[ 8]-@XMM[ 9]}, [r0,:128]! | ||
1813 | vld1.64 {@XMM[10]-@XMM[11]}, [r0,:128]! | ||
1814 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
1815 | vld1.64 {@XMM[12]-@XMM[13]}, [r0,:128]! | ||
1816 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
1817 | veor @XMM[8], @XMM[4], @XMM[10] | ||
1818 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
1819 | veor @XMM[9], @XMM[6], @XMM[11] | ||
1820 | veor @XMM[10], @XMM[3], @XMM[12] | ||
1821 | vst1.8 {@XMM[8]-@XMM[9]}, [$out]! | ||
1822 | veor @XMM[11], @XMM[7], @XMM[13] | ||
1823 | vst1.8 {@XMM[10]-@XMM[11]}, [$out]! | ||
1824 | |||
1825 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
1826 | b .Lxts_enc_done | ||
1827 | |||
1828 | @ put this in range for both ARM and Thumb mode adr instructions | ||
1829 | .align 5 | ||
1830 | .Lxts_magic: | ||
1831 | .quad 1, 0x87 | ||
1832 | |||
1833 | .align 5 | ||
1834 | .Lxts_enc_5: | ||
1835 | vst1.64 {@XMM[13]}, [r0,:128] @ next round tweak | ||
1836 | |||
1837 | veor @XMM[3], @XMM[3], @XMM[11] | ||
1838 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1839 | add r4, sp, #0x90 @ pass key schedule | ||
1840 | #else | ||
1841 | add r4, $key, #248 @ pass key schedule | ||
1842 | #endif | ||
1843 | veor @XMM[4], @XMM[4], @XMM[12] | ||
1844 | mov r5, $rounds @ pass rounds | ||
1845 | mov r0, sp | ||
1846 | |||
1847 | bl _bsaes_encrypt8 | ||
1848 | |||
1849 | vld1.64 {@XMM[ 8]-@XMM[ 9]}, [r0,:128]! | ||
1850 | vld1.64 {@XMM[10]-@XMM[11]}, [r0,:128]! | ||
1851 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
1852 | vld1.64 {@XMM[12]}, [r0,:128]! | ||
1853 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
1854 | veor @XMM[8], @XMM[4], @XMM[10] | ||
1855 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
1856 | veor @XMM[9], @XMM[6], @XMM[11] | ||
1857 | veor @XMM[10], @XMM[3], @XMM[12] | ||
1858 | vst1.8 {@XMM[8]-@XMM[9]}, [$out]! | ||
1859 | vst1.8 {@XMM[10]}, [$out]! | ||
1860 | |||
1861 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
1862 | b .Lxts_enc_done | ||
1863 | .align 4 | ||
1864 | .Lxts_enc_4: | ||
1865 | vst1.64 {@XMM[12]}, [r0,:128] @ next round tweak | ||
1866 | |||
1867 | veor @XMM[2], @XMM[2], @XMM[10] | ||
1868 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1869 | add r4, sp, #0x90 @ pass key schedule | ||
1870 | #else | ||
1871 | add r4, $key, #248 @ pass key schedule | ||
1872 | #endif | ||
1873 | veor @XMM[3], @XMM[3], @XMM[11] | ||
1874 | mov r5, $rounds @ pass rounds | ||
1875 | mov r0, sp | ||
1876 | |||
1877 | bl _bsaes_encrypt8 | ||
1878 | |||
1879 | vld1.64 {@XMM[ 8]-@XMM[ 9]}, [r0,:128]! | ||
1880 | vld1.64 {@XMM[10]-@XMM[11]}, [r0,:128]! | ||
1881 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
1882 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
1883 | veor @XMM[8], @XMM[4], @XMM[10] | ||
1884 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
1885 | veor @XMM[9], @XMM[6], @XMM[11] | ||
1886 | vst1.8 {@XMM[8]-@XMM[9]}, [$out]! | ||
1887 | |||
1888 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
1889 | b .Lxts_enc_done | ||
1890 | .align 4 | ||
1891 | .Lxts_enc_3: | ||
1892 | vst1.64 {@XMM[11]}, [r0,:128] @ next round tweak | ||
1893 | |||
1894 | veor @XMM[1], @XMM[1], @XMM[9] | ||
1895 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1896 | add r4, sp, #0x90 @ pass key schedule | ||
1897 | #else | ||
1898 | add r4, $key, #248 @ pass key schedule | ||
1899 | #endif | ||
1900 | veor @XMM[2], @XMM[2], @XMM[10] | ||
1901 | mov r5, $rounds @ pass rounds | ||
1902 | mov r0, sp | ||
1903 | |||
1904 | bl _bsaes_encrypt8 | ||
1905 | |||
1906 | vld1.64 {@XMM[8]-@XMM[9]}, [r0,:128]! | ||
1907 | vld1.64 {@XMM[10]}, [r0,:128]! | ||
1908 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
1909 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
1910 | veor @XMM[8], @XMM[4], @XMM[10] | ||
1911 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
1912 | vst1.8 {@XMM[8]}, [$out]! | ||
1913 | |||
1914 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
1915 | b .Lxts_enc_done | ||
1916 | .align 4 | ||
1917 | .Lxts_enc_2: | ||
1918 | vst1.64 {@XMM[10]}, [r0,:128] @ next round tweak | ||
1919 | |||
1920 | veor @XMM[0], @XMM[0], @XMM[8] | ||
1921 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
1922 | add r4, sp, #0x90 @ pass key schedule | ||
1923 | #else | ||
1924 | add r4, $key, #248 @ pass key schedule | ||
1925 | #endif | ||
1926 | veor @XMM[1], @XMM[1], @XMM[9] | ||
1927 | mov r5, $rounds @ pass rounds | ||
1928 | mov r0, sp | ||
1929 | |||
1930 | bl _bsaes_encrypt8 | ||
1931 | |||
1932 | vld1.64 {@XMM[8]-@XMM[9]}, [r0,:128]! | ||
1933 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
1934 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
1935 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
1936 | |||
1937 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
1938 | b .Lxts_enc_done | ||
1939 | .align 4 | ||
1940 | .Lxts_enc_1: | ||
1941 | mov r0, sp | ||
1942 | veor @XMM[0], @XMM[8] | ||
1943 | mov r1, sp | ||
1944 | vst1.8 {@XMM[0]}, [sp,:128] | ||
1945 | mov r2, $key | ||
1946 | mov r4, $fp @ preserve fp | ||
1947 | |||
1948 | bl AES_encrypt | ||
1949 | |||
1950 | vld1.8 {@XMM[0]}, [sp,:128] | ||
1951 | veor @XMM[0], @XMM[0], @XMM[8] | ||
1952 | vst1.8 {@XMM[0]}, [$out]! | ||
1953 | mov $fp, r4 | ||
1954 | |||
1955 | vmov @XMM[8], @XMM[9] @ next round tweak | ||
1956 | |||
1957 | .Lxts_enc_done: | ||
1958 | #ifndef XTS_CHAIN_TWEAK | ||
1959 | adds $len, #0x10 | ||
1960 | beq .Lxts_enc_ret | ||
1961 | sub r6, $out, #0x10 | ||
1962 | |||
1963 | .Lxts_enc_steal: | ||
1964 | ldrb r0, [$inp], #1 | ||
1965 | ldrb r1, [$out, #-0x10] | ||
1966 | strb r0, [$out, #-0x10] | ||
1967 | strb r1, [$out], #1 | ||
1968 | |||
1969 | subs $len, #1 | ||
1970 | bhi .Lxts_enc_steal | ||
1971 | |||
1972 | vld1.8 {@XMM[0]}, [r6] | ||
1973 | mov r0, sp | ||
1974 | veor @XMM[0], @XMM[0], @XMM[8] | ||
1975 | mov r1, sp | ||
1976 | vst1.8 {@XMM[0]}, [sp,:128] | ||
1977 | mov r2, $key | ||
1978 | mov r4, $fp @ preserve fp | ||
1979 | |||
1980 | bl AES_encrypt | ||
1981 | |||
1982 | vld1.8 {@XMM[0]}, [sp,:128] | ||
1983 | veor @XMM[0], @XMM[0], @XMM[8] | ||
1984 | vst1.8 {@XMM[0]}, [r6] | ||
1985 | mov $fp, r4 | ||
1986 | #endif | ||
1987 | |||
1988 | .Lxts_enc_ret: | ||
1989 | bic r0, $fp, #0xf | ||
1990 | vmov.i32 q0, #0 | ||
1991 | vmov.i32 q1, #0 | ||
1992 | #ifdef XTS_CHAIN_TWEAK | ||
1993 | ldr r1, [$fp, #0x20+VFP_ABI_FRAME] @ chain tweak | ||
1994 | #endif | ||
1995 | .Lxts_enc_bzero: @ wipe key schedule [if any] | ||
1996 | vstmia sp!, {q0-q1} | ||
1997 | cmp sp, r0 | ||
1998 | bne .Lxts_enc_bzero | ||
1999 | |||
2000 | mov sp, $fp | ||
2001 | #ifdef XTS_CHAIN_TWEAK | ||
2002 | vst1.8 {@XMM[8]}, [r1] | ||
2003 | #endif | ||
2004 | VFP_ABI_POP | ||
2005 | ldmia sp!, {r4-r10, pc} @ return | ||
2006 | |||
2007 | .size bsaes_xts_encrypt,.-bsaes_xts_encrypt | ||
2008 | |||
2009 | .globl bsaes_xts_decrypt | ||
2010 | .type bsaes_xts_decrypt,%function | ||
2011 | .align 4 | ||
2012 | bsaes_xts_decrypt: | ||
2013 | mov ip, sp | ||
2014 | stmdb sp!, {r4-r10, lr} @ 0x20 | ||
2015 | VFP_ABI_PUSH | ||
2016 | mov r6, sp @ future $fp | ||
2017 | |||
2018 | mov $inp, r0 | ||
2019 | mov $out, r1 | ||
2020 | mov $len, r2 | ||
2021 | mov $key, r3 | ||
2022 | |||
2023 | sub r0, sp, #0x10 @ 0x10 | ||
2024 | bic r0, #0xf @ align at 16 bytes | ||
2025 | mov sp, r0 | ||
2026 | |||
2027 | #ifdef XTS_CHAIN_TWEAK | ||
2028 | ldr r0, [ip] @ pointer to input tweak | ||
2029 | #else | ||
2030 | @ generate initial tweak | ||
2031 | ldr r0, [ip, #4] @ iv[] | ||
2032 | mov r1, sp | ||
2033 | ldr r2, [ip, #0] @ key2 | ||
2034 | bl AES_encrypt | ||
2035 | mov r0, sp @ pointer to initial tweak | ||
2036 | #endif | ||
2037 | |||
2038 | ldr $rounds, [$key, #240] @ get # of rounds | ||
2039 | mov $fp, r6 | ||
2040 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2041 | @ allocate the key schedule on the stack | ||
2042 | sub r12, sp, $rounds, lsl#7 @ 128 bytes per inner round key | ||
2043 | @ add r12, #`128-32` @ size of bit-sliced key schedule | ||
2044 | sub r12, #`32+16` @ place for tweak[9] | ||
2045 | |||
2046 | @ populate the key schedule | ||
2047 | mov r4, $key @ pass key | ||
2048 | mov r5, $rounds @ pass # of rounds | ||
2049 | mov sp, r12 | ||
2050 | add r12, #0x90 @ pass key schedule | ||
2051 | bl _bsaes_key_convert | ||
2052 | add r4, sp, #0x90 | ||
2053 | vldmia r4, {@XMM[6]} | ||
2054 | vstmia r12, {@XMM[15]} @ save last round key | ||
2055 | veor @XMM[7], @XMM[7], @XMM[6] @ fix up round 0 key | ||
2056 | vstmia r4, {@XMM[7]} | ||
2057 | #else | ||
2058 | ldr r12, [$key, #244] | ||
2059 | eors r12, #1 | ||
2060 | beq 0f | ||
2061 | |||
2062 | str r12, [$key, #244] | ||
2063 | mov r4, $key @ pass key | ||
2064 | mov r5, $rounds @ pass # of rounds | ||
2065 | add r12, $key, #248 @ pass key schedule | ||
2066 | bl _bsaes_key_convert | ||
2067 | add r4, $key, #248 | ||
2068 | vldmia r4, {@XMM[6]} | ||
2069 | vstmia r12, {@XMM[15]} @ save last round key | ||
2070 | veor @XMM[7], @XMM[7], @XMM[6] @ fix up round 0 key | ||
2071 | vstmia r4, {@XMM[7]} | ||
2072 | |||
2073 | .align 2 | ||
2074 | 0: sub sp, #0x90 @ place for tweak[9] | ||
2075 | #endif | ||
2076 | vld1.8 {@XMM[8]}, [r0] @ initial tweak | ||
2077 | adr $magic, .Lxts_magic | ||
2078 | |||
2079 | tst $len, #0xf @ if not multiple of 16 | ||
2080 | it ne @ Thumb2 thing, sanity check in ARM | ||
2081 | subne $len, #0x10 @ subtract another 16 bytes | ||
2082 | subs $len, #0x80 | ||
2083 | |||
2084 | blo .Lxts_dec_short | ||
2085 | b .Lxts_dec_loop | ||
2086 | |||
2087 | .align 4 | ||
2088 | .Lxts_dec_loop: | ||
2089 | vldmia $magic, {$twmask} @ load XTS magic | ||
2090 | vshr.s64 @T[0], @XMM[8], #63 | ||
2091 | mov r0, sp | ||
2092 | vand @T[0], @T[0], $twmask | ||
2093 | ___ | ||
2094 | for($i=9;$i<16;$i++) { | ||
2095 | $code.=<<___; | ||
2096 | vadd.u64 @XMM[$i], @XMM[$i-1], @XMM[$i-1] | ||
2097 | vst1.64 {@XMM[$i-1]}, [r0,:128]! | ||
2098 | vswp `&Dhi("@T[0]")`,`&Dlo("@T[0]")` | ||
2099 | vshr.s64 @T[1], @XMM[$i], #63 | ||
2100 | veor @XMM[$i], @XMM[$i], @T[0] | ||
2101 | vand @T[1], @T[1], $twmask | ||
2102 | ___ | ||
2103 | @T=reverse(@T); | ||
2104 | |||
2105 | $code.=<<___ if ($i>=10); | ||
2106 | vld1.8 {@XMM[$i-10]}, [$inp]! | ||
2107 | ___ | ||
2108 | $code.=<<___ if ($i>=11); | ||
2109 | veor @XMM[$i-11], @XMM[$i-11], @XMM[$i-3] | ||
2110 | ___ | ||
2111 | } | ||
2112 | $code.=<<___; | ||
2113 | vadd.u64 @XMM[8], @XMM[15], @XMM[15] | ||
2114 | vst1.64 {@XMM[15]}, [r0,:128]! | ||
2115 | vswp `&Dhi("@T[0]")`,`&Dlo("@T[0]")` | ||
2116 | veor @XMM[8], @XMM[8], @T[0] | ||
2117 | vst1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
2118 | |||
2119 | vld1.8 {@XMM[6]-@XMM[7]}, [$inp]! | ||
2120 | veor @XMM[5], @XMM[5], @XMM[13] | ||
2121 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2122 | add r4, sp, #0x90 @ pass key schedule | ||
2123 | #else | ||
2124 | add r4, $key, #248 @ pass key schedule | ||
2125 | #endif | ||
2126 | veor @XMM[6], @XMM[6], @XMM[14] | ||
2127 | mov r5, $rounds @ pass rounds | ||
2128 | veor @XMM[7], @XMM[7], @XMM[15] | ||
2129 | mov r0, sp | ||
2130 | |||
2131 | bl _bsaes_decrypt8 | ||
2132 | |||
2133 | vld1.64 {@XMM[ 8]-@XMM[ 9]}, [r0,:128]! | ||
2134 | vld1.64 {@XMM[10]-@XMM[11]}, [r0,:128]! | ||
2135 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
2136 | vld1.64 {@XMM[12]-@XMM[13]}, [r0,:128]! | ||
2137 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
2138 | veor @XMM[8], @XMM[6], @XMM[10] | ||
2139 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
2140 | veor @XMM[9], @XMM[4], @XMM[11] | ||
2141 | vld1.64 {@XMM[14]-@XMM[15]}, [r0,:128]! | ||
2142 | veor @XMM[10], @XMM[2], @XMM[12] | ||
2143 | vst1.8 {@XMM[8]-@XMM[9]}, [$out]! | ||
2144 | veor @XMM[11], @XMM[7], @XMM[13] | ||
2145 | veor @XMM[12], @XMM[3], @XMM[14] | ||
2146 | vst1.8 {@XMM[10]-@XMM[11]}, [$out]! | ||
2147 | veor @XMM[13], @XMM[5], @XMM[15] | ||
2148 | vst1.8 {@XMM[12]-@XMM[13]}, [$out]! | ||
2149 | |||
2150 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
2151 | |||
2152 | subs $len, #0x80 | ||
2153 | bpl .Lxts_dec_loop | ||
2154 | |||
2155 | .Lxts_dec_short: | ||
2156 | adds $len, #0x70 | ||
2157 | bmi .Lxts_dec_done | ||
2158 | |||
2159 | vldmia $magic, {$twmask} @ load XTS magic | ||
2160 | vshr.s64 @T[0], @XMM[8], #63 | ||
2161 | mov r0, sp | ||
2162 | vand @T[0], @T[0], $twmask | ||
2163 | ___ | ||
2164 | for($i=9;$i<16;$i++) { | ||
2165 | $code.=<<___; | ||
2166 | vadd.u64 @XMM[$i], @XMM[$i-1], @XMM[$i-1] | ||
2167 | vst1.64 {@XMM[$i-1]}, [r0,:128]! | ||
2168 | vswp `&Dhi("@T[0]")`,`&Dlo("@T[0]")` | ||
2169 | vshr.s64 @T[1], @XMM[$i], #63 | ||
2170 | veor @XMM[$i], @XMM[$i], @T[0] | ||
2171 | vand @T[1], @T[1], $twmask | ||
2172 | ___ | ||
2173 | @T=reverse(@T); | ||
2174 | |||
2175 | $code.=<<___ if ($i>=10); | ||
2176 | vld1.8 {@XMM[$i-10]}, [$inp]! | ||
2177 | subs $len, #0x10 | ||
2178 | bmi .Lxts_dec_`$i-9` | ||
2179 | ___ | ||
2180 | $code.=<<___ if ($i>=11); | ||
2181 | veor @XMM[$i-11], @XMM[$i-11], @XMM[$i-3] | ||
2182 | ___ | ||
2183 | } | ||
2184 | $code.=<<___; | ||
2185 | sub $len, #0x10 | ||
2186 | vst1.64 {@XMM[15]}, [r0,:128] @ next round tweak | ||
2187 | |||
2188 | vld1.8 {@XMM[6]}, [$inp]! | ||
2189 | veor @XMM[5], @XMM[5], @XMM[13] | ||
2190 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2191 | add r4, sp, #0x90 @ pass key schedule | ||
2192 | #else | ||
2193 | add r4, $key, #248 @ pass key schedule | ||
2194 | #endif | ||
2195 | veor @XMM[6], @XMM[6], @XMM[14] | ||
2196 | mov r5, $rounds @ pass rounds | ||
2197 | mov r0, sp | ||
2198 | |||
2199 | bl _bsaes_decrypt8 | ||
2200 | |||
2201 | vld1.64 {@XMM[ 8]-@XMM[ 9]}, [r0,:128]! | ||
2202 | vld1.64 {@XMM[10]-@XMM[11]}, [r0,:128]! | ||
2203 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
2204 | vld1.64 {@XMM[12]-@XMM[13]}, [r0,:128]! | ||
2205 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
2206 | veor @XMM[8], @XMM[6], @XMM[10] | ||
2207 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
2208 | veor @XMM[9], @XMM[4], @XMM[11] | ||
2209 | vld1.64 {@XMM[14]}, [r0,:128]! | ||
2210 | veor @XMM[10], @XMM[2], @XMM[12] | ||
2211 | vst1.8 {@XMM[8]-@XMM[9]}, [$out]! | ||
2212 | veor @XMM[11], @XMM[7], @XMM[13] | ||
2213 | veor @XMM[12], @XMM[3], @XMM[14] | ||
2214 | vst1.8 {@XMM[10]-@XMM[11]}, [$out]! | ||
2215 | vst1.8 {@XMM[12]}, [$out]! | ||
2216 | |||
2217 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
2218 | b .Lxts_dec_done | ||
2219 | .align 4 | ||
2220 | .Lxts_dec_6: | ||
2221 | vst1.64 {@XMM[14]}, [r0,:128] @ next round tweak | ||
2222 | |||
2223 | veor @XMM[4], @XMM[4], @XMM[12] | ||
2224 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2225 | add r4, sp, #0x90 @ pass key schedule | ||
2226 | #else | ||
2227 | add r4, $key, #248 @ pass key schedule | ||
2228 | #endif | ||
2229 | veor @XMM[5], @XMM[5], @XMM[13] | ||
2230 | mov r5, $rounds @ pass rounds | ||
2231 | mov r0, sp | ||
2232 | |||
2233 | bl _bsaes_decrypt8 | ||
2234 | |||
2235 | vld1.64 {@XMM[ 8]-@XMM[ 9]}, [r0,:128]! | ||
2236 | vld1.64 {@XMM[10]-@XMM[11]}, [r0,:128]! | ||
2237 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
2238 | vld1.64 {@XMM[12]-@XMM[13]}, [r0,:128]! | ||
2239 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
2240 | veor @XMM[8], @XMM[6], @XMM[10] | ||
2241 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
2242 | veor @XMM[9], @XMM[4], @XMM[11] | ||
2243 | veor @XMM[10], @XMM[2], @XMM[12] | ||
2244 | vst1.8 {@XMM[8]-@XMM[9]}, [$out]! | ||
2245 | veor @XMM[11], @XMM[7], @XMM[13] | ||
2246 | vst1.8 {@XMM[10]-@XMM[11]}, [$out]! | ||
2247 | |||
2248 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
2249 | b .Lxts_dec_done | ||
2250 | .align 4 | ||
2251 | .Lxts_dec_5: | ||
2252 | vst1.64 {@XMM[13]}, [r0,:128] @ next round tweak | ||
2253 | |||
2254 | veor @XMM[3], @XMM[3], @XMM[11] | ||
2255 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2256 | add r4, sp, #0x90 @ pass key schedule | ||
2257 | #else | ||
2258 | add r4, $key, #248 @ pass key schedule | ||
2259 | #endif | ||
2260 | veor @XMM[4], @XMM[4], @XMM[12] | ||
2261 | mov r5, $rounds @ pass rounds | ||
2262 | mov r0, sp | ||
2263 | |||
2264 | bl _bsaes_decrypt8 | ||
2265 | |||
2266 | vld1.64 {@XMM[ 8]-@XMM[ 9]}, [r0,:128]! | ||
2267 | vld1.64 {@XMM[10]-@XMM[11]}, [r0,:128]! | ||
2268 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
2269 | vld1.64 {@XMM[12]}, [r0,:128]! | ||
2270 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
2271 | veor @XMM[8], @XMM[6], @XMM[10] | ||
2272 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
2273 | veor @XMM[9], @XMM[4], @XMM[11] | ||
2274 | veor @XMM[10], @XMM[2], @XMM[12] | ||
2275 | vst1.8 {@XMM[8]-@XMM[9]}, [$out]! | ||
2276 | vst1.8 {@XMM[10]}, [$out]! | ||
2277 | |||
2278 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
2279 | b .Lxts_dec_done | ||
2280 | .align 4 | ||
2281 | .Lxts_dec_4: | ||
2282 | vst1.64 {@XMM[12]}, [r0,:128] @ next round tweak | ||
2283 | |||
2284 | veor @XMM[2], @XMM[2], @XMM[10] | ||
2285 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2286 | add r4, sp, #0x90 @ pass key schedule | ||
2287 | #else | ||
2288 | add r4, $key, #248 @ pass key schedule | ||
2289 | #endif | ||
2290 | veor @XMM[3], @XMM[3], @XMM[11] | ||
2291 | mov r5, $rounds @ pass rounds | ||
2292 | mov r0, sp | ||
2293 | |||
2294 | bl _bsaes_decrypt8 | ||
2295 | |||
2296 | vld1.64 {@XMM[ 8]-@XMM[ 9]}, [r0,:128]! | ||
2297 | vld1.64 {@XMM[10]-@XMM[11]}, [r0,:128]! | ||
2298 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
2299 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
2300 | veor @XMM[8], @XMM[6], @XMM[10] | ||
2301 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
2302 | veor @XMM[9], @XMM[4], @XMM[11] | ||
2303 | vst1.8 {@XMM[8]-@XMM[9]}, [$out]! | ||
2304 | |||
2305 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
2306 | b .Lxts_dec_done | ||
2307 | .align 4 | ||
2308 | .Lxts_dec_3: | ||
2309 | vst1.64 {@XMM[11]}, [r0,:128] @ next round tweak | ||
2310 | |||
2311 | veor @XMM[1], @XMM[1], @XMM[9] | ||
2312 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2313 | add r4, sp, #0x90 @ pass key schedule | ||
2314 | #else | ||
2315 | add r4, $key, #248 @ pass key schedule | ||
2316 | #endif | ||
2317 | veor @XMM[2], @XMM[2], @XMM[10] | ||
2318 | mov r5, $rounds @ pass rounds | ||
2319 | mov r0, sp | ||
2320 | |||
2321 | bl _bsaes_decrypt8 | ||
2322 | |||
2323 | vld1.64 {@XMM[8]-@XMM[9]}, [r0,:128]! | ||
2324 | vld1.64 {@XMM[10]}, [r0,:128]! | ||
2325 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
2326 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
2327 | veor @XMM[8], @XMM[6], @XMM[10] | ||
2328 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
2329 | vst1.8 {@XMM[8]}, [$out]! | ||
2330 | |||
2331 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
2332 | b .Lxts_dec_done | ||
2333 | .align 4 | ||
2334 | .Lxts_dec_2: | ||
2335 | vst1.64 {@XMM[10]}, [r0,:128] @ next round tweak | ||
2336 | |||
2337 | veor @XMM[0], @XMM[0], @XMM[8] | ||
2338 | #ifndef BSAES_ASM_EXTENDED_KEY | ||
2339 | add r4, sp, #0x90 @ pass key schedule | ||
2340 | #else | ||
2341 | add r4, $key, #248 @ pass key schedule | ||
2342 | #endif | ||
2343 | veor @XMM[1], @XMM[1], @XMM[9] | ||
2344 | mov r5, $rounds @ pass rounds | ||
2345 | mov r0, sp | ||
2346 | |||
2347 | bl _bsaes_decrypt8 | ||
2348 | |||
2349 | vld1.64 {@XMM[8]-@XMM[9]}, [r0,:128]! | ||
2350 | veor @XMM[0], @XMM[0], @XMM[ 8] | ||
2351 | veor @XMM[1], @XMM[1], @XMM[ 9] | ||
2352 | vst1.8 {@XMM[0]-@XMM[1]}, [$out]! | ||
2353 | |||
2354 | vld1.64 {@XMM[8]}, [r0,:128] @ next round tweak | ||
2355 | b .Lxts_dec_done | ||
2356 | .align 4 | ||
2357 | .Lxts_dec_1: | ||
2358 | mov r0, sp | ||
2359 | veor @XMM[0], @XMM[8] | ||
2360 | mov r1, sp | ||
2361 | vst1.8 {@XMM[0]}, [sp,:128] | ||
2362 | mov r2, $key | ||
2363 | mov r4, $fp @ preserve fp | ||
2364 | mov r5, $magic @ preserve magic | ||
2365 | |||
2366 | bl AES_decrypt | ||
2367 | |||
2368 | vld1.8 {@XMM[0]}, [sp,:128] | ||
2369 | veor @XMM[0], @XMM[0], @XMM[8] | ||
2370 | vst1.8 {@XMM[0]}, [$out]! | ||
2371 | mov $fp, r4 | ||
2372 | mov $magic, r5 | ||
2373 | |||
2374 | vmov @XMM[8], @XMM[9] @ next round tweak | ||
2375 | |||
2376 | .Lxts_dec_done: | ||
2377 | #ifndef XTS_CHAIN_TWEAK | ||
2378 | adds $len, #0x10 | ||
2379 | beq .Lxts_dec_ret | ||
2380 | |||
2381 | @ calculate one round of extra tweak for the stolen ciphertext | ||
2382 | vldmia $magic, {$twmask} | ||
2383 | vshr.s64 @XMM[6], @XMM[8], #63 | ||
2384 | vand @XMM[6], @XMM[6], $twmask | ||
2385 | vadd.u64 @XMM[9], @XMM[8], @XMM[8] | ||
2386 | vswp `&Dhi("@XMM[6]")`,`&Dlo("@XMM[6]")` | ||
2387 | veor @XMM[9], @XMM[9], @XMM[6] | ||
2388 | |||
2389 | @ perform the final decryption with the last tweak value | ||
2390 | vld1.8 {@XMM[0]}, [$inp]! | ||
2391 | mov r0, sp | ||
2392 | veor @XMM[0], @XMM[0], @XMM[9] | ||
2393 | mov r1, sp | ||
2394 | vst1.8 {@XMM[0]}, [sp,:128] | ||
2395 | mov r2, $key | ||
2396 | mov r4, $fp @ preserve fp | ||
2397 | |||
2398 | bl AES_decrypt | ||
2399 | |||
2400 | vld1.8 {@XMM[0]}, [sp,:128] | ||
2401 | veor @XMM[0], @XMM[0], @XMM[9] | ||
2402 | vst1.8 {@XMM[0]}, [$out] | ||
2403 | |||
2404 | mov r6, $out | ||
2405 | .Lxts_dec_steal: | ||
2406 | ldrb r1, [$out] | ||
2407 | ldrb r0, [$inp], #1 | ||
2408 | strb r1, [$out, #0x10] | ||
2409 | strb r0, [$out], #1 | ||
2410 | |||
2411 | subs $len, #1 | ||
2412 | bhi .Lxts_dec_steal | ||
2413 | |||
2414 | vld1.8 {@XMM[0]}, [r6] | ||
2415 | mov r0, sp | ||
2416 | veor @XMM[0], @XMM[8] | ||
2417 | mov r1, sp | ||
2418 | vst1.8 {@XMM[0]}, [sp,:128] | ||
2419 | mov r2, $key | ||
2420 | |||
2421 | bl AES_decrypt | ||
2422 | |||
2423 | vld1.8 {@XMM[0]}, [sp,:128] | ||
2424 | veor @XMM[0], @XMM[0], @XMM[8] | ||
2425 | vst1.8 {@XMM[0]}, [r6] | ||
2426 | mov $fp, r4 | ||
2427 | #endif | ||
2428 | |||
2429 | .Lxts_dec_ret: | ||
2430 | bic r0, $fp, #0xf | ||
2431 | vmov.i32 q0, #0 | ||
2432 | vmov.i32 q1, #0 | ||
2433 | #ifdef XTS_CHAIN_TWEAK | ||
2434 | ldr r1, [$fp, #0x20+VFP_ABI_FRAME] @ chain tweak | ||
2435 | #endif | ||
2436 | .Lxts_dec_bzero: @ wipe key schedule [if any] | ||
2437 | vstmia sp!, {q0-q1} | ||
2438 | cmp sp, r0 | ||
2439 | bne .Lxts_dec_bzero | ||
2440 | |||
2441 | mov sp, $fp | ||
2442 | #ifdef XTS_CHAIN_TWEAK | ||
2443 | vst1.8 {@XMM[8]}, [r1] | ||
2444 | #endif | ||
2445 | VFP_ABI_POP | ||
2446 | ldmia sp!, {r4-r10, pc} @ return | ||
2447 | |||
2448 | .size bsaes_xts_decrypt,.-bsaes_xts_decrypt | ||
2449 | ___ | ||
2450 | } | ||
2451 | $code.=<<___; | ||
2452 | #endif | ||
2453 | ___ | ||
2454 | |||
2455 | $code =~ s/\`([^\`]*)\`/eval($1)/gem; | ||
2456 | |||
2457 | open SELF,$0; | ||
2458 | while(<SELF>) { | ||
2459 | next if (/^#!/); | ||
2460 | last if (!s/^#/@/ and !/^$/); | ||
2461 | print; | ||
2462 | } | ||
2463 | close SELF; | ||
2464 | |||
2465 | print $code; | ||
2466 | |||
2467 | close STDOUT; | ||