memcpy.S 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243
  1. /*
  2. * linux/arch/arm/lib/memcpy.S
  3. *
  4. * Author: Nicolas Pitre
  5. * Created: Sep 28, 2005
  6. * Copyright: MontaVista Software, Inc.
  7. *
  8. * This program is free software; you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License version 2 as
  10. * published by the Free Software Foundation.
  11. */
  12. #include <asm/assembler.h>
  13. #define W(instr) instr
  14. #define LDR1W_SHIFT 0
  15. #define STR1W_SHIFT 0
  16. .macro ldr1w ptr reg abort
  17. W(ldr) \reg, [\ptr], #4
  18. .endm
  19. .macro ldr4w ptr reg1 reg2 reg3 reg4 abort
  20. ldmia \ptr!, {\reg1, \reg2, \reg3, \reg4}
  21. .endm
  22. .macro ldr8w ptr reg1 reg2 reg3 reg4 reg5 reg6 reg7 reg8 abort
  23. ldmia \ptr!, {\reg1, \reg2, \reg3, \reg4, \reg5, \reg6, \reg7, \reg8}
  24. .endm
  25. .macro ldr1b ptr reg cond=al abort
  26. ldr\cond\()b \reg, [\ptr], #1
  27. .endm
  28. .macro str1w ptr reg abort
  29. W(str) \reg, [\ptr], #4
  30. .endm
  31. .macro str8w ptr reg1 reg2 reg3 reg4 reg5 reg6 reg7 reg8 abort
  32. stmia \ptr!, {\reg1, \reg2, \reg3, \reg4, \reg5, \reg6, \reg7, \reg8}
  33. .endm
  34. .macro str1b ptr reg cond=al abort
  35. str\cond\()b \reg, [\ptr], #1
  36. .endm
  37. .macro enter reg1 reg2
  38. stmdb sp!, {r0, \reg1, \reg2}
  39. .endm
  40. .macro exit reg1 reg2
  41. ldmfd sp!, {r0, \reg1, \reg2}
  42. .endm
  43. .text
  44. /* Prototype: void *memcpy(void *dest, const void *src, size_t n); */
  45. .globl memcpy
  46. memcpy:
  47. cmp r0, r1
  48. moveq pc, lr
  49. enter r4, lr
  50. subs r2, r2, #4
  51. blt 8f
  52. ands ip, r0, #3
  53. PLD( pld [r1, #0] )
  54. bne 9f
  55. ands ip, r1, #3
  56. bne 10f
  57. 1: subs r2, r2, #(28)
  58. stmfd sp!, {r5 - r8}
  59. blt 5f
  60. CALGN( ands ip, r0, #31 )
  61. CALGN( rsb r3, ip, #32 )
  62. CALGN( sbcnes r4, r3, r2 ) @ C is always set here
  63. CALGN( bcs 2f )
  64. CALGN( adr r4, 6f )
  65. CALGN( subs r2, r2, r3 ) @ C gets set
  66. CALGN( add pc, r4, ip )
  67. PLD( pld [r1, #0] )
  68. 2: PLD( subs r2, r2, #96 )
  69. PLD( pld [r1, #28] )
  70. PLD( blt 4f )
  71. PLD( pld [r1, #60] )
  72. PLD( pld [r1, #92] )
  73. 3: PLD( pld [r1, #124] )
  74. 4: ldr8w r1, r3, r4, r5, r6, r7, r8, ip, lr, abort=20f
  75. subs r2, r2, #32
  76. str8w r0, r3, r4, r5, r6, r7, r8, ip, lr, abort=20f
  77. bge 3b
  78. PLD( cmn r2, #96 )
  79. PLD( bge 4b )
  80. 5: ands ip, r2, #28
  81. rsb ip, ip, #32
  82. #if LDR1W_SHIFT > 0
  83. lsl ip, ip, #LDR1W_SHIFT
  84. #endif
  85. addne pc, pc, ip @ C is always clear here
  86. b 7f
  87. 6:
  88. .rept (1 << LDR1W_SHIFT)
  89. W(nop)
  90. .endr
  91. ldr1w r1, r3, abort=20f
  92. ldr1w r1, r4, abort=20f
  93. ldr1w r1, r5, abort=20f
  94. ldr1w r1, r6, abort=20f
  95. ldr1w r1, r7, abort=20f
  96. ldr1w r1, r8, abort=20f
  97. ldr1w r1, lr, abort=20f
  98. #if LDR1W_SHIFT < STR1W_SHIFT
  99. lsl ip, ip, #STR1W_SHIFT - LDR1W_SHIFT
  100. #elif LDR1W_SHIFT > STR1W_SHIFT
  101. lsr ip, ip, #LDR1W_SHIFT - STR1W_SHIFT
  102. #endif
  103. add pc, pc, ip
  104. nop
  105. .rept (1 << STR1W_SHIFT)
  106. W(nop)
  107. .endr
  108. str1w r0, r3, abort=20f
  109. str1w r0, r4, abort=20f
  110. str1w r0, r5, abort=20f
  111. str1w r0, r6, abort=20f
  112. str1w r0, r7, abort=20f
  113. str1w r0, r8, abort=20f
  114. str1w r0, lr, abort=20f
  115. CALGN( bcs 2b )
  116. 7: ldmfd sp!, {r5 - r8}
  117. 8: movs r2, r2, lsl #31
  118. ldr1b r1, r3, ne, abort=21f
  119. ldr1b r1, r4, cs, abort=21f
  120. ldr1b r1, ip, cs, abort=21f
  121. str1b r0, r3, ne, abort=21f
  122. str1b r0, r4, cs, abort=21f
  123. str1b r0, ip, cs, abort=21f
  124. exit r4, pc
  125. 9: rsb ip, ip, #4
  126. cmp ip, #2
  127. ldr1b r1, r3, gt, abort=21f
  128. ldr1b r1, r4, ge, abort=21f
  129. ldr1b r1, lr, abort=21f
  130. str1b r0, r3, gt, abort=21f
  131. str1b r0, r4, ge, abort=21f
  132. subs r2, r2, ip
  133. str1b r0, lr, abort=21f
  134. blt 8b
  135. ands ip, r1, #3
  136. beq 1b
  137. 10: bic r1, r1, #3
  138. cmp ip, #2
  139. ldr1w r1, lr, abort=21f
  140. beq 17f
  141. bgt 18f
  142. .macro forward_copy_shift pull push
  143. subs r2, r2, #28
  144. blt 14f
  145. CALGN( ands ip, r0, #31 )
  146. CALGN( rsb ip, ip, #32 )
  147. CALGN( sbcnes r4, ip, r2 ) @ C is always set here
  148. CALGN( subcc r2, r2, ip )
  149. CALGN( bcc 15f )
  150. 11: stmfd sp!, {r5 - r9}
  151. PLD( pld [r1, #0] )
  152. PLD( subs r2, r2, #96 )
  153. PLD( pld [r1, #28] )
  154. PLD( blt 13f )
  155. PLD( pld [r1, #60] )
  156. PLD( pld [r1, #92] )
  157. 12: PLD( pld [r1, #124] )
  158. 13: ldr4w r1, r4, r5, r6, r7, abort=19f
  159. mov r3, lr, pull #\pull
  160. subs r2, r2, #32
  161. ldr4w r1, r8, r9, ip, lr, abort=19f
  162. orr r3, r3, r4, push #\push
  163. mov r4, r4, pull #\pull
  164. orr r4, r4, r5, push #\push
  165. mov r5, r5, pull #\pull
  166. orr r5, r5, r6, push #\push
  167. mov r6, r6, pull #\pull
  168. orr r6, r6, r7, push #\push
  169. mov r7, r7, pull #\pull
  170. orr r7, r7, r8, push #\push
  171. mov r8, r8, pull #\pull
  172. orr r8, r8, r9, push #\push
  173. mov r9, r9, pull #\pull
  174. orr r9, r9, ip, push #\push
  175. mov ip, ip, pull #\pull
  176. orr ip, ip, lr, push #\push
  177. str8w r0, r3, r4, r5, r6, r7, r8, r9, ip, , abort=19f
  178. bge 12b
  179. PLD( cmn r2, #96 )
  180. PLD( bge 13b )
  181. ldmfd sp!, {r5 - r9}
  182. 14: ands ip, r2, #28
  183. beq 16f
  184. 15: mov r3, lr, pull #\pull
  185. ldr1w r1, lr, abort=21f
  186. subs ip, ip, #4
  187. orr r3, r3, lr, push #\push
  188. str1w r0, r3, abort=21f
  189. bgt 15b
  190. CALGN( cmp r2, #0 )
  191. CALGN( bge 11b )
  192. 16: sub r1, r1, #(\push / 8)
  193. b 8b
  194. .endm
  195. forward_copy_shift pull=8 push=24
  196. 17: forward_copy_shift pull=16 push=16
  197. 18: forward_copy_shift pull=24 push=8