// SPDX-License-Identifier: GPL-2.0-or-later1/*2* sm3-neon-core.S - SM3 secure hash using NEON instructions3*4* Linux/arm64 port of the libgcrypt SM3 implementation for AArch645*6* Copyright (C) 2021 Jussi Kivilinna <[email protected]>7* Copyright (c) 2022 Tianjia Zhang <[email protected]>8*/910#include <linux/linkage.h>11#include <linux/cfi_types.h>12#include <asm/assembler.h>1314/* Context structure */1516#define state_h0 017#define state_h1 418#define state_h2 819#define state_h3 1220#define state_h4 1621#define state_h5 2022#define state_h6 2423#define state_h7 282425/* Stack structure */2627#define STACK_W_SIZE (32 * 2 * 3)2829#define STACK_W (0)30#define STACK_SIZE (STACK_W + STACK_W_SIZE)3132/* Register macros */3334#define RSTATE x035#define RDATA x136#define RNBLKS x237#define RKPTR x2838#define RFRAME x293940#define ra w341#define rb w442#define rc w543#define rd w644#define re w745#define rf w846#define rg w947#define rh w104849#define t0 w1150#define t1 w1251#define t2 w1352#define t3 w1453#define t4 w1554#define t5 w1655#define t6 w175657#define k_even w1958#define k_odd w205960#define addr0 x2161#define addr1 x226263#define s0 w2364#define s1 w2465#define s2 w2566#define s3 w266768#define W0 v069#define W1 v170#define W2 v271#define W3 v372#define W4 v473#define W5 v57475#define XTMP0 v676#define XTMP1 v777#define XTMP2 v1678#define XTMP3 v1779#define XTMP4 v1880#define XTMP5 v1981#define XTMP6 v208283/* Helper macros. */8485#define _(...) /*_*/8687#define clear_vec(x) \88movi x.8h, #0;8990#define rolw(o, a, n) \91ror o, a, #(32 - n);9293/* Round function macros. */9495#define GG1_1(x, y, z, o, t) \96eor o, x, y;97#define GG1_2(x, y, z, o, t) \98eor o, o, z;99#define GG1_3(x, y, z, o, t)100101#define FF1_1(x, y, z, o, t) GG1_1(x, y, z, o, t)102#define FF1_2(x, y, z, o, t)103#define FF1_3(x, y, z, o, t) GG1_2(x, y, z, o, t)104105#define GG2_1(x, y, z, o, t) \106bic o, z, x;107#define GG2_2(x, y, z, o, t) \108and t, y, x;109#define GG2_3(x, y, z, o, t) \110eor o, o, t;111112#define FF2_1(x, y, z, o, t) \113eor o, x, y;114#define FF2_2(x, y, z, o, t) \115and t, x, y; \116and o, o, z;117#define FF2_3(x, y, z, o, t) \118eor o, o, t;119120#define R(i, a, b, c, d, e, f, g, h, k, K_LOAD, round, widx, wtype, IOP, iop_param) \121K_LOAD(round); \122ldr t5, [sp, #(wtype##_W1_ADDR(round, widx))]; \123rolw(t0, a, 12); /* rol(a, 12) => t0 */ \124IOP(1, iop_param); \125FF##i##_1(a, b, c, t1, t2); \126ldr t6, [sp, #(wtype##_W1W2_ADDR(round, widx))]; \127add k, k, e; \128IOP(2, iop_param); \129GG##i##_1(e, f, g, t3, t4); \130FF##i##_2(a, b, c, t1, t2); \131IOP(3, iop_param); \132add k, k, t0; \133add h, h, t5; \134add d, d, t6; /* w1w2 + d => d */ \135IOP(4, iop_param); \136rolw(k, k, 7); /* rol (t0 + e + t), 7) => k */ \137GG##i##_2(e, f, g, t3, t4); \138add h, h, k; /* h + w1 + k => h */ \139IOP(5, iop_param); \140FF##i##_3(a, b, c, t1, t2); \141eor t0, t0, k; /* k ^ t0 => t0 */ \142GG##i##_3(e, f, g, t3, t4); \143add d, d, t1; /* FF(a,b,c) + d => d */ \144IOP(6, iop_param); \145add t3, t3, h; /* GG(e,f,g) + h => t3 */ \146rolw(b, b, 9); /* rol(b, 9) => b */ \147eor h, t3, t3, ror #(32-9); \148IOP(7, iop_param); \149add d, d, t0; /* t0 + d => d */ \150rolw(f, f, 19); /* rol(f, 19) => f */ \151IOP(8, iop_param); \152eor h, h, t3, ror #(32-17); /* P0(t3) => h */153154#define R1(a, b, c, d, e, f, g, h, k, K_LOAD, round, widx, wtype, IOP, iop_param) \155R(1, ##a, ##b, ##c, ##d, ##e, ##f, ##g, ##h, ##k, K_LOAD, round, widx, wtype, IOP, iop_param)156157#define R2(a, b, c, d, e, f, g, h, k, K_LOAD, round, widx, wtype, IOP, iop_param) \158R(2, ##a, ##b, ##c, ##d, ##e, ##f, ##g, ##h, ##k, K_LOAD, round, widx, wtype, IOP, iop_param)159160#define KL(round) \161ldp k_even, k_odd, [RKPTR, #(4*(round))];162163/* Input expansion macros. */164165/* Byte-swapped input address. */166#define IW_W_ADDR(round, widx, offs) \167(STACK_W + ((round) / 4) * 64 + (offs) + ((widx) * 4))168169/* Expanded input address. */170#define XW_W_ADDR(round, widx, offs) \171(STACK_W + ((((round) / 3) - 4) % 2) * 64 + (offs) + ((widx) * 4))172173/* Rounds 1-12, byte-swapped input block addresses. */174#define IW_W1_ADDR(round, widx) IW_W_ADDR(round, widx, 32)175#define IW_W1W2_ADDR(round, widx) IW_W_ADDR(round, widx, 48)176177/* Rounds 1-12, expanded input block addresses. */178#define XW_W1_ADDR(round, widx) XW_W_ADDR(round, widx, 0)179#define XW_W1W2_ADDR(round, widx) XW_W_ADDR(round, widx, 16)180181/* Input block loading.182* Interleaving within round function needed for in-order CPUs. */183#define LOAD_W_VEC_1_1() \184add addr0, sp, #IW_W1_ADDR(0, 0);185#define LOAD_W_VEC_1_2() \186add addr1, sp, #IW_W1_ADDR(4, 0);187#define LOAD_W_VEC_1_3() \188ld1 {W0.16b}, [RDATA], #16;189#define LOAD_W_VEC_1_4() \190ld1 {W1.16b}, [RDATA], #16;191#define LOAD_W_VEC_1_5() \192ld1 {W2.16b}, [RDATA], #16;193#define LOAD_W_VEC_1_6() \194ld1 {W3.16b}, [RDATA], #16;195#define LOAD_W_VEC_1_7() \196rev32 XTMP0.16b, W0.16b;197#define LOAD_W_VEC_1_8() \198rev32 XTMP1.16b, W1.16b;199#define LOAD_W_VEC_2_1() \200rev32 XTMP2.16b, W2.16b;201#define LOAD_W_VEC_2_2() \202rev32 XTMP3.16b, W3.16b;203#define LOAD_W_VEC_2_3() \204eor XTMP4.16b, XTMP1.16b, XTMP0.16b;205#define LOAD_W_VEC_2_4() \206eor XTMP5.16b, XTMP2.16b, XTMP1.16b;207#define LOAD_W_VEC_2_5() \208st1 {XTMP0.16b}, [addr0], #16;209#define LOAD_W_VEC_2_6() \210st1 {XTMP4.16b}, [addr0]; \211add addr0, sp, #IW_W1_ADDR(8, 0);212#define LOAD_W_VEC_2_7() \213eor XTMP6.16b, XTMP3.16b, XTMP2.16b;214#define LOAD_W_VEC_2_8() \215ext W0.16b, XTMP0.16b, XTMP0.16b, #8; /* W0: xx, w0, xx, xx */216#define LOAD_W_VEC_3_1() \217mov W2.16b, XTMP1.16b; /* W2: xx, w6, w5, w4 */218#define LOAD_W_VEC_3_2() \219st1 {XTMP1.16b}, [addr1], #16;220#define LOAD_W_VEC_3_3() \221st1 {XTMP5.16b}, [addr1]; \222ext W1.16b, XTMP0.16b, XTMP0.16b, #4; /* W1: xx, w3, w2, w1 */223#define LOAD_W_VEC_3_4() \224ext W3.16b, XTMP1.16b, XTMP2.16b, #12; /* W3: xx, w9, w8, w7 */225#define LOAD_W_VEC_3_5() \226ext W4.16b, XTMP2.16b, XTMP3.16b, #8; /* W4: xx, w12, w11, w10 */227#define LOAD_W_VEC_3_6() \228st1 {XTMP2.16b}, [addr0], #16;229#define LOAD_W_VEC_3_7() \230st1 {XTMP6.16b}, [addr0];231#define LOAD_W_VEC_3_8() \232ext W5.16b, XTMP3.16b, XTMP3.16b, #4; /* W5: xx, w15, w14, w13 */233234#define LOAD_W_VEC_1(iop_num, ...) \235LOAD_W_VEC_1_##iop_num()236#define LOAD_W_VEC_2(iop_num, ...) \237LOAD_W_VEC_2_##iop_num()238#define LOAD_W_VEC_3(iop_num, ...) \239LOAD_W_VEC_3_##iop_num()240241/* Message scheduling. Note: 3 words per vector register.242* Interleaving within round function needed for in-order CPUs. */243#define SCHED_W_1_1(round, w0, w1, w2, w3, w4, w5) \244/* Load (w[i - 16]) => XTMP0 */ \245/* Load (w[i - 13]) => XTMP5 */ \246ext XTMP0.16b, w0.16b, w0.16b, #12; /* XTMP0: w0, xx, xx, xx */247#define SCHED_W_1_2(round, w0, w1, w2, w3, w4, w5) \248ext XTMP5.16b, w1.16b, w1.16b, #12;249#define SCHED_W_1_3(round, w0, w1, w2, w3, w4, w5) \250ext XTMP0.16b, XTMP0.16b, w1.16b, #12; /* XTMP0: xx, w2, w1, w0 */251#define SCHED_W_1_4(round, w0, w1, w2, w3, w4, w5) \252ext XTMP5.16b, XTMP5.16b, w2.16b, #12;253#define SCHED_W_1_5(round, w0, w1, w2, w3, w4, w5) \254/* w[i - 9] == w3 */ \255/* W3 ^ XTMP0 => XTMP0 */ \256eor XTMP0.16b, XTMP0.16b, w3.16b;257#define SCHED_W_1_6(round, w0, w1, w2, w3, w4, w5) \258/* w[i - 3] == w5 */ \259/* rol(XMM5, 15) ^ XTMP0 => XTMP0 */ \260/* rol(XTMP5, 7) => XTMP1 */ \261add addr0, sp, #XW_W1_ADDR((round), 0); \262shl XTMP2.4s, w5.4s, #15;263#define SCHED_W_1_7(round, w0, w1, w2, w3, w4, w5) \264shl XTMP1.4s, XTMP5.4s, #7;265#define SCHED_W_1_8(round, w0, w1, w2, w3, w4, w5) \266sri XTMP2.4s, w5.4s, #(32-15);267#define SCHED_W_2_1(round, w0, w1, w2, w3, w4, w5) \268sri XTMP1.4s, XTMP5.4s, #(32-7);269#define SCHED_W_2_2(round, w0, w1, w2, w3, w4, w5) \270eor XTMP0.16b, XTMP0.16b, XTMP2.16b;271#define SCHED_W_2_3(round, w0, w1, w2, w3, w4, w5) \272/* w[i - 6] == W4 */ \273/* W4 ^ XTMP1 => XTMP1 */ \274eor XTMP1.16b, XTMP1.16b, w4.16b;275#define SCHED_W_2_4(round, w0, w1, w2, w3, w4, w5) \276/* P1(XTMP0) ^ XTMP1 => W0 */ \277shl XTMP3.4s, XTMP0.4s, #15;278#define SCHED_W_2_5(round, w0, w1, w2, w3, w4, w5) \279shl XTMP4.4s, XTMP0.4s, #23;280#define SCHED_W_2_6(round, w0, w1, w2, w3, w4, w5) \281eor w0.16b, XTMP1.16b, XTMP0.16b;282#define SCHED_W_2_7(round, w0, w1, w2, w3, w4, w5) \283sri XTMP3.4s, XTMP0.4s, #(32-15);284#define SCHED_W_2_8(round, w0, w1, w2, w3, w4, w5) \285sri XTMP4.4s, XTMP0.4s, #(32-23);286#define SCHED_W_3_1(round, w0, w1, w2, w3, w4, w5) \287eor w0.16b, w0.16b, XTMP3.16b;288#define SCHED_W_3_2(round, w0, w1, w2, w3, w4, w5) \289/* Load (w[i - 3]) => XTMP2 */ \290ext XTMP2.16b, w4.16b, w4.16b, #12;291#define SCHED_W_3_3(round, w0, w1, w2, w3, w4, w5) \292eor w0.16b, w0.16b, XTMP4.16b;293#define SCHED_W_3_4(round, w0, w1, w2, w3, w4, w5) \294ext XTMP2.16b, XTMP2.16b, w5.16b, #12;295#define SCHED_W_3_5(round, w0, w1, w2, w3, w4, w5) \296/* W1 ^ W2 => XTMP3 */ \297eor XTMP3.16b, XTMP2.16b, w0.16b;298#define SCHED_W_3_6(round, w0, w1, w2, w3, w4, w5)299#define SCHED_W_3_7(round, w0, w1, w2, w3, w4, w5) \300st1 {XTMP2.16b-XTMP3.16b}, [addr0];301#define SCHED_W_3_8(round, w0, w1, w2, w3, w4, w5)302303#define SCHED_W_W0W1W2W3W4W5_1(iop_num, round) \304SCHED_W_1_##iop_num(round, W0, W1, W2, W3, W4, W5)305#define SCHED_W_W0W1W2W3W4W5_2(iop_num, round) \306SCHED_W_2_##iop_num(round, W0, W1, W2, W3, W4, W5)307#define SCHED_W_W0W1W2W3W4W5_3(iop_num, round) \308SCHED_W_3_##iop_num(round, W0, W1, W2, W3, W4, W5)309310#define SCHED_W_W1W2W3W4W5W0_1(iop_num, round) \311SCHED_W_1_##iop_num(round, W1, W2, W3, W4, W5, W0)312#define SCHED_W_W1W2W3W4W5W0_2(iop_num, round) \313SCHED_W_2_##iop_num(round, W1, W2, W3, W4, W5, W0)314#define SCHED_W_W1W2W3W4W5W0_3(iop_num, round) \315SCHED_W_3_##iop_num(round, W1, W2, W3, W4, W5, W0)316317#define SCHED_W_W2W3W4W5W0W1_1(iop_num, round) \318SCHED_W_1_##iop_num(round, W2, W3, W4, W5, W0, W1)319#define SCHED_W_W2W3W4W5W0W1_2(iop_num, round) \320SCHED_W_2_##iop_num(round, W2, W3, W4, W5, W0, W1)321#define SCHED_W_W2W3W4W5W0W1_3(iop_num, round) \322SCHED_W_3_##iop_num(round, W2, W3, W4, W5, W0, W1)323324#define SCHED_W_W3W4W5W0W1W2_1(iop_num, round) \325SCHED_W_1_##iop_num(round, W3, W4, W5, W0, W1, W2)326#define SCHED_W_W3W4W5W0W1W2_2(iop_num, round) \327SCHED_W_2_##iop_num(round, W3, W4, W5, W0, W1, W2)328#define SCHED_W_W3W4W5W0W1W2_3(iop_num, round) \329SCHED_W_3_##iop_num(round, W3, W4, W5, W0, W1, W2)330331#define SCHED_W_W4W5W0W1W2W3_1(iop_num, round) \332SCHED_W_1_##iop_num(round, W4, W5, W0, W1, W2, W3)333#define SCHED_W_W4W5W0W1W2W3_2(iop_num, round) \334SCHED_W_2_##iop_num(round, W4, W5, W0, W1, W2, W3)335#define SCHED_W_W4W5W0W1W2W3_3(iop_num, round) \336SCHED_W_3_##iop_num(round, W4, W5, W0, W1, W2, W3)337338#define SCHED_W_W5W0W1W2W3W4_1(iop_num, round) \339SCHED_W_1_##iop_num(round, W5, W0, W1, W2, W3, W4)340#define SCHED_W_W5W0W1W2W3W4_2(iop_num, round) \341SCHED_W_2_##iop_num(round, W5, W0, W1, W2, W3, W4)342#define SCHED_W_W5W0W1W2W3W4_3(iop_num, round) \343SCHED_W_3_##iop_num(round, W5, W0, W1, W2, W3, W4)344345346/*347* Transform blocks*64 bytes (blocks*16 32-bit words) at 'src'.348*349* void sm3_neon_transform(struct sm3_state *sst, u8 const *src,350* int blocks)351*/352.text353.align 3354SYM_TYPED_FUNC_START(sm3_neon_transform)355ldp ra, rb, [RSTATE, #0]356ldp rc, rd, [RSTATE, #8]357ldp re, rf, [RSTATE, #16]358ldp rg, rh, [RSTATE, #24]359360stp x28, x29, [sp, #-16]!361stp x19, x20, [sp, #-16]!362stp x21, x22, [sp, #-16]!363stp x23, x24, [sp, #-16]!364stp x25, x26, [sp, #-16]!365mov RFRAME, sp366367sub addr0, sp, #STACK_SIZE368adr_l RKPTR, .LKtable369and sp, addr0, #(~63)370371/* Preload first block. */372LOAD_W_VEC_1(1, 0)373LOAD_W_VEC_1(2, 0)374LOAD_W_VEC_1(3, 0)375LOAD_W_VEC_1(4, 0)376LOAD_W_VEC_1(5, 0)377LOAD_W_VEC_1(6, 0)378LOAD_W_VEC_1(7, 0)379LOAD_W_VEC_1(8, 0)380LOAD_W_VEC_2(1, 0)381LOAD_W_VEC_2(2, 0)382LOAD_W_VEC_2(3, 0)383LOAD_W_VEC_2(4, 0)384LOAD_W_VEC_2(5, 0)385LOAD_W_VEC_2(6, 0)386LOAD_W_VEC_2(7, 0)387LOAD_W_VEC_2(8, 0)388LOAD_W_VEC_3(1, 0)389LOAD_W_VEC_3(2, 0)390LOAD_W_VEC_3(3, 0)391LOAD_W_VEC_3(4, 0)392LOAD_W_VEC_3(5, 0)393LOAD_W_VEC_3(6, 0)394LOAD_W_VEC_3(7, 0)395LOAD_W_VEC_3(8, 0)396397.balign 16398.Loop:399/* Transform 0-3 */400R1(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 0, 0, IW, _, 0)401R1(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 1, 1, IW, _, 0)402R1(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 2, 2, IW, _, 0)403R1(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 3, 3, IW, _, 0)404405/* Transform 4-7 + Precalc 12-14 */406R1(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 4, 0, IW, _, 0)407R1(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 5, 1, IW, _, 0)408R1(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 6, 2, IW, SCHED_W_W0W1W2W3W4W5_1, 12)409R1(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 7, 3, IW, SCHED_W_W0W1W2W3W4W5_2, 12)410411/* Transform 8-11 + Precalc 12-17 */412R1(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 8, 0, IW, SCHED_W_W0W1W2W3W4W5_3, 12)413R1(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 9, 1, IW, SCHED_W_W1W2W3W4W5W0_1, 15)414R1(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 10, 2, IW, SCHED_W_W1W2W3W4W5W0_2, 15)415R1(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 11, 3, IW, SCHED_W_W1W2W3W4W5W0_3, 15)416417/* Transform 12-14 + Precalc 18-20 */418R1(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 12, 0, XW, SCHED_W_W2W3W4W5W0W1_1, 18)419R1(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 13, 1, XW, SCHED_W_W2W3W4W5W0W1_2, 18)420R1(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 14, 2, XW, SCHED_W_W2W3W4W5W0W1_3, 18)421422/* Transform 15-17 + Precalc 21-23 */423R1(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 15, 0, XW, SCHED_W_W3W4W5W0W1W2_1, 21)424R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 16, 1, XW, SCHED_W_W3W4W5W0W1W2_2, 21)425R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 17, 2, XW, SCHED_W_W3W4W5W0W1W2_3, 21)426427/* Transform 18-20 + Precalc 24-26 */428R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 18, 0, XW, SCHED_W_W4W5W0W1W2W3_1, 24)429R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 19, 1, XW, SCHED_W_W4W5W0W1W2W3_2, 24)430R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 20, 2, XW, SCHED_W_W4W5W0W1W2W3_3, 24)431432/* Transform 21-23 + Precalc 27-29 */433R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 21, 0, XW, SCHED_W_W5W0W1W2W3W4_1, 27)434R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 22, 1, XW, SCHED_W_W5W0W1W2W3W4_2, 27)435R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 23, 2, XW, SCHED_W_W5W0W1W2W3W4_3, 27)436437/* Transform 24-26 + Precalc 30-32 */438R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 24, 0, XW, SCHED_W_W0W1W2W3W4W5_1, 30)439R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 25, 1, XW, SCHED_W_W0W1W2W3W4W5_2, 30)440R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 26, 2, XW, SCHED_W_W0W1W2W3W4W5_3, 30)441442/* Transform 27-29 + Precalc 33-35 */443R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 27, 0, XW, SCHED_W_W1W2W3W4W5W0_1, 33)444R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 28, 1, XW, SCHED_W_W1W2W3W4W5W0_2, 33)445R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 29, 2, XW, SCHED_W_W1W2W3W4W5W0_3, 33)446447/* Transform 30-32 + Precalc 36-38 */448R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 30, 0, XW, SCHED_W_W2W3W4W5W0W1_1, 36)449R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 31, 1, XW, SCHED_W_W2W3W4W5W0W1_2, 36)450R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 32, 2, XW, SCHED_W_W2W3W4W5W0W1_3, 36)451452/* Transform 33-35 + Precalc 39-41 */453R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 33, 0, XW, SCHED_W_W3W4W5W0W1W2_1, 39)454R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 34, 1, XW, SCHED_W_W3W4W5W0W1W2_2, 39)455R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 35, 2, XW, SCHED_W_W3W4W5W0W1W2_3, 39)456457/* Transform 36-38 + Precalc 42-44 */458R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 36, 0, XW, SCHED_W_W4W5W0W1W2W3_1, 42)459R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 37, 1, XW, SCHED_W_W4W5W0W1W2W3_2, 42)460R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 38, 2, XW, SCHED_W_W4W5W0W1W2W3_3, 42)461462/* Transform 39-41 + Precalc 45-47 */463R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 39, 0, XW, SCHED_W_W5W0W1W2W3W4_1, 45)464R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 40, 1, XW, SCHED_W_W5W0W1W2W3W4_2, 45)465R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 41, 2, XW, SCHED_W_W5W0W1W2W3W4_3, 45)466467/* Transform 42-44 + Precalc 48-50 */468R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 42, 0, XW, SCHED_W_W0W1W2W3W4W5_1, 48)469R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 43, 1, XW, SCHED_W_W0W1W2W3W4W5_2, 48)470R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 44, 2, XW, SCHED_W_W0W1W2W3W4W5_3, 48)471472/* Transform 45-47 + Precalc 51-53 */473R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 45, 0, XW, SCHED_W_W1W2W3W4W5W0_1, 51)474R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 46, 1, XW, SCHED_W_W1W2W3W4W5W0_2, 51)475R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 47, 2, XW, SCHED_W_W1W2W3W4W5W0_3, 51)476477/* Transform 48-50 + Precalc 54-56 */478R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 48, 0, XW, SCHED_W_W2W3W4W5W0W1_1, 54)479R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 49, 1, XW, SCHED_W_W2W3W4W5W0W1_2, 54)480R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 50, 2, XW, SCHED_W_W2W3W4W5W0W1_3, 54)481482/* Transform 51-53 + Precalc 57-59 */483R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 51, 0, XW, SCHED_W_W3W4W5W0W1W2_1, 57)484R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 52, 1, XW, SCHED_W_W3W4W5W0W1W2_2, 57)485R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 53, 2, XW, SCHED_W_W3W4W5W0W1W2_3, 57)486487/* Transform 54-56 + Precalc 60-62 */488R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 54, 0, XW, SCHED_W_W4W5W0W1W2W3_1, 60)489R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 55, 1, XW, SCHED_W_W4W5W0W1W2W3_2, 60)490R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 56, 2, XW, SCHED_W_W4W5W0W1W2W3_3, 60)491492/* Transform 57-59 + Precalc 63 */493R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 57, 0, XW, SCHED_W_W5W0W1W2W3W4_1, 63)494R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 58, 1, XW, SCHED_W_W5W0W1W2W3W4_2, 63)495R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 59, 2, XW, SCHED_W_W5W0W1W2W3W4_3, 63)496497/* Transform 60 */498R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 60, 0, XW, _, _)499subs RNBLKS, RNBLKS, #1500b.eq .Lend501502/* Transform 61-63 + Preload next block */503R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 61, 1, XW, LOAD_W_VEC_1, _)504ldp s0, s1, [RSTATE, #0]505R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 62, 2, XW, LOAD_W_VEC_2, _)506ldp s2, s3, [RSTATE, #8]507R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 63, 0, XW, LOAD_W_VEC_3, _)508509/* Update the chaining variables. */510eor ra, ra, s0511eor rb, rb, s1512ldp s0, s1, [RSTATE, #16]513eor rc, rc, s2514ldp k_even, k_odd, [RSTATE, #24]515eor rd, rd, s3516eor re, re, s0517stp ra, rb, [RSTATE, #0]518eor rf, rf, s1519stp rc, rd, [RSTATE, #8]520eor rg, rg, k_even521stp re, rf, [RSTATE, #16]522eor rh, rh, k_odd523stp rg, rh, [RSTATE, #24]524b .Loop525526.Lend:527/* Transform 61-63 */528R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 61, 1, XW, _, _)529ldp s0, s1, [RSTATE, #0]530R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 62, 2, XW, _, _)531ldp s2, s3, [RSTATE, #8]532R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 63, 0, XW, _, _)533534/* Update the chaining variables. */535eor ra, ra, s0536clear_vec(W0)537eor rb, rb, s1538clear_vec(W1)539ldp s0, s1, [RSTATE, #16]540clear_vec(W2)541eor rc, rc, s2542clear_vec(W3)543ldp k_even, k_odd, [RSTATE, #24]544clear_vec(W4)545eor rd, rd, s3546clear_vec(W5)547eor re, re, s0548clear_vec(XTMP0)549stp ra, rb, [RSTATE, #0]550clear_vec(XTMP1)551eor rf, rf, s1552clear_vec(XTMP2)553stp rc, rd, [RSTATE, #8]554clear_vec(XTMP3)555eor rg, rg, k_even556clear_vec(XTMP4)557stp re, rf, [RSTATE, #16]558clear_vec(XTMP5)559eor rh, rh, k_odd560clear_vec(XTMP6)561stp rg, rh, [RSTATE, #24]562563/* Clear message expansion area */564add addr0, sp, #STACK_W565st1 {W0.16b-W3.16b}, [addr0], #64566st1 {W0.16b-W3.16b}, [addr0], #64567st1 {W0.16b-W3.16b}, [addr0]568569mov sp, RFRAME570571ldp x25, x26, [sp], #16572ldp x23, x24, [sp], #16573ldp x21, x22, [sp], #16574ldp x19, x20, [sp], #16575ldp x28, x29, [sp], #16576577ret578SYM_FUNC_END(sm3_neon_transform)579580581.section ".rodata", "a"582583.align 4584.LKtable:585.long 0x79cc4519, 0xf3988a32, 0xe7311465, 0xce6228cb586.long 0x9cc45197, 0x3988a32f, 0x7311465e, 0xe6228cbc587.long 0xcc451979, 0x988a32f3, 0x311465e7, 0x6228cbce588.long 0xc451979c, 0x88a32f39, 0x11465e73, 0x228cbce6589.long 0x9d8a7a87, 0x3b14f50f, 0x7629ea1e, 0xec53d43c590.long 0xd8a7a879, 0xb14f50f3, 0x629ea1e7, 0xc53d43ce591.long 0x8a7a879d, 0x14f50f3b, 0x29ea1e76, 0x53d43cec592.long 0xa7a879d8, 0x4f50f3b1, 0x9ea1e762, 0x3d43cec5593.long 0x7a879d8a, 0xf50f3b14, 0xea1e7629, 0xd43cec53594.long 0xa879d8a7, 0x50f3b14f, 0xa1e7629e, 0x43cec53d595.long 0x879d8a7a, 0x0f3b14f5, 0x1e7629ea, 0x3cec53d4596.long 0x79d8a7a8, 0xf3b14f50, 0xe7629ea1, 0xcec53d43597.long 0x9d8a7a87, 0x3b14f50f, 0x7629ea1e, 0xec53d43c598.long 0xd8a7a879, 0xb14f50f3, 0x629ea1e7, 0xc53d43ce599.long 0x8a7a879d, 0x14f50f3b, 0x29ea1e76, 0x53d43cec600.long 0xa7a879d8, 0x4f50f3b1, 0x9ea1e762, 0x3d43cec5601602603