1 /* Optimized strcpy implementation for PowerPC64.
2 Copyright (C) 1997, 1999, 2000, 2002, 2003 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, write to the Free
17 Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
24 /* See strlen.s for comments on how the end-of-string testing works. */
26 /* char * [r3] strcpy (char *dest [r3], const char *src [r4]) */
28 EALIGN (BP_SYM (strcpy), 4, 0)
32 #define rRTN r3 /* incoming DEST arg preserved as result */
33 /* Note. The Bounded pointer support in this code is broken. This code
34 was inherited from PPC32 and and that support was never completed.
35 Current PPC gcc does not support -fbounds-check or -fbounded-pointers.
36 These artifacts are left in the code as a reminder in case we need
37 bounded pointer support in the future. */
38 #if __BOUNDED_POINTERS__
39 # define rDEST r4 /* pointer to previous word in dest */
40 # define rSRC r5 /* pointer to previous word in src */
44 # define rSRC r4 /* pointer to previous word in src */
45 # define rDEST r5 /* pointer to previous word in dest */
47 #define rWORD r6 /* current word from src */
48 #define rFEFE r7 /* constant 0xfefefefefefefeff (-0x0101010101010101) */
49 #define r7F7F r8 /* constant 0x7f7f7f7f7f7f7f7f */
50 #define rNEG r9 /* ~(word in s1 | 0x7f7f7f7f7f7f7f7f) */
51 #define rALT r10 /* alternate word from src */
53 CHECK_BOUNDS_LOW (rSRC, rLOW, rHIGH)
54 CHECK_BOUNDS_LOW (rDEST, rLOW, rHIGH)
55 STORE_RETURN_BOUNDS (rLOW, rHIGH)
59 clrldi. rTMP, rTMP, 61
60 #if __BOUNDED_POINTERS__
71 addi rFEFE, rFEFE, -0x101
72 addi r7F7F, r7F7F, 0x7f7f
74 insrdi r7F7F, r7F7F, 32, 0
75 add rFEFE, rFEFE, rTMP
78 L(g0): ldu rALT, 8(rSRC)
86 L(g2): add rTMP, rFEFE, rWORD
87 nor rNEG, r7F7F, rWORD
92 /* We've hit the end of the string. Do the rest byte-by-byte. */
94 extrdi. rTMP, rALT, 8, 0
97 extrdi. rTMP, rALT, 8, 8
100 extrdi. rTMP, rALT, 8, 16
103 extrdi. rTMP, rALT, 8, 24
106 extrdi. rTMP, rALT, 8, 32
109 extrdi. rTMP, rALT, 8, 40
112 extrdi. rTMP, rALT, 8, 48
116 /* GKM FIXME: check high bound. */
119 /* Oh well. In this case, we just do a byte-by-byte copy. */
128 L(u0): lbzu rALT, 1(rSRC)
132 nop /* Let 601 load start of loop. */
137 L(u2): stb rWORD, 1(rDEST)
138 /* GKM FIXME: check high bound. */
140 L(u1): stb rALT, 1(rDEST)
141 /* GKM FIXME: check high bound. */
144 END (BP_SYM (strcpy))
145 libc_hidden_builtin_def (strcpy)