// 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 <asm/assembler.h>1213/* Context structure */1415#define state_h0 016#define state_h1 417#define state_h2 818#define state_h3 1219#define state_h4 1620#define state_h5 2021#define state_h6 2422#define state_h7 282324/* Stack structure */2526#define STACK_W_SIZE (32 * 2 * 3)2728#define STACK_W (0)29#define STACK_SIZE (STACK_W + STACK_W_SIZE)3031/* Register macros */3233#define RSTATE x034#define RDATA x135#define RNBLKS x236#define RKPTR x2837#define RFRAME x293839#define ra w340#define rb w441#define rc w542#define rd w643#define re w744#define rf w845#define rg w946#define rh w104748#define t0 w1149#define t1 w1250#define t2 w1351#define t3 w1452#define t4 w1553#define t5 w1654#define t6 w175556#define k_even w1957#define k_odd w205859#define addr0 x2160#define addr1 x226162#define s0 w2363#define s1 w2464#define s2 w2565#define s3 w266667#define W0 v068#define W1 v169#define W2 v270#define W3 v371#define W4 v472#define W5 v57374#define XTMP0 v675#define XTMP1 v776#define XTMP2 v1677#define XTMP3 v1778#define XTMP4 v1879#define XTMP5 v1980#define XTMP6 v208182/* Helper macros. */8384#define _(...) /*_*/8586#define clear_vec(x) \87movi x.8h, #0;8889#define rolw(o, a, n) \90ror o, a, #(32 - n);9192/* Round function macros. */9394#define GG1_1(x, y, z, o, t) \95eor o, x, y;96#define GG1_2(x, y, z, o, t) \97eor o, o, z;98#define GG1_3(x, y, z, o, t)99100#define FF1_1(x, y, z, o, t) GG1_1(x, y, z, o, t)101#define FF1_2(x, y, z, o, t)102#define FF1_3(x, y, z, o, t) GG1_2(x, y, z, o, t)103104#define GG2_1(x, y, z, o, t) \105bic o, z, x;106#define GG2_2(x, y, z, o, t) \107and t, y, x;108#define GG2_3(x, y, z, o, t) \109eor o, o, t;110111#define FF2_1(x, y, z, o, t) \112eor o, x, y;113#define FF2_2(x, y, z, o, t) \114and t, x, y; \115and o, o, z;116#define FF2_3(x, y, z, o, t) \117eor o, o, t;118119#define R(i, a, b, c, d, e, f, g, h, k, K_LOAD, round, widx, wtype, IOP, iop_param) \120K_LOAD(round); \121ldr t5, [sp, #(wtype##_W1_ADDR(round, widx))]; \122rolw(t0, a, 12); /* rol(a, 12) => t0 */ \123IOP(1, iop_param); \124FF##i##_1(a, b, c, t1, t2); \125ldr t6, [sp, #(wtype##_W1W2_ADDR(round, widx))]; \126add k, k, e; \127IOP(2, iop_param); \128GG##i##_1(e, f, g, t3, t4); \129FF##i##_2(a, b, c, t1, t2); \130IOP(3, iop_param); \131add k, k, t0; \132add h, h, t5; \133add d, d, t6; /* w1w2 + d => d */ \134IOP(4, iop_param); \135rolw(k, k, 7); /* rol (t0 + e + t), 7) => k */ \136GG##i##_2(e, f, g, t3, t4); \137add h, h, k; /* h + w1 + k => h */ \138IOP(5, iop_param); \139FF##i##_3(a, b, c, t1, t2); \140eor t0, t0, k; /* k ^ t0 => t0 */ \141GG##i##_3(e, f, g, t3, t4); \142add d, d, t1; /* FF(a,b,c) + d => d */ \143IOP(6, iop_param); \144add t3, t3, h; /* GG(e,f,g) + h => t3 */ \145rolw(b, b, 9); /* rol(b, 9) => b */ \146eor h, t3, t3, ror #(32-9); \147IOP(7, iop_param); \148add d, d, t0; /* t0 + d => d */ \149rolw(f, f, 19); /* rol(f, 19) => f */ \150IOP(8, iop_param); \151eor h, h, t3, ror #(32-17); /* P0(t3) => h */152153#define R1(a, b, c, d, e, f, g, h, k, K_LOAD, round, widx, wtype, IOP, iop_param) \154R(1, ##a, ##b, ##c, ##d, ##e, ##f, ##g, ##h, ##k, K_LOAD, round, widx, wtype, IOP, iop_param)155156#define R2(a, b, c, d, e, f, g, h, k, K_LOAD, round, widx, wtype, IOP, iop_param) \157R(2, ##a, ##b, ##c, ##d, ##e, ##f, ##g, ##h, ##k, K_LOAD, round, widx, wtype, IOP, iop_param)158159#define KL(round) \160ldp k_even, k_odd, [RKPTR, #(4*(round))];161162/* Input expansion macros. */163164/* Byte-swapped input address. */165#define IW_W_ADDR(round, widx, offs) \166(STACK_W + ((round) / 4) * 64 + (offs) + ((widx) * 4))167168/* Expanded input address. */169#define XW_W_ADDR(round, widx, offs) \170(STACK_W + ((((round) / 3) - 4) % 2) * 64 + (offs) + ((widx) * 4))171172/* Rounds 1-12, byte-swapped input block addresses. */173#define IW_W1_ADDR(round, widx) IW_W_ADDR(round, widx, 32)174#define IW_W1W2_ADDR(round, widx) IW_W_ADDR(round, widx, 48)175176/* Rounds 1-12, expanded input block addresses. */177#define XW_W1_ADDR(round, widx) XW_W_ADDR(round, widx, 0)178#define XW_W1W2_ADDR(round, widx) XW_W_ADDR(round, widx, 16)179180/* Input block loading.181* Interleaving within round function needed for in-order CPUs. */182#define LOAD_W_VEC_1_1() \183add addr0, sp, #IW_W1_ADDR(0, 0);184#define LOAD_W_VEC_1_2() \185add addr1, sp, #IW_W1_ADDR(4, 0);186#define LOAD_W_VEC_1_3() \187ld1 {W0.16b}, [RDATA], #16;188#define LOAD_W_VEC_1_4() \189ld1 {W1.16b}, [RDATA], #16;190#define LOAD_W_VEC_1_5() \191ld1 {W2.16b}, [RDATA], #16;192#define LOAD_W_VEC_1_6() \193ld1 {W3.16b}, [RDATA], #16;194#define LOAD_W_VEC_1_7() \195rev32 XTMP0.16b, W0.16b;196#define LOAD_W_VEC_1_8() \197rev32 XTMP1.16b, W1.16b;198#define LOAD_W_VEC_2_1() \199rev32 XTMP2.16b, W2.16b;200#define LOAD_W_VEC_2_2() \201rev32 XTMP3.16b, W3.16b;202#define LOAD_W_VEC_2_3() \203eor XTMP4.16b, XTMP1.16b, XTMP0.16b;204#define LOAD_W_VEC_2_4() \205eor XTMP5.16b, XTMP2.16b, XTMP1.16b;206#define LOAD_W_VEC_2_5() \207st1 {XTMP0.16b}, [addr0], #16;208#define LOAD_W_VEC_2_6() \209st1 {XTMP4.16b}, [addr0]; \210add addr0, sp, #IW_W1_ADDR(8, 0);211#define LOAD_W_VEC_2_7() \212eor XTMP6.16b, XTMP3.16b, XTMP2.16b;213#define LOAD_W_VEC_2_8() \214ext W0.16b, XTMP0.16b, XTMP0.16b, #8; /* W0: xx, w0, xx, xx */215#define LOAD_W_VEC_3_1() \216mov W2.16b, XTMP1.16b; /* W2: xx, w6, w5, w4 */217#define LOAD_W_VEC_3_2() \218st1 {XTMP1.16b}, [addr1], #16;219#define LOAD_W_VEC_3_3() \220st1 {XTMP5.16b}, [addr1]; \221ext W1.16b, XTMP0.16b, XTMP0.16b, #4; /* W1: xx, w3, w2, w1 */222#define LOAD_W_VEC_3_4() \223ext W3.16b, XTMP1.16b, XTMP2.16b, #12; /* W3: xx, w9, w8, w7 */224#define LOAD_W_VEC_3_5() \225ext W4.16b, XTMP2.16b, XTMP3.16b, #8; /* W4: xx, w12, w11, w10 */226#define LOAD_W_VEC_3_6() \227st1 {XTMP2.16b}, [addr0], #16;228#define LOAD_W_VEC_3_7() \229st1 {XTMP6.16b}, [addr0];230#define LOAD_W_VEC_3_8() \231ext W5.16b, XTMP3.16b, XTMP3.16b, #4; /* W5: xx, w15, w14, w13 */232233#define LOAD_W_VEC_1(iop_num, ...) \234LOAD_W_VEC_1_##iop_num()235#define LOAD_W_VEC_2(iop_num, ...) \236LOAD_W_VEC_2_##iop_num()237#define LOAD_W_VEC_3(iop_num, ...) \238LOAD_W_VEC_3_##iop_num()239240/* Message scheduling. Note: 3 words per vector register.241* Interleaving within round function needed for in-order CPUs. */242#define SCHED_W_1_1(round, w0, w1, w2, w3, w4, w5) \243/* Load (w[i - 16]) => XTMP0 */ \244/* Load (w[i - 13]) => XTMP5 */ \245ext XTMP0.16b, w0.16b, w0.16b, #12; /* XTMP0: w0, xx, xx, xx */246#define SCHED_W_1_2(round, w0, w1, w2, w3, w4, w5) \247ext XTMP5.16b, w1.16b, w1.16b, #12;248#define SCHED_W_1_3(round, w0, w1, w2, w3, w4, w5) \249ext XTMP0.16b, XTMP0.16b, w1.16b, #12; /* XTMP0: xx, w2, w1, w0 */250#define SCHED_W_1_4(round, w0, w1, w2, w3, w4, w5) \251ext XTMP5.16b, XTMP5.16b, w2.16b, #12;252#define SCHED_W_1_5(round, w0, w1, w2, w3, w4, w5) \253/* w[i - 9] == w3 */ \254/* W3 ^ XTMP0 => XTMP0 */ \255eor XTMP0.16b, XTMP0.16b, w3.16b;256#define SCHED_W_1_6(round, w0, w1, w2, w3, w4, w5) \257/* w[i - 3] == w5 */ \258/* rol(XMM5, 15) ^ XTMP0 => XTMP0 */ \259/* rol(XTMP5, 7) => XTMP1 */ \260add addr0, sp, #XW_W1_ADDR((round), 0); \261shl XTMP2.4s, w5.4s, #15;262#define SCHED_W_1_7(round, w0, w1, w2, w3, w4, w5) \263shl XTMP1.4s, XTMP5.4s, #7;264#define SCHED_W_1_8(round, w0, w1, w2, w3, w4, w5) \265sri XTMP2.4s, w5.4s, #(32-15);266#define SCHED_W_2_1(round, w0, w1, w2, w3, w4, w5) \267sri XTMP1.4s, XTMP5.4s, #(32-7);268#define SCHED_W_2_2(round, w0, w1, w2, w3, w4, w5) \269eor XTMP0.16b, XTMP0.16b, XTMP2.16b;270#define SCHED_W_2_3(round, w0, w1, w2, w3, w4, w5) \271/* w[i - 6] == W4 */ \272/* W4 ^ XTMP1 => XTMP1 */ \273eor XTMP1.16b, XTMP1.16b, w4.16b;274#define SCHED_W_2_4(round, w0, w1, w2, w3, w4, w5) \275/* P1(XTMP0) ^ XTMP1 => W0 */ \276shl XTMP3.4s, XTMP0.4s, #15;277#define SCHED_W_2_5(round, w0, w1, w2, w3, w4, w5) \278shl XTMP4.4s, XTMP0.4s, #23;279#define SCHED_W_2_6(round, w0, w1, w2, w3, w4, w5) \280eor w0.16b, XTMP1.16b, XTMP0.16b;281#define SCHED_W_2_7(round, w0, w1, w2, w3, w4, w5) \282sri XTMP3.4s, XTMP0.4s, #(32-15);283#define SCHED_W_2_8(round, w0, w1, w2, w3, w4, w5) \284sri XTMP4.4s, XTMP0.4s, #(32-23);285#define SCHED_W_3_1(round, w0, w1, w2, w3, w4, w5) \286eor w0.16b, w0.16b, XTMP3.16b;287#define SCHED_W_3_2(round, w0, w1, w2, w3, w4, w5) \288/* Load (w[i - 3]) => XTMP2 */ \289ext XTMP2.16b, w4.16b, w4.16b, #12;290#define SCHED_W_3_3(round, w0, w1, w2, w3, w4, w5) \291eor w0.16b, w0.16b, XTMP4.16b;292#define SCHED_W_3_4(round, w0, w1, w2, w3, w4, w5) \293ext XTMP2.16b, XTMP2.16b, w5.16b, #12;294#define SCHED_W_3_5(round, w0, w1, w2, w3, w4, w5) \295/* W1 ^ W2 => XTMP3 */ \296eor XTMP3.16b, XTMP2.16b, w0.16b;297#define SCHED_W_3_6(round, w0, w1, w2, w3, w4, w5)298#define SCHED_W_3_7(round, w0, w1, w2, w3, w4, w5) \299st1 {XTMP2.16b-XTMP3.16b}, [addr0];300#define SCHED_W_3_8(round, w0, w1, w2, w3, w4, w5)301302#define SCHED_W_W0W1W2W3W4W5_1(iop_num, round) \303SCHED_W_1_##iop_num(round, W0, W1, W2, W3, W4, W5)304#define SCHED_W_W0W1W2W3W4W5_2(iop_num, round) \305SCHED_W_2_##iop_num(round, W0, W1, W2, W3, W4, W5)306#define SCHED_W_W0W1W2W3W4W5_3(iop_num, round) \307SCHED_W_3_##iop_num(round, W0, W1, W2, W3, W4, W5)308309#define SCHED_W_W1W2W3W4W5W0_1(iop_num, round) \310SCHED_W_1_##iop_num(round, W1, W2, W3, W4, W5, W0)311#define SCHED_W_W1W2W3W4W5W0_2(iop_num, round) \312SCHED_W_2_##iop_num(round, W1, W2, W3, W4, W5, W0)313#define SCHED_W_W1W2W3W4W5W0_3(iop_num, round) \314SCHED_W_3_##iop_num(round, W1, W2, W3, W4, W5, W0)315316#define SCHED_W_W2W3W4W5W0W1_1(iop_num, round) \317SCHED_W_1_##iop_num(round, W2, W3, W4, W5, W0, W1)318#define SCHED_W_W2W3W4W5W0W1_2(iop_num, round) \319SCHED_W_2_##iop_num(round, W2, W3, W4, W5, W0, W1)320#define SCHED_W_W2W3W4W5W0W1_3(iop_num, round) \321SCHED_W_3_##iop_num(round, W2, W3, W4, W5, W0, W1)322323#define SCHED_W_W3W4W5W0W1W2_1(iop_num, round) \324SCHED_W_1_##iop_num(round, W3, W4, W5, W0, W1, W2)325#define SCHED_W_W3W4W5W0W1W2_2(iop_num, round) \326SCHED_W_2_##iop_num(round, W3, W4, W5, W0, W1, W2)327#define SCHED_W_W3W4W5W0W1W2_3(iop_num, round) \328SCHED_W_3_##iop_num(round, W3, W4, W5, W0, W1, W2)329330#define SCHED_W_W4W5W0W1W2W3_1(iop_num, round) \331SCHED_W_1_##iop_num(round, W4, W5, W0, W1, W2, W3)332#define SCHED_W_W4W5W0W1W2W3_2(iop_num, round) \333SCHED_W_2_##iop_num(round, W4, W5, W0, W1, W2, W3)334#define SCHED_W_W4W5W0W1W2W3_3(iop_num, round) \335SCHED_W_3_##iop_num(round, W4, W5, W0, W1, W2, W3)336337#define SCHED_W_W5W0W1W2W3W4_1(iop_num, round) \338SCHED_W_1_##iop_num(round, W5, W0, W1, W2, W3, W4)339#define SCHED_W_W5W0W1W2W3W4_2(iop_num, round) \340SCHED_W_2_##iop_num(round, W5, W0, W1, W2, W3, W4)341#define SCHED_W_W5W0W1W2W3W4_3(iop_num, round) \342SCHED_W_3_##iop_num(round, W5, W0, W1, W2, W3, W4)343344345/*346* Transform nblocks*64 bytes (nblocks*16 32-bit words) at 'data'.347*348* void sm3_neon_transform(struct sm3_block_state *state,349* const u8 *data, size_t nblocks)350*/351.text352.align 3353SYM_FUNC_START(sm3_neon_transform)354ldp ra, rb, [RSTATE, #0]355ldp rc, rd, [RSTATE, #8]356ldp re, rf, [RSTATE, #16]357ldp rg, rh, [RSTATE, #24]358359stp x28, x29, [sp, #-16]!360stp x19, x20, [sp, #-16]!361stp x21, x22, [sp, #-16]!362stp x23, x24, [sp, #-16]!363stp x25, x26, [sp, #-16]!364mov RFRAME, sp365366sub addr0, sp, #STACK_SIZE367adr_l RKPTR, .LKtable368and sp, addr0, #(~63)369370/* Preload first block. */371LOAD_W_VEC_1(1, 0)372LOAD_W_VEC_1(2, 0)373LOAD_W_VEC_1(3, 0)374LOAD_W_VEC_1(4, 0)375LOAD_W_VEC_1(5, 0)376LOAD_W_VEC_1(6, 0)377LOAD_W_VEC_1(7, 0)378LOAD_W_VEC_1(8, 0)379LOAD_W_VEC_2(1, 0)380LOAD_W_VEC_2(2, 0)381LOAD_W_VEC_2(3, 0)382LOAD_W_VEC_2(4, 0)383LOAD_W_VEC_2(5, 0)384LOAD_W_VEC_2(6, 0)385LOAD_W_VEC_2(7, 0)386LOAD_W_VEC_2(8, 0)387LOAD_W_VEC_3(1, 0)388LOAD_W_VEC_3(2, 0)389LOAD_W_VEC_3(3, 0)390LOAD_W_VEC_3(4, 0)391LOAD_W_VEC_3(5, 0)392LOAD_W_VEC_3(6, 0)393LOAD_W_VEC_3(7, 0)394LOAD_W_VEC_3(8, 0)395396.balign 16397.Loop:398/* Transform 0-3 */399R1(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 0, 0, IW, _, 0)400R1(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 1, 1, IW, _, 0)401R1(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 2, 2, IW, _, 0)402R1(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 3, 3, IW, _, 0)403404/* Transform 4-7 + Precalc 12-14 */405R1(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 4, 0, IW, _, 0)406R1(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 5, 1, IW, _, 0)407R1(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 6, 2, IW, SCHED_W_W0W1W2W3W4W5_1, 12)408R1(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 7, 3, IW, SCHED_W_W0W1W2W3W4W5_2, 12)409410/* Transform 8-11 + Precalc 12-17 */411R1(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 8, 0, IW, SCHED_W_W0W1W2W3W4W5_3, 12)412R1(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 9, 1, IW, SCHED_W_W1W2W3W4W5W0_1, 15)413R1(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 10, 2, IW, SCHED_W_W1W2W3W4W5W0_2, 15)414R1(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 11, 3, IW, SCHED_W_W1W2W3W4W5W0_3, 15)415416/* Transform 12-14 + Precalc 18-20 */417R1(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 12, 0, XW, SCHED_W_W2W3W4W5W0W1_1, 18)418R1(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 13, 1, XW, SCHED_W_W2W3W4W5W0W1_2, 18)419R1(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 14, 2, XW, SCHED_W_W2W3W4W5W0W1_3, 18)420421/* Transform 15-17 + Precalc 21-23 */422R1(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 15, 0, XW, SCHED_W_W3W4W5W0W1W2_1, 21)423R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 16, 1, XW, SCHED_W_W3W4W5W0W1W2_2, 21)424R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 17, 2, XW, SCHED_W_W3W4W5W0W1W2_3, 21)425426/* Transform 18-20 + Precalc 24-26 */427R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 18, 0, XW, SCHED_W_W4W5W0W1W2W3_1, 24)428R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 19, 1, XW, SCHED_W_W4W5W0W1W2W3_2, 24)429R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 20, 2, XW, SCHED_W_W4W5W0W1W2W3_3, 24)430431/* Transform 21-23 + Precalc 27-29 */432R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 21, 0, XW, SCHED_W_W5W0W1W2W3W4_1, 27)433R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 22, 1, XW, SCHED_W_W5W0W1W2W3W4_2, 27)434R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 23, 2, XW, SCHED_W_W5W0W1W2W3W4_3, 27)435436/* Transform 24-26 + Precalc 30-32 */437R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 24, 0, XW, SCHED_W_W0W1W2W3W4W5_1, 30)438R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 25, 1, XW, SCHED_W_W0W1W2W3W4W5_2, 30)439R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 26, 2, XW, SCHED_W_W0W1W2W3W4W5_3, 30)440441/* Transform 27-29 + Precalc 33-35 */442R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 27, 0, XW, SCHED_W_W1W2W3W4W5W0_1, 33)443R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 28, 1, XW, SCHED_W_W1W2W3W4W5W0_2, 33)444R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 29, 2, XW, SCHED_W_W1W2W3W4W5W0_3, 33)445446/* Transform 30-32 + Precalc 36-38 */447R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 30, 0, XW, SCHED_W_W2W3W4W5W0W1_1, 36)448R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 31, 1, XW, SCHED_W_W2W3W4W5W0W1_2, 36)449R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 32, 2, XW, SCHED_W_W2W3W4W5W0W1_3, 36)450451/* Transform 33-35 + Precalc 39-41 */452R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 33, 0, XW, SCHED_W_W3W4W5W0W1W2_1, 39)453R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 34, 1, XW, SCHED_W_W3W4W5W0W1W2_2, 39)454R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 35, 2, XW, SCHED_W_W3W4W5W0W1W2_3, 39)455456/* Transform 36-38 + Precalc 42-44 */457R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 36, 0, XW, SCHED_W_W4W5W0W1W2W3_1, 42)458R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 37, 1, XW, SCHED_W_W4W5W0W1W2W3_2, 42)459R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 38, 2, XW, SCHED_W_W4W5W0W1W2W3_3, 42)460461/* Transform 39-41 + Precalc 45-47 */462R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 39, 0, XW, SCHED_W_W5W0W1W2W3W4_1, 45)463R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 40, 1, XW, SCHED_W_W5W0W1W2W3W4_2, 45)464R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 41, 2, XW, SCHED_W_W5W0W1W2W3W4_3, 45)465466/* Transform 42-44 + Precalc 48-50 */467R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 42, 0, XW, SCHED_W_W0W1W2W3W4W5_1, 48)468R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 43, 1, XW, SCHED_W_W0W1W2W3W4W5_2, 48)469R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 44, 2, XW, SCHED_W_W0W1W2W3W4W5_3, 48)470471/* Transform 45-47 + Precalc 51-53 */472R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 45, 0, XW, SCHED_W_W1W2W3W4W5W0_1, 51)473R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 46, 1, XW, SCHED_W_W1W2W3W4W5W0_2, 51)474R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 47, 2, XW, SCHED_W_W1W2W3W4W5W0_3, 51)475476/* Transform 48-50 + Precalc 54-56 */477R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 48, 0, XW, SCHED_W_W2W3W4W5W0W1_1, 54)478R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 49, 1, XW, SCHED_W_W2W3W4W5W0W1_2, 54)479R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 50, 2, XW, SCHED_W_W2W3W4W5W0W1_3, 54)480481/* Transform 51-53 + Precalc 57-59 */482R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 51, 0, XW, SCHED_W_W3W4W5W0W1W2_1, 57)483R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 52, 1, XW, SCHED_W_W3W4W5W0W1W2_2, 57)484R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 53, 2, XW, SCHED_W_W3W4W5W0W1W2_3, 57)485486/* Transform 54-56 + Precalc 60-62 */487R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 54, 0, XW, SCHED_W_W4W5W0W1W2W3_1, 60)488R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 55, 1, XW, SCHED_W_W4W5W0W1W2W3_2, 60)489R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 56, 2, XW, SCHED_W_W4W5W0W1W2W3_3, 60)490491/* Transform 57-59 + Precalc 63 */492R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 57, 0, XW, SCHED_W_W5W0W1W2W3W4_1, 63)493R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 58, 1, XW, SCHED_W_W5W0W1W2W3W4_2, 63)494R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 59, 2, XW, SCHED_W_W5W0W1W2W3W4_3, 63)495496/* Transform 60 */497R2(ra, rb, rc, rd, re, rf, rg, rh, k_even, KL, 60, 0, XW, _, _)498subs RNBLKS, RNBLKS, #1499b.eq .Lend500501/* Transform 61-63 + Preload next block */502R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 61, 1, XW, LOAD_W_VEC_1, _)503ldp s0, s1, [RSTATE, #0]504R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 62, 2, XW, LOAD_W_VEC_2, _)505ldp s2, s3, [RSTATE, #8]506R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 63, 0, XW, LOAD_W_VEC_3, _)507508/* Update the chaining variables. */509eor ra, ra, s0510eor rb, rb, s1511ldp s0, s1, [RSTATE, #16]512eor rc, rc, s2513ldp k_even, k_odd, [RSTATE, #24]514eor rd, rd, s3515eor re, re, s0516stp ra, rb, [RSTATE, #0]517eor rf, rf, s1518stp rc, rd, [RSTATE, #8]519eor rg, rg, k_even520stp re, rf, [RSTATE, #16]521eor rh, rh, k_odd522stp rg, rh, [RSTATE, #24]523b .Loop524525.Lend:526/* Transform 61-63 */527R2(rd, ra, rb, rc, rh, re, rf, rg, k_odd, _, 61, 1, XW, _, _)528ldp s0, s1, [RSTATE, #0]529R2(rc, rd, ra, rb, rg, rh, re, rf, k_even, KL, 62, 2, XW, _, _)530ldp s2, s3, [RSTATE, #8]531R2(rb, rc, rd, ra, rf, rg, rh, re, k_odd, _, 63, 0, XW, _, _)532533/* Update the chaining variables. */534eor ra, ra, s0535clear_vec(W0)536eor rb, rb, s1537clear_vec(W1)538ldp s0, s1, [RSTATE, #16]539clear_vec(W2)540eor rc, rc, s2541clear_vec(W3)542ldp k_even, k_odd, [RSTATE, #24]543clear_vec(W4)544eor rd, rd, s3545clear_vec(W5)546eor re, re, s0547clear_vec(XTMP0)548stp ra, rb, [RSTATE, #0]549clear_vec(XTMP1)550eor rf, rf, s1551clear_vec(XTMP2)552stp rc, rd, [RSTATE, #8]553clear_vec(XTMP3)554eor rg, rg, k_even555clear_vec(XTMP4)556stp re, rf, [RSTATE, #16]557clear_vec(XTMP5)558eor rh, rh, k_odd559clear_vec(XTMP6)560stp rg, rh, [RSTATE, #24]561562/* Clear message expansion area */563add addr0, sp, #STACK_W564st1 {W0.16b-W3.16b}, [addr0], #64565st1 {W0.16b-W3.16b}, [addr0], #64566st1 {W0.16b-W3.16b}, [addr0]567568mov sp, RFRAME569570ldp x25, x26, [sp], #16571ldp x23, x24, [sp], #16572ldp x21, x22, [sp], #16573ldp x19, x20, [sp], #16574ldp x28, x29, [sp], #16575576ret577SYM_FUNC_END(sm3_neon_transform)578579580.section ".rodata", "a"581582.align 4583.LKtable:584.long 0x79cc4519, 0xf3988a32, 0xe7311465, 0xce6228cb585.long 0x9cc45197, 0x3988a32f, 0x7311465e, 0xe6228cbc586.long 0xcc451979, 0x988a32f3, 0x311465e7, 0x6228cbce587.long 0xc451979c, 0x88a32f39, 0x11465e73, 0x228cbce6588.long 0x9d8a7a87, 0x3b14f50f, 0x7629ea1e, 0xec53d43c589.long 0xd8a7a879, 0xb14f50f3, 0x629ea1e7, 0xc53d43ce590.long 0x8a7a879d, 0x14f50f3b, 0x29ea1e76, 0x53d43cec591.long 0xa7a879d8, 0x4f50f3b1, 0x9ea1e762, 0x3d43cec5592.long 0x7a879d8a, 0xf50f3b14, 0xea1e7629, 0xd43cec53593.long 0xa879d8a7, 0x50f3b14f, 0xa1e7629e, 0x43cec53d594.long 0x879d8a7a, 0x0f3b14f5, 0x1e7629ea, 0x3cec53d4595.long 0x79d8a7a8, 0xf3b14f50, 0xe7629ea1, 0xcec53d43596.long 0x9d8a7a87, 0x3b14f50f, 0x7629ea1e, 0xec53d43c597.long 0xd8a7a879, 0xb14f50f3, 0x629ea1e7, 0xc53d43ce598.long 0x8a7a879d, 0x14f50f3b, 0x29ea1e76, 0x53d43cec599.long 0xa7a879d8, 0x4f50f3b1, 0x9ea1e762, 0x3d43cec5600601602