ce426f
# commit db9b4570c5dc550074140ac1d1677077fba29a26
ce426f
# Author: Alan Modra <amodra@gmail.com>
ce426f
# Date:   Sat Aug 17 18:40:11 2013 +0930
ce426f
# 
ce426f
#     PowerPC LE strlen
ce426f
#     http://sourceware.org/ml/libc-alpha/2013-08/msg00097.html
ce426f
#     
ce426f
#     This is the first of nine patches adding little-endian support to the
ce426f
#     existing optimised string and memory functions.  I did spend some
ce426f
#     time with a power7 simulator looking at cycle by cycle behaviour for
ce426f
#     memchr, but most of these patches have not been run on cpu simulators
ce426f
#     to check that we are going as fast as possible.  I'm sure PowerPC can
ce426f
#     do better.  However, the little-endian support mostly leaves main
ce426f
#     loops unchanged, so I'm banking on previous authors having done a
ce426f
#     good job on big-endian..  As with most code you stare at long enough,
ce426f
#     I found some improvements for big-endian too.
ce426f
#     
ce426f
#     Little-endian support for strlen.  Like most of the string functions,
ce426f
#     I leave the main word or multiple-word loops substantially unchanged,
ce426f
#     just needing to modify the tail.
ce426f
#     
ce426f
#     Removing the branch in the power7 functions is just a tidy.  .align
ce426f
#     produces a branch anyway.  Modifying regs in the non-power7 functions
ce426f
#     is to suit the new little-endian tail.
ce426f
#     
ce426f
#         * sysdeps/powerpc/powerpc64/power7/strlen.S (strlen): Add little-endian
ce426f
#         support.  Don't branch over align.
ce426f
#         * sysdeps/powerpc/powerpc32/power7/strlen.S: Likewise.
ce426f
#         * sysdeps/powerpc/powerpc64/strlen.S (strlen): Add little-endian support.
ce426f
#         Rearrange tmp reg use to suit.  Comment.
ce426f
#         * sysdeps/powerpc/powerpc32/strlen.S: Likewise.
ce426f
# 
ce426f
diff -urN glibc-2.17-c758a686/sysdeps/powerpc/powerpc32/power7/strlen.S glibc-2.17-c758a686/sysdeps/powerpc/powerpc32/power7/strlen.S
ce426f
--- glibc-2.17-c758a686/sysdeps/powerpc/powerpc32/power7/strlen.S	2014-05-28 12:28:44.000000000 -0500
ce426f
+++ glibc-2.17-c758a686/sysdeps/powerpc/powerpc32/power7/strlen.S	2014-05-28 12:28:45.000000000 -0500
ce426f
@@ -31,7 +31,11 @@
ce426f
 	li	r0,0	      /* Word with null chars to use with cmpb.  */
ce426f
 	li	r5,-1	      /* MASK = 0xffffffffffffffff.  */
ce426f
 	lwz	r12,0(r4)     /* Load word from memory.  */
ce426f
+#ifdef __LITTLE_ENDIAN__
ce426f
+	slw	r5,r5,r6
ce426f
+#else
ce426f
 	srw	r5,r5,r6      /* MASK = MASK >> padding.  */
ce426f
+#endif
ce426f
 	orc	r9,r12,r5     /* Mask bits that are not part of the string.  */
ce426f
 	cmpb	r10,r9,r0     /* Check for null bytes in WORD1.  */
ce426f
 	cmpwi	cr7,r10,0     /* If r10 == 0, no null's have been found.  */
ce426f
@@ -49,9 +53,6 @@
ce426f
 	cmpb	r10,r12,r0
ce426f
 	cmpwi	cr7,r10,0
ce426f
 	bne	cr7,L(done)
ce426f
-	b	L(loop)	      /* We branch here (rather than falling through)
ce426f
-				 to skip the nops due to heavy alignment
ce426f
-				 of the loop below.  */
ce426f
 
ce426f
 	/* Main loop to look for the end of the string.  Since it's a
ce426f
 	   small loop (< 8 instructions), align it to 32-bytes.  */
ce426f
@@ -88,9 +89,15 @@
ce426f
 	   0xff in the same position as the null byte in the original
ce426f
 	   word from the string.  Use that to calculate the length.  */
ce426f
 L(done):
ce426f
-	cntlzw	r0,r10	      /* Count leading zeroes before the match.  */
ce426f
+#ifdef __LITTLE_ENDIAN__
ce426f
+	addi	r9, r10, -1   /* Form a mask from trailing zeros.  */
ce426f
+	andc	r9, r9, r10
ce426f
+	popcntw r0, r9	      /* Count the bits in the mask.  */
ce426f
+#else
ce426f
+	cntlzw	r0,r10	      /* Count leading zeros before the match.  */
ce426f
+#endif
ce426f
 	subf	r5,r3,r4
ce426f
-	srwi	r0,r0,3	      /* Convert leading zeroes to bytes.  */
ce426f
+	srwi	r0,r0,3	      /* Convert leading zeros to bytes.  */
ce426f
 	add	r3,r5,r0      /* Compute final length.  */
ce426f
 	blr
ce426f
 END (BP_SYM (strlen))
ce426f
diff -urN glibc-2.17-c758a686/sysdeps/powerpc/powerpc32/strlen.S glibc-2.17-c758a686/sysdeps/powerpc/powerpc32/strlen.S
ce426f
--- glibc-2.17-c758a686/sysdeps/powerpc/powerpc32/strlen.S	2014-05-28 12:28:44.000000000 -0500
ce426f
+++ glibc-2.17-c758a686/sysdeps/powerpc/powerpc32/strlen.S	2014-05-28 12:32:24.000000000 -0500
ce426f
@@ -31,7 +31,12 @@
ce426f
       1 is subtracted you get a value in the range 0x00-0x7f, none of which
ce426f
       have their high bit set. The expression here is
ce426f
       (x + 0xfefefeff) & ~(x | 0x7f7f7f7f), which gives 0x00000000 when
ce426f
-      there were no 0x00 bytes in the word.
ce426f
+      there were no 0x00 bytes in the word.  You get 0x80 in bytes that
ce426f
+      match, but possibly false 0x80 matches in the next more significant
ce426f
+      byte to a true match due to carries.  For little-endian this is
ce426f
+      of no consequence since the least significant match is the one
ce426f
+      we're interested in, but big-endian needs method 2 to find which
ce426f
+      byte matches.
ce426f
 
ce426f
    2) Given a word 'x', we can test to see _which_ byte was zero by
ce426f
       calculating ~(((x & 0x7f7f7f7f) + 0x7f7f7f7f) | x | 0x7f7f7f7f).
ce426f
@@ -74,7 +79,7 @@
ce426f
 
ce426f
 ENTRY (BP_SYM (strlen))
ce426f
 
ce426f
-#define rTMP1	r0
ce426f
+#define rTMP4	r0
ce426f
 #define rRTN	r3	/* incoming STR arg, outgoing result */
ce426f
 #define rSTR	r4	/* current string position */
ce426f
 #define rPADN	r5	/* number of padding bits we prepend to the
ce426f
@@ -84,9 +89,9 @@
ce426f
 #define rWORD1	r8	/* current string word */
ce426f
 #define rWORD2	r9	/* next string word */
ce426f
 #define rMASK	r9	/* mask for first string word */
ce426f
-#define rTMP2	r10
ce426f
-#define rTMP3	r11
ce426f
-#define rTMP4	r12
ce426f
+#define rTMP1	r10
ce426f
+#define rTMP2	r11
ce426f
+#define rTMP3	r12
ce426f
 
ce426f
 	CHECK_BOUNDS_LOW (rRTN, rTMP1, rTMP2)
ce426f
 
ce426f
@@ -96,15 +101,20 @@
ce426f
 	lwz	rWORD1, 0(rSTR)
ce426f
 	li	rMASK, -1
ce426f
 	addi	r7F7F, r7F7F, 0x7f7f
ce426f
-/* That's the setup done, now do the first pair of words.
ce426f
-   We make an exception and use method (2) on the first two words, to reduce
ce426f
-   overhead.  */
ce426f
+/* We use method (2) on the first two words, because rFEFE isn't
ce426f
+   required which reduces setup overhead.  Also gives a faster return
ce426f
+   for small strings on big-endian due to needing to recalculate with
ce426f
+   method (2) anyway.  */
ce426f
+#ifdef __LITTLE_ENDIAN__
ce426f
+	slw	rMASK, rMASK, rPADN
ce426f
+#else
ce426f
 	srw	rMASK, rMASK, rPADN
ce426f
+#endif
ce426f
 	and	rTMP1, r7F7F, rWORD1
ce426f
 	or	rTMP2, r7F7F, rWORD1
ce426f
 	add	rTMP1, rTMP1, r7F7F
ce426f
-	nor	rTMP1, rTMP2, rTMP1
ce426f
-	and.	rWORD1, rTMP1, rMASK
ce426f
+	nor	rTMP3, rTMP2, rTMP1
ce426f
+	and.	rTMP3, rTMP3, rMASK
ce426f
 	mtcrf	0x01, rRTN
ce426f
 	bne	L(done0)
ce426f
 	lis	rFEFE, -0x101
ce426f
@@ -113,11 +123,12 @@
ce426f
 	bt	29, L(loop)
ce426f
 
ce426f
 /* Handle second word of pair.  */
ce426f
+/* Perhaps use method (1) here for little-endian, saving one instruction?  */
ce426f
 	lwzu	rWORD1, 4(rSTR)
ce426f
 	and	rTMP1, r7F7F, rWORD1
ce426f
 	or	rTMP2, r7F7F, rWORD1
ce426f
 	add	rTMP1, rTMP1, r7F7F
ce426f
-	nor.	rWORD1, rTMP2, rTMP1
ce426f
+	nor.	rTMP3, rTMP2, rTMP1
ce426f
 	bne	L(done0)
ce426f
 
ce426f
 /* The loop.  */
ce426f
@@ -131,29 +142,53 @@
ce426f
 	add	rTMP3, rFEFE, rWORD2
ce426f
 	nor	rTMP4, r7F7F, rWORD2
ce426f
 	bne	L(done1)
ce426f
-	and.	rTMP1, rTMP3, rTMP4
ce426f
+	and.	rTMP3, rTMP3, rTMP4
ce426f
 	beq	L(loop)
ce426f
 
ce426f
+#ifndef __LITTLE_ENDIAN__
ce426f
 	and	rTMP1, r7F7F, rWORD2
ce426f
 	add	rTMP1, rTMP1, r7F7F
ce426f
-	andc	rWORD1, rTMP4, rTMP1
ce426f
+	andc	rTMP3, rTMP4, rTMP1
ce426f
 	b	L(done0)
ce426f
 
ce426f
 L(done1):
ce426f
 	and	rTMP1, r7F7F, rWORD1
ce426f
 	subi	rSTR, rSTR, 4
ce426f
 	add	rTMP1, rTMP1, r7F7F
ce426f
-	andc	rWORD1, rTMP2, rTMP1
ce426f
+	andc	rTMP3, rTMP2, rTMP1
ce426f
 
ce426f
 /* When we get to here, rSTR points to the first word in the string that
ce426f
-   contains a zero byte, and the most significant set bit in rWORD1 is in that
ce426f
-   byte.  */
ce426f
+   contains a zero byte, and rTMP3 has 0x80 for bytes that are zero,
ce426f
+   and 0x00 otherwise.  */
ce426f
 L(done0):
ce426f
-	cntlzw	rTMP3, rWORD1
ce426f
+	cntlzw	rTMP3, rTMP3
ce426f
 	subf	rTMP1, rRTN, rSTR
ce426f
 	srwi	rTMP3, rTMP3, 3
ce426f
 	add	rRTN, rTMP1, rTMP3
ce426f
 	/* GKM FIXME: check high bound.  */
ce426f
 	blr
ce426f
+#else
ce426f
+
ce426f
+L(done0):
ce426f
+	addi	rTMP1, rTMP3, -1	/* Form a mask from trailing zeros.  */
ce426f
+	andc	rTMP1, rTMP1, rTMP3
ce426f
+	cntlzw	rTMP1, rTMP1		/* Count bits not in the mask.  */
ce426f
+	subf	rTMP3, rRTN, rSTR
ce426f
+	subfic	rTMP1, rTMP1, 32-7
ce426f
+	srwi	rTMP1, rTMP1, 3
ce426f
+	add	rRTN, rTMP1, rTMP3
ce426f
+	blr
ce426f
+
ce426f
+L(done1):
ce426f
+	addi	rTMP3, rTMP1, -1
ce426f
+	andc	rTMP3, rTMP3, rTMP1
ce426f
+	cntlzw	rTMP3, rTMP3
ce426f
+	subf	rTMP1, rRTN, rSTR
ce426f
+	subfic	rTMP3, rTMP3, 32-7-32
ce426f
+	srawi	rTMP3, rTMP3, 3
ce426f
+	add	rRTN, rTMP1, rTMP3
ce426f
+	blr
ce426f
+#endif
ce426f
+
ce426f
 END (BP_SYM (strlen))
ce426f
 libc_hidden_builtin_def (strlen)
ce426f
diff -urN glibc-2.17-c758a686/sysdeps/powerpc/powerpc64/power7/strlen.S glibc-2.17-c758a686/sysdeps/powerpc/powerpc64/power7/strlen.S
ce426f
--- glibc-2.17-c758a686/sysdeps/powerpc/powerpc64/power7/strlen.S	2014-05-28 12:28:44.000000000 -0500
ce426f
+++ glibc-2.17-c758a686/sysdeps/powerpc/powerpc64/power7/strlen.S	2014-05-28 12:28:45.000000000 -0500
ce426f
@@ -32,7 +32,11 @@
ce426f
 				 with cmpb.  */
ce426f
 	li	r5,-1	      /* MASK = 0xffffffffffffffff.  */
ce426f
 	ld	r12,0(r4)     /* Load doubleword from memory.  */
ce426f
+#ifdef __LITTLE_ENDIAN__
ce426f
+	sld	r5,r5,r6
ce426f
+#else
ce426f
 	srd	r5,r5,r6      /* MASK = MASK >> padding.  */
ce426f
+#endif
ce426f
 	orc	r9,r12,r5     /* Mask bits that are not part of the string.  */
ce426f
 	cmpb	r10,r9,r0     /* Check for null bytes in DWORD1.  */
ce426f
 	cmpdi	cr7,r10,0     /* If r10 == 0, no null's have been found.  */
ce426f
@@ -50,9 +54,6 @@
ce426f
 	cmpb	r10,r12,r0
ce426f
 	cmpdi	cr7,r10,0
ce426f
 	bne	cr7,L(done)
ce426f
-	b	L(loop)	      /* We branch here (rather than falling through)
ce426f
-				 to skip the nops due to heavy alignment
ce426f
-				 of the loop below.  */
ce426f
 
ce426f
 	/* Main loop to look for the end of the string.  Since it's a
ce426f
 	   small loop (< 8 instructions), align it to 32-bytes.  */
ce426f
@@ -89,9 +90,15 @@
ce426f
 	   0xff in the same position as the null byte in the original
ce426f
 	   doubleword from the string.  Use that to calculate the length.  */
ce426f
 L(done):
ce426f
-	cntlzd	r0,r10	      /* Count leading zeroes before the match.  */
ce426f
+#ifdef __LITTLE_ENDIAN__
ce426f
+	addi	r9, r10, -1   /* Form a mask from trailing zeros.  */
ce426f
+	andc	r9, r9, r10
ce426f
+	popcntd r0, r9	      /* Count the bits in the mask.  */
ce426f
+#else
ce426f
+	cntlzd	r0,r10	      /* Count leading zeros before the match.  */
ce426f
+#endif
ce426f
 	subf	r5,r3,r4
ce426f
-	srdi	r0,r0,3	      /* Convert leading zeroes to bytes.  */
ce426f
+	srdi	r0,r0,3	      /* Convert leading/trailing zeros to bytes.  */
ce426f
 	add	r3,r5,r0      /* Compute final length.  */
ce426f
 	blr
ce426f
 END (BP_SYM (strlen))
ce426f
diff -urN glibc-2.17-c758a686/sysdeps/powerpc/powerpc64/strlen.S glibc-2.17-c758a686/sysdeps/powerpc/powerpc64/strlen.S
ce426f
--- glibc-2.17-c758a686/sysdeps/powerpc/powerpc64/strlen.S	2014-05-28 12:28:44.000000000 -0500
ce426f
+++ glibc-2.17-c758a686/sysdeps/powerpc/powerpc64/strlen.S	2014-05-28 12:38:17.000000000 -0500
ce426f
@@ -31,7 +31,12 @@
ce426f
       1 is subtracted you get a value in the range 0x00-0x7f, none of which
ce426f
       have their high bit set. The expression here is
ce426f
       (x + 0xfefefeff) & ~(x | 0x7f7f7f7f), which gives 0x00000000 when
ce426f
-      there were no 0x00 bytes in the word.
ce426f
+      there were no 0x00 bytes in the word.  You get 0x80 in bytes that
ce426f
+      match, but possibly false 0x80 matches in the next more significant
ce426f
+      byte to a true match due to carries.  For little-endian this is
ce426f
+      of no consequence since the least significant match is the one
ce426f
+      we're interested in, but big-endian needs method 2 to find which
ce426f
+      byte matches.
ce426f
 
ce426f
    2) Given a word 'x', we can test to see _which_ byte was zero by
ce426f
       calculating ~(((x & 0x7f7f7f7f) + 0x7f7f7f7f) | x | 0x7f7f7f7f).
ce426f
@@ -64,7 +69,7 @@
ce426f
    Answer:
ce426f
    1) Added a Data Cache Block Touch early to prefetch the first 128 
ce426f
    byte cache line. Adding dcbt instructions to the loop would not be 
ce426f
-   effective since most strings will be shorter than the cache line.*/
ce426f
+   effective since most strings will be shorter than the cache line.  */
ce426f
 
ce426f
 /* Some notes on register usage: Under the SVR4 ABI, we can use registers
ce426f
    0 and 3 through 12 (so long as we don't call any procedures) without
ce426f
@@ -80,7 +85,7 @@
ce426f
 ENTRY (BP_SYM (strlen))
ce426f
 	CALL_MCOUNT 1
ce426f
 
ce426f
-#define rTMP1	r0
ce426f
+#define rTMP4	r0
ce426f
 #define rRTN	r3	/* incoming STR arg, outgoing result */
ce426f
 #define rSTR	r4	/* current string position */
ce426f
 #define rPADN	r5	/* number of padding bits we prepend to the
ce426f
@@ -90,9 +95,9 @@
ce426f
 #define rWORD1	r8	/* current string doubleword */
ce426f
 #define rWORD2	r9	/* next string doubleword */
ce426f
 #define rMASK	r9	/* mask for first string doubleword */
ce426f
-#define rTMP2	r10
ce426f
-#define rTMP3	r11
ce426f
-#define rTMP4	r12
ce426f
+#define rTMP1	r10
ce426f
+#define rTMP2	r11
ce426f
+#define rTMP3	r12
ce426f
 
ce426f
 /* Note:  The Bounded pointer support in this code is broken.  This code
ce426f
    was inherited from PPC32 and that support was never completed.
ce426f
@@ -109,30 +114,36 @@
ce426f
 	addi	r7F7F, r7F7F, 0x7f7f
ce426f
 	li	rMASK, -1
ce426f
 	insrdi	r7F7F, r7F7F, 32, 0
ce426f
-/* That's the setup done, now do the first pair of doublewords.
ce426f
-   We make an exception and use method (2) on the first two doublewords, 
ce426f
-   to reduce overhead.  */
ce426f
-	srd	rMASK, rMASK, rPADN
ce426f
+/* We use method (2) on the first two doublewords, because rFEFE isn't
ce426f
+   required which reduces setup overhead.  Also gives a faster return
ce426f
+   for small strings on big-endian due to needing to recalculate with
ce426f
+   method (2) anyway.  */
ce426f
+#ifdef __LITTLE_ENDIAN__
ce426f
+	sld	rMASK, rMASK, rPADN
ce426f
+#else
ce426f
+ 	srd	rMASK, rMASK, rPADN
ce426f
+#endif
ce426f
 	and	rTMP1, r7F7F, rWORD1
ce426f
 	or	rTMP2, r7F7F, rWORD1
ce426f
 	lis	rFEFE, -0x101
ce426f
 	add	rTMP1, rTMP1, r7F7F
ce426f
 	addi	rFEFE, rFEFE, -0x101
ce426f
-	nor	rTMP1, rTMP2, rTMP1
ce426f
-	and.	rWORD1, rTMP1, rMASK
ce426f
+	nor	rTMP3, rTMP2, rTMP1
ce426f
+	and.	rTMP3, rTMP3, rMASK
ce426f
 	mtcrf	0x01, rRTN
ce426f
 	bne	L(done0)
ce426f
-	sldi  rTMP1, rFEFE, 32
ce426f
-	add  rFEFE, rFEFE, rTMP1
ce426f
+	sldi	rTMP1, rFEFE, 32
ce426f
+	add	rFEFE, rFEFE, rTMP1
ce426f
 /* Are we now aligned to a doubleword boundary?  */
ce426f
 	bt	28, L(loop)
ce426f
 
ce426f
 /* Handle second doubleword of pair.  */
ce426f
+/* Perhaps use method (1) here for little-endian, saving one instruction?  */
ce426f
 	ldu	rWORD1, 8(rSTR)
ce426f
 	and	rTMP1, r7F7F, rWORD1
ce426f
 	or	rTMP2, r7F7F, rWORD1
ce426f
 	add	rTMP1, rTMP1, r7F7F
ce426f
-	nor.	rWORD1, rTMP2, rTMP1
ce426f
+	nor.	rTMP3, rTMP2, rTMP1
ce426f
 	bne	L(done0)
ce426f
 
ce426f
 /* The loop.  */
ce426f
@@ -146,29 +157,53 @@
ce426f
 	add	rTMP3, rFEFE, rWORD2
ce426f
 	nor	rTMP4, r7F7F, rWORD2
ce426f
 	bne	L(done1)
ce426f
-	and.	rTMP1, rTMP3, rTMP4
ce426f
+	and.	rTMP3, rTMP3, rTMP4
ce426f
 	beq	L(loop)
ce426f
 
ce426f
+#ifndef __LITTLE_ENDIAN__
ce426f
 	and	rTMP1, r7F7F, rWORD2
ce426f
 	add	rTMP1, rTMP1, r7F7F
ce426f
-	andc	rWORD1, rTMP4, rTMP1
ce426f
+	andc	rTMP3, rTMP4, rTMP1
ce426f
 	b	L(done0)
ce426f
 
ce426f
 L(done1):
ce426f
 	and	rTMP1, r7F7F, rWORD1
ce426f
 	subi	rSTR, rSTR, 8
ce426f
 	add	rTMP1, rTMP1, r7F7F
ce426f
-	andc	rWORD1, rTMP2, rTMP1
ce426f
+	andc	rTMP3, rTMP2, rTMP1
ce426f
 
ce426f
 /* When we get to here, rSTR points to the first doubleword in the string that
ce426f
-   contains a zero byte, and the most significant set bit in rWORD1 is in that
ce426f
-   byte.  */
ce426f
+   contains a zero byte, and rTMP3 has 0x80 for bytes that are zero, and 0x00
ce426f
+   otherwise.  */
ce426f
 L(done0):
ce426f
-	cntlzd	rTMP3, rWORD1
ce426f
+	cntlzd	rTMP3, rTMP3
ce426f
 	subf	rTMP1, rRTN, rSTR
ce426f
 	srdi	rTMP3, rTMP3, 3
ce426f
 	add	rRTN, rTMP1, rTMP3
ce426f
 	/* GKM FIXME: check high bound.  */
ce426f
 	blr
ce426f
+#else
ce426f
+
ce426f
+L(done0):
ce426f
+	addi	rTMP1, rTMP3, -1	/* Form a mask from trailing zeros.  */
ce426f
+	andc	rTMP1, rTMP1, rTMP3
ce426f
+	cntlzd	rTMP1, rTMP1		/* Count bits not in the mask.  */
ce426f
+	subf	rTMP3, rRTN, rSTR
ce426f
+	subfic	rTMP1, rTMP1, 64-7
ce426f
+	srdi	rTMP1, rTMP1, 3
ce426f
+	add	rRTN, rTMP1, rTMP3
ce426f
+	blr
ce426f
+
ce426f
+L(done1):
ce426f
+	addi	rTMP3, rTMP1, -1
ce426f
+	andc	rTMP3, rTMP3, rTMP1
ce426f
+	cntlzd	rTMP3, rTMP3
ce426f
+	subf	rTMP1, rRTN, rSTR
ce426f
+	subfic	rTMP3, rTMP3, 64-7-64
ce426f
+	sradi	rTMP3, rTMP3, 3
ce426f
+	add	rRTN, rTMP1, rTMP3
ce426f
+	blr
ce426f
+#endif
ce426f
+
ce426f
 END (BP_SYM (strlen))
ce426f
 libc_hidden_builtin_def (strlen)