1 /* $NetBSD: bcopy.c,v 1.9 2009/03/18 12:25:06 tsutsui 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.9 2009/03/18 12:25:06 tsutsui 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 #ifdef _FORTIFY_SOURCE
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
!= 0);
93 _DIAGASSERT(src0
!= 0);
96 if (length
== 0 || dst
== src
) /* nothing to do */
100 * Macros: loop-t-times; and loop-t-times, t>0
102 #define TLOOP(s) if (t) TLOOP1(s)
103 #define TLOOP1(s) do { s; } while (--t)
105 if ((unsigned long)dst
< (unsigned long)src
) {
109 u
= (unsigned long)src
; /* only need low bits */
110 if ((u
| (unsigned long)dst
) & wmask
) {
112 * Try to align operands. This cannot be done
113 * unless the low bits match.
115 if ((u
^ (unsigned long)dst
) & wmask
|| length
< wsize
)
118 t
= wsize
- (size_t)(u
& wmask
);
120 TLOOP1(*dst
++ = *src
++);
123 * Copy whole words, then mop up any trailing bytes.
126 TLOOP(*(word
*)(void *)dst
= *(const word
*)(const void *)src
; src
+= wsize
; dst
+= wsize
);
128 TLOOP(*dst
++ = *src
++);
131 * Copy backwards. Otherwise essentially the same.
132 * Alignment works as before, except that it takes
133 * (t&wmask) bytes to align, not wsize-(t&wmask).
137 _DIAGASSERT((unsigned long)dst
>= (unsigned long)dst0
);
138 _DIAGASSERT((unsigned long)src
>= (unsigned long)src0
);
139 u
= (unsigned long)src
;
140 if ((u
| (unsigned long)dst
) & wmask
) {
141 if ((u
^ (unsigned long)dst
) & wmask
|| length
<= wsize
)
144 t
= (size_t)(u
& wmask
);
146 TLOOP1(*--dst
= *--src
);
149 TLOOP(src
-= wsize
; dst
-= wsize
; *(word
*)(void *)dst
= *(const word
*)(const void *)src
);
151 TLOOP(*--dst
= *--src
);
154 #if defined(MEMCOPY) || defined(MEMMOVE)
160 #else /* __OPTIMIZE_SIZE__ */
163 * This is designed to be small, not fast.
166 memcpy(void *s1
, const void *s2
, size_t n
)
175 #elif defined(MEMMOVE)
177 * This is designed to be small, not fast.
180 memmove(void *s1
, const void *s2
, size_t n
)
198 * This is designed to be small, not fast.
201 bcopy(const void *s2
, void *s1
, size_t n
)
210 #endif /* __OPTIMIZE_SIZE__ */