OSDN Git Service

- trim any trailing whitespace
[uclinux-h8/uClibc.git] / libc / string / powerpc / memcpy.c
1 /*
2  * Copyright (C) 2004 Joakim Tjernlund
3  * Copyright (C) 2000-2005 Erik Andersen <andersen@uclibc.org>
4  *
5  * Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball.
6  */
7
8 /* These are carefully optimized mem*() functions for PPC written in C.
9  * Don't muck around with these function without checking the generated
10  * assmbler code.
11  * It is possible to optimize these significantly more by using specific
12  * data cache instructions(mainly dcbz). However that requires knownledge
13  * about the CPU's cache line size.
14  *
15  * BUG ALERT!
16  * The cache instructions on MPC8xx CPU's are buggy(they don't update
17  * the DAR register when causing a DTLB Miss/Error) and cannot be
18  * used on 8xx CPU's without a kernel patch to work around this
19  * problem.
20  */
21
22 #include <string.h>
23
24 /* Experimentally off - libc_hidden_proto(memcpy) */
25 void *memcpy(void *to, const void *from, size_t n)
26 /* PPC can do pre increment and load/store, but not post increment and load/store.
27    Therefore use *++ptr instead of *ptr++. */
28 {
29         unsigned long rem, chunks, tmp1, tmp2;
30         unsigned char *tmp_to;
31         unsigned char *tmp_from = (unsigned char *)from;
32
33         chunks = n / 8;
34         tmp_from -= 4;
35         tmp_to = to - 4;
36         if (!chunks)
37                 goto lessthan8;
38         rem = (unsigned long )tmp_to % 4;
39         if (rem)
40                 goto align;
41  copy_chunks:
42         do {
43                 /* make gcc to load all data, then store it */
44                 tmp1 = *(unsigned long *)(tmp_from+4);
45                 tmp_from += 8;
46                 tmp2 = *(unsigned long *)tmp_from;
47                 *(unsigned long *)(tmp_to+4) = tmp1;
48                 tmp_to += 8;
49                 *(unsigned long *)tmp_to = tmp2;
50         } while (--chunks);
51  lessthan8:
52         n = n % 8;
53         if (n >= 4) {
54                 *(unsigned long *)(tmp_to+4) = *(unsigned long *)(tmp_from+4);
55                 tmp_from += 4;
56                 tmp_to += 4;
57                 n = n-4;
58         }
59         if (!n ) return to;
60         tmp_from += 3;
61         tmp_to += 3;
62         do {
63                 *++tmp_to = *++tmp_from;
64         } while (--n);
65
66         return to;
67  align:
68         rem = 4 - rem;
69         n = n - rem;
70         do {
71                 *(tmp_to+4) = *(tmp_from+4);
72                 ++tmp_from;
73                 ++tmp_to;
74         } while (--rem);
75         chunks = n / 8;
76         if (chunks)
77                 goto copy_chunks;
78         goto lessthan8;
79 }
80 libc_hidden_def(memcpy)