Blame string/memmove.c

Packit 6c4009
/* Copy memory to memory until the specified number of bytes
Packit 6c4009
   has been copied.  Overlap is handled correctly.
Packit 6c4009
   Copyright (C) 1991-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
   Contributed by Torbjorn Granlund (tege@sics.se).
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <string.h>
Packit 6c4009
#include <memcopy.h>
Packit 6c4009
Packit 6c4009
/* All this is so that bcopy.c can #include
Packit 6c4009
   this file after defining some things.  */
Packit 6c4009
#ifndef	a1
Packit 6c4009
#define	a1	dest	/* First arg is DEST.  */
Packit 6c4009
#define	a1const
Packit 6c4009
#define	a2	src	/* Second arg is SRC.  */
Packit 6c4009
#define	a2const	const
Packit 6c4009
#undef memmove
Packit 6c4009
#endif
Packit 6c4009
#if	!defined(RETURN) || !defined(rettype)
Packit 6c4009
#define	RETURN(s)	return (s)	/* Return DEST.  */
Packit 6c4009
#define	rettype		void *
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#ifndef MEMMOVE
Packit 6c4009
#define MEMMOVE memmove
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
rettype
Packit 6c4009
inhibit_loop_to_libcall
Packit 6c4009
MEMMOVE (a1const void *a1, a2const void *a2, size_t len)
Packit 6c4009
{
Packit 6c4009
  unsigned long int dstp = (long int) dest;
Packit 6c4009
  unsigned long int srcp = (long int) src;
Packit 6c4009
Packit 6c4009
  /* This test makes the forward copying code be used whenever possible.
Packit 6c4009
     Reduces the working set.  */
Packit 6c4009
  if (dstp - srcp >= len)	/* *Unsigned* compare!  */
Packit 6c4009
    {
Packit 6c4009
      /* Copy from the beginning to the end.  */
Packit 6c4009
Packit 6c4009
#if MEMCPY_OK_FOR_FWD_MEMMOVE
Packit 6c4009
      dest = memcpy (dest, src, len);
Packit 6c4009
#else
Packit 6c4009
      /* If there not too few bytes to copy, use word copy.  */
Packit 6c4009
      if (len >= OP_T_THRES)
Packit 6c4009
	{
Packit 6c4009
	  /* Copy just a few bytes to make DSTP aligned.  */
Packit 6c4009
	  len -= (-dstp) % OPSIZ;
Packit 6c4009
	  BYTE_COPY_FWD (dstp, srcp, (-dstp) % OPSIZ);
Packit 6c4009
Packit 6c4009
	  /* Copy whole pages from SRCP to DSTP by virtual address
Packit 6c4009
	     manipulation, as much as possible.  */
Packit 6c4009
Packit 6c4009
	  PAGE_COPY_FWD_MAYBE (dstp, srcp, len, len);
Packit 6c4009
Packit 6c4009
	  /* Copy from SRCP to DSTP taking advantage of the known
Packit 6c4009
	     alignment of DSTP.  Number of bytes remaining is put
Packit 6c4009
	     in the third argument, i.e. in LEN.  This number may
Packit 6c4009
	     vary from machine to machine.  */
Packit 6c4009
Packit 6c4009
	  WORD_COPY_FWD (dstp, srcp, len, len);
Packit 6c4009
Packit 6c4009
	  /* Fall out and copy the tail.  */
Packit 6c4009
	}
Packit 6c4009
Packit 6c4009
      /* There are just a few bytes to copy.  Use byte memory operations.  */
Packit 6c4009
      BYTE_COPY_FWD (dstp, srcp, len);
Packit 6c4009
#endif /* MEMCPY_OK_FOR_FWD_MEMMOVE */
Packit 6c4009
    }
Packit 6c4009
  else
Packit 6c4009
    {
Packit 6c4009
      /* Copy from the end to the beginning.  */
Packit 6c4009
      srcp += len;
Packit 6c4009
      dstp += len;
Packit 6c4009
Packit 6c4009
      /* If there not too few bytes to copy, use word copy.  */
Packit 6c4009
      if (len >= OP_T_THRES)
Packit 6c4009
	{
Packit 6c4009
	  /* Copy just a few bytes to make DSTP aligned.  */
Packit 6c4009
	  len -= dstp % OPSIZ;
Packit 6c4009
	  BYTE_COPY_BWD (dstp, srcp, dstp % OPSIZ);
Packit 6c4009
Packit 6c4009
	  /* Copy from SRCP to DSTP taking advantage of the known
Packit 6c4009
	     alignment of DSTP.  Number of bytes remaining is put
Packit 6c4009
	     in the third argument, i.e. in LEN.  This number may
Packit 6c4009
	     vary from machine to machine.  */
Packit 6c4009
Packit 6c4009
	  WORD_COPY_BWD (dstp, srcp, len, len);
Packit 6c4009
Packit 6c4009
	  /* Fall out and copy the tail.  */
Packit 6c4009
	}
Packit 6c4009
Packit 6c4009
      /* There are just a few bytes to copy.  Use byte memory operations.  */
Packit 6c4009
      BYTE_COPY_BWD (dstp, srcp, len);
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  RETURN (dest);
Packit 6c4009
}
Packit 6c4009
#ifndef memmove
Packit 6c4009
libc_hidden_builtin_def (memmove)
Packit 6c4009
#endif