1 /* $NetBSD: bcopy.c,v 1.11 2014/04/16 20:39:55 joerg Exp $ */
4 * Copyright (c) 1990, 1993
5 * The Regents of the University of California. All rights reserved.
7 * This code is derived from software contributed to Berkeley by
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
13 * 1. Redistributions of source code must retain the above copyright
14 * notice, this list of conditions and the following disclaimer.
15 * 2. Redistributions in binary form must reproduce the above copyright
16 * notice, this list of conditions and the following disclaimer in the
17 * documentation and/or other materials provided with the distribution.
18 * 3. Neither the name of the University nor the names of its contributors
19 * may be used to endorse or promote products derived from this software
20 * without specific prior written permission.
22 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
23 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
24 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
25 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
26 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
27 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
28 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
29 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
31 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
35 #include <sys/cdefs.h>
36 #if defined(LIBC_SCCS) && !defined(lint)
38 static char sccsid
[] = "@(#)bcopy.c 8.1 (Berkeley) 6/4/93";
40 __RCSID("$NetBSD: bcopy.c,v 1.11 2014/04/16 20:39:55 joerg Exp $");
42 #endif /* LIBC_SCCS and not lint */
44 #if !defined(_KERNEL) && !defined(_STANDALONE)
48 #include <lib/libkern/libkern.h>
49 #if !defined(MEMCOPY) && defined(_STANDALONE)
50 #include <lib/libsa/stand.h>
54 #if defined(_FORTIFY_SOURCE) || defined(_STANDALONE) || defined(_KERNEL)
60 #ifndef __OPTIMIZE_SIZE__
62 * sizeof(word) MUST BE A POWER OF TWO
63 * SO THAT wmask BELOW IS ALL ONES
65 typedef long word
; /* "word" used for optimal copy speed */
67 #define wsize sizeof(word)
68 #define wmask (wsize - 1)
71 * Copy a block of memory, handling overlap.
72 * This is the routine that actually implements
73 * (the portable versions of) bcopy, memcpy, and memmove.
77 memcpy(void *dst0
, const void *src0
, size_t length
)
78 #elif defined(MEMMOVE)
80 memmove(void *dst0
, const void *src0
, size_t length
)
83 bcopy(const void *src0
, void *dst0
, size_t length
)
87 const char *src
= src0
;
92 _DIAGASSERT((dst0
&& src0
) || length
== 0);
95 if (length
== 0 || dst
== src
) /* nothing to do */
99 * Macros: loop-t-times; and loop-t-times, t>0
101 #define TLOOP(s) if (t) TLOOP1(s)
102 #define TLOOP1(s) do { s; } while (--t)
104 if ((unsigned long)dst
< (unsigned long)src
) {
108 u
= (unsigned long)src
; /* only need low bits */
109 if ((u
| (unsigned long)dst
) & wmask
) {
111 * Try to align operands. This cannot be done
112 * unless the low bits match.
114 if ((u
^ (unsigned long)dst
) & wmask
|| length
< wsize
)
117 t
= wsize
- (size_t)(u
& wmask
);
119 TLOOP1(*dst
++ = *src
++);
122 * Copy whole words, then mop up any trailing bytes.
125 TLOOP(*(word
*)(void *)dst
= *(const word
*)(const void *)src
; src
+= wsize
; dst
+= wsize
);
127 TLOOP(*dst
++ = *src
++);
130 * Copy backwards. Otherwise essentially the same.
131 * Alignment works as before, except that it takes
132 * (t&wmask) bytes to align, not wsize-(t&wmask).
136 _DIAGASSERT((unsigned long)dst
>= (unsigned long)dst0
);
137 _DIAGASSERT((unsigned long)src
>= (unsigned long)src0
);
138 u
= (unsigned long)src
;
139 if ((u
| (unsigned long)dst
) & wmask
) {
140 if ((u
^ (unsigned long)dst
) & wmask
|| length
<= wsize
)
143 t
= (size_t)(u
& wmask
);
145 TLOOP1(*--dst
= *--src
);
148 TLOOP(src
-= wsize
; dst
-= wsize
; *(word
*)(void *)dst
= *(const word
*)(const void *)src
);
150 TLOOP(*--dst
= *--src
);
153 #if defined(MEMCOPY) || defined(MEMMOVE)
159 #else /* __OPTIMIZE_SIZE__ */
162 * This is designed to be small, not fast.
165 memcpy(void *s1
, const void *s2
, size_t n
)
174 #elif defined(MEMMOVE)
176 * This is designed to be small, not fast.
179 memmove(void *s1
, const void *s2
, size_t n
)
197 * This is designed to be small, not fast.
200 bcopy(const void *s2
, void *s1
, size_t n
)
209 #endif /* __OPTIMIZE_SIZE__ */