1*0a6a1f1dSLionel Sambuc /* $NetBSD: memset.c,v 1.10 2013/12/02 21:21:33 joerg Exp $ */
2b6cbf720SGianluca Guida
3b6cbf720SGianluca Guida /*-
4b6cbf720SGianluca Guida * Copyright (c) 1990, 1993
5b6cbf720SGianluca Guida * The Regents of the University of California. All rights reserved.
6b6cbf720SGianluca Guida *
7b6cbf720SGianluca Guida * This code is derived from software contributed to Berkeley by
8b6cbf720SGianluca Guida * Mike Hibler and Chris Torek.
9b6cbf720SGianluca Guida *
10b6cbf720SGianluca Guida * Redistribution and use in source and binary forms, with or without
11b6cbf720SGianluca Guida * modification, are permitted provided that the following conditions
12b6cbf720SGianluca Guida * are met:
13b6cbf720SGianluca Guida * 1. Redistributions of source code must retain the above copyright
14b6cbf720SGianluca Guida * notice, this list of conditions and the following disclaimer.
15b6cbf720SGianluca Guida * 2. Redistributions in binary form must reproduce the above copyright
16b6cbf720SGianluca Guida * notice, this list of conditions and the following disclaimer in the
17b6cbf720SGianluca Guida * documentation and/or other materials provided with the distribution.
18b6cbf720SGianluca Guida * 3. Neither the name of the University nor the names of its contributors
19b6cbf720SGianluca Guida * may be used to endorse or promote products derived from this software
20b6cbf720SGianluca Guida * without specific prior written permission.
21b6cbf720SGianluca Guida *
22b6cbf720SGianluca Guida * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
23b6cbf720SGianluca Guida * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
24b6cbf720SGianluca Guida * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
25b6cbf720SGianluca Guida * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
26b6cbf720SGianluca Guida * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
27b6cbf720SGianluca Guida * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
28b6cbf720SGianluca Guida * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
29b6cbf720SGianluca Guida * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30b6cbf720SGianluca Guida * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
31b6cbf720SGianluca Guida * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32b6cbf720SGianluca Guida * SUCH DAMAGE.
33b6cbf720SGianluca Guida */
34b6cbf720SGianluca Guida
35b6cbf720SGianluca Guida #include <sys/cdefs.h>
36b6cbf720SGianluca Guida #if defined(LIBC_SCCS) && !defined(lint)
37b6cbf720SGianluca Guida #if 0
38b6cbf720SGianluca Guida static char sccsid[] = "@(#)memset.c 8.1 (Berkeley) 6/4/93";
39b6cbf720SGianluca Guida #else
40*0a6a1f1dSLionel Sambuc __RCSID("$NetBSD: memset.c,v 1.10 2013/12/02 21:21:33 joerg Exp $");
41b6cbf720SGianluca Guida #endif
42b6cbf720SGianluca Guida #endif /* LIBC_SCCS and not lint */
43b6cbf720SGianluca Guida
44b6cbf720SGianluca Guida #include <sys/types.h>
45b6cbf720SGianluca Guida
46b6cbf720SGianluca Guida #if !defined(_KERNEL) && !defined(_STANDALONE)
47b6cbf720SGianluca Guida #include <assert.h>
48b6cbf720SGianluca Guida #include <limits.h>
49b6cbf720SGianluca Guida #include <string.h>
50b6cbf720SGianluca Guida #else
51b6cbf720SGianluca Guida #include <lib/libkern/libkern.h>
52b6cbf720SGianluca Guida #if defined(BZERO) && defined(_STANDALONE)
53b6cbf720SGianluca Guida #include <lib/libsa/stand.h>
54b6cbf720SGianluca Guida #endif
55b6cbf720SGianluca Guida #include <machine/limits.h>
56b6cbf720SGianluca Guida #endif
57b6cbf720SGianluca Guida
58b6cbf720SGianluca Guida #define wsize sizeof(u_int)
59b6cbf720SGianluca Guida #define wmask (wsize - 1)
60b6cbf720SGianluca Guida
61b6cbf720SGianluca Guida #ifdef _FORTIFY_SOURCE
62b6cbf720SGianluca Guida #undef bzero
63b6cbf720SGianluca Guida #endif
64f14fb602SLionel Sambuc #undef memset
65b6cbf720SGianluca Guida
66b6cbf720SGianluca Guida #ifndef __OPTIMIZE_SIZE__
67b6cbf720SGianluca Guida #ifdef BZERO
68b6cbf720SGianluca Guida #define RETURN return
69b6cbf720SGianluca Guida #define VAL 0
70b6cbf720SGianluca Guida #define WIDEVAL 0
71b6cbf720SGianluca Guida
72b6cbf720SGianluca Guida void
bzero(void * dst0,size_t length)73b6cbf720SGianluca Guida bzero(void *dst0, size_t length)
74b6cbf720SGianluca Guida #else
75b6cbf720SGianluca Guida #define RETURN return (dst0)
76b6cbf720SGianluca Guida #define VAL c0
77b6cbf720SGianluca Guida #define WIDEVAL c
78b6cbf720SGianluca Guida
79*0a6a1f1dSLionel Sambuc #if defined(__ARM_EABI__)
80*0a6a1f1dSLionel Sambuc void __aeabi_memset(void *, size_t, int);
81*0a6a1f1dSLionel Sambuc void __aeabi_memclr(void *, size_t);
82*0a6a1f1dSLionel Sambuc
83*0a6a1f1dSLionel Sambuc __strong_alias(__aeabi_memset4, __aebi_memset)
84*0a6a1f1dSLionel Sambuc __strong_alias(__aeabi_memset8, __aebi_memset)
85*0a6a1f1dSLionel Sambuc
86*0a6a1f1dSLionel Sambuc void
87*0a6a1f1dSLionel Sambuc __aeabi_memset(void *dst0, size_t length, int c)
88*0a6a1f1dSLionel Sambuc {
89*0a6a1f1dSLionel Sambuc memset(dst0, c, length);
90*0a6a1f1dSLionel Sambuc }
91*0a6a1f1dSLionel Sambuc
92*0a6a1f1dSLionel Sambuc void
93*0a6a1f1dSLionel Sambuc __aeabi_memclr(void *dst0, size_t length)
94*0a6a1f1dSLionel Sambuc {
95*0a6a1f1dSLionel Sambuc memset(dst0, 0, length);
96*0a6a1f1dSLionel Sambuc }
97*0a6a1f1dSLionel Sambuc #endif
98*0a6a1f1dSLionel Sambuc
99b6cbf720SGianluca Guida void *
100b6cbf720SGianluca Guida memset(void *dst0, int c0, size_t length)
101b6cbf720SGianluca Guida #endif
102b6cbf720SGianluca Guida {
103b6cbf720SGianluca Guida size_t t;
104b6cbf720SGianluca Guida #ifndef BZERO
105b6cbf720SGianluca Guida u_int c;
106b6cbf720SGianluca Guida #endif
107b6cbf720SGianluca Guida u_char *dst;
108b6cbf720SGianluca Guida
109b6cbf720SGianluca Guida _DIAGASSERT(dst0 != 0);
110b6cbf720SGianluca Guida
111b6cbf720SGianluca Guida dst = dst0;
112b6cbf720SGianluca Guida /*
113b6cbf720SGianluca Guida * If not enough words, just fill bytes. A length >= 2 words
114b6cbf720SGianluca Guida * guarantees that at least one of them is `complete' after
115b6cbf720SGianluca Guida * any necessary alignment. For instance:
116b6cbf720SGianluca Guida *
117b6cbf720SGianluca Guida * |-----------|-----------|-----------|
118b6cbf720SGianluca Guida * |00|01|02|03|04|05|06|07|08|09|0A|00|
119b6cbf720SGianluca Guida * ^---------------------^
120b6cbf720SGianluca Guida * dst dst+length-1
121b6cbf720SGianluca Guida *
122b6cbf720SGianluca Guida * but we use a minimum of 3 here since the overhead of the code
123b6cbf720SGianluca Guida * to do word writes is substantial.
124b6cbf720SGianluca Guida */
125b6cbf720SGianluca Guida if (length < 3 * wsize) {
126b6cbf720SGianluca Guida while (length != 0) {
127b6cbf720SGianluca Guida *dst++ = VAL;
128b6cbf720SGianluca Guida --length;
129b6cbf720SGianluca Guida }
130b6cbf720SGianluca Guida RETURN;
131b6cbf720SGianluca Guida }
132b6cbf720SGianluca Guida
133b6cbf720SGianluca Guida #ifndef BZERO
134b6cbf720SGianluca Guida if ((c = (u_char)c0) != 0) { /* Fill the word. */
135b6cbf720SGianluca Guida c = (c << 8) | c; /* u_int is 16 bits. */
136b6cbf720SGianluca Guida #if UINT_MAX > 0xffff
137b6cbf720SGianluca Guida c = (c << 16) | c; /* u_int is 32 bits. */
138b6cbf720SGianluca Guida #endif
139b6cbf720SGianluca Guida #if UINT_MAX > 0xffffffff
140b6cbf720SGianluca Guida c = (c << 32) | c; /* u_int is 64 bits. */
141b6cbf720SGianluca Guida #endif
142b6cbf720SGianluca Guida }
143b6cbf720SGianluca Guida #endif
144b6cbf720SGianluca Guida /* Align destination by filling in bytes. */
145b6cbf720SGianluca Guida if ((t = (size_t)((u_long)dst & wmask)) != 0) {
146b6cbf720SGianluca Guida t = wsize - t;
147b6cbf720SGianluca Guida length -= t;
148b6cbf720SGianluca Guida do {
149b6cbf720SGianluca Guida *dst++ = VAL;
150b6cbf720SGianluca Guida } while (--t != 0);
151b6cbf720SGianluca Guida }
152b6cbf720SGianluca Guida
153b6cbf720SGianluca Guida /* Fill words. Length was >= 2*words so we know t >= 1 here. */
154b6cbf720SGianluca Guida t = length / wsize;
155b6cbf720SGianluca Guida do {
156b6cbf720SGianluca Guida *(u_int *)(void *)dst = WIDEVAL;
157b6cbf720SGianluca Guida dst += wsize;
158b6cbf720SGianluca Guida } while (--t != 0);
159b6cbf720SGianluca Guida
160b6cbf720SGianluca Guida /* Mop up trailing bytes, if any. */
161b6cbf720SGianluca Guida t = length & wmask;
162b6cbf720SGianluca Guida if (t != 0)
163b6cbf720SGianluca Guida do {
164b6cbf720SGianluca Guida *dst++ = VAL;
165b6cbf720SGianluca Guida } while (--t != 0);
166b6cbf720SGianluca Guida RETURN;
167b6cbf720SGianluca Guida }
168b6cbf720SGianluca Guida #else /* __OPTIMIZE_SIZE__ */
169b6cbf720SGianluca Guida #ifdef BZERO
170b6cbf720SGianluca Guida void
bzero(void * dstv,size_t length)171b6cbf720SGianluca Guida bzero(void *dstv, size_t length)
172b6cbf720SGianluca Guida {
173b6cbf720SGianluca Guida u_char *dst = dstv;
174b6cbf720SGianluca Guida while (length-- > 0)
175b6cbf720SGianluca Guida *dst++ = 0;
176b6cbf720SGianluca Guida }
177b6cbf720SGianluca Guida #else
178b6cbf720SGianluca Guida void *
memset(void * dstv,int c,size_t length)179b6cbf720SGianluca Guida memset(void *dstv, int c, size_t length)
180b6cbf720SGianluca Guida {
181b6cbf720SGianluca Guida u_char *dst = dstv;
182b6cbf720SGianluca Guida while (length-- > 0)
183b6cbf720SGianluca Guida *dst++ = c;
184b6cbf720SGianluca Guida return dstv;
185b6cbf720SGianluca Guida }
186b6cbf720SGianluca Guida #endif /* BZERO */
187b6cbf720SGianluca Guida #endif /* __OPTIMIZE_SIZE__ */
188