2009-03-27 14:25:21 +01:00
/*
* Copyright ( C ) 2008 - 2009 Michal Simek < monstr @ monstr . eu >
* Copyright ( C ) 2008 - 2009 PetaLogix
* Copyright ( C ) 2007 John Williams
*
* Reasonably optimised generic C - code for memcpy on Microblaze
* This is generic C code to do efficient , alignment - aware memcpy .
*
* It is based on demo code originally Copyright 2001 by Intel Corp , taken from
* http : //www.embedded.com/showArticle.jhtml?articleID=19205567
*
tree-wide: fix assorted typos all over the place
That is "success", "unknown", "through", "performance", "[re|un]mapping"
, "access", "default", "reasonable", "[con]currently", "temperature"
, "channel", "[un]used", "application", "example","hierarchy", "therefore"
, "[over|under]flow", "contiguous", "threshold", "enough" and others.
Signed-off-by: André Goddard Rosa <andre.goddard@gmail.com>
Signed-off-by: Jiri Kosina <jkosina@suse.cz>
2009-11-14 13:09:05 -02:00
* Attempts were made , unsuccessfully , to contact the original
2009-03-27 14:25:21 +01:00
* author of this code ( Michael Morrow , Intel ) . Below is the original
* copyright notice .
*
* This software has been developed by Intel Corporation .
* Intel specifically disclaims all warranties , express or
* implied , and all liability , including consequential and
* other indirect damages , for the use of this program , including
* liability for infringement of any proprietary rights ,
* and including the warranties of merchantability and fitness
* for a particular purpose . Intel does not assume any
* responsibility for and errors which may appear in this program
* not any responsibility to update it .
*/
2013-02-01 13:10:35 +01:00
# include <linux/export.h>
2009-03-27 14:25:21 +01:00
# include <linux/types.h>
# include <linux/stddef.h>
# include <linux/compiler.h>
# include <linux/string.h>
# ifdef __HAVE_ARCH_MEMCPY
2010-10-09 13:58:24 +10:00
# ifndef CONFIG_OPT_LIB_FUNCTION
2009-03-27 14:25:21 +01:00
void * memcpy ( void * v_dst , const void * v_src , __kernel_size_t c )
{
const char * src = v_src ;
char * dst = v_dst ;
2010-10-09 13:58:24 +10:00
2009-03-27 14:25:21 +01:00
/* Simple, byte oriented memcpy. */
while ( c - - )
* dst + + = * src + + ;
return v_dst ;
2010-10-09 13:58:24 +10:00
}
# else /* CONFIG_OPT_LIB_FUNCTION */
void * memcpy ( void * v_dst , const void * v_src , __kernel_size_t c )
{
const char * src = v_src ;
char * dst = v_dst ;
2009-03-27 14:25:21 +01:00
/* The following code tries to optimize the copy by using unsigned
* alignment . This will work fine if both source and destination are
* aligned on the same boundary . However , if they are aligned on
* different boundaries shifts will be necessary . This might result in
* bad performance on MicroBlaze systems without a barrel shifter .
*/
const uint32_t * i_src ;
uint32_t * i_dst ;
2010-03-23 15:37:02 +01:00
if ( likely ( c > = 4 ) ) {
2009-03-27 14:25:21 +01:00
unsigned value , buf_hold ;
2011-03-30 22:57:33 -03:00
/* Align the destination to a word boundary. */
/* This is done in an endian independent manner. */
2009-03-27 14:25:21 +01:00
switch ( ( unsigned long ) dst & 3 ) {
case 1 :
* dst + + = * src + + ;
- - c ;
2021-04-20 19:20:41 -07:00
fallthrough ;
2009-03-27 14:25:21 +01:00
case 2 :
* dst + + = * src + + ;
- - c ;
2021-04-20 19:20:41 -07:00
fallthrough ;
2009-03-27 14:25:21 +01:00
case 3 :
* dst + + = * src + + ;
- - c ;
}
i_dst = ( void * ) dst ;
/* Choose a copy scheme based on the source */
2011-03-30 22:57:33 -03:00
/* alignment relative to destination. */
2009-03-27 14:25:21 +01:00
switch ( ( unsigned long ) src & 3 ) {
case 0x0 : /* Both byte offsets are aligned */
i_src = ( const void * ) src ;
for ( ; c > = 4 ; c - = 4 )
* i_dst + + = * i_src + + ;
src = ( const void * ) i_src ;
break ;
case 0x1 : /* Unaligned - Off by 1 */
/* Word align the source */
i_src = ( const void * ) ( ( unsigned ) src & ~ 3 ) ;
2010-10-09 14:05:58 +10:00
# ifndef __MICROBLAZEEL__
2009-03-27 14:25:21 +01:00
/* Load the holding buffer */
buf_hold = * i_src + + < < 8 ;
for ( ; c > = 4 ; c - = 4 ) {
value = * i_src + + ;
* i_dst + + = buf_hold | value > > 24 ;
buf_hold = value < < 8 ;
}
2010-10-09 14:05:58 +10:00
# else
/* Load the holding buffer */
2012-12-27 10:40:38 +01:00
buf_hold = ( * i_src + + & 0xFFFFFF00 ) > > 8 ;
2009-03-27 14:25:21 +01:00
2010-10-09 14:05:58 +10:00
for ( ; c > = 4 ; c - = 4 ) {
value = * i_src + + ;
* i_dst + + = buf_hold | ( ( value & 0xFF ) < < 24 ) ;
2012-12-27 10:40:38 +01:00
buf_hold = ( value & 0xFFFFFF00 ) > > 8 ;
2010-10-09 14:05:58 +10:00
}
# endif
2009-03-27 14:25:21 +01:00
/* Realign the source */
src = ( const void * ) i_src ;
src - = 3 ;
break ;
case 0x2 : /* Unaligned - Off by 2 */
/* Word align the source */
i_src = ( const void * ) ( ( unsigned ) src & ~ 3 ) ;
2010-10-09 14:05:58 +10:00
# ifndef __MICROBLAZEEL__
2009-03-27 14:25:21 +01:00
/* Load the holding buffer */
buf_hold = * i_src + + < < 16 ;
for ( ; c > = 4 ; c - = 4 ) {
value = * i_src + + ;
* i_dst + + = buf_hold | value > > 16 ;
buf_hold = value < < 16 ;
}
2010-10-09 14:05:58 +10:00
# else
/* Load the holding buffer */
2012-12-27 10:40:38 +01:00
buf_hold = ( * i_src + + & 0xFFFF0000 ) > > 16 ;
2009-03-27 14:25:21 +01:00
2010-10-09 14:05:58 +10:00
for ( ; c > = 4 ; c - = 4 ) {
value = * i_src + + ;
2012-12-27 10:40:38 +01:00
* i_dst + + = buf_hold | ( ( value & 0xFFFF ) < < 16 ) ;
buf_hold = ( value & 0xFFFF0000 ) > > 16 ;
2010-10-09 14:05:58 +10:00
}
# endif
2009-03-27 14:25:21 +01:00
/* Realign the source */
src = ( const void * ) i_src ;
src - = 2 ;
break ;
case 0x3 : /* Unaligned - Off by 3 */
/* Word align the source */
i_src = ( const void * ) ( ( unsigned ) src & ~ 3 ) ;
2010-10-09 14:05:58 +10:00
# ifndef __MICROBLAZEEL__
2009-03-27 14:25:21 +01:00
/* Load the holding buffer */
buf_hold = * i_src + + < < 24 ;
for ( ; c > = 4 ; c - = 4 ) {
value = * i_src + + ;
* i_dst + + = buf_hold | value > > 8 ;
buf_hold = value < < 24 ;
}
2010-10-09 14:05:58 +10:00
# else
/* Load the holding buffer */
buf_hold = ( * i_src + + & 0xFF000000 ) > > 24 ;
2009-03-27 14:25:21 +01:00
2010-10-09 14:05:58 +10:00
for ( ; c > = 4 ; c - = 4 ) {
value = * i_src + + ;
* i_dst + + = buf_hold | ( ( value & 0xFFFFFF ) < < 8 ) ;
buf_hold = ( value & 0xFF000000 ) > > 24 ;
}
# endif
2009-03-27 14:25:21 +01:00
/* Realign the source */
src = ( const void * ) i_src ;
src - = 1 ;
break ;
}
dst = ( void * ) i_dst ;
}
/* Finish off any remaining bytes */
2011-03-30 22:57:33 -03:00
/* simple fast copy, ... unless a cache boundary is crossed */
2009-03-27 14:25:21 +01:00
switch ( c ) {
case 3 :
* dst + + = * src + + ;
2021-04-20 19:20:41 -07:00
fallthrough ;
2009-03-27 14:25:21 +01:00
case 2 :
* dst + + = * src + + ;
2021-04-20 19:20:41 -07:00
fallthrough ;
2009-03-27 14:25:21 +01:00
case 1 :
* dst + + = * src + + ;
}
return v_dst ;
}
2010-10-09 13:58:24 +10:00
# endif /* CONFIG_OPT_LIB_FUNCTION */
2009-03-27 14:25:21 +01:00
EXPORT_SYMBOL ( memcpy ) ;
# endif /* __HAVE_ARCH_MEMCPY */