strlen.S 3.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283
  1. /* Copyright (C) 1998 Free Software Foundation, Inc.
  2. This file is part of the GNU C Library.
  3. Code contributed by Matthew Wilcox <willy@odie.barnet.ac.uk>
  4. The GNU C Library is free software; you can redistribute it and/or
  5. modify it under the terms of the GNU Lesser General Public
  6. License as published by the Free Software Foundation; either
  7. version 2.1 of the License, or (at your option) any later version.
  8. The GNU C Library is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  11. Lesser General Public License for more details.
  12. You should have received a copy of the GNU Lesser General Public
  13. License along with the GNU C Library; if not, write to the Free
  14. Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
  15. 02111-1307 USA. */
  16. #include <features.h>
  17. #include <endian.h>
  18. #include <sys/syscall.h>
  19. /* size_t strlen(const char *S)
  20. * entry: r0 -> string
  21. * exit: r0 = len
  22. */
  23. .text
  24. .global __strlen
  25. .hidden __strlen
  26. .type __strlen,%function
  27. .align 4
  28. __strlen:
  29. bic r1, r0, $3 @ addr of word containing first byte
  30. ldr r2, [r1], $4 @ get the first word
  31. ands r3, r0, $3 @ how many bytes are duff?
  32. rsb r0, r3, $0 @ get - that number into counter.
  33. beq Laligned @ skip into main check routine if no
  34. @ more
  35. #if __BYTE_ORDER == __BIG_ENDIAN
  36. orr r2, r2, $0xff000000 @ set this byte to non-zero
  37. subs r3, r3, $1 @ any more to do?
  38. orrgt r2, r2, $0x00ff0000 @ if so, set this byte
  39. subs r3, r3, $1 @ more?
  40. orrgt r2, r2, $0x0000ff00 @ then set.
  41. #else
  42. orr r2, r2, $0x000000ff @ set this byte to non-zero
  43. subs r3, r3, $1 @ any more to do?
  44. orrgt r2, r2, $0x0000ff00 @ if so, set this byte
  45. subs r3, r3, $1 @ more?
  46. orrgt r2, r2, $0x00ff0000 @ then set.
  47. #endif
  48. Laligned: @ here, we have a word in r2. Does it
  49. tst r2, $0x000000ff @ contain any zeroes?
  50. tstne r2, $0x0000ff00 @
  51. tstne r2, $0x00ff0000 @
  52. tstne r2, $0xff000000 @
  53. addne r0, r0, $4 @ if not, the string is 4 bytes longer
  54. ldrne r2, [r1], $4 @ and we continue to the next word
  55. bne Laligned @
  56. Llastword: @ drop through to here once we find a
  57. #if __BYTE_ORDER == __BIG_ENDIAN
  58. tst r2, $0xff000000 @ word that has a zero byte in it
  59. addne r0, r0, $1 @
  60. tstne r2, $0x00ff0000 @ and add up to 3 bytes on to it
  61. addne r0, r0, $1 @
  62. tstne r2, $0x0000ff00 @ (if first three all non-zero, 4th
  63. addne r0, r0, $1 @ must be zero)
  64. #else
  65. tst r2, $0x000000ff @
  66. addne r0, r0, $1 @
  67. tstne r2, $0x0000ff00 @ and add up to 3 bytes on to it
  68. addne r0, r0, $1 @
  69. tstne r2, $0x00ff0000 @ (if first three all non-zero, 4th
  70. addne r0, r0, $1 @ must be zero)
  71. #endif
  72. mov pc,lr
  73. .size __strlen,.-__strlen
  74. strong_alias(__strlen,strlen)