18c2ecf20Sopenharmony_ci/* 28c2ecf20Sopenharmony_ci * This file is subject to the terms and conditions of the GNU General Public 38c2ecf20Sopenharmony_ci * License. See the file COPYING in the main directory of this archive 48c2ecf20Sopenharmony_ci * for more details. 58c2ecf20Sopenharmony_ci */ 68c2ecf20Sopenharmony_ci 78c2ecf20Sopenharmony_ci#include <linux/module.h> 88c2ecf20Sopenharmony_ci#include <linux/string.h> 98c2ecf20Sopenharmony_ci 108c2ecf20Sopenharmony_civoid *memcpy(void *to, const void *from, size_t n) 118c2ecf20Sopenharmony_ci{ 128c2ecf20Sopenharmony_ci void *xto = to; 138c2ecf20Sopenharmony_ci size_t temp; 148c2ecf20Sopenharmony_ci 158c2ecf20Sopenharmony_ci if (!n) 168c2ecf20Sopenharmony_ci return xto; 178c2ecf20Sopenharmony_ci if ((long)to & 1) { 188c2ecf20Sopenharmony_ci char *cto = to; 198c2ecf20Sopenharmony_ci const char *cfrom = from; 208c2ecf20Sopenharmony_ci *cto++ = *cfrom++; 218c2ecf20Sopenharmony_ci to = cto; 228c2ecf20Sopenharmony_ci from = cfrom; 238c2ecf20Sopenharmony_ci n--; 248c2ecf20Sopenharmony_ci } 258c2ecf20Sopenharmony_ci#if defined(CONFIG_M68000) 268c2ecf20Sopenharmony_ci if ((long)from & 1) { 278c2ecf20Sopenharmony_ci char *cto = to; 288c2ecf20Sopenharmony_ci const char *cfrom = from; 298c2ecf20Sopenharmony_ci for (; n; n--) 308c2ecf20Sopenharmony_ci *cto++ = *cfrom++; 318c2ecf20Sopenharmony_ci return xto; 328c2ecf20Sopenharmony_ci } 338c2ecf20Sopenharmony_ci#endif 348c2ecf20Sopenharmony_ci if (n > 2 && (long)to & 2) { 358c2ecf20Sopenharmony_ci short *sto = to; 368c2ecf20Sopenharmony_ci const short *sfrom = from; 378c2ecf20Sopenharmony_ci *sto++ = *sfrom++; 388c2ecf20Sopenharmony_ci to = sto; 398c2ecf20Sopenharmony_ci from = sfrom; 408c2ecf20Sopenharmony_ci n -= 2; 418c2ecf20Sopenharmony_ci } 428c2ecf20Sopenharmony_ci temp = n >> 2; 438c2ecf20Sopenharmony_ci if (temp) { 448c2ecf20Sopenharmony_ci long *lto = to; 458c2ecf20Sopenharmony_ci const long *lfrom = from; 468c2ecf20Sopenharmony_ci#if defined(CONFIG_M68000) || defined(CONFIG_COLDFIRE) 478c2ecf20Sopenharmony_ci for (; temp; temp--) 488c2ecf20Sopenharmony_ci *lto++ = *lfrom++; 498c2ecf20Sopenharmony_ci#else 508c2ecf20Sopenharmony_ci size_t temp1; 518c2ecf20Sopenharmony_ci asm volatile ( 528c2ecf20Sopenharmony_ci " movel %2,%3\n" 538c2ecf20Sopenharmony_ci " andw #7,%3\n" 548c2ecf20Sopenharmony_ci " lsrl #3,%2\n" 558c2ecf20Sopenharmony_ci " negw %3\n" 568c2ecf20Sopenharmony_ci " jmp %%pc@(1f,%3:w:2)\n" 578c2ecf20Sopenharmony_ci "4: movel %0@+,%1@+\n" 588c2ecf20Sopenharmony_ci " movel %0@+,%1@+\n" 598c2ecf20Sopenharmony_ci " movel %0@+,%1@+\n" 608c2ecf20Sopenharmony_ci " movel %0@+,%1@+\n" 618c2ecf20Sopenharmony_ci " movel %0@+,%1@+\n" 628c2ecf20Sopenharmony_ci " movel %0@+,%1@+\n" 638c2ecf20Sopenharmony_ci " movel %0@+,%1@+\n" 648c2ecf20Sopenharmony_ci " movel %0@+,%1@+\n" 658c2ecf20Sopenharmony_ci "1: dbra %2,4b\n" 668c2ecf20Sopenharmony_ci " clrw %2\n" 678c2ecf20Sopenharmony_ci " subql #1,%2\n" 688c2ecf20Sopenharmony_ci " jpl 4b" 698c2ecf20Sopenharmony_ci : "=a" (lfrom), "=a" (lto), "=d" (temp), "=&d" (temp1) 708c2ecf20Sopenharmony_ci : "0" (lfrom), "1" (lto), "2" (temp)); 718c2ecf20Sopenharmony_ci#endif 728c2ecf20Sopenharmony_ci to = lto; 738c2ecf20Sopenharmony_ci from = lfrom; 748c2ecf20Sopenharmony_ci } 758c2ecf20Sopenharmony_ci if (n & 2) { 768c2ecf20Sopenharmony_ci short *sto = to; 778c2ecf20Sopenharmony_ci const short *sfrom = from; 788c2ecf20Sopenharmony_ci *sto++ = *sfrom++; 798c2ecf20Sopenharmony_ci to = sto; 808c2ecf20Sopenharmony_ci from = sfrom; 818c2ecf20Sopenharmony_ci } 828c2ecf20Sopenharmony_ci if (n & 1) { 838c2ecf20Sopenharmony_ci char *cto = to; 848c2ecf20Sopenharmony_ci const char *cfrom = from; 858c2ecf20Sopenharmony_ci *cto = *cfrom; 868c2ecf20Sopenharmony_ci } 878c2ecf20Sopenharmony_ci return xto; 888c2ecf20Sopenharmony_ci} 898c2ecf20Sopenharmony_ciEXPORT_SYMBOL(memcpy); 90