| /* 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> |
| #include <linux/module.h> |
| |
| #undef memmove |
| void *memmove(void *dest, const void *src, size_t count) |
| { |
| unsigned long d0,d1,d2,d3,d4,d5,d6,d7; |
| char *ret; |
| |
| __asm__ __volatile__( |
| /* Handle more 32bytes in loop */ |
| "mov %2, %3\n\t" |
| "cmp $0x20, %0\n\t" |
| "jb 1f\n\t" |
| |
| /* Decide forward/backward copy mode */ |
| "cmp %2, %1\n\t" |
| "jb 2f\n\t" |
| |
| /* |
| * movsq instruction have many startup latency |
| * so we handle small size by general register. |
| */ |
| "cmp $680, %0\n\t" |
| "jb 3f\n\t" |
| /* |
| * movsq instruction is only good for aligned case. |
| */ |
| "cmpb %%dil, %%sil\n\t" |
| "je 4f\n\t" |
| "3:\n\t" |
| "sub $0x20, %0\n\t" |
| /* |
| * We gobble 32byts forward in each loop. |
| */ |
| "5:\n\t" |
| "sub $0x20, %0\n\t" |
| "movq 0*8(%1), %4\n\t" |
| "movq 1*8(%1), %5\n\t" |
| "movq 2*8(%1), %6\n\t" |
| "movq 3*8(%1), %7\n\t" |
| "leaq 4*8(%1), %1\n\t" |
| |
| "movq %4, 0*8(%2)\n\t" |
| "movq %5, 1*8(%2)\n\t" |
| "movq %6, 2*8(%2)\n\t" |
| "movq %7, 3*8(%2)\n\t" |
| "leaq 4*8(%2), %2\n\t" |
| "jae 5b\n\t" |
| "addq $0x20, %0\n\t" |
| "jmp 1f\n\t" |
| /* |
| * Handle data forward by movsq. |
| */ |
| ".p2align 4\n\t" |
| "4:\n\t" |
| "movq %0, %8\n\t" |
| "movq -8(%1, %0), %4\n\t" |
| "lea -8(%2, %0), %5\n\t" |
| "shrq $3, %8\n\t" |
| "rep movsq\n\t" |
| "movq %4, (%5)\n\t" |
| "jmp 13f\n\t" |
| /* |
| * Handle data backward by movsq. |
| */ |
| ".p2align 4\n\t" |
| "7:\n\t" |
| "movq %0, %8\n\t" |
| "movq (%1), %4\n\t" |
| "movq %2, %5\n\t" |
| "leaq -8(%1, %0), %1\n\t" |
| "leaq -8(%2, %0), %2\n\t" |
| "shrq $3, %8\n\t" |
| "std\n\t" |
| "rep movsq\n\t" |
| "cld\n\t" |
| "movq %4, (%5)\n\t" |
| "jmp 13f\n\t" |
| |
| /* |
| * Start to prepare for backward copy. |
| */ |
| ".p2align 4\n\t" |
| "2:\n\t" |
| "cmp $680, %0\n\t" |
| "jb 6f \n\t" |
| "cmp %%dil, %%sil\n\t" |
| "je 7b \n\t" |
| "6:\n\t" |
| /* |
| * Calculate copy position to tail. |
| */ |
| "addq %0, %1\n\t" |
| "addq %0, %2\n\t" |
| "subq $0x20, %0\n\t" |
| /* |
| * We gobble 32byts backward in each loop. |
| */ |
| "8:\n\t" |
| "subq $0x20, %0\n\t" |
| "movq -1*8(%1), %4\n\t" |
| "movq -2*8(%1), %5\n\t" |
| "movq -3*8(%1), %6\n\t" |
| "movq -4*8(%1), %7\n\t" |
| "leaq -4*8(%1), %1\n\t" |
| |
| "movq %4, -1*8(%2)\n\t" |
| "movq %5, -2*8(%2)\n\t" |
| "movq %6, -3*8(%2)\n\t" |
| "movq %7, -4*8(%2)\n\t" |
| "leaq -4*8(%2), %2\n\t" |
| "jae 8b\n\t" |
| /* |
| * Calculate copy position to head. |
| */ |
| "addq $0x20, %0\n\t" |
| "subq %0, %1\n\t" |
| "subq %0, %2\n\t" |
| "1:\n\t" |
| "cmpq $16, %0\n\t" |
| "jb 9f\n\t" |
| /* |
| * Move data from 16 bytes to 31 bytes. |
| */ |
| "movq 0*8(%1), %4\n\t" |
| "movq 1*8(%1), %5\n\t" |
| "movq -2*8(%1, %0), %6\n\t" |
| "movq -1*8(%1, %0), %7\n\t" |
| "movq %4, 0*8(%2)\n\t" |
| "movq %5, 1*8(%2)\n\t" |
| "movq %6, -2*8(%2, %0)\n\t" |
| "movq %7, -1*8(%2, %0)\n\t" |
| "jmp 13f\n\t" |
| ".p2align 4\n\t" |
| "9:\n\t" |
| "cmpq $8, %0\n\t" |
| "jb 10f\n\t" |
| /* |
| * Move data from 8 bytes to 15 bytes. |
| */ |
| "movq 0*8(%1), %4\n\t" |
| "movq -1*8(%1, %0), %5\n\t" |
| "movq %4, 0*8(%2)\n\t" |
| "movq %5, -1*8(%2, %0)\n\t" |
| "jmp 13f\n\t" |
| "10:\n\t" |
| "cmpq $4, %0\n\t" |
| "jb 11f\n\t" |
| /* |
| * Move data from 4 bytes to 7 bytes. |
| */ |
| "movl (%1), %4d\n\t" |
| "movl -4(%1, %0), %5d\n\t" |
| "movl %4d, (%2)\n\t" |
| "movl %5d, -4(%2, %0)\n\t" |
| "jmp 13f\n\t" |
| "11:\n\t" |
| "cmp $2, %0\n\t" |
| "jb 12f\n\t" |
| /* |
| * Move data from 2 bytes to 3 bytes. |
| */ |
| "movw (%1), %4w\n\t" |
| "movw -2(%1, %0), %5w\n\t" |
| "movw %4w, (%2)\n\t" |
| "movw %5w, -2(%2, %0)\n\t" |
| "jmp 13f\n\t" |
| "12:\n\t" |
| "cmp $1, %0\n\t" |
| "jb 13f\n\t" |
| /* |
| * Move data for 1 byte. |
| */ |
| "movb (%1), %4b\n\t" |
| "movb %4b, (%2)\n\t" |
| "13:\n\t" |
| : "=&d" (d0), "=&S" (d1), "=&D" (d2), "=&a" (ret) , |
| "=r"(d3), "=r"(d4), "=r"(d5), "=r"(d6), "=&c" (d7) |
| :"0" (count), |
| "1" (src), |
| "2" (dest) |
| :"memory"); |
| |
| return ret; |
| |
| } |
| EXPORT_SYMBOL(memmove); |