mirror of
git://sourceware.org/git/newlib-cygwin.git
synced 2025-01-15 10:19:09 +08:00
13f7d71fee
Fix clobbering bytes before destination if src and dst have same non-zero misalignment.
135 lines
2.5 KiB
ArmAsm
135 lines
2.5 KiB
ArmAsm
! Entry: arg0: destination
|
|
! arg1: source
|
|
! Exit: result: destination
|
|
!
|
|
! SH5 code Copyright 2002 SuperH Ltd.
|
|
|
|
#include "asm.h"
|
|
|
|
ENTRY(strcpy)
|
|
|
|
#if __SHMEDIA__
|
|
|
|
pta/l shortstring,tr1
|
|
ldlo.q r3,0,r4
|
|
ptabs r18,tr4
|
|
shlli r3,3,r7
|
|
addi r2, 8, r0
|
|
mcmpeq.b r4,r63,r6
|
|
SHHI r6,r7,r6
|
|
bnei/u r6,0,tr1 // shortstring
|
|
pta/l no_lddst, tr2
|
|
ori r3,-8,r23
|
|
sub r2, r23, r0
|
|
sub r3, r2, r21
|
|
addi r21, 8, r20
|
|
ldx.q r0, r21, r5
|
|
pta/l loop, tr0
|
|
ori r2,-8,r22
|
|
mcmpeq.b r5, r63, r6
|
|
bgt/u r22, r23, tr2 // no_lddst
|
|
|
|
// r22 < r23 : Need to do a load from the destination.
|
|
// r22 == r23 : Doesn't actually need to load from destination,
|
|
// but still can be handled here.
|
|
ldlo.q r2, 0, r9
|
|
movi -1, r8
|
|
SHLO r8, r7, r8
|
|
mcmv r4, r8, r9
|
|
stlo.q r2, 0, r9
|
|
beqi/l r6, 0, tr0 // loop
|
|
|
|
add r5, r63, r4
|
|
addi r0, 8, r0
|
|
blink tr1, r63 // shortstring
|
|
no_lddst:
|
|
// r22 > r23: note that for r22 == r23 the sthi.q would clobber
|
|
// bytes before the destination region.
|
|
stlo.q r2, 0, r4
|
|
SHHI r4, r7, r4
|
|
sthi.q r0, -1, r4
|
|
beqi/l r6, 0, tr0 // loop
|
|
|
|
add r5, r63, r4
|
|
addi r0, 8, r0
|
|
shortstring:
|
|
#ifndef __LITTLE_ENDIAN__
|
|
pta/l shortstring2,tr1
|
|
byterev r4,r4
|
|
#endif
|
|
shortstring2:
|
|
st.b r0,-8,r4
|
|
andi r4,0xff,r5
|
|
shlri r4,8,r4
|
|
addi r0,1,r0
|
|
bnei/l r5,0,tr1
|
|
blink tr4,r63 // return
|
|
|
|
.balign 8
|
|
loop:
|
|
stlo.q r0, 0, r5
|
|
ldx.q r0, r20, r4
|
|
addi r0, 16, r0
|
|
sthi.q r0, -9, r5
|
|
mcmpeq.b r4, r63, r6
|
|
bnei/u r6, 0, tr1 // shortstring
|
|
ldx.q r0, r21, r5
|
|
stlo.q r0, -8, r4
|
|
sthi.q r0, -1, r4
|
|
mcmpeq.b r5, r63, r6
|
|
beqi/l r6, 0, tr0 // loop
|
|
|
|
add r5, r63, r4
|
|
addi r0, 8, r0
|
|
blink tr1, r63 // shortstring
|
|
|
|
#else /* ! __SHMEDIA__, i.e. SH 1..4 / SHcompact */
|
|
|
|
#ifdef __SH5__
|
|
#define DST r2
|
|
#define SRC r3
|
|
#define TMP r4
|
|
#define RESULT R2
|
|
! r0,r1,r3,r4: clobbered
|
|
#else
|
|
#define DST r4
|
|
#define SRC r5
|
|
#define TMP r2
|
|
#define RESULT r0
|
|
! r1-r2,r5: clobbered
|
|
#endif
|
|
mov DST,r0
|
|
or SRC,r0
|
|
tst #3,r0
|
|
SL(bf, L_setup_char_loop, mov DST,r0)
|
|
mov.l @SRC+,r1
|
|
mov #0,TMP
|
|
cmp/str TMP,r1
|
|
SL(bt, Longword_loop_end, sub SRC,r0)
|
|
.align 2
|
|
Longword_loop:
|
|
mov.l r1,@(r0,SRC)
|
|
mov.l @SRC+,r1
|
|
cmp/str TMP,r1
|
|
bt Longword_loop_end
|
|
mov.l r1,@(r0,SRC)
|
|
mov.l @SRC+,r1
|
|
cmp/str TMP,r1
|
|
bf Longword_loop
|
|
Longword_loop_end:
|
|
add #-4,SRC
|
|
add #3,r0
|
|
.align 2
|
|
L_char_loop:
|
|
mov.b @SRC+,r1
|
|
L_char_loop_start:
|
|
tst r1,r1
|
|
SL(bf, L_char_loop, mov.b r1,@(r0,SRC))
|
|
rts
|
|
mov DST,RESULT
|
|
L_setup_char_loop:
|
|
mov.b @SRC+,r1
|
|
bra L_char_loop_start
|
|
sub SRC,r0
|
|
#endif /* ! __SHMEDIA__ */
|