sparkle384_arm.S 11.6 KB
Newer Older
lwc-tester committed
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301
.syntax unified
.thumb
.cpu cortex-m3
.align 8

///////////////////////////////////////////////////////////////////////////////
// sparkle384_arm.asm: ARM Asm implementation of the SPARKLE384 permutation. //
// This file is part of the SPARKLE submission to NIST's LW Crypto Project.  //
// Version 1.0.1 (2019-06-29), see <http://www.cryptolux.org/> for updates.  //
// Authors: The SPARKLE Group (C. Beierle, A. Biryukov, L. Cardoso dos       //
// Santos, J. Groszschaedl, L. Perrin, A. Udovenko, V. Velichkov, Q. Wang).  //
// License: GPLv3 (see LICENSE file), other licenses available upon request. //
// Copyright (C) 2019 University of Luxembourg <http://www.uni.lu/>.         //
// ------------------------------------------------------------------------- //
// This program is free software: you can redistribute it and/or modify it   //
// under the terms of the GNU General Public License as published by the     //
// Free Software Foundation, either version 3 of the License, or (at your    //
// option) any later version. This program is distributed in the hope that   //
// it will be useful, but WITHOUT ANY WARRANTY/ without even the implied     //
// warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the  //
// GNU General Public License for more details. You should have received a   //
// copy of the GNU General Public License along with this program. If not,   //
// see <http://www.gnu.org/licenses/>.                                       //
///////////////////////////////////////////////////////////////////////////////
    

/*----------------------------------------------------------------------------*/
/*                        Register names and constants                        */
/*----------------------------------------------------------------------------*/

// register sta holds the start address of array <state>
#define sta r0
// register scnt holds the step counter (for loop termination)
#define scnt r0
// register c0w holds a round constant (i.e. a word of array <rcon>)
#define c0w r0
// register ns holds the parameter <ns>, i.e. the number of steps
#define ns r1
// register rca holds the start address of array <rcon>
#define rca r1
// register x0w holds the 1st word of the array <state>
#define x0w r2
// register y0w holds the 2nd word of the array <state>
#define y0w r3
// register x1w holds the 3rd word of the array <state>
#define x1w r4
// register y1w holds the 4th word of the array <state>
#define y1w r5
// register x2w holds the 5th word of the array <state>
#define x2w r6
// register y2w holds the 6th word of the array <state>
#define y2w r7
// register x3w holds the 7th word of the array <state>
#define x3w r8
// register y3w holds the 8th word of the array <state>
#define y3w r9
// register x4w holds the 9th word of the array <state>
#define x4w r10
// register y4w holds the 10th word of the array <state>
#define y4w r11
// register x5w holds the 11th word of the array <state>
#define x5w r12
// register y5w holds the 12th word of the array <state>
#define y5w lr
// register tmpx holds a temporary value
#define tmpx r0
// register tmpy holds another temorary value
#define tmpy r1

/*----------------------------------------------------------------------------*/
/*                              Round Constants                               */
/*----------------------------------------------------------------------------*/

RCON:
    .word 0xB7E15162
    .word 0xBF715880
    .word 0x38B4DA56
    .word 0x324E7738
    .word 0xBB1185EB
    .word 0x4F7C7B57
    .word 0xCFBFA1C8
    .word 0xC2B3293D

/*----------------------------------------------------------------------------*/
/*                                   .macroS                                   */
/*----------------------------------------------------------------------------*/

.macro    PROLOGUE_384
    push    {r4-r12,lr}
    ldmia.w sta!, {x0w-x5w,y5w}
    push    {sta}
.endm

.macro    EPILOGUE_384
    pop     {sta}
    stmdb.w sta!, {x0w-x5w,y5w}
    pop     {r4-r12,lr}
    bx      lr
.endm

.macro    ADD_STEP_CNT_384
    ldr     rca, =RCON
    eor     y1w, y1w, scnt
    and     scnt, scnt, #7
    ldr.w   c0w, [rca, scnt, lsl #2]
    eor     y0w, y0w, c0w
.endm

.macro    ARX_BOX xi:req, yi:req, ci:req
    add     \xi, \xi, \yi, ror #31
    eor     \yi, \yi, \xi, ror #24
    eor     \xi, \xi, \ci
    add     \xi, \xi, \yi, ror #17
    eor     \yi, \yi, \xi, ror #17
    eor     \xi, \xi, \ci
    add     \xi, \xi, \yi
    eor     \yi, \yi, \xi, ror #31
    eor     \xi, \xi, \ci
    add     \xi, \xi, \yi, ror #24
    eor     \yi, \yi, \xi, ror #16
    eor     \xi, \xi, \ci
.endm

.macro   TRI_XOR tx:req, x0:req, x1:req, x2:req
    eor     \tx, \x0, \x1
    eor     \tx, \tx, \x2
.endm

.macro   ARXBOX_LAYER_384
    // ARX-box computations for the three left-side branches (i.e. x[0]-y[2]).
    // ldr.w   c0w, [rca, #0]
    movw    c0w, #0x5162
    movt    c0w, #0xB7E1
    ARX_BOX x0w, y0w, c0w
    // ldr.w   c0w, [rca, #4]
    movw    c0w, #0x5880
    movt    c0w, #0xBF71
    ARX_BOX x1w, y1w, c0w
    // ldr.w   c0w, [rca, #8]
    movw    c0w, #0xDA56
    movt    c0w, #0x38B4
    ARX_BOX x2w, y2w, c0w
    // ARX-box computations for the three right-side branches (i.e. x[3]-y[5]).
    // ldr.w   c0w, [rca, #12]
    movw    c0w, #0x7738
    movt    c0w, #0x324E
    ARX_BOX x3w, y3w, c0w
    // ldr.w   c0w, [rca, #16]
    movw    c0w, #0x85EB
    movt    c0w, #0xBB11
    ARX_BOX x4w, y4w, c0w
    // ldr.w   c0w, [rca, #20]
    movw    c0w, #0x7B57
    movt    c0w, #0x4F7C
    ARX_BOX x5w, y5w, c0w
.endm

.macro   LINEAR_LAYER_384
    // First part of Feistel round: tmpx and tmpy are computed and XORED to the
    // y-words and x-words of the right-side branches (i.e. to y[3], y[4], y[5]
    // and to x[3], x[4], x[5]). Note that y[5] and x[5] are stored in register
    // tmpx and tmpy (and not in register y5w and x5w) to reduce the execution
    // time of the subsequent branch permutation.
    eor     tmpx, x0w, x1w
    eor     tmpx, tmpx, x2w
    eor     tmpx, tmpx, tmpx, lsl #16
    eor     y3w, y3w, tmpx, ror #16
    eor     y4w, y4w, tmpx, ror #16
    eor     tmpx, y5w, tmpx, ror #16
    eor     tmpy, y0w, y1w
    eor     tmpy, tmpy, y2w
    eor     tmpy, tmpy, tmpy, lsl #16
    eor     x3w, x3w, tmpy, ror #16
    eor     x4w, x4w, tmpy, ror #16
    eor     tmpy, x5w, tmpy, ror #16
    // Branch permutation: 1-branch left-rotation of the right-side branches
    // along with a swap of the left and right branches (via register writes).
    // Also combined with the branch permutation is the second Feistel part,
    // in which the left-side branches are XORed with the result of the first
    // Feistel part.
    mov     y5w, y2w
    eor     y2w, y3w, y0w
    mov     y3w, y0w
    eor     y0w, y4w, y1w
    mov     y4w, y1w
    eor     y1w, tmpx, y5w
    mov     x5w, x2w
    eor     x2w, x3w, x0w
    mov     x3w, x0w
    eor     x0w, x4w, x1w
    mov     x4w, x1w
    eor     x1w, tmpy, x5w
.endm

/*----------------------------------------------------------------------------*/
/*                 SPARKLE384 PERMUTATIONS (BRANCH-UNROLLED)                  */
/*----------------------------------------------------------------------------*/


// Function prototype:
// -------------------
// void sparkle384_arm(uint32_t *state, int ns)
//
// Parameters:
// -----------
// state: pointer to an uint32-array containing the 12 state words
// ns: number of steps
//
// Return value:
// -------------
// None

/*.align	1
.p2align 2,,3
.syntax unified
.thumb
.thumb_func
.fpu softvfp
*/

.global sparkle384_arm
.type sparkle384_arm, %function

sparkle384_arm:
    PROLOGUE_384           // push callee-saved registers
    mov scnt, #0           // clear step-counter
.L1:
    push {scnt-ns}         // push ns and step-counter (we need registers!)
    ADD_STEP_CNT_384       // macro to add step-counter to state
    ARXBOX_LAYER_384       // macro for the arxbox layer
    LINEAR_LAYER_384       // macro for the linear layer
    pop {scnt-ns}          // restore ns and step-counter from stack
    add scnt, #1           // increment step-counter
    teq scnt, ns           // test whether step-counter equals ns
    bne .L1                // if not then jump back to start of loop
    EPILOGUE_384           // pop callee-saved registers

/* 
// hand unrolled version
sparkle384_arm:
    //quick and dirty unroll
    PROLOGUE_384           // push callee-saved registers
    mov scnt, #0           // clear step-counter
    // first iteration
    ADD_STEP_CNT_384       // macro to add step-counter to state
    push {scnt-ns}         // push ns and step-counter (we need regs!)
    ARXBOX_LAYER_384       // macro for the arxbox layer
    LINEAR_LAYER_384       // macro for the linear layer
    pop {scnt-ns}          // restore ns and step-counter from stack
    add scnt, #1           // increment step-counter
    teq scnt, ns           // test whether step-counter equals ns
    // second iteration
    ADD_STEP_CNT_384       // macro to add step-counter to state
    push {scnt-ns}         // push ns and step-counter (we need regs!)
    ARXBOX_LAYER_384       // macro for the arxbox layer
    LINEAR_LAYER_384       // macro for the linear layer
    pop {scnt-ns}          // restore ns and step-counter from stack
    add scnt, #1           // increment step-counter
    teq scnt, ns           // test whether step-counter equals ns
    // third iteration
    ADD_STEP_CNT_384       // macro to add step-counter to state
    push {scnt-ns}         // push ns and step-counter (we need regs!)
    ARXBOX_LAYER_384       // macro for the arxbox layer
    LINEAR_LAYER_384       // macro for the linear layer
    pop {scnt-ns}          // restore ns and step-counter from stack
    add scnt, #1           // increment step-counter
    teq scnt, ns           // test whether step-counter equals ns
    // fourth iteration
    ADD_STEP_CNT_384       // macro to add step-counter to state
    push {scnt-ns}         // push ns and step-counter (we need regs!)
    ARXBOX_LAYER_384       // macro for the arxbox layer
    LINEAR_LAYER_384       // macro for the linear layer
    pop {scnt-ns}          // restore ns and step-counter from stack
    add scnt, #1           // increment step-counter
    teq scnt, ns           // test whether step-counter equals ns
    // fifth iteration
    ADD_STEP_CNT_384       // macro to add step-counter to state
    push {scnt-ns}         // push ns and step-counter (we need regs!)
    ARXBOX_LAYER_384       // macro for the arxbox layer
    LINEAR_LAYER_384       // macro for the linear layer
    pop {scnt-ns}          // restore ns and step-counter from stack
    add scnt, #1           // increment step-counter
    teq scnt, ns           // test whether step-counter equals ns
    // sixth iteration
    ADD_STEP_CNT_384       // macro to add step-counter to state
    push {scnt-ns}         // push ns and step-counter (we need regs!)
    ARXBOX_LAYER_384       // macro for the arxbox layer
    LINEAR_LAYER_384       // macro for the linear layer
    pop {scnt-ns}          // restore ns and step-counter from stack
    add scnt, #1           // increment step-counter
    teq scnt, ns           // test whether step-counter equals ns
    // seventh iteration
    ADD_STEP_CNT_384       // macro to add step-counter to state
    push {scnt-ns}         // push ns and step-counter (we need regs!)
    ARXBOX_LAYER_384       // macro for the arxbox layer
    LINEAR_LAYER_384       // macro for the linear layer
    pop {scnt-ns}          // restore ns and step-counter from stack
    add scnt, #1           // increment step-counter
    teq scnt, ns           // test whether step-counter equals ns
    EPILOGUE_384           // pop callee-saved registers
*/