2000-02-18 03:39:52 +08:00
|
|
|
/*
|
|
|
|
FUNCTION
|
|
|
|
<<memset>>---set an area of memory
|
|
|
|
|
|
|
|
INDEX
|
|
|
|
memset
|
|
|
|
|
|
|
|
ANSI_SYNOPSIS
|
|
|
|
#include <string.h>
|
|
|
|
void *memset(const void *<[dst]>, int <[c]>, size_t <[length]>);
|
|
|
|
|
|
|
|
TRAD_SYNOPSIS
|
|
|
|
#include <string.h>
|
|
|
|
void *memset(<[dst]>, <[c]>, <[length]>)
|
|
|
|
void *<[dst]>;
|
|
|
|
int <[c]>;
|
|
|
|
size_t <[length]>;
|
|
|
|
|
|
|
|
DESCRIPTION
|
|
|
|
This function converts the argument <[c]> into an unsigned
|
|
|
|
char and fills the first <[length]> characters of the array
|
|
|
|
pointed to by <[dst]> to the value.
|
|
|
|
|
|
|
|
RETURNS
|
2008-05-27 07:23:15 +08:00
|
|
|
<<memset>> returns the value of <[dst]>.
|
2000-02-18 03:39:52 +08:00
|
|
|
|
|
|
|
PORTABILITY
|
|
|
|
<<memset>> is ANSI C.
|
|
|
|
|
|
|
|
<<memset>> requires no supporting OS subroutines.
|
|
|
|
|
|
|
|
QUICKREF
|
|
|
|
memset ansi pure
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#define LBLOCKSIZE (sizeof(long))
|
|
|
|
#define UNALIGNED(X) ((long)X & (LBLOCKSIZE - 1))
|
|
|
|
#define TOO_SMALL(LEN) ((LEN) < LBLOCKSIZE)
|
|
|
|
|
2008-05-27 07:23:15 +08:00
|
|
|
_PTR
|
2000-02-18 03:39:52 +08:00
|
|
|
_DEFUN (memset, (m, c, n),
|
|
|
|
_PTR m _AND
|
|
|
|
int c _AND
|
|
|
|
size_t n)
|
|
|
|
{
|
|
|
|
char *s = (char *) m;
|
|
|
|
|
2008-05-27 07:23:15 +08:00
|
|
|
#if !defined(PREFER_SIZE_OVER_SPEED) && !defined(__OPTIMIZE_SIZE__)
|
2000-08-25 00:25:36 +08:00
|
|
|
int i;
|
2000-02-18 03:39:52 +08:00
|
|
|
unsigned long buffer;
|
|
|
|
unsigned long *aligned_addr;
|
2002-11-28 02:10:16 +08:00
|
|
|
unsigned int d = c & 0xff; /* To avoid sign extension, copy C to an
|
|
|
|
unsigned variable. */
|
2000-02-18 03:39:52 +08:00
|
|
|
|
2008-05-27 07:23:15 +08:00
|
|
|
while (UNALIGNED (s))
|
2000-02-18 03:39:52 +08:00
|
|
|
{
|
2008-05-27 07:23:15 +08:00
|
|
|
if (n--)
|
|
|
|
*s++ = (char) 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 = (unsigned long *) s;
|
2000-02-18 03:39:52 +08:00
|
|
|
|
2002-11-26 04:56:17 +08:00
|
|
|
/* Store D into each char sized location in BUFFER so that
|
2000-02-18 03:39:52 +08:00
|
|
|
we can set large blocks quickly. */
|
2008-05-27 07:23:15 +08:00
|
|
|
buffer = (d << 8) | d;
|
|
|
|
buffer |= (buffer << 16);
|
|
|
|
for (i = 32; i < LBLOCKSIZE * 8; i <<= 1)
|
|
|
|
buffer = (buffer << i) | buffer;
|
2000-02-18 03:39:52 +08:00
|
|
|
|
2008-05-27 07:23:15 +08:00
|
|
|
/* Unroll the loop. */
|
2000-02-18 03:39:52 +08:00
|
|
|
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 = (char*)aligned_addr;
|
|
|
|
}
|
|
|
|
|
2008-05-27 07:23:15 +08:00
|
|
|
#endif /* not PREFER_SIZE_OVER_SPEED */
|
|
|
|
|
2000-02-18 03:39:52 +08:00
|
|
|
while (n--)
|
2008-05-27 07:23:15 +08:00
|
|
|
*s++ = (char) c;
|
2000-02-18 03:39:52 +08:00
|
|
|
|
|
|
|
return m;
|
|
|
|
}
|