blob: 21ca7d6b05883f66b89604242aa0e5e05d79a541 [file] [log] [blame]
/****************************************************************************
* 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 <nuttx/config.h>
#include <sys/types.h>
#include <string.h>
#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