/**************************************************************************** * libs/libc/string/lib_bsdmemset.c * * SPDX-License-Identifier: BSD * SPDX-FileCopyrightText: 1994-2009 Red Hat, Inc. All rights reserved * * Copyright (c) 1994-2009 Red Hat, Inc. All rights reserved. * * This copyrighted material is made available to anyone wishing to use, * modify, copy, or redistribute it subject to the terms and conditions * of the BSD License. This program is distributed in the hope that * it will be useful, but WITHOUT ANY WARRANTY expressed or implied, * including the implied warranties of MERCHANTABILITY or FITNESS FOR * A PARTICULAR PURPOSE. A copy of this license is available at * http://www.opensource.org/licenses. Any Red Hat trademarks that are * incorporated in the source code or documentation are not subject to * the BSD License and may only be used or replicated with the express * permission of Red Hat, Inc. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include "libc.h" /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ #define LBLOCKSIZE (sizeof(long)) #define UNALIGNED(x) ((long)(x) & (LBLOCKSIZE - 1)) #define TOO_SMALL(len) ((len) < LBLOCKSIZE) /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: memset ****************************************************************************/ #if !defined(CONFIG_LIBC_ARCH_MEMSET) && defined(LIBC_BUILD_MEMSET) #undef memset no_builtin("memset") FAR void *memset(FAR void *m, int c, size_t n) { FAR unsigned long *aligned_addr; FAR char *s = (FAR char *)m; unsigned long buffer; int i; /* To avoid sign extension, copy C to an unsigned variable. */ while (UNALIGNED(s)) { if (n--) { *s++ = c; } else { return m; } } if (!TOO_SMALL(n)) { /* If we get this far, we know that n is large and s is word-aligned. */ aligned_addr = (FAR unsigned long *)s; buffer = ((unsigned int)c << 8) | c; buffer |= (buffer << 16); for (i = 32; i < LBLOCKSIZE * 8; i <<= 1) { buffer = (buffer << i) | buffer; } /* Unroll the loop. */ while (n >= LBLOCKSIZE * 4) { *aligned_addr++ = buffer; *aligned_addr++ = buffer; *aligned_addr++ = buffer; *aligned_addr++ = buffer; n -= 4 * LBLOCKSIZE; } while (n >= LBLOCKSIZE) { *aligned_addr++ = buffer; n -= LBLOCKSIZE; } /* Pick up the remainder with a bytewise loop. */ s = (FAR char *)aligned_addr; } while (n--) { *s++ = c; } return m; } #endif