| Greg Ungerer | 66d83ab | 2011-03-28 16:53:37 +1000 | [diff] [blame] | 1 | /* | 
|  | 2 | * This file is subject to the terms and conditions of the GNU General Public | 
|  | 3 | * License.  See the file COPYING in the main directory of this archive | 
|  | 4 | * for more details. | 
|  | 5 | */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 6 |  | 
| Greg Ungerer | 66d83ab | 2011-03-28 16:53:37 +1000 | [diff] [blame] | 7 | #include <linux/module.h> | 
|  | 8 | #include <linux/string.h> | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 9 |  | 
| Greg Ungerer | 66d83ab | 2011-03-28 16:53:37 +1000 | [diff] [blame] | 10 | void *memcpy(void *to, const void *from, size_t n) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 11 | { | 
| Greg Ungerer | 66d83ab | 2011-03-28 16:53:37 +1000 | [diff] [blame] | 12 | void *xto = to; | 
|  | 13 | size_t temp, temp1; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 14 |  | 
| Greg Ungerer | 66d83ab | 2011-03-28 16:53:37 +1000 | [diff] [blame] | 15 | if (!n) | 
|  | 16 | return xto; | 
|  | 17 | if ((long)to & 1) { | 
|  | 18 | char *cto = to; | 
|  | 19 | const char *cfrom = from; | 
|  | 20 | *cto++ = *cfrom++; | 
|  | 21 | to = cto; | 
|  | 22 | from = cfrom; | 
|  | 23 | n--; | 
|  | 24 | } | 
| Greg Ungerer | f230e80 | 2011-08-05 14:41:29 +1000 | [diff] [blame] | 25 | #if defined(CONFIG_M68000) | 
|  | 26 | if ((long)from & 1) { | 
|  | 27 | char *cto = to; | 
|  | 28 | const char *cfrom = from; | 
|  | 29 | for (; n; n--) | 
|  | 30 | *cto++ = *cfrom++; | 
|  | 31 | return xto; | 
|  | 32 | } | 
|  | 33 | #endif | 
| Greg Ungerer | 66d83ab | 2011-03-28 16:53:37 +1000 | [diff] [blame] | 34 | if (n > 2 && (long)to & 2) { | 
|  | 35 | short *sto = to; | 
|  | 36 | const short *sfrom = from; | 
|  | 37 | *sto++ = *sfrom++; | 
|  | 38 | to = sto; | 
|  | 39 | from = sfrom; | 
|  | 40 | n -= 2; | 
|  | 41 | } | 
|  | 42 | temp = n >> 2; | 
|  | 43 | if (temp) { | 
|  | 44 | long *lto = to; | 
|  | 45 | const long *lfrom = from; | 
| Greg Ungerer | 734c3ce | 2011-06-02 16:07:33 +1000 | [diff] [blame] | 46 | #if defined(CONFIG_M68000) || defined(CONFIG_COLDFIRE) | 
|  | 47 | for (; temp; temp--) | 
|  | 48 | *lto++ = *lfrom++; | 
|  | 49 | #else | 
| Greg Ungerer | 66d83ab | 2011-03-28 16:53:37 +1000 | [diff] [blame] | 50 | asm volatile ( | 
|  | 51 | "	movel %2,%3\n" | 
|  | 52 | "	andw  #7,%3\n" | 
|  | 53 | "	lsrl  #3,%2\n" | 
|  | 54 | "	negw  %3\n" | 
|  | 55 | "	jmp   %%pc@(1f,%3:w:2)\n" | 
|  | 56 | "4:	movel %0@+,%1@+\n" | 
|  | 57 | "	movel %0@+,%1@+\n" | 
|  | 58 | "	movel %0@+,%1@+\n" | 
|  | 59 | "	movel %0@+,%1@+\n" | 
|  | 60 | "	movel %0@+,%1@+\n" | 
|  | 61 | "	movel %0@+,%1@+\n" | 
|  | 62 | "	movel %0@+,%1@+\n" | 
|  | 63 | "	movel %0@+,%1@+\n" | 
|  | 64 | "1:	dbra  %2,4b\n" | 
|  | 65 | "	clrw  %2\n" | 
|  | 66 | "	subql #1,%2\n" | 
|  | 67 | "	jpl   4b" | 
|  | 68 | : "=a" (lfrom), "=a" (lto), "=d" (temp), "=&d" (temp1) | 
|  | 69 | : "0" (lfrom), "1" (lto), "2" (temp)); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 70 | #endif | 
| Greg Ungerer | 66d83ab | 2011-03-28 16:53:37 +1000 | [diff] [blame] | 71 | to = lto; | 
|  | 72 | from = lfrom; | 
|  | 73 | } | 
|  | 74 | if (n & 2) { | 
|  | 75 | short *sto = to; | 
|  | 76 | const short *sfrom = from; | 
|  | 77 | *sto++ = *sfrom++; | 
|  | 78 | to = sto; | 
|  | 79 | from = sfrom; | 
|  | 80 | } | 
|  | 81 | if (n & 1) { | 
|  | 82 | char *cto = to; | 
|  | 83 | const char *cfrom = from; | 
|  | 84 | *cto = *cfrom; | 
|  | 85 | } | 
|  | 86 | return xto; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 87 | } | 
| Greg Ungerer | 66d83ab | 2011-03-28 16:53:37 +1000 | [diff] [blame] | 88 | EXPORT_SYMBOL(memcpy); |