Replace FSF snail mail address with URLs.
[glibc.git] / sysdeps / powerpc / powerpc32 / power4 / wordcopy.c
blobe8a35eaeef6e92febbda7f9b8ccea161db5a82de
1 /* _memcopy.c -- subroutines for memory copy functions.
2 Copyright (C) 1991, 1996 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
4 Contributed by Torbjorn Granlund (tege@sics.se).
6 The GNU C Library is free software; you can redistribute it and/or
7 modify it under the terms of the GNU Lesser General Public
8 License as published by the Free Software Foundation; either
9 version 2.1 of the License, or (at your option) any later version.
11 The GNU C Library is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 Lesser General Public License for more details.
16 You should have received a copy of the GNU Lesser General Public
17 License along with the GNU C Library; if not, see
18 <http://www.gnu.org/licenses/>. */
20 /* BE VERY CAREFUL IF YOU CHANGE THIS CODE...! */
22 #include <stddef.h>
23 #include <memcopy.h>
25 /* _wordcopy_fwd_aligned -- Copy block beginning at SRCP to
26 block beginning at DSTP with LEN `op_t' words (not LEN bytes!).
27 Both SRCP and DSTP should be aligned for memory operations on `op_t's. */
29 void
30 _wordcopy_fwd_aligned (dstp, srcp, len)
31 long int dstp;
32 long int srcp;
33 size_t len;
35 op_t a0, a1;
37 if (len & 1)
39 ((op_t *) dstp)[0] = ((op_t *) srcp)[0];
41 if (len == 1)
42 return;
43 srcp += OPSIZ;
44 dstp += OPSIZ;
45 len -= 1;
50 a0 = ((op_t *) srcp)[0];
51 a1 = ((op_t *) srcp)[1];
52 ((op_t *) dstp)[0] = a0;
53 ((op_t *) dstp)[1] = a1;
55 srcp += 2 * OPSIZ;
56 dstp += 2 * OPSIZ;
57 len -= 2;
59 while (len != 0);
62 /* _wordcopy_fwd_dest_aligned -- Copy block beginning at SRCP to
63 block beginning at DSTP with LEN `op_t' words (not LEN bytes!).
64 DSTP should be aligned for memory operations on `op_t's, but SRCP must
65 *not* be aligned. */
67 void
68 _wordcopy_fwd_dest_aligned (dstp, srcp, len)
69 long int dstp;
70 long int srcp;
71 size_t len;
73 op_t a0, a1, a2;
74 int sh_1, sh_2;
76 /* Calculate how to shift a word read at the memory operation
77 aligned srcp to make it aligned for copy. */
79 sh_1 = 8 * (srcp % OPSIZ);
80 sh_2 = 8 * OPSIZ - sh_1;
82 /* Make SRCP aligned by rounding it down to the beginning of the `op_t'
83 it points in the middle of. */
84 srcp &= -OPSIZ;
85 a0 = ((op_t *) srcp)[0];
87 if (len & 1)
89 a1 = ((op_t *) srcp)[1];
90 ((op_t *) dstp)[0] = MERGE (a0, sh_1, a1, sh_2);
92 if (len == 1)
93 return;
95 a0 = a1;
96 srcp += OPSIZ;
97 dstp += OPSIZ;
98 len -= 1;
103 a1 = ((op_t *) srcp)[1];
104 a2 = ((op_t *) srcp)[2];
105 ((op_t *) dstp)[0] = MERGE (a0, sh_1, a1, sh_2);
106 ((op_t *) dstp)[1] = MERGE (a1, sh_1, a2, sh_2);
107 a0 = a2;
109 srcp += 2 * OPSIZ;
110 dstp += 2 * OPSIZ;
111 len -= 2;
113 while (len != 0);
116 /* _wordcopy_bwd_aligned -- Copy block finishing right before
117 SRCP to block finishing right before DSTP with LEN `op_t' words
118 (not LEN bytes!). Both SRCP and DSTP should be aligned for memory
119 operations on `op_t's. */
121 void
122 _wordcopy_bwd_aligned (dstp, srcp, len)
123 long int dstp;
124 long int srcp;
125 size_t len;
127 op_t a0, a1;
129 if (len & 1)
131 srcp -= OPSIZ;
132 dstp -= OPSIZ;
133 ((op_t *) dstp)[0] = ((op_t *) srcp)[0];
135 if (len == 1)
136 return;
137 len -= 1;
142 srcp -= 2 * OPSIZ;
143 dstp -= 2 * OPSIZ;
145 a1 = ((op_t *) srcp)[1];
146 a0 = ((op_t *) srcp)[0];
147 ((op_t *) dstp)[1] = a1;
148 ((op_t *) dstp)[0] = a0;
150 len -= 2;
152 while (len != 0);
155 /* _wordcopy_bwd_dest_aligned -- Copy block finishing right
156 before SRCP to block finishing right before DSTP with LEN `op_t'
157 words (not LEN bytes!). DSTP should be aligned for memory
158 operations on `op_t', but SRCP must *not* be aligned. */
160 void
161 _wordcopy_bwd_dest_aligned (dstp, srcp, len)
162 long int dstp;
163 long int srcp;
164 size_t len;
166 op_t a0, a1, a2;
167 int sh_1, sh_2;
169 /* Calculate how to shift a word read at the memory operation
170 aligned srcp to make it aligned for copy. */
172 sh_1 = 8 * (srcp % OPSIZ);
173 sh_2 = 8 * OPSIZ - sh_1;
175 /* Make srcp aligned by rounding it down to the beginning of the op_t
176 it points in the middle of. */
177 srcp &= -OPSIZ;
178 a2 = ((op_t *) srcp)[0];
180 if (len & 1)
182 srcp -= OPSIZ;
183 dstp -= OPSIZ;
184 a1 = ((op_t *) srcp)[0];
185 ((op_t *) dstp)[0] = MERGE (a1, sh_1, a2, sh_2);
187 if (len == 1)
188 return;
190 a2 = a1;
191 len -= 1;
196 srcp -= 2 * OPSIZ;
197 dstp -= 2 * OPSIZ;
199 a1 = ((op_t *) srcp)[1];
200 a0 = ((op_t *) srcp)[0];
201 ((op_t *) dstp)[1] = MERGE (a1, sh_1, a2, sh_2);
202 ((op_t *) dstp)[0] = MERGE (a0, sh_1, a1, sh_2);
203 a2 = a0;
205 len -= 2;
207 while (len != 0);