ack/mach/i80/libem/cii.s
George Koehler c95bcac91d Correct the stack pointer when i80 shrinks an integer.
The code used `sphl` to set the stack pointer, but the correct value
was in de, not hl.  Fix by swapping the values of de and hl, so `sphl`
is now correct.  When we shrink an integer from 4 to 2 bytes, both
registers de and hl point to copies of the result, but only one
register preserves the stack below the result.

This fixes writehex() in tests/plat/lib/test.c, when I compile it with
ack -mcpm, so it preserves the pointer to "0123456789abcdef", so it
writes hexadecimal digits and not garbage.

This bug goes back to commit 157b243 of Mar 18, 1985, so the bug is
32 years old, and probably the oldest bug that I ever fixed.
2017-12-07 15:39:41 -05:00

90 lines
1.5 KiB
ArmAsm

.define .cii
.sect .text
.sect .rom
.sect .data
.sect .bss
.sect .text
! Convert integer to integer
! Expects in a-reg: 1 for signed integer to signed integer (cii)
! 0 for unsigned integer to unsigned integer (cuu)
! Expects on stack: destination size
! source size
! source
! Yields on stack: result
.cii: pop h
shld .retadr
mov h,b
mov l,c
shld .bcreg
sta .areg ! save a-register
pop b
mov e,c
pop b ! c = source size
mov b,e ! b = destination size
mov a,b
cmp c
jz 3f ! destination size = source size
jc shrink ! destination size < source size
! if destination size > source size only:
lxi h,0
dad sp
mov e,l
mov d,h ! de = stackpointer
mov a,b
sub c ! c = (still) source size
mov b,a ! b = destination size - source size
cma
mov l,a
mvi h,255
inx h
dad d ! hl = stackpointer - (dest. size - source size)
sphl ! new stackpointer
1: ldax d ! move source downwards
mov m,a
inx d
inx h
dcr c
jnz 1b
ral ! a-reg still contains most significant byte of source
jnc 1f ! jump if is a positive integer
lda .areg
ora a
jz 1f ! jump if it is a cuu
mvi c,255 ! c-reg contains filler byte
1: mov m,c ! fill
inx h
dcr b
jnz 1b
jmp 3f ! done
!if destination size < source size only:
shrink: mov l,b ! load destination size in hl
mvi h,0
dad sp
mov d,h
mov e,l ! de points just above lowest bytes of source
mov l,c ! load source size in hl
mvi h,0
dad sp ! hl points just above "destination"
1: dcx d ! move upwards
dcx h
ldax d
mov m,a
dcr b
jnz 1b
sphl
3: lhld .bcreg
mov b,h
mov c,l
lhld .retadr
pchl