2005-04-17 06:20:36 +08:00
|
|
|
/* Normally compiler builtins are used, but sometimes the compiler calls out
|
|
|
|
of line code. Based on asm-i386/string.h.
|
|
|
|
*/
|
|
|
|
#define _STRING_C
|
|
|
|
#include <linux/string.h>
|
2006-06-26 19:59:44 +08:00
|
|
|
#include <linux/module.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#undef memmove
|
2008-02-20 03:53:38 +08:00
|
|
|
void *memmove(void *dest, const void *src, size_t count)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2010-08-24 05:11:12 +08:00
|
|
|
unsigned long d0, d1, d2, d3;
|
2008-02-20 03:53:38 +08:00
|
|
|
if (dest < src) {
|
2010-08-24 05:11:12 +08:00
|
|
|
if ((dest + count) < src)
|
|
|
|
return memcpy(dest, src, count);
|
|
|
|
else
|
|
|
|
__asm__ __volatile__(
|
|
|
|
"movq %0, %3\n\t"
|
|
|
|
"shr $3, %0\n\t"
|
|
|
|
"andq $7, %3\n\t"
|
|
|
|
"rep\n\t"
|
|
|
|
"movsq\n\t"
|
|
|
|
"movq %3, %0\n\t"
|
|
|
|
"rep\n\t"
|
|
|
|
"movsb"
|
|
|
|
: "=&c" (d0), "=&S" (d1), "=&D" (d2), "=r" (d3)
|
|
|
|
:"0" (count),
|
|
|
|
"1" (src),
|
|
|
|
"2" (dest)
|
|
|
|
:"memory");
|
2005-04-17 06:20:36 +08:00
|
|
|
} else {
|
2010-08-24 05:11:12 +08:00
|
|
|
if((src + count) < dest)
|
|
|
|
return memcpy(dest, src, count);
|
|
|
|
else
|
|
|
|
__asm__ __volatile__(
|
|
|
|
"movq %0, %3\n\t"
|
|
|
|
"lea -8(%1, %0), %1\n\t"
|
|
|
|
"lea -8(%2, %0), %2\n\t"
|
|
|
|
"shr $3, %0\n\t"
|
|
|
|
"andq $7, %3\n\t"
|
|
|
|
"std\n\t"
|
|
|
|
"rep\n\t"
|
|
|
|
"movsq\n\t"
|
|
|
|
"lea 7(%1), %1\n\t"
|
|
|
|
"lea 7(%2), %2\n\t"
|
|
|
|
"movq %3, %0\n\t"
|
|
|
|
"rep\n\t"
|
|
|
|
"movsb\n\t"
|
|
|
|
"cld"
|
|
|
|
: "=&c" (d0), "=&S" (d1), "=&D" (d2), "=r" (d3)
|
|
|
|
:"0" (count),
|
|
|
|
"1" (src),
|
|
|
|
"2" (dest)
|
|
|
|
:"memory");
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
return dest;
|
2008-02-20 03:53:38 +08:00
|
|
|
}
|
2006-06-26 19:59:44 +08:00
|
|
|
EXPORT_SYMBOL(memmove);
|