sha2-ce-core.S 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  1. /*
  2. * sha2-ce-core.S - core SHA-224/SHA-256 transform using v8 Crypto Extensions
  3. *
  4. * Copyright (C) 2014 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. #include <linux/linkage.h>
  11. #include <asm/assembler.h>
  12. .text
  13. .arch armv8-a+crypto
  14. dga .req q20
  15. dgav .req v20
  16. dgb .req q21
  17. dgbv .req v21
  18. t0 .req v22
  19. t1 .req v23
  20. dg0q .req q24
  21. dg0v .req v24
  22. dg1q .req q25
  23. dg1v .req v25
  24. dg2q .req q26
  25. dg2v .req v26
  26. .macro add_only, ev, rc, s0
  27. mov dg2v.16b, dg0v.16b
  28. .ifeq \ev
  29. add t1.4s, v\s0\().4s, \rc\().4s
  30. sha256h dg0q, dg1q, t0.4s
  31. sha256h2 dg1q, dg2q, t0.4s
  32. .else
  33. .ifnb \s0
  34. add t0.4s, v\s0\().4s, \rc\().4s
  35. .endif
  36. sha256h dg0q, dg1q, t1.4s
  37. sha256h2 dg1q, dg2q, t1.4s
  38. .endif
  39. .endm
  40. .macro add_update, ev, rc, s0, s1, s2, s3
  41. sha256su0 v\s0\().4s, v\s1\().4s
  42. add_only \ev, \rc, \s1
  43. sha256su1 v\s0\().4s, v\s2\().4s, v\s3\().4s
  44. .endm
  45. /*
  46. * The SHA-256 round constants
  47. */
  48. .section ".rodata", "a"
  49. .align 4
  50. .Lsha2_rcon:
  51. .word 0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5
  52. .word 0x3956c25b, 0x59f111f1, 0x923f82a4, 0xab1c5ed5
  53. .word 0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3
  54. .word 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174
  55. .word 0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc
  56. .word 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da
  57. .word 0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7
  58. .word 0xc6e00bf3, 0xd5a79147, 0x06ca6351, 0x14292967
  59. .word 0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13
  60. .word 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85
  61. .word 0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3
  62. .word 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070
  63. .word 0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5
  64. .word 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f, 0x682e6ff3
  65. .word 0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208
  66. .word 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2
  67. /*
  68. * void sha2_ce_transform(struct sha256_ce_state *sst, u8 const *src,
  69. * int blocks)
  70. */
  71. .text
  72. ENTRY(sha2_ce_transform)
  73. frame_push 3
  74. mov x19, x0
  75. mov x20, x1
  76. mov x21, x2
  77. /* load round constants */
  78. 0: adr_l x8, .Lsha2_rcon
  79. ld1 { v0.4s- v3.4s}, [x8], #64
  80. ld1 { v4.4s- v7.4s}, [x8], #64
  81. ld1 { v8.4s-v11.4s}, [x8], #64
  82. ld1 {v12.4s-v15.4s}, [x8]
  83. /* load state */
  84. ld1 {dgav.4s, dgbv.4s}, [x19]
  85. /* load sha256_ce_state::finalize */
  86. ldr_l w4, sha256_ce_offsetof_finalize, x4
  87. ldr w4, [x19, x4]
  88. /* load input */
  89. 1: ld1 {v16.4s-v19.4s}, [x20], #64
  90. sub w21, w21, #1
  91. CPU_LE( rev32 v16.16b, v16.16b )
  92. CPU_LE( rev32 v17.16b, v17.16b )
  93. CPU_LE( rev32 v18.16b, v18.16b )
  94. CPU_LE( rev32 v19.16b, v19.16b )
  95. 2: add t0.4s, v16.4s, v0.4s
  96. mov dg0v.16b, dgav.16b
  97. mov dg1v.16b, dgbv.16b
  98. add_update 0, v1, 16, 17, 18, 19
  99. add_update 1, v2, 17, 18, 19, 16
  100. add_update 0, v3, 18, 19, 16, 17
  101. add_update 1, v4, 19, 16, 17, 18
  102. add_update 0, v5, 16, 17, 18, 19
  103. add_update 1, v6, 17, 18, 19, 16
  104. add_update 0, v7, 18, 19, 16, 17
  105. add_update 1, v8, 19, 16, 17, 18
  106. add_update 0, v9, 16, 17, 18, 19
  107. add_update 1, v10, 17, 18, 19, 16
  108. add_update 0, v11, 18, 19, 16, 17
  109. add_update 1, v12, 19, 16, 17, 18
  110. add_only 0, v13, 17
  111. add_only 1, v14, 18
  112. add_only 0, v15, 19
  113. add_only 1
  114. /* update state */
  115. add dgav.4s, dgav.4s, dg0v.4s
  116. add dgbv.4s, dgbv.4s, dg1v.4s
  117. /* handled all input blocks? */
  118. cbz w21, 3f
  119. if_will_cond_yield_neon
  120. st1 {dgav.4s, dgbv.4s}, [x19]
  121. do_cond_yield_neon
  122. b 0b
  123. endif_yield_neon
  124. b 1b
  125. /*
  126. * Final block: add padding and total bit count.
  127. * Skip if the input size was not a round multiple of the block size,
  128. * the padding is handled by the C code in that case.
  129. */
  130. 3: cbz x4, 4f
  131. ldr_l w4, sha256_ce_offsetof_count, x4
  132. ldr x4, [x19, x4]
  133. movi v17.2d, #0
  134. mov x8, #0x80000000
  135. movi v18.2d, #0
  136. ror x7, x4, #29 // ror(lsl(x4, 3), 32)
  137. fmov d16, x8
  138. mov x4, #0
  139. mov v19.d[0], xzr
  140. mov v19.d[1], x7
  141. b 2b
  142. /* store new state */
  143. 4: st1 {dgav.4s, dgbv.4s}, [x19]
  144. frame_pop
  145. ret
  146. ENDPROC(sha2_ce_transform)