xref: /minix3/common/lib/libc/string/memset.c (revision 0a6a1f1d05b60e214de2f05a7310ddd1f0e590e7)
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