1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140
|
/*
Copyright (c) 1994 Cygnus Support.
All rights reserved.
Redistribution and use in source and binary forms are permitted
provided that the above copyright notice and this paragraph are
duplicated in all such forms and that any documentation,
and/or other materials related to such
distribution and use acknowledge that the software was developed
at Cygnus Support, Inc. Cygnus Support, Inc. may not be used to
endorse or promote products derived from this software without
specific prior written permission.
THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
*/
/*
FUNCTION
<<memcpy>>---copy memory regions
SYNOPSIS
#include <string.h>
void* memcpy(void *restrict <[out]>, const void *restrict <[in]>,
size_t <[n]>);
DESCRIPTION
This function copies <[n]> bytes from the memory region
pointed to by <[in]> to the memory region pointed to by
<[out]>.
If the regions overlap, the behavior is undefined.
RETURNS
<<memcpy>> returns a pointer to the first byte of the <[out]>
region.
PORTABILITY
<<memcpy>> is ANSI C.
<<memcpy>> requires no supporting OS subroutines.
QUICKREF
memcpy ansi pure
*/
#include <string.h>
#include "local.h"
#include <stdint.h>
#undef memcpy
void * __no_builtin
memcpy(void * __restrict dst0, const void * __restrict src0, size_t len0)
{
char *dst = dst0;
const char *src = src0;
#if !(defined(__PREFER_SIZE_OVER_SPEED) || defined(__OPTIMIZE_SIZE__))
unsigned long *aligned_dst;
const unsigned long *aligned_src;
/* If the size is small, or either SRC or DST is unaligned,
then punt into the byte copy loop. This should be rare. */
if (!TOO_SMALL_BIG_BLOCK(len0)) {
/*
* Align dst and make sure
* we won't fetch anything before src
*/
unsigned start = LITTLE_BLOCK_SIZE * 2 - UNALIGNED_X(dst);
while (start--) {
*dst++ = *src++;
len0--;
}
aligned_dst = (unsigned long *)dst;
int byte_shift = UNALIGNED_X(src);
if (!byte_shift) {
aligned_src = (unsigned long *)src;
/* Copy 4X long words at a time if possible. */
while (len0 >= BIG_BLOCK_SIZE) {
*aligned_dst++ = *aligned_src++;
*aligned_dst++ = *aligned_src++;
*aligned_dst++ = *aligned_src++;
*aligned_dst++ = *aligned_src++;
len0 -= BIG_BLOCK_SIZE;
}
/* Copy one long word at a time if possible. */
while (len0 >= LITTLE_BLOCK_SIZE) {
*aligned_dst++ = *aligned_src++;
len0 -= LITTLE_BLOCK_SIZE;
}
src = (char *)aligned_src;
} else {
/*
* Fetch source words and then merge two of them for each
* dest word:
*
* byte_shift remain
* | | |
* xxxxxxxL RRRRRRRy
* D DDDDDDD
*
* We don't want to fetch past the source at all, so stop
* when we have fewer than 'remain' bytes left
*/
/* bytes used from the left word */
int remain = LITTLE_BLOCK_SIZE - byte_shift;
/* bit shifts for the left and right words */
int left_shift = byte_shift << 3;
int right_shift = remain << 3;
aligned_src = (unsigned long *)(src - byte_shift);
unsigned long left = *aligned_src++, right;
while (len0 >= LITTLE_BLOCK_SIZE + remain) {
right = *aligned_src++;
#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
*aligned_dst++ = (left >> left_shift) | (right << right_shift);
#else
*aligned_dst++ = (left << left_shift) | (right >> right_shift);
#endif
left = right;
len0 -= LITTLE_BLOCK_SIZE;
}
src = (char *)aligned_src - remain;
}
/* Pick up any residual with a byte copier. */
dst = (char *)aligned_dst;
}
#endif
while (len0--)
*dst++ = *src++;
return dst0;
}
|