string.S 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. /*
  2. * Copyright (C) Paul Mackerras 1997.
  3. *
  4. * This program is free software; you can redistribute it and/or
  5. * modify it under the terms of the GNU General Public License
  6. * as published by the Free Software Foundation; either version
  7. * 2 of the License, or (at your option) any later version.
  8. *
  9. * NOTE: this code runs in 32 bit mode and is packaged as ELF32.
  10. */
  11. #include "ppc_asm.h"
  12. .text
  13. .globl strcpy
  14. strcpy:
  15. addi r5,r3,-1
  16. addi r4,r4,-1
  17. 1: lbzu r0,1(r4)
  18. cmpwi 0,r0,0
  19. stbu r0,1(r5)
  20. bne 1b
  21. blr
  22. .globl strncpy
  23. strncpy:
  24. cmpwi 0,r5,0
  25. beqlr
  26. mtctr r5
  27. addi r6,r3,-1
  28. addi r4,r4,-1
  29. 1: lbzu r0,1(r4)
  30. cmpwi 0,r0,0
  31. stbu r0,1(r6)
  32. bdnzf 2,1b /* dec ctr, branch if ctr != 0 && !cr0.eq */
  33. blr
  34. .globl strcat
  35. strcat:
  36. addi r5,r3,-1
  37. addi r4,r4,-1
  38. 1: lbzu r0,1(r5)
  39. cmpwi 0,r0,0
  40. bne 1b
  41. addi r5,r5,-1
  42. 1: lbzu r0,1(r4)
  43. cmpwi 0,r0,0
  44. stbu r0,1(r5)
  45. bne 1b
  46. blr
  47. .globl strcmp
  48. strcmp:
  49. addi r5,r3,-1
  50. addi r4,r4,-1
  51. 1: lbzu r3,1(r5)
  52. cmpwi 1,r3,0
  53. lbzu r0,1(r4)
  54. subf. r3,r0,r3
  55. beqlr 1
  56. beq 1b
  57. blr
  58. .globl strlen
  59. strlen:
  60. addi r4,r3,-1
  61. 1: lbzu r0,1(r4)
  62. cmpwi 0,r0,0
  63. bne 1b
  64. subf r3,r3,r4
  65. blr
  66. .globl memset
  67. memset:
  68. rlwimi r4,r4,8,16,23
  69. rlwimi r4,r4,16,0,15
  70. addi r6,r3,-4
  71. cmplwi 0,r5,4
  72. blt 7f
  73. stwu r4,4(r6)
  74. beqlr
  75. andi. r0,r6,3
  76. add r5,r0,r5
  77. subf r6,r0,r6
  78. rlwinm r0,r5,32-2,2,31
  79. mtctr r0
  80. bdz 6f
  81. 1: stwu r4,4(r6)
  82. bdnz 1b
  83. 6: andi. r5,r5,3
  84. 7: cmpwi 0,r5,0
  85. beqlr
  86. mtctr r5
  87. addi r6,r6,3
  88. 8: stbu r4,1(r6)
  89. bdnz 8b
  90. blr
  91. .globl memmove
  92. memmove:
  93. cmplw 0,r3,r4
  94. bgt backwards_memcpy
  95. /* fall through */
  96. .globl memcpy
  97. memcpy:
  98. rlwinm. r7,r5,32-3,3,31 /* r0 = r5 >> 3 */
  99. addi r6,r3,-4
  100. addi r4,r4,-4
  101. beq 2f /* if less than 8 bytes to do */
  102. andi. r0,r6,3 /* get dest word aligned */
  103. mtctr r7
  104. bne 5f
  105. 1: lwz r7,4(r4)
  106. lwzu r8,8(r4)
  107. stw r7,4(r6)
  108. stwu r8,8(r6)
  109. bdnz 1b
  110. andi. r5,r5,7
  111. 2: cmplwi 0,r5,4
  112. blt 3f
  113. lwzu r0,4(r4)
  114. addi r5,r5,-4
  115. stwu r0,4(r6)
  116. 3: cmpwi 0,r5,0
  117. beqlr
  118. mtctr r5
  119. addi r4,r4,3
  120. addi r6,r6,3
  121. 4: lbzu r0,1(r4)
  122. stbu r0,1(r6)
  123. bdnz 4b
  124. blr
  125. 5: subfic r0,r0,4
  126. mtctr r0
  127. 6: lbz r7,4(r4)
  128. addi r4,r4,1
  129. stb r7,4(r6)
  130. addi r6,r6,1
  131. bdnz 6b
  132. subf r5,r0,r5
  133. rlwinm. r7,r5,32-3,3,31
  134. beq 2b
  135. mtctr r7
  136. b 1b
  137. .globl backwards_memcpy
  138. backwards_memcpy:
  139. rlwinm. r7,r5,32-3,3,31 /* r0 = r5 >> 3 */
  140. add r6,r3,r5
  141. add r4,r4,r5
  142. beq 2f
  143. andi. r0,r6,3
  144. mtctr r7
  145. bne 5f
  146. 1: lwz r7,-4(r4)
  147. lwzu r8,-8(r4)
  148. stw r7,-4(r6)
  149. stwu r8,-8(r6)
  150. bdnz 1b
  151. andi. r5,r5,7
  152. 2: cmplwi 0,r5,4
  153. blt 3f
  154. lwzu r0,-4(r4)
  155. subi r5,r5,4
  156. stwu r0,-4(r6)
  157. 3: cmpwi 0,r5,0
  158. beqlr
  159. mtctr r5
  160. 4: lbzu r0,-1(r4)
  161. stbu r0,-1(r6)
  162. bdnz 4b
  163. blr
  164. 5: mtctr r0
  165. 6: lbzu r7,-1(r4)
  166. stbu r7,-1(r6)
  167. bdnz 6b
  168. subf r5,r0,r5
  169. rlwinm. r7,r5,32-3,3,31
  170. beq 2b
  171. mtctr r7
  172. b 1b
  173. .globl memcmp
  174. memcmp:
  175. cmpwi 0,r5,0
  176. blelr
  177. mtctr r5
  178. addi r6,r3,-1
  179. addi r4,r4,-1
  180. 1: lbzu r3,1(r6)
  181. lbzu r0,1(r4)
  182. subf. r3,r0,r3
  183. bdnzt 2,1b
  184. blr
  185. /*
  186. * Flush the dcache and invalidate the icache for a range of addresses.
  187. *
  188. * flush_cache(addr, len)
  189. */
  190. .global flush_cache
  191. flush_cache:
  192. addi 4,4,0x1f /* len = (len + 0x1f) / 0x20 */
  193. rlwinm. 4,4,27,5,31
  194. mtctr 4
  195. beqlr
  196. 1: dcbf 0,3
  197. icbi 0,3
  198. addi 3,3,0x20
  199. bdnz 1b
  200. sync
  201. isync
  202. blr