2000-05-28 13:40:48 +00:00
|
|
|
;
|
2003-08-20 10:17:53 +00:00
|
|
|
; Ullrich von Bassewitz, 2003-08-20
|
2009-09-20 14:32:25 +00:00
|
|
|
; Performance increase (about 20%) by
|
|
|
|
; Christian Krueger, 2009-09-13
|
2000-05-28 13:40:48 +00:00
|
|
|
;
|
2003-08-20 10:17:53 +00:00
|
|
|
; void* __fastcall__ memcpy (void* dest, const void* src, size_t n);
|
|
|
|
;
|
|
|
|
; NOTE: This function contains entry points for memmove, which will ressort
|
|
|
|
; to memcpy for an upwards copy. Don't change this module without looking
|
|
|
|
; at memmove!
|
2000-05-28 13:40:48 +00:00
|
|
|
;
|
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
.export _memcpy, memcpy_upwards, memcpy_getparams
|
2018-05-20 15:30:18 +02:00
|
|
|
.import popax, popptr1
|
2013-05-09 13:56:54 +02:00
|
|
|
.importzp sp, ptr1, ptr2, ptr3
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
; ----------------------------------------------------------------------
|
|
|
|
_memcpy:
|
2003-08-20 10:17:53 +00:00
|
|
|
jsr memcpy_getparams
|
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
memcpy_upwards: ; assert Y = 0
|
|
|
|
ldx ptr3+1 ; Get high byte of n
|
|
|
|
beq L2 ; Jump if zero
|
2003-08-20 10:17:53 +00:00
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
L1: .repeat 2 ; Unroll this a bit to make it faster...
|
|
|
|
lda (ptr1),Y ; copy a byte
|
|
|
|
sta (ptr2),Y
|
|
|
|
iny
|
|
|
|
.endrepeat
|
|
|
|
bne L1
|
|
|
|
inc ptr1+1
|
|
|
|
inc ptr2+1
|
|
|
|
dex ; Next 256 byte block
|
|
|
|
bne L1 ; Repeat if any
|
2003-08-20 10:17:53 +00:00
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
; the following section could be 10% faster if we were able to copy
|
|
|
|
; back to front - unfortunately we are forced to copy strict from
|
|
|
|
; low to high since this function is also used for
|
|
|
|
; memmove and blocks could be overlapping!
|
|
|
|
; {
|
|
|
|
L2: ; assert Y = 0
|
|
|
|
ldx ptr3 ; Get the low byte of n
|
|
|
|
beq done ; something to copy
|
2003-08-20 10:17:53 +00:00
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
L3: lda (ptr1),Y ; copy a byte
|
|
|
|
sta (ptr2),Y
|
|
|
|
iny
|
|
|
|
dex
|
|
|
|
bne L3
|
2003-08-20 10:17:53 +00:00
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
; }
|
2009-09-20 14:32:25 +00:00
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
done: jmp popax ; Pop ptr and return as result
|
2000-05-28 13:40:48 +00:00
|
|
|
|
|
|
|
; ----------------------------------------------------------------------
|
2003-08-20 10:17:53 +00:00
|
|
|
; Get the parameters from stack as follows:
|
|
|
|
;
|
2013-05-09 13:56:54 +02:00
|
|
|
; size --> ptr3
|
2003-08-20 10:17:53 +00:00
|
|
|
; src --> ptr1
|
|
|
|
; dest --> ptr2
|
2013-05-09 13:56:54 +02:00
|
|
|
; First argument (dest) will remain on stack and is returned in a/x!
|
2003-08-20 10:17:53 +00:00
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
memcpy_getparams: ; IMPORTANT! Function has to leave with Y=0!
|
|
|
|
sta ptr3
|
2009-09-20 14:32:25 +00:00
|
|
|
stx ptr3+1 ; save n to ptr3
|
2000-05-28 13:40:48 +00:00
|
|
|
|
2018-05-20 15:30:18 +02:00
|
|
|
jsr popptr1 ; save src to ptr1
|
2000-05-28 13:40:48 +00:00
|
|
|
|
2013-05-09 13:56:54 +02:00
|
|
|
; save dest to ptr2
|
2018-05-20 15:30:18 +02:00
|
|
|
iny ; Y=0 guaranteed by popptr1, we need '1' here...
|
|
|
|
; (direct stack access is three cycles faster
|
2009-09-20 14:32:25 +00:00
|
|
|
; (total cycle count with return))
|
|
|
|
lda (sp),y
|
2013-05-09 13:56:54 +02:00
|
|
|
tax
|
|
|
|
stx ptr2+1 ; save high byte of ptr2
|
|
|
|
dey ; Y = 0
|
2009-09-20 14:32:25 +00:00
|
|
|
lda (sp),y ; Get ptr2 low
|
2013-05-09 13:56:54 +02:00
|
|
|
sta ptr2
|
|
|
|
rts
|