📄 memset.c
字号:
/*===========================================================================
//
// memset.c
//
// ANSI standard memset() routine
//
//==========================================================================
//####COPYRIGHTBEGIN####
//
// -------------------------------------------
// The contents of this file are subject to the Red Hat eCos Public License
// Version 1.1 (the "License"); you may not use this file except in
// compliance with the License. You may obtain a copy of the License at
// http://www.redhat.com/
//
// Software distributed under the License is distributed on an "AS IS"
// basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the
// License for the specific language governing rights and limitations under
// the License.
//
// The Original Code is eCos - Embedded Configurable Operating System,
// released September 30, 1998.
//
// The Initial Developer of the Original Code is Red Hat.
// Portions created by Red Hat are
// Copyright (C) 1998, 1999, 2000 Red Hat, Inc.
// All Rights Reserved.
// -------------------------------------------
//
//####COPYRIGHTEND####
//==========================================================================
//#####DESCRIPTIONBEGIN####
//
// Author(s): jlarmour
// Contributors: jlarmour
// Date: 1998-06-04
// Purpose: This file implements the ANSI memset() function
// Description: This file implements the memset() function defined in ANSI para
// 7.11.6.1. This is implemented in the kernel rather than the
// C library due to it being required by gcc whether or not the
// C library has been configured in.
//
//####DESCRIPTIONEND####
//
//==========================================================================*/
/* INCLUDES */
#include <pkgconf/infra.h> /* Configuration of infra package */
#include <cyg/infra/cyg_type.h> /* Common type definitions */
#include <cyg/infra/cyg_trac.h> /* Tracing support */
#include <cyg/infra/cyg_ass.h> /* Assertion support */
#include <stddef.h> /* Compiler defns such as size_t, NULL etc. */
/* MACROS */
/* Nonzero if X is not aligned on a word boundary. */
#define CYG_STR_UNALIGNED(X) ((CYG_WORD)(X) & (sizeof (CYG_WORD) - 1))
/* How many bytes are copied each iteration of the word copy loop in the
* optimised string implementation
*/
#define CYG_STR_OPT_LITTLEBLOCKSIZE (sizeof (CYG_WORD))
/* EXPORTED SYMBOLS */
externC void *
memset( void *s, int c, size_t n ) __attribute__ ((weak, alias("_memset")));
/* FUNCTIONS */
void *
_memset( void *s, int c, size_t n )
{
#if defined(CYGIMP_INFRA_PREFER_SMALL_TO_FAST_MEMSET) || defined(__OPTIMIZE_SIZE__)
char *char_ptr = (char *)s;
#ifdef CYG_TRACING_FIXED
CYG_REPORT_FUNCNAMETYPE( "_memset", "returning %08x" );
CYG_REPORT_FUNCARG3( "s=%08x, c=%d, n=%d", s, c, n );
if (n != 0)
{
CYG_CHECK_DATA_PTR( char_ptr, "s is not a valid pointer!" );
CYG_CHECK_DATA_PTR( (&char_ptr[n-1]), "s+n-1 is not a valid address!" );
}
#endif
while (n-- != 0)
{
*char_ptr++ = (char) c;
}
#ifdef CYG_TRACING_FIXED
CYG_REPORT_RETVAL( s );
#endif
return s;
#else
char *char_ptr = (char *)s;
cyg_ucount8 count;
CYG_WORD buffer;
CYG_WORD *aligned_addr;
char *unaligned_addr;
#ifdef CYG_TRACING_FIXED
CYG_REPORT_FUNCNAMETYPE( "_memset", "returning %08x" );
CYG_REPORT_FUNCARG3( "s=%08x, c=%d, n=%d", s, c, n );
if (n != 0)
{
CYG_CHECK_DATA_PTR( s, "s is not a valid pointer!" );
CYG_CHECK_DATA_PTR( (char *)s+n-1, "s+n-1 is not a valid address!" );
}
#endif
if (n < sizeof(CYG_WORD) || CYG_STR_UNALIGNED (s))
{
while (n-- != 0)
{
*char_ptr++ = (char) c;
}
#ifdef CYG_TRACING_FIXED
CYG_REPORT_RETVAL( s );
#endif
return s;
}
/* If we get this far, we know that n is large and s is word-aligned. */
aligned_addr = (CYG_WORD *)s;
/* Store C into each char sized location in BUFFER so that
* we can set large blocks quickly.
*/
c &= 0xff;
if (CYG_STR_OPT_LITTLEBLOCKSIZE == 4)
{
buffer = (c << 8) | c;
buffer |= (buffer << 16);
}
else
{
buffer = 0;
for (count = 0; count < CYG_STR_OPT_LITTLEBLOCKSIZE; count++)
buffer = (buffer << 8) | c;
}
while (n >= CYG_STR_OPT_LITTLEBLOCKSIZE*4)
{
*aligned_addr++ = buffer;
*aligned_addr++ = buffer;
*aligned_addr++ = buffer;
*aligned_addr++ = buffer;
n -= 4*CYG_STR_OPT_LITTLEBLOCKSIZE;
}
while (n >= CYG_STR_OPT_LITTLEBLOCKSIZE)
{
*aligned_addr++ = buffer;
n -= CYG_STR_OPT_LITTLEBLOCKSIZE;
}
/* Pick up the remainder with a bytewise loop. */
unaligned_addr = (char*)aligned_addr;
while (n)
{
*unaligned_addr++ = (char)c;
n--;
}
#ifdef CYG_TRACING_FIXED
CYG_REPORT_RETVAL( s );
#endif
return s;
#endif /* not defined(CYGIMP_PREFER_SMALL_TO_FAST_MEMSET) ||
* defined(__OPTIMIZE_SIZE__) */
} /* _memset() */
/* EOF memset.c */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -