ack/mach/z80/int/em.s
1987-02-03 13:23:38 +00:00

4678 lines
54 KiB
ArmAsm

#
.sect .text
.sect .rom
.sect .data
.sect .bss
.sect .text
! This program is an EM interpreter for the Z80.
! Register pair bc is used to hold lb.
! Register ix is used to hold the EM program counter.
! The interpreter assumes 16-bit words and 16-bit pointers.
! #define CPM1 1
! Definitions:
zone = 8 ! size of subroutine call block (address + old lb)
bdos = 5 ! standard entry into I/O-routines
boot = 0
fcb = 0x5c ! file descriptor of EM-1 file (5C hex)
reset=0
delete=19
makefile=22
close=16
readconsole = 10
writeconsole = 2
open = 15
read = 20
write = 21
setdma = 26
printstring = 9
seqread = 20
randomread = 33
seqwrite = 21
randomwrite = 34
consolein = 1
diconio = 6
RAW=0 !0 for cooked,1 for raw io
timebuf=0xFFDE
b_lolp = 176
b_loln = 179
b_lof = 161
b_loi = 168
b_lal = 130
b_lil = 146
b_stlm = 227
b_stf = 214
b_sti = 218
b_inl = 112
b_cal = 63
b_asp = 44
b_zrl = 249
EARRAY = 0
ERANGE = 1
EILLINS=18
EILLSIZE=19
ECASE=20
EMON=25
!--------------------------- Initialization ---------------------------
jp init ! 3 byte instruction.
!------------------------- MAIN DISPATCH ------------------------------
!
! must be put in a suitable place in memory to reduce memory usage
! must be put on a page boundary
.data1 0 ! fourth byte
dispat = . - 4 ! base of dispatch table
! .data2 loc.0
! .data2 loc.1
.data2 loc.2
.data2 loc.3
.data2 loc.4
.data2 loc.5
.data2 loc.6
.data2 loc.7
.data2 loc.8
.data2 loc.9
.data2 loc.10
.data2 loc.11
.data2 loc.12
.data2 loc.13
.data2 loc.14
.data2 loc.15
.data2 loc.16
.data2 loc.17
.data2 loc.18
.data2 loc.19
.data2 loc.20
.data2 loc.21
.data2 loc.22
.data2 loc.23
.data2 loc.24
.data2 loc.25
.data2 loc.26
.data2 loc.27
.data2 loc.28
.data2 loc.29
.data2 loc.30
.data2 loc.31
.data2 loc.32
.data2 loc.33
.data2 aar.2
.data2 adf.s0
.data2 adi.2
.data2 adi.4
.data2 adp.l
.data2 adp.1
.data2 adp.2
.data2 adp.s0
.data2 adp.sm1
.data2 ads.2
.data2 and.2
.data2 asp.2
.data2 asp.4
.data2 asp.6
.data2 asp.8
.data2 asp.10
.data2 asp.w0
.data2 beq.l
.data2 beq.s0
.data2 bge.s0
.data2 bgt.s0
.data2 ble.s0
.data2 blm.s0
.data2 blt.s0
.data2 bne.s0
.data2 bra.l
.data2 bra.sm1
.data2 bra.sm2
.data2 bra.s0
.data2 bra.s1
.data2 cal.1
.data2 cal.2
.data2 cal.3
.data2 cal.4
.data2 cal.5
.data2 cal.6
.data2 cal.7
.data2 cal.8
.data2 cal.9
.data2 cal.10
.data2 cal.11
.data2 cal.12
.data2 cal.13
.data2 cal.14
.data2 cal.15
.data2 cal.16
.data2 cal.17
.data2 cal.18
.data2 cal.19
.data2 cal.20
.data2 cal.21
.data2 cal.22
.data2 cal.23
.data2 cal.24
.data2 cal.25
.data2 cal.26
.data2 cal.27
.data2 cal.28
.data2 cal.s0
.data2 cff.z
.data2 cif.z
.data2 cii.z
.data2 cmf.s0
.data2 cmi.2
.data2 cmi.4
.data2 cmp.z
.data2 cms.s0
.data2 csa.2
.data2 csb.2
.data2 dec.z
.data2 dee.w0
.data2 del.wm1
.data2 dup.2
.data2 dvf.s0
.data2 dvi.2
.data2 fil.l
.data2 inc.z
.data2 ine.l
.data2 ine.w0
.data2 inl.m2
.data2 inl.m4
.data2 inl.m6
.data2 inl.wm1
.data2 inn.s0
.data2 ior.2
.data2 ior.s0
.data2 lae.l
.data2 lae.w0
.data2 lae.w1
.data2 lae.w2
.data2 lae.w3
.data2 lae.w4
.data2 lae.w5
.data2 lae.w6
.data2 lal.p
.data2 lal.n
.data2 lal.0
.data2 lal.m1
.data2 lal.w0
.data2 lal.wm1
.data2 lal.wm2
.data2 lar.2
.data2 ldc.0
.data2 lde.l
.data2 lde.w0
.data2 ldl.0
.data2 ldl.wm1
.data2 lfr.2
.data2 lfr.4
.data2 lfr.s0
.data2 lil.wm1
.data2 lil.w0
.data2 lil.0
.data2 lil.2
.data2 lin.l
.data2 lin.s0
.data2 lni.z
.data2 loc.l
.data2 loc.m1
.data2 loc.s0
.data2 loc.sm1
.data2 loe.l
.data2 loe.w0
.data2 loe.w1
.data2 loe.w2
.data2 loe.w3
.data2 loe.w4
.data2 lof.l
.data2 lof.2
.data2 lof.4
.data2 lof.6
.data2 lof.8
.data2 lof.s0
.data2 loi.l
.data2 loi.1
.data2 loi.2
.data2 loi.4
.data2 loi.6
.data2 loi.8
.data2 loi.s0
.data2 lol.p
.data2 lol.n
.data2 lol.0
.data2 lol.2
.data2 lol.4
.data2 lol.6
.data2 lol.m2
.data2 lol.m4
.data2 lol.m6
.data2 lol.m8
.data2 lol.m10
.data2 lol.m12
.data2 lol.m14
.data2 lol.m16
.data2 lol.w0
.data2 lol.wm1
.data2 lxa.1
.data2 lxl.1
.data2 lxl.2
.data2 mlf.s0
.data2 mli.2
.data2 mli.4
.data2 rck.2
.data2 ret.0
.data2 ret.2
.data2 ret.s0
.data2 rmi.2
.data2 sar.2
.data2 sbf.s0
.data2 sbi.2
.data2 sbi.4
.data2 sdl.wm1
.data2 set.s0
.data2 sil.wm1
.data2 sil.w0
.data2 sli.2
.data2 ste.l
.data2 ste.w0
.data2 ste.w1
.data2 ste.w2
.data2 stf.l
.data2 stf.2
.data2 stf.4
.data2 stf.s0
.data2 sti.1
.data2 sti.2
.data2 sti.4
.data2 sti.6
.data2 sti.8
.data2 sti.s0
.data2 stl.p
.data2 stl.n
.data2 stl.p0
.data2 stl.p2
.data2 stl.m2
.data2 stl.m4
.data2 stl.m6
.data2 stl.m8
.data2 stl.m10
.data2 stl.wm1
.data2 teq.z
.data2 tgt.z
.data2 tlt.z
.data2 tne.z
.data2 zeq.l
.data2 zeq.s0
.data2 zeq.s1
.data2 zer.s0
.data2 zge.s0
.data2 zgt.s0
.data2 zle.s0
.data2 zlt.s0
.data2 zne.s0
.data2 zne.sm1
.data2 zre.l
.data2 zre.w0
.data2 zrl.m2
.data2 zrl.m4
.data2 zrl.wm1
.data2 zrl.n
.data2 loop1
.data2 loop2
!----------------- END OF MAIN DISPATCH -------------------------------
xxx:
.data2 loc.0
.data2 loc.1
init:
ld sp,(bdos+1) ! address of fbase
ld hl,xxx
ld de,dispat
ld bc,4
ldir
call uxinit
warmstart:
ld sp,(bdos+1) ! address of fbase
call makeargv
ld de,0x80
ld c,setdma
call bdos
ld c,open
ld de,fcb
call bdos
inc a
jr z,bademfile
ld c,read
ld de,fcb
call bdos
or a
jr nz,bademfile ! no file
ld de,header
ld hl,0x90 ! start of 2nd half of header
ld bc,10 ! we copy only first 5 words
ldir
ld de,(ntext) ! size of program text in bytes
ld hl,0
sbc hl,de
add hl,sp
ld sp,hl ! save space for program
ld (pb),hl ! set procedure base
ld a,0xa0
ld (nextp),a
ld de,(ntext)
xor a
ld h,a
ld l,a
sbc hl,de
ex de,hl
ld h,a
ld l,a
add hl,sp
1: call getb
ld (hl),c
inc hl
inc e
jr nz,1b
inc d
jr nz,1b
! now program text has been read,so start read-
ld iy,0 ! ing data descriptors, (nextp) (was hl) is
ld ix,eb ! pointer into DMA,ix is pointer into global
! data area,iy is #bytes pushed in last instr (used for repeat)
rddata: ld hl,(ndata)
ld a,h
or l
jr z,prdes ! no data left
dec hl
ld (ndata),hl
call getb ! read 1 byte (here:init type) into register c
dec c
jp p,2f
call getw
push iy
pop hl
ld a,h
or l
jr z,5f ! size of block is zero, so no work
push hl
push bc
3: pop hl ! #repeats
pop bc ! block size
push bc
ld a,h
or l
jr z,4f ! ready
dec hl
push hl
push ix
pop hl
add ix,bc
dec hl
ld d,h
ld e,l
add hl,bc
ex de,hl
lddr
jr 3b
4: pop bc
5: ld iy,0 ! now last instruction = repeat = type 0
jr rddata
2: ld b,c ! here other types come
jr nz,2f ! Z-flag was (re-)set when decrementing c
call getb ! uninitialized words, fetch #words
sla c
rl b
ld iy,0
add iy,bc
add ix,bc
4: jr rddata
2: call getb ! remaining types, first fetch #bytes/words
ld a,b
cp 7
jr z,rdflt
jp p,bademfile ! floats are not accepted,nor are illegal types
ld b,0
cp 1
jr z,2f
cp 5
jp m,1f
2: ld iy,0 ! initialized bytes, simply copy from EM-1 file
add iy,bc
ld b,c ! #bytes
3:
call getb
ld (ix),c
inc ix
djnz 3b
jr 4b
1: cp 2
jr z,2f
cp 3
jr z,3f
ld hl,(pb)
jr 4f
3: ld hl,eb
jr 4f
2: ld hl,0
4: ld (ntext),hl ! ntext is used here to hold base address of
ld iy,0 ! correct type: data,instr or 0 (plain numbers)
add iy,bc
add iy,bc
ld b,c
1:
push bc
ex de,hl ! save e into l
call getw
ex de,hl
ld hl,(ntext)
add hl,bc
ld (ix),l
inc ix
ld (ix),h
inc ix
pop bc
djnz 1b
2: jr rddata
rdflt:
ld a,c
cp 4
jr nz,bademfile
push ix
pop hl
1: call getb
ld a,c
ld (hl),a
inc hl
or a
jr nz,1b
push ix
pop hl
call atof
ld b,4
1: ld a,(hl)
ld (ix),a
inc ix
inc hl
djnz 1b
jr rddata
bademfile:
ld c,printstring
ld de,1f
call bdos
jp 0
1: .ascii 'load file error\r\n$'
! now all data has been read,so on with the procedure descriptors
prdes:
ld (hp),ix ! initialize heap pointer
ld de,(nproc)
ld hl,0
xor a
sbc hl,de
add hl,hl
add hl,hl ! 4 bytes per proc-descriptor
add hl,sp
ld sp,hl ! save space for procedure descriptors
push hl
pop ix
ld (pd),hl ! initialize base
ld hl,(nproc)
1: ld a,h
or l
jr z,2f
dec hl
call getb
ld (ix),c
inc ix
call getb
ld (ix),c
inc ix
call getw
ex de,hl
ld hl,(pb)
add hl,bc
ld (ix),l
inc ix
ld (ix),h
inc ix
ex de,hl
jr 1b
2:
ld de,(entry) ! get ready for start of program
ld ix,0 ! reta, jumping here will stop execution
push ix
ld hl,argv
push hl
ld hl,(argc)
push hl
jr cal ! call EM-1 main program
getw: call getb
ld b,c
call getb
ld a,b
ld b,c
ld c,a
ret
getb: push hl ! getb reads 1 byte in register c from standard
push de
ld a,(nextp) ! DMA buffer and refills if necessary
or a
jr nz,1f
push bc
ld c,read
ld de,fcb
call bdos
or a
jr nz,bademfile
pop bc
ld a,0x80
1: ld l,a
ld h,0
ld c,(hl)
inc a
ld (nextp),a
pop de
pop hl
ret
!------------------------- Main loop of the interpreter ---------------
phl: push hl
loop: ld e,(ix) ! e = opcode byte
inc ix ! advance EM program counter to next byte
ld hl,dispat ! hl = address of dispatching table
xor a
ld d,a
add hl,de ! compute address of routine for this opcode
add hl,de ! hl = address of routine to dispatch to
ld d,(hl) ! e = low byte of routine address
inc hl ! hl now points to 2nd byte of routine address
ld h,(hl) ! h = high byte of routine address
ld l,d ! hl = address of routine
ld d,a
jp (hl) ! go execute the routine
loop1: ld e,(ix) ! e = opcode byte
inc ix ! advance EM program counter to next byte
ld hl,dispat1 ! hl = address of dispatching table
xor a
ld d,a
add hl,de ! compute address of routine for this opcode
add hl,de ! hl = address of routine to dispatch to
ld d,(hl) ! e = low byte of routine address
inc hl ! hl now points to 2nd byte of routine address
ld h,(hl) ! h = high byte of routine address
ld l,d ! hl = address of routine
ld d,a
jp (hl) ! go execute the routine
loop2: ld e,(ix) ! e = opcode byte
inc ix ! advance EM program counter to next byte
ld hl,dispat2 ! hl = address of dispatching table
xor a
ld d,a
add hl,de ! compute address of routine for this opcode
add hl,de ! hl = address of routine to dispatch to
ld d,(hl) ! e = low byte of routine address
inc hl ! hl now points to 2nd byte of routine address
ld h,(hl) ! h = high byte of routine address
ld l,d ! hl = address of routine
ld d,a
jp (hl) ! go execute the routine
! Note that d and a are both still 0, and the carry bit is cleared.
! The execution routines make heavy use of these properties.
! The reason that the carry bit is cleared is a little subtle, since the
! two instructions add hl,de affect it. However, since dispat is being
! added twice a number < 256, no carry can occur.
!---------------------- Routines to compute addresses of locals -------
! There are four addressing routines, corresponding to four ways the
! offset can be represented:
! loml: 16-bit offset. Codes 1-32767 mean offsets -2 to -65534 bytes
! loms: 8-bit offset. Codes 1-255 mean offsets -2 to -510 bytes
! lopl: 16-bit offset. Codes 0-32767 mean offsets 0 to +65534 bytes
! lops: 8-bit offset. Codes 0-255 mean offsets 0 to +510 bytes
loml: ld d,(ix) ! loml is for 16-bit offsets with implied minus
inc ix
jr 1f
loms:
dec d
1: ld e,(ix) ! loms is for 8-bit offsets with implied minus
inc ix
ld h,b
ld l,c ! hl = bc
add hl,de
add hl,de ! hl now equals lb - byte offset
jp (iy)
lopl: ld d,(ix) ! lopl is for 16-bit offsets >= 0
inc ix
lops: ld h,d
ld l,(ix) ! fetch low order byte of offset
inc ix
add hl,hl ! convert offset to bytes
ld de,zone ! to account of return address zone
add hl,de
add hl,bc ! hl now equals lb - byte offset
jp (iy)
!---------------------------- LOADS -----------------------------------
! LOC, LPI
loc.l: lpi.l:
ld d,(ix) ! loc with 16-bit offset
inc ix
loc.s0: ld e,(ix) ! loc with 8-bit offset
inc ix
loc.0: loc.1: loc.2: loc.3: loc.4: loc.5: loc.6: loc.7:
loc.8: loc.9: loc.10: loc.11: loc.12: loc.13: loc.14: loc.15:
loc.16: loc.17: loc.18: loc.19: loc.20: loc.21: loc.22: loc.23:
loc.24: loc.25: loc.26: loc.27: loc.28: loc.29: loc.30: loc.31:
loc.32: loc.33:
push de
jr loop
loc.m1: ld hl,-1
jr phl
loc.sm1:dec d ! for constants -256...-1
jr loc.s0
! LDC
ldc.f: ld h,(ix)
inc ix
ld l,(ix)
inc ix
push hl
ld h,(ix)
inc ix
ld l,(ix)
inc ix
jr phl
ldc.l: ld h,(ix)
inc ix
ld l,(ix)
inc ix
ld e,d
bit 7,h
jr z,1f
dec de
1:
push de
jr phl
ldc.0: ld e,d
push de
push de
jr loop
! LOL
lol.0: lol.1: lol.2: lol.3: lol.4: lol.5: lol.6:
ld hl,-b_lolp-b_lolp+zone
add hl,de
add hl,de
add hl,bc
jr ipsh
lol.m2: lol.m4: lol.m6: lol.m8: lol.m10: lol.m12: lol.m14: lol.m16:
ld hl,b_loln+b_loln
sbc hl,de
xor a ! clear carry bit
sbc hl,de
add hl,bc ! hl = lb - byte offset
ipsh: ld e,(hl)
inc hl
ld d,(hl)
push de
jr loop
lol.wm1:ld iy,ipsh
jr loms
lol.n: ld iy,ipsh
jr loml
lol.w0: ld iy,ipsh
jr lops
lol.p: ld iy,ipsh
jr lopl
! LOE
loe.w4: inc d
loe.w3: inc d
loe.w2: inc d
loe.w1: inc d
loe.w0: ld e,(ix)
inc ix
ld hl,eb
add hl,de
add hl,de
jr ipsh
loe.l: ld d,(ix)
inc ix
jr loe.w0
! LOF
lof.2: lof.4: lof.6: lof.8:
ld hl,-b_lof-b_lof ! assume lof 1 means stack +2, not -2
add hl,de
add hl,de
1: pop de
add hl,de
jr ipsh
lof.s0: ld h,d
2: ld l,(ix)
inc ix
jr 1b
lof.l: ld h,(ix)
inc ix
jr 2b
! LAL
lal.m1: ld h,b
ld l,c
dec hl
jr phl
lal.0: ld h,b
ld l,c
ld de,zone
add hl,de
jr phl
lal.wm2:dec d
lal.wm1:ld iy,phl
jr loms
lal.w0: ld iy,phl
jr lops
lal.n: ld h,(ix)
inc ix
ld l,(ix)
inc ix
add hl,bc
jr phl
lal.p: ld h,(ix)
inc ix
ld l,(ix)
inc ix
add hl,bc
ld de,zone
add hl,de
jr phl
! LAE
lae.w8: inc d
lae.w7: inc d
lae.w6: inc d
lae.w5: inc d
lae.w4: inc d
lae.w3: inc d
lae.w2: inc d
lae.w1: inc d
lae.w0: ld e,(ix)
inc ix
ld hl,eb
add hl,de
add hl,de
jr phl
lae.l: ld d,(ix)
inc ix
ld e,(ix)
inc ix
ld hl,eb
add hl,de
jr phl
! LIL
lil.0: lil.2:
ld hl,-b_lil-b_lil+zone
add hl,de
add hl,de
add hl,bc
1: ld e,(hl)
inc hl
ld h,(hl)
ld l,e
jr ipsh
lil.wm1:ld iy,1b
jr loms
lil.n: ld iy,1b
jr loml
lil.w0: ld iy,1b
jr lops
lil.p: ld iy,1b
jr lopl
! LXL, LXA
lxl.1:
ld a,1
jr 7f
lxl.2:
ld a,2
jr 7f
lxl.l: ld d,(ix)
inc ix
lxl.s: ld a,(ix)
inc ix
7: ld iy,phl
5: ld h,b
ld l,c
or a
jr z,3f
2: inc hl
inc hl
inc hl
inc hl
inc hl
inc hl
inc hl
inc hl
.assert [ .-2b-zone] == 0
ld e,(hl)
inc hl
ld h,(hl)
ld l,e
dec a
jr nz,2b
3: cp d
jr z,4f
dec d
jr 2b
4: jp (iy)
lxa.1:
ld a,1
jr 7f
lxa.l: ld d,(ix)
inc ix
lxa.s: ld a,(ix)
inc ix
7: ld iy,1f
jr 5b
1: ld de,zone
add hl,de
jr phl
lpb.z:
pop hl
.assert [ zone/256] == 0
ld e,zone
add hl,de
jr phl
dch.z:
ld e,2
jr loi
exg.z:
pop de
jr exg
exg.l:
ld d,(ix)
inc ix
exg.s0:
ld e,(ix)
inc ix
exg:
push bc
pop iy
ld hl,0
add hl,sp
ld b,h
ld c,l
add hl,de
1:
ld a,(bc)
ex af,af2
ld a,(hl)
ld (bc),a
ex af,af2
ld (hl),a
inc bc
inc hl
dec de
ld a,d
or e
jr nz,1b
push iy
pop bc
jr loop
! LDL
ldl.0: ld de,zone
ld h,b
ld l,c
add hl,de
dipsh: inc hl
inc hl
inc hl
ld d,(hl)
dec hl
ld e,(hl)
dec hl
push de
ld d,(hl)
dec hl
ld e,(hl)
push de
jr loop
ldl.wm1:ld iy,dipsh
jr loms
ldl.n: ld iy,dipsh
jr loml
ldl.w0: ld iy,dipsh
jr lops
ldl.p: ld iy,dipsh
jr lopl
! LDE
lde.l: ld d,(ix)
inc ix
jr lde.w0
lde.w3: inc d
lde.w2: inc d
lde.w1: inc d
lde.w0: ld e,(ix)
inc ix
ld hl,eb
add hl,de
add hl,de
jr dipsh
! LDF
ldf.l: ld d,(ix)
inc ix
ld e,(ix)
inc ix
pop hl
add hl,de
jr dipsh
! LOI,LOS
los.z:
ld iy,los.2
jr pop2
los.l: call long2
los.2: pop de
loi: pop hl
add hl,de
dec hl
srl d
rr e
jr nc,1f
ld a,e
or d
jr nz,eilsize
ld e,(hl) ! here the 1-byte case is caught
push de
jr loop
1: push bc
pop iy
2: ld b,(hl)
dec hl
ld c,(hl)
dec hl
push bc
dec de
ld a,d
or e
jr nz,2b
loiend: push iy
pop bc
jr loop
loi.1: loi.2: loi.4: loi.6: loi.8:
ld hl,-b_loi-b_loi
add hl,de
adc hl,de ! again we use that the carry is cleared
jr nz,1f
inc hl ! in case loi.0 object size is 1 byte!
1: ex de,hl
jr loi
loi.l: ld d,(ix)
inc ix
loi.s0: ld e,(ix)
inc ix
jr loi
! ------------------------------ STORES --------------------------------
! STL
stl.p2: ld hl,2
jr 4f
stl.p0: ld hl,0
4: ld de,zone
add hl,de
add hl,bc
jr ipop
stl.m2: stl.m4: stl.m6: stl.m8: stl.m10:
ld hl,b_stlm+b_stlm
stl.zrl:sbc hl,de
xor a
sbc hl,de
add hl,bc
ipop: pop de
ld (hl),e
inc hl
ld (hl),d
jr loop
stl.wm1:ld iy,ipop
jr loms
stl.n: ld iy,ipop
jr loml
stl.w0: ld iy,ipop
jr lops
stl.p: ld iy,ipop
jr lopl
! STE
ste.w3: inc d
ste.w2: inc d
ste.w1: inc d
ste.w0: ld e,(ix)
inc ix
ld hl,eb
add hl,de
add hl,de
jr ipop
ste.l: ld d,(ix)
inc ix
jr ste.w0
! STF
stf.2: stf.4: stf.6:
ld hl,-b_stf-b_stf
add hl,de
add hl,de
1: pop de
add hl,de
jr ipop
stf.s0: ld h,d
2: ld l,(ix)
inc ix
jr 1b
stf.l: ld h,(ix)
inc ix
jr 2b
! SIL
1: ld e,(hl)
inc hl
ld h,(hl)
ld l,e
jr ipop
sil.wm1:ld iy,1b
jr loms
sil.n: ld iy,1b
jr loml
sil.w0: ld iy,1b
jr lops
sil.p: ld iy,1b
jr lopl
! STI, STS
sts.z:
ld iy,sts.2
jr pop2
sts.l: call long2
sts.2: pop de
sti: pop hl
srl d
rr e
jr nc,1f
ld a,e
or d
jr nz,eilsize
pop de ! here the 1-byte case is caught
ld (hl),e
jr loop
1: push bc
pop iy
2: pop bc
ld (hl),c
inc hl
ld (hl),b
inc hl
dec de
ld a,e
or d
jr nz,2b
jr loiend
sti.1: sti.2: sti.4: sti.6: sti.8:
ld hl,-b_sti-b_sti
add hl,de
adc hl,de ! again we use that the carry is cleared
jr nz,1f
inc hl ! in case sti.0 object size is 1 byte!
1: ex de,hl
jr sti
sti.l: ld d,(ix)
inc ix
sti.s0: ld e,(ix)
inc ix
jr sti
! SDL
sdl.wm1:ld iy,1f
jr loms
sdl.n: ld iy,1f
jr loml
sdl.w0: ld iy,1f
jr lops
sdl.p: ld iy,1f
jr lopl
! SDE
sde.l: ld d,(ix)
inc ix
ld e,(ix)
inc ix
ld hl,eb
2: add hl,de
1: pop de
ld (hl),e
inc hl
ld (hl),d
inc hl
jr ipop
! SDF
sdf.l: ld d,(ix)
inc ix
ld e,(ix)
inc ix
pop hl
jr 2b
!------------------------- SINGLE PRECISION ARITHMETIC ---------------
! ADI, ADP, ADS, ADU
adi.z: adu.z:
pop de
9:
call chk24
.data2 adi.2,adi.4
adi.l: adu.l:
ld d,(ix) ! I guess a routine chk24.l could do this job
inc ix
ld e,(ix)
inc ix
jr 9b
ads.z:
ld iy,adi.2
jr pop2
ads.l:
call long2
ads.2: adi.2: adu.2:
pop de
1: pop hl
add hl,de
jr phl
adp.l: ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr 1b
adp.sm1:dec d
adp.s0: ld e,(ix)
inc ix
jr 1b
adp.2: pop hl
inc hl
jr 1f
inc.z:
adp.1: pop hl
1: inc hl
jr phl
! SBI, SBP, SBS, SBU (but what is SBP?)
sbi.z: sbu.z:
pop de
9:
call chk24
.data2 sbi.2,sbi.4
sbi.l: sbu.l:
ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr 9b
sbs.z:
ld iy,sbi.2
jr pop2
sbs.l:
call long2
sbi.2:
pop de
pop hl
sbc hl,de
jr phl
! NGI
ngi.z:
pop de
9:
call chk24
.data2 ngi.2,ngi.4
ngi.l:
ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr 9b
ngi.2: ld hl,0
pop de
sbc hl,de
jr phl
! MLI, MLU Johan version
mli.z: mlu.z:
pop de
9:
call chk24
.data2 mli.2,mli.4
mli.l: mlu.l:
ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr 9b
mli.2: mlu.2:
ld iy,loop
mliint: pop de
pop hl
push bc
ld b,h
ld c,l
ld hl,0
ld a,16
0:
bit 7,d
jr z,1f
add hl,bc
1:
dec a
jr z,2f
ex de,hl
add hl,hl
ex de,hl
add hl,hl
jr 0b
2:
pop bc
push hl
jp (iy)
! DVI, DVU
dvi.z:
pop de
9:
call chk24
.data2 dvi.2,dvi.4
dvi.l:
ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr 9b
dvi.2:
pop hl
pop de
push bc
ld b,h
ld c,l
xor a
ld h,a
ld l,a
sbc hl,bc
jp m,1f
ld b,h
ld c,l
cpl
1:
or a
ld hl,0
sbc hl,de
jp m,1f
ex de,hl
cpl
1:
push af
ld hl,0
ld a,16
0:
add hl,hl
ex de,hl
add hl,hl
ex de,hl
jr nc,1f
inc hl
or a
1:
sbc hl,bc
inc de
jp p,2f
add hl,bc
dec de
2:
dec a
jr nz,0b
pop af
or a
jr z,1f
ld hl,0
sbc hl,de
ex de,hl
1:
pop bc
push de
jr loop
dvu.z:
pop de
9:
call chk24
.data2 dvu.2,dvu.4
dvu.l:
ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr 9b
dvu.2:
pop hl
pop de
push bc
ld b,h
ld c,l
ld hl,0
ld a,16
0:
add hl,hl
ex de,hl
add hl,hl
ex de,hl
jr nc,1f
inc hl
or a
1:
sbc hl,bc
inc de
jp p,2f
add hl,bc
dec de
2:
dec a
jr nz,0b
pop bc
push de
jr loop
! RMI, RMU
rmi.z:
pop de
9:
call chk24
.data2 rmi.2,rmi.4
rmi.l:
ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr 9b
rmi.2:
pop hl
pop de
push bc
ld b,h
ld c,l
xor a
ld h,a
ld l,a
sbc hl,bc
jp m,1f
ld b,h
ld c,l
1:
or a
ld hl,0
sbc hl,de
jp m,1f
ex de,hl
cpl
1:
push af
ld hl,0
ld a,16
0:
add hl,hl
ex de,hl
add hl,hl
ex de,hl
jr nc,1f
inc hl
or a
1:
sbc hl,bc
inc de
jp p,2f
add hl,bc
dec de
2:
dec a
jr nz,0b
ex de,hl
pop af
or a
jr z,1f
ld hl,0
sbc hl,de
ex de,hl
1:
pop bc
push de
jr loop
rmu.4:
ld iy,.dvu4
jr 1f
rmi.4:
ld iy,.dvi4
1:
ld (retarea),bc
ld (retarea+2),ix
ld hl,1f
push hl
push iy
ret
1:
pop hl
pop hl
push bc
push de
ld bc,(retarea)
ld ix,(retarea+2)
jr loop
rmu.z:
pop de
9:
call chk24
.data2 rmu.2,rmu.4
rmu.l:
ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr 9b
rmu.2:
pop hl
pop de
push bc
ld b,h
ld c,l
ld hl,0
ld a,16
0:
add hl,hl
ex de,hl
add hl,hl
ex de,hl
jr nc,1f
inc hl
or a
1:
sbc hl,bc
inc de
jp p,2f
add hl,bc
dec de
2:
dec a
jr nz,0b
pop bc
jr phl
! SLI, SLU
slu.z: sli.z:
pop de
9:
call chk24
.data2 sli.2,sli.4
slu.l:
sli.l:
ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr 9b
sli.2:
pop de
pop hl
ld a,d
or a
jr z,1f
ld e,15
2: add hl,hl
1: dec e
jp m,phl
jr 2b
sli.4:
slu.4:
pop de
pop iy
pop hl
inc d
dec d
jr z,1f
ld e,31
1:
dec e
jp m,2f
add iy,iy
adc hl,hl
jr 1b
2:
push hl
push iy
jr loop
! SRI, SRU
sri.z:
pop de
9:
call chk24
.data2 sri.2,sri.4
sri.l:
ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr 9b
sri.2: pop de
pop hl
ld a,d
or a
jr z,1f
ld e,15
2: sra h
rr l
1: dec e
jp m,phl
jr 2b
sri.4:
pop de
ld a,e
inc d
dec d
pop de
pop hl
jr z,1f
ld a,31
1:
dec a
jp m,2f
sra h
rr l
rr d
rr e
jr 1b
2:
push hl
push de
jr loop
sru.z:
pop de
9:
call chk24
.data2 sru.2,sru.4
sru.l:
ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr 9b
sru.2: pop de
pop hl
ld a,d
or a
jr z,1f
ld e,15
2: srl h
rr l
1: dec e
jp m,phl
jr 2b
sru.4:
pop de
ld a,e
inc d
dec d
pop de
pop hl
jr z,1f
ld a,31
1:
dec a
jp m,2f
srl h
rr l
rr d
rr e
jr 1b
2:
push hl
push de
jr loop
! ROL, ROR
rol.z:
pop de
9:
call chk24
.data2 rol.2,rol.4
rol.l:
ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr 9b
rol.2: pop de
pop hl
ld a,e
and 15
jr z,phl
ld de,0
1: add hl,hl
adc hl,de
dec a
jr nz,1b
jr phl
rol.4:
pop de
pop iy
pop hl
ld a,e
and 31
jr z,3f
1:
add iy,iy
adc hl,hl
jr nc,2f
inc iy
2:
dec a
jr nz,1b
3:
push hl
push iy
ror.z:
pop de
9:
call chk24
.data2 ror.2,ror.4
ror.l:
ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr 9b
ror.2: pop de
pop hl
ld a,e
and 15
jr z,phl
1: srl h
rr l
jr nc,2f
set 7,h
2: dec a
jr nz,1b
jr phl
ror.4:
pop de
ld a,e
pop de
pop hl
and 31
jr z,0f
1:
srl h
rr l
rr d
rr e
jr nc,2f
set 7,h
2:
dec a
jr nz,1b
0:
push hl
push de
jr loop
pop2: ld de,2
pop hl
sbc hl,de
jr nz,eilsize
xor a
ld d,a
jp (iy)
chk24:
! this routine is used to call indirectly
! a routine for either 2 or 4 byte operation
! ( e.g. mli.2 or mli.4)
! de contains 2 or 4
! iy points to a descriptor containing
! the addresses of both routines
pop iy ! address of descriptor
ld a,d ! high byte must be 0
or a
jr nz,unimpld
ld a,e
cp 2
jr z,1f
inc iy
inc iy ! points to word containing
! address of 4 byte routine
cp 4
jr nz,unimpld
1:
ld h,(iy+1)
ld l,(iy)
xor a
jp (hl)
!--------------------- INCREMENT, DECREMENT, ZERO ----------------------
! INC
inl.m2: inl.m4: inl.m6:
ld hl, b_inl+b_inl
sbc hl,de
xor a
sbc hl,de
add hl,bc
1: inc (hl)
jr nz,loop
inc hl
inc (hl)
jr loop
inl.wm1:ld iy,1b
jr loms
inl.n: ld iy,1b
jr loml
inl.p: ld iy,1b
jr lopl
! INE
ine.w3: inc d
ine.w2: inc d
ine.w1: inc d
ine.w0: ld e,(ix)
inc ix
ld hl,eb
add hl,de
add hl,de
jr 1b
ine.l: ld d,(ix)
inc ix
jr ine.w0
! DEC
dec.z: pop hl
dec hl
push hl
jr loop
1: ld e,(hl)
inc hl
ld d,(hl)
dec de
ld (hl),d
dec hl
ld (hl),e
jr loop
del.wm1:ld iy,1b
jr loms
del.n: ld iy,1b
jr loml
del.p: ld iy,1b
jr lopl
! DEE
dee.w3: inc d
dee.w2: inc d
dee.w1: inc d
dee.w0: ld e,(ix)
inc ix
ld hl,eb
add hl,de
add hl,de
jr 1b
dee.l: ld d,(ix)
inc ix
jr dee.w0
! ZERO
zri2: zru2:
ld h,d
ld l,d
jr phl
zrf.z:
zer.z: pop de
2: ld hl,0
sra d
rr e
1: push hl
dec de
ld a,e
or d
jr nz,1b
jr loop
zrf.l:
zer.l: ld d,(ix)
inc ix
zer.s0: ld e,(ix)
inc ix
jr 2b
zrl.m2: zrl.m4:
ld h,d
ld l,d
push hl
ld hl,b_zrl+b_zrl
jr stl.zrl
zrl.wm1:
ld h,d
ld l,d
push hl
jr stl.wm1
zrl.n:
ld h,d
ld l,d
push hl
jr stl.n
zrl.w0:
ld h,d
ld l,d
push hl
jr stl.w0
zrl.p:
ld h,d
ld l,d
push hl
jr stl.p
zre.w0:
ld h,d
ld l,d
push hl
jr ste.w0
zre.l:
ld h,d
ld l,d
push hl
jr ste.l
! ------------------------- CONVERT GROUP ------------------------------
! CII, CIU
cii.z: ciu.z:
pop hl
pop de
sbc hl,de ! hl and de can only have values 2 or 4, that's
! why a single subtract can split the 3 cases
jr z,loop ! equal, so do nothing
jp p,2f
3: pop hl ! smaller, so shrink size from double to single
pop de
jr phl
2: pop hl ! larger, so expand (for cii with sign extend)
res 1,e
bit 7,h
jr z,1f
dec de
1: push de
jr phl
! CUI, CUU
cui.z: cuu.z:
pop hl
pop de
sbc hl,de
jr z,loop
jp m,3b
res 1,e
pop hl
jr 1b
! ------------------------------ SETS ---------------------------------
! SET
set.z: pop hl
doset: pop de
push bc
pop iy
ld b,h
ld c,l
xor a
0: push af
inc sp
dec c
jr nz,0b
dec b
jp p,0b
push iy
pop bc
ex de,hl
ld a,l
sra h
jp m,unimpld
rr l
sra h
rr l
sra h
rr l
push hl
or a
sbc hl,de
pop hl
jp p,unimpld
add hl,sp
ld (hl),1
and 7
jr 1f
0: sla (hl)
dec a
1: jr nz,0b
jr loop
set.l: ld d,(ix)
inc ix
set.s0: ld e,(ix)
inc ix
ex de,hl
jr doset
! INN
inn.z: pop hl
jr 1f
inn.l: ld d,(ix)
inc ix
inn.s0: ld e,(ix)
inc ix
ex de,hl
1:
pop de
add hl,sp
push hl
pop iy
ex de,hl
ld a,l
sra h
jp m,0f
rr l
sra h
rr l
sra h
rr l
add hl,sp
push hl
or a ! clear carry
sbc hl,de
pop hl
jp m,1f
0: xor a
jr 4f
1: ld e,(hl)
and 7
jr 2f
3: rrc e
dec a
2: jr nz,3b
ld a,e
and 1
4: ld l,a
ld h,0
ld sp,iy
jr phl
! ------------------------- LOGICAL GROUP -----------------------------
! AND
and.z: pop de
doand: ld h,d
ld l,e
add hl,sp
push bc
ld b,h
ld c,l
ex de,hl
add hl,de
1: dec hl
dec de
ld a,(de)
and (hl)
ld (hl),a
xor a
sbc hl,bc
jr z,2f
add hl,bc
jr 1b
2: ld h,b
ld l,c
pop bc
ld sp,hl
jr loop
and.l: ld d,(ix)
inc ix
and.s0: ld e,(ix)
inc ix
jr doand
and.2: ld e,2
jr doand
! IOR
ior.z: pop de
ior: ld h,d
ld l,e
add hl,sp
push bc
ld b,h
ld c,l
ex de,hl
add hl,de
1: dec hl
dec de
ld a,(de)
or (hl)
ld (hl),a
xor a
sbc hl,bc
jr z,2f
add hl,bc
jr 1b
2: ld h,b
ld l,c
pop bc
ld sp,hl
jr loop
ior.l: ld d,(ix)
inc ix
ior.s0: ld e,(ix)
inc ix
jr ior
ior.2: ld e,2
jr ior
! XOR
xor.z: pop de
exor: ld h,d
ld l,e
add hl,sp
push bc
ld b,h
ld c,l
ex de,hl
add hl,de
1: dec hl
dec de
ld a,(de)
xor (hl)
ld (hl),a
xor a
sbc hl,bc
jr z,2f
add hl,bc
jr 1b
2: ld h,b
ld l,c
pop bc
ld sp,hl
jr loop
xor.l: ld d,(ix)
inc ix
ld e,(ix)
inc ix
jr exor
! COM
com.z: pop hl
com: add hl,sp
1: dec hl
ld a,(hl)
cpl
ld (hl),a
xor a
sbc hl,sp
jr z,loop
add hl,sp
jr 1b
com.l: ld d,(ix)
inc ix
ld e,(ix)
inc ix
ex de,hl
jr com
! ------------------------- COMPARE GROUP ------------------------------
! CMI
cmi.2: pop de
pop hl
ld a,h
xor d ! check sign bit to catch overflow with subtract
jp m,1f
sbc hl,de
jr phl
1: xor d ! now a equals (original) h again
jp m,phl
set 0,l ! to catch case hl=0>de bit 0 is set explicitly
jr phl
! CMU, CMP
cmi.4: inc a
ld de,4
jr docmu
cmp.z: ld de,2
jr docmu
cmi.z: inc a
cmu.z:
pop de
jr docmu
cmi.l: inc a
cmu.l: ld d,(ix)
inc ix
ld e,(ix)
inc ix
docmu: push bc
pop iy
ld b,d
ld c,e
ld hl,0
add hl,sp
add hl,bc
dec hl
ld d,h
ld e,l
add hl,bc
ld (retarea),hl ! save new sp-1
or a
jr z,1f
ld a,(de)
cp (hl)
dec hl
dec de
dec bc
jr z,1f
jp p,4f
jp pe,5f
jr 6f
1:
ld a,(de)
cp (hl)
dec de
dec hl
dec bc
jr nz,2f
ld a,b
or c
jr nz,1b
ld d,a
ld e,a
jr 3f
2:
jr nc,5f
6:
ld de,1
jr 3f
4:
jp pe,6b
5:
ld de,-1
3:
ld hl,(retarea)
inc hl
ld sp,hl
push de
push iy
pop bc
jr loop
! CMS
cms.z: pop hl
jr 1f
cms.l: ld d,(ix)
inc ix
cms.s0: ld e,(ix)
inc ix
ex de,hl
1: push bc
pop iy
ld b,h
ld c,l
add hl,sp
0:
dec sp
pop af
cpi
jr nz,1f
ld a,b
or c
jr nz,0b
ld de,0
jr 2f
1:
add hl,bc
ld de,1
2:
ld sp,hl
push de
push iy
pop bc
jr loop
! TLT, TLE, TEQ, TNE, TGE, TGT
tlt.z:
ld h,d
ld l,d
pop de
bit 7,d
jr z,1f
inc l
1:
jr phl
tle.z: ld hl,1
pop de
xor a
add a,d
jp m,phl
jr nz,1f
xor a
add a,e
jr z,2f
1: dec l
2:
jr phl
teq.z:
ld h,d
ld l,d
pop de
ld a,d
or e
jr nz,1f
inc l
1:
jr phl
tne.z:
ld h,d
ld l,d
pop de
ld a,d
or e
jr z,1f
inc l
1:
jr phl
tge.z:
ld h,d
ld l,d
pop de
bit 7,d
jr nz,1f
inc l
1:
jr phl
tgt.z:
ld h,d
ld l,d
pop de
xor a
add a,d
jp m,phl
jr nz,1f
xor a
add a,e
jr z,2f
1: inc l
2:
jr phl
! ------------------------- BRANCH GROUP -------------------------------
! BLT, BLE, BEQ, BNE, BGE, BGT, BRA
b.pl: ld d,(ix)
inc ix
b.ps: ld e,(ix)
inc ix
push ix
pop hl
add hl,de
pop de
ex (sp),hl
xor a
jp (iy)
bra.l: ld d,(ix)
inc ix
jr bra.s0
bra.sm2:dec d
bra.sm1:dec d
dec d
bra.s1: inc d
bra.s0: ld e,(ix)
inc ix
add ix,de
jr loop
bgo: pop ix ! take branch
jr loop
blt.s0: ld iy,blt
jr b.ps
blt.l: ld iy,blt
jr b.pl
blt: ld a,h
xor d
jp m,1f
sbc hl,de
jr 2f
1: xor d
2: jp m,bgo
pop de
jr loop
ble.s0: ld iy,ble
jr b.ps
ble.l: ld iy,ble
jr b.pl
ble: ex de,hl
jr bge
beq.s0: ld iy,beq
jr b.ps
beq.l: ld iy,beq
jr b.pl
beq: sbc hl,de
jr z,bgo
pop de ! keep stack clean, so dump unused jump address
jr loop
bne.s0: ld iy,bne
jr b.ps
bne.l: ld iy,bne
jr b.pl
bne: sbc hl,de
jr nz,bgo
pop de ! keep stack clean, so dump unused jump address
jr loop
bge.s0: ld iy,bge
jr b.ps
bge.l: ld iy,bge
jr b.pl
bge: ld a,h
xor d ! check sign bit to catch overflow with subtract
jp m,1f
sbc hl,de
jr 2f
1: xor d ! now a equals (original) h again
2: jp p,bgo
pop de ! keep stack clean, so dump unused jump address
jr loop
bgt.s0: ld iy,bgt
jr b.ps
bgt.l: ld iy,bgt
jr b.pl
bgt: ex de,hl
jr blt
! ZLT, ZLE, ZEQ, ZNE, ZGE, ZGT
z.pl: ld d,(ix)
inc ix
z.ps: ld e,(ix)
inc ix
push ix
pop hl
add hl,de
ex de,hl
pop hl
xor a
add a,h
jp (iy)
zlt.l: ld iy,zlt
jr z.pl
zlt.s0: ld iy,zlt
jr z.ps
zlt: jp m,zgo
jr loop
zle.l: ld iy,zle
jr z.pl
zle.s0: ld iy,zle
jr z.ps
zle: jp m,zgo
jr nz,loop
xor a
add a,l
jr z,zgo
jr loop
zeq.l: ld iy,zeq
jr z.pl
zeq.s1: inc d
zeq.s0: ld iy,zeq
jr z.ps
zeq: ld a,l
or h
jr nz,loop
zgo: push de
pop ix
jr loop
zne.sm1:dec d
jr zne.s0
zne.l: ld iy,zne
jr z.pl
zne.s0: ld iy,zne
jr z.ps
zne: ld a,l
or h
jr nz,zgo
jr loop
zge.l: ld iy,zge
jr z.pl
zge.s0: ld iy,zge
jr z.ps
zge: jp m,loop
jr zgo
zgt.l: ld iy,zgt
jr z.pl
zgt.s0: ld iy,zgt
jr z.ps
zgt: jp m,loop
jr nz,zgo
xor a
add a,l
jr z,loop
jr zgo
! ------------------- ARRAY REFERENCE GROUP ---------------------------
! AAR
aar.z:
ld iy,aar.2
jr pop2
aar.l: call long2
aar.2: ld hl,loop
aarint: pop iy ! descriptor
ex (sp),hl ! save return address and hl:=index
ld e,(iy+0)
ld d,(iy+1) ! de := lwb
ld a,h
xor d
jp m,1f
sbc hl,de
jr 2f
1: sbc hl,de
xor d
2: call m,e.array
ld e,(iy+2)
ld d,(iy+3) ! de := upb - lwb
push hl
ex de,hl
ld a,h
xor d
jp m,1f
sbc hl,de
jr 2f
1: xor d
2: ex de,hl
pop hl
call m,e.array
1: ld e,(iy+4)
ld d,(iy+5)
pop iy
ex (sp),iy
push iy ! exchange base address and return address
push de
push de
push hl
ld iy,1f
jr mliint
1: pop de
pop iy
pop hl
push iy
add hl,de
pop de
ex (sp),hl
jp (hl)
lar.l: call long2
lar.2: ld hl,loi
jr aarint
lar.z:
ld iy,lar.2
jr pop2
sar.l: call long2
sar.2: ld hl,sti
jr aarint
sar.z:
ld iy,sar.2
jr pop2
! --------------------- PROCEDURE CALL/RETURN --------------------------
! CAL
cal.1: cal.2: cal.3: cal.4: cal.5: cal.6: cal.7: cal.8:
cal.9: cal.10: cal.11: cal.12: cal.13: cal.14: cal.15: cal.16:
cal.17: cal.18: cal.19: cal.20: cal.21: cal.22: cal.23: cal.24:
cal.25: cal.26: cal.27: cal.28:
ld hl,-b_cal
add hl,de
ex de,hl
jr cal
cal.l: ld d,(ix)
inc ix
cal.s0: ld e,(ix)
inc ix
cal: push ix ! entry point for main program of interpreter
push bc
ld hl,(eb)
push hl
ld hl,(ebp4)
push hl
! temporary tracing facility
! NOP it if you don't want it
push de
ld de,(ebp4)
ld hl,(eb)
call prline
pop de
! end of temporary tracing
ld hl,0
add hl,sp
ld b,h
ld c,l
ld hl,(pd)
ex de,hl
add hl,hl
add hl,hl
add hl,de
push hl
pop iy
ld e,(iy+0)
ld d,(iy+1)
ld l,c
ld h,b
xor a
sbc hl,de
ld sp,hl
ld e,(iy+2)
ld d,(iy+3)
ld ix,0
add ix,de
jr loop
! CAI
cai.z: pop de
jr cal
! LFR
lfr.z: pop de
2: ld a,e
rr a
cp 5
jp p,eilsize ! only result sizes <= 8 are allowed
ld hl,retarea
add hl,de
1: dec hl
ld d,(hl)
dec hl
ld e,(hl)
push de
dec a
jr nz,1b
jr loop
lfr.l: ld d,(ix)
inc ix
lfr.s0: ld e,(ix)
inc ix
jr 2b
lfr.2: ld hl,(retarea)
jr phl
lfr.4: ld de,4
jr 2b
! RET
ret.2: ld a,1
jr 3f
ret.z: pop de
2: ld a,d
or e
jr z,ret.0
rr a
cp 5
jp p,eilsize ! only result sizes <= 8 bytes are allowed
3: ld hl,retarea
1: pop de
ld (hl),e
inc hl
ld (hl),d
inc hl
dec a
jr nz,1b
ret.0:
ld h,b
ld l,c
ld sp,hl
pop hl
ld (ebp4),hl
pop hl
ld (eb),hl
pop bc ! old LB
pop ix ! reta
push ix ! check to see if reta = boot (= 0)
pop hl
ld a,l
or h
jr nz,loop ! not done yet
call uxfinish
jr boot
ret.l: ld d,(ix)
inc ix
ret.s0: ld e,(ix)
inc ix
jr 2b
! ------------------------- MISCELLANEOUS -----------------------------
! SIG, TRP, RTT
sig.z:
ld hl,(trapproc)
ex (sp),hl
ld (trapproc),hl
jr loop
trp.z:
ex (sp),hl
push de
push af
push ix
push iy
push bc
! ld iy,trapproc
! ld a,(iy)
! or (iy+1)
! jr nz,1f
ld iy,2f+13
call octnr
ld c,printstring
ld de,2f
call bdos
ld de,(ebp4)
ld hl,(eb)
call prline
0:
pop iy ! LB
ld a,(iy+6)
or (iy+7) ! reta
jr nz,3f
call uxfinish
jp boot
3:
ld c,(iy+4)
ld b,(iy+5)
push bc ! next LB
ld e,(iy)
ld d,(iy+1) ! file name
ld l,(iy+2)
ld h,(iy+3) ! lineno
call prline
jr 0b
!1:
! ld ix,0
! push hl
! ld hl,(trapproc)
! push hl
! ld hl,0
! ld (trapproc),hl
! jr cai.z
2: .ascii 'error 0xxxxxx\r\n$'
prline:
! prints lineno (hl) and filename (de)
push de
ld iy,2f+12
call octnr
ld c,printstring
ld de,2f
call bdos
pop de
ld hl,4f
0:
ld a,(de)
or a
jr z,1f
ld (hl),a
inc de
inc hl
jr 0b
1:
ld (hl),36 ! '$'
ld de,4f
ld c,printstring
call bdos
ld de,3f
ld c,printstring
call bdos
ret
2: .ascii 'line 0xxxxxx in $'
3: .ascii '\r\n$'
4: .space 12
rtt.z=ret.0
! NOP
! changed into output routine to print linenumber
! in octal (6 digits)
nop.z: push bc
ld iy,1f+12
ld hl,(eb)
call octnr
ld iy,1f+20
ld hl,0
add hl,sp
call octnr
ld c,printstring
ld de,1f
call bdos
pop bc
jr loop
1: .ascii 'test 0xxxxxx 0xxxxxx\r\n$'
octnr:
ld b,6
1: ld a,7
and l
add a,'0'
dec iy
ld (iy+0),a
srl h
rr l
srl h
rr l
srl h
rr l
djnz 1b
ret
! DUP
dup.2: pop hl
push hl
jr phl
dus.z:
ld iy,1f
jr pop2
dus.l: call long2
1: push bc
pop iy
pop bc
jr dodup
dup.l:
push bc
pop iy
ld b,(ix)
inc ix
ld c,(ix)
inc ix
dodup: ld h,d
ld l,d ! ld hl,0
add hl,sp
ld d,h
ld e,l
xor a
sbc hl,bc
ld sp,hl
ex de,hl
ldir
push iy
pop bc
jr loop
! BLM, BLS
bls.z:
ld iy,blm
jr pop2
bls.l: call long2
blm:
push bc
pop iy
pop bc
pop de
pop hl
ldir
push iy
pop bc
jr loop
blm.l:
ld d,(ix)
inc ix
blm.s0: ld e,(ix)
inc ix
push de
jr blm
! ASP, ASS
ass.z:
ld iy,1f
jr pop2
ass.l: call long2
1: pop hl
jr 1f
asp.l:
ld h,(ix)
inc ix
ld l,(ix)
inc ix
asp: add hl,hl
1: add hl,sp
ld sp,hl
jr loop
asp.2: asp.4: asp.6: asp.8: asp.10:
ld hl,-b_asp
add hl,de
jr asp
asp.w0: ld e,(ix)
inc ix
ex de,hl
jr asp
! CSA
csa.z:
ld iy,csa.2
jr pop2
csa.l: call long2
csa.2:
!! temporary version while bug in cem remains
! pop iy
! pop de
! push bc
! ld c,(iy)
! ld b,(iy+1)
! ld l,(iy+4)
! ld h,(iy+5)
! xor a
! sbc hl,de
! jp m,1f
! ex de,hl
! ld e,(iy+2)
! ld d,(iy+3)
! xor a
! sbc hl,de
! jp m,1f
! end of temporary piece
pop iy
pop hl
push bc
ld c,(iy)
ld b,(iy+1)
ld e,(iy+2)
ld d,(iy+3)
xor a
sbc hl,de
jp m,1f
ex de,hl
ld l,(iy+4)
ld h,(iy+5)
xor a
sbc hl,de
jp m,1f
ex de,hl
add hl,hl
ld de,6
add hl,de
ex de,hl
add iy,de
ld l,(iy)
ld h,(iy+1)
ld a,h
or l
jr nz,2f
1: ld a,b
or c
jr z,e.case
ld l,c
ld h,b
2: pop bc
push hl
pop ix
jr loop
! CSB
csb.z:
ld iy,csb.2
jr pop2
csb.l: call long2
csb.2:
pop ix
pop iy
ld e,(ix)
inc ix
ld d,(ix)
inc ix
push de
ex (sp),iy
pop de
push bc
ld c,(ix)
inc ix
ld b,(ix)
inc ix
1:
ld a,b
or c
jr z,noteq
ld a,(ix+0)
cp e
jr nz,2f
ld a,(ix+1)
cp d
jr nz,2f
ld l,(ix+2)
ld h,(ix+3)
jr 3f
2: inc ix
inc ix
inc ix
inc ix
dec bc
jr 1b
noteq: push iy
pop hl
3: ld a,l
or h
jr z,e.case
2:
pop bc
push hl
pop ix
jr loop
! LIN
lin.l: ld d,(ix)
inc ix
lin.s0: ld e,(ix)
inc ix
ld (eb),de
jr loop
! FIL
fil.z: pop hl
1:
ld (ebp4),hl
jr loop
fil.l: ld h,(ix)
inc ix
ld l,(ix)
inc ix
ld de,eb
add hl,de
jr 1b
! LNI
lni.z: ld hl,(eb)
inc hl
ld (eb),hl
jr loop
! RCK
rck.z:
ld iy,rck.2
jr pop2
rck.l: call long2
rck.2:
pop iy
3: pop hl
push hl
ld e,(iy)
ld d,(iy+1)
ld a,h
xor d ! check sign bit to catch overflow with subtract
jp m,1f
sbc hl,de
jr 2f
1: xor d ! now a equals (original) h again
2: call m,e.rck
pop de
push de
ld l,(iy+2)
ld h,(iy+3)
ld a,h
xor d ! check sign bit to catch overflow with subtract
jp m,1f
sbc hl,de
jr 2f
1: xor d ! now a equals (original) h again
2: call m,e.rck
jr loop
! LIM
lim.z: ld hl,(ignmask)
jr phl
! SIM
sim.z: pop de
ld (ignmask),de
jr loop
! LOR
lor.s0: ld e,(ix)
inc ix
ld a,d
or e
jr nz,1f
push bc
jr loop
1: ld hl,-1
adc hl,de
jr nz,1f
add hl,sp
jr phl
1: ld hl,(hp)
jr phl
! STR
str.s0: ld e,(ix)
inc ix
ld a,d
or e
jr nz,1f
pop bc
jr loop
1: pop hl
dec de
ld a,d
or e
jr nz,1f
ld sp,hl
jr loop
1: ld (hp),hl
jr loop
! Floating point calling routines
loadfregs:
pop hl
pop de
ld (fpac),de
pop de
ld (fpac+2),de
pop de
ld (fpop),de
pop de
ld (fpop+2),de
jp (hl)
dofltop:
call loadfregs
push bc
push ix
ld hl,1f
push hl
push iy
ret ! really a call
1:
pop ix
pop bc
ld hl,(fpac+2)
push hl
ld hl,(fpac)
jr phl
pop4:
pop hl
or h
jr nz,9f
ld a,l
cp 4
jr nz,9f
jp (iy)
arg4:
or d
jr nz,9f
ld a,(ix)
inc ix
cp 4
jr nz,9f
jp (iy)
9: jr unimpld
adf.z: ld iy,doadf
jr pop4
adf.l: ld d,(ix)
inc ix
adf.s0: ld iy,doadf
jr arg4
doadf:
ld iy,fpadd ! routine to call
jr dofltop
sbf.z: ld iy,dosbf
jr pop4
sbf.l: ld d,(ix)
inc ix
sbf.s0: ld iy,dosbf
jr arg4
dosbf:
ld iy,fpsub ! routine to call
jr dofltop
mlf.z: ld iy,domlf
jr pop4
mlf.l: ld d,(ix)
inc ix
mlf.s0: ld iy,domlf
jr arg4
domlf:
ld iy,fpmult ! routine to call
jr dofltop
dvf.z: ld iy,dodvf
jr pop4
dvf.l: ld d,(ix)
inc ix
dvf.s0: ld iy,dodvf
jr arg4
dodvf:
ld iy,fpdiv ! routine to call
jr dofltop
cmf.z: ld iy,docmf
jr pop4
cmf.l: ld d,(ix)
inc ix
cmf.s0: ld iy,docmf
jr arg4
docmf:
call loadfregs
push bc
push ix
call fpcmf
pop ix
pop bc
ld hl,(fpac)
jr phl
cfi.z:
pop de
call chk24
.data2 1f,0f
1: ld iy,1f
jr pop4
1: pop hl
ld (fpac),hl
pop hl
ld (fpac+2),hl
push bc
push ix
call fpcfi
pop ix
pop bc
ld hl,(fpac)
jr phl
0: ld iy,1f
jr pop4
1: pop hl
ld (fpac),hl
pop hl
ld (fpac+2),hl!
push bc
push ix
call fpcfd
jr 8f
cif.z:
ld iy,1f
jr pop4
1:
pop de
call chk24
.data2 1f,0f
1: pop hl
ld (fpac),hl
push bc
push ix
call fpcif
8: pop ix
pop bc
ld hl,(fpac+2)
push hl
ld hl,(fpac)
jr phl
0: pop hl
ld (fpac),hl
pop hl
ld (fpac+2),hl
push bc
push ix
call fpcdf
jr 8b
ngf.l: ld d,(ix)
inc ix
ld iy,1f
jr arg4
ngf.z:
ld iy,1f
jr pop4
1: pop hl
ld (fpac),hl
pop hl
ld (fpac+2),hl
push bc
push ix
call fpcomp
jr 8b
fif.z:
ld iy,1f
jr pop4
fif.l:
ld d,(ix)
inc ix
ld iy,1f
jr arg4
1: call loadfregs
push bc
push ix
call fpfif
pop ix
pop bc
ld hl,(fpac+2)
push hl
ld hl,(fpac)
push hl
ld hl,(fpop+2)
push hl
ld hl,(fpop)
jr phl
fef.z:
ld iy,1f
jr pop4
fef.l:
ld d,(ix)
inc ix
ld iy,1f
jr arg4
1: pop hl
ld (fpop),hl
pop hl
ld (fpop+2),hl
push bc
push ix
call fpfef
pop ix
pop bc
ld hl,(fpop+2)
push hl
ld hl,(fpop)
push hl
ld hl,(fpac)
jr phl
! double aritmetic
adi.4:
push bc
pop iy
pop hl
pop de
pop bc
add hl,bc
ex de,hl
pop bc
adc hl,bc
push hl
push de
push iy
pop bc
jr loop
sbi.4:
push bc
pop iy
pop bc
pop de
pop hl
sbc hl,bc
ex de,hl
ld b,h
ld c,l
pop hl
9:
sbc hl,bc
push hl
push de
push iy
pop bc
jr loop
ngi.4:
push bc
pop iy
ld hl,0
pop de
sbc hl,de
ex de,hl
ld hl,0
pop bc
jr 9b
mli.4:
ld iy,.mli4
0:
ld (retarea),bc
ld (retarea+2),ix
ld hl,1f
push hl
push iy
ret
1:
ld bc,(retarea)
ld ix,(retarea+2)
jr loop
dvu.4:
ld iy,.dvu4
jr 0b
dvi.4:
ld iy,.dvi4
jr 0b
! list of not yet implemented instructions
cuf.z:
cff.z:
cfu.z:
unimpld: ! used in dispatch table to
! catch unimplemented instructions
ld hl,EILLINS
9: push hl
jr trp.z
eilsize:
ld hl,EILLSIZE
jr 9b
e.case:
ld hl,ECASE
jr 9b
e.mon:
ld hl,EMON
jr 9b
e.array:
push af
ld a,(ignmask)
bit 0,a
jr nz,8f
ld hl,EARRAY
jr 9b
e.rck:
push af
ld a,(ignmask)
bit 1,a
jr nz,8f
ld hl,ERANGE
jr 9b
8:
pop af
ret
long2: ld a,(ix)
inc ix
or a
jr nz,unimpld
ld a,(ix)
inc ix
cp 2
jr nz,unimpld
xor a ! clear carry
ret
! monitor instruction
! a small collection of UNIX system calls implemented under CP/M
ux_indir=e.mon
ux_fork=e.mon
ux_wait=e.mon
ux_link=e.mon
ux_exec=e.mon
ux_chdir=e.mon
ux_mknod=e.mon
ux_chmod=e.mon
ux_chown=e.mon
ux_break=e.mon
ux_stat=e.mon
ux_seek=e.mon
ux_mount=e.mon
ux_umount=e.mon
ux_setuid=e.mon
ux_getuid=e.mon
ux_stime=e.mon
ux_ptrace=e.mon
ux_alarm=e.mon
ux_fstat=e.mon
ux_pause=e.mon
ux_utime=e.mon
ux_stty=e.mon
ux_gtty=e.mon
ux_access=e.mon
ux_nice=e.mon
ux_sync=e.mon
ux_kill=e.mon
ux_dup=e.mon
ux_pipe=e.mon
ux_times=e.mon
ux_prof=e.mon
ux_unused=e.mon
ux_setgid=e.mon
ux_getgid=e.mon
ux_sig=e.mon
ux_umask=e.mon
ux_chroot=e.mon
EPERM = 1
ENOENT = 2
ESRCH = 3
EINTR = 4
EIO = 5
ENXIO = 6
E2BIG = 7
ENOEXEC = 8
EBADF = 9
ECHILD = 10
EAGAIN = 11
ENOMEM = 12
EACCES = 13
EFAULT = 14
ENOTBLK = 15
EBUSY = 16
EEXIST = 17
EXDEV = 18
ENODEV = 19
ENOTDIR = 20
EISDIR = 21
EINVAL = 22
ENFILE = 23
EMFILE = 24
ENOTTY = 25
ETXTBSY = 26
EFBIG = 27
ENOSPC = 28
ESPIPE = 29
EROFS = 30
EMLINK = 31
EPIPE = 32
EDOM = 33
! Structure of filearea maintained by this implementation
! First iobuffer of 128 bytes
! Then the fcb area of 36 bytes
! The number of bytes left in the buffer, 1 byte
! The iopointer into the buffer, 2 bytes
! The openflag 0 unused, 1 reading, 2 writing, 1 byte
! The filedescriptor starting at 3, 1 byte
! The number of CTRL-Zs that have been absorbed, 1 byte
! The byte read after a sequence of CTRL-Zs, 1 byte
maxfiles=8
filesize=128+36+1+2+1+1+1+1
filefcb=0 ! pointers point to fcb
position=33
nleft=36
iopointer=37
openflag=39
fildes=40
zcount=41
zsave=42
.assert [ filefcb] == 0
0: .space maxfiles*filesize
filearea = 0b+128
sibuf:
.data2 0
.space 82
siptr: .space 2
saveargs:
.space 128
argv: .space 40 ! not more than 20 args
argc: .space 2
ttymode:.data1 9,9,8,21;.data2 06310+RAW*040 ! raw = 040
uxinit:
xor a
ld c,maxfiles
ld hl,0b
1: ld b,filesize
2: ld (hl),a
inc hl
djnz 2b
dec c
jr nz,1b
ret
uxfinish:
ld a,maxfiles-1
1: push af
call closefil
pop af
dec a
cp 0377
jr nz,1b
ret
makeargv:
ld hl,0x80
ld de,saveargs
ld bc,128
ldir
ld hl,saveargs
ld e,(hl)
inc hl
ld d,0
add hl,de
ld (hl),0
ld hl,saveargs+1
ld ix,argv
1: ld a,(hl)
or a
jr z,9f
cp ' '
jr nz,2f
4: ld (hl),0
inc hl
jr 1b
2: ld (ix),l
inc ix
ld (ix),h
inc ix
3: inc hl
ld a,(hl)
or a
jr z,9f
cp ' '
jr nz,3b
jr 4b
9: push ix
pop hl
ld de,argv
or a
sbc hl,de
srl h;rr l
ld (argc),hl
ld (ix+0),0
ld (ix+1),0
ret
mon.z:
pop de ! system call number
xor a
or d
jr nz,unimpld ! too big
ld a,e
and 0300 ! only 64 system calls
jr nz,unimpld
sla e
ld hl,systab
add hl,de
ld e,(hl)
inc hl
ld d,(hl)
ex de,hl
jp (hl)
systab:
.data2 ux_indir
.data2 ux_exit
.data2 ux_fork
.data2 ux_read
.data2 ux_write
.data2 ux_open
.data2 ux_close
.data2 ux_wait
.data2 ux_creat
.data2 ux_link
.data2 ux_unlink
.data2 ux_exec
.data2 ux_chdir
.data2 ux_time
.data2 ux_mknod
.data2 ux_chmod
.data2 ux_chown
.data2 ux_break
.data2 ux_stat
.data2 ux_seek
.data2 ux_getpid
.data2 ux_mount
.data2 ux_umount
.data2 ux_setuid
.data2 ux_getuid
.data2 ux_stime
.data2 ux_ptrace
.data2 ux_alarm
.data2 ux_fstat
.data2 ux_pause
.data2 ux_utime
.data2 ux_stty
.data2 ux_gtty
.data2 ux_access
.data2 ux_nice
.data2 ux_ftime
.data2 ux_sync
.data2 ux_kill
.data2 unimpld
.data2 unimpld
.data2 unimpld
.data2 ux_dup
.data2 ux_pipe
.data2 ux_times
.data2 ux_prof
.data2 ux_unused
.data2 ux_setgid
.data2 ux_getgid
.data2 ux_sig
.data2 unimpld
.data2 unimpld
.data2 unimpld
.data2 unimpld
.data2 unimpld
.data2 ux_ioctl
.data2 unimpld
.data2 unimpld
.data2 unimpld
.data2 unimpld
.data2 ux_exece
.data2 ux_umask
.data2 ux_chroot
.data2 unimpld
.data2 unimpld
emptyfile:
! searches for a free filestructure
! returns pointer in iy, 0 if not found
ld iy,filearea
ld l,maxfiles
1:
xor a
or (iy+openflag)
jr nz,3f
ld a,maxfiles+3
sub l
ld (iy+fildes),a
#ifdef CPM1
push bc
push iy
ld de,-128
add iy,de
push iy
pop de
ld c,setdma
call bdos
pop iy
pop bc
or a ! to clear C
#endif
ret
3:
ld de,filesize
add iy,de
dec l
jr nz,1b
scf
ret
findfile:
ld iy,filearea
ld de,filesize
0:
dec a
ret m
add iy,de
jr 0b
getchar:
push bc
push de
push hl
dec (iy+nleft)
jp p,0f
push iy
pop hl
ld de,-128
add hl,de
ld (iy+iopointer),l
ld (iy+iopointer+1),h
ex de,hl
push iy
ld c,setdma
call bdos
#ifdef CPM1
ld c,seqread
#else
ld c,randomread
#endif
pop de
call bdos
or a
jr z,1f
ld (iy+zcount),0
pop hl
pop de
pop bc
scf
ret
1:
inc (iy+position)
jr nz,2f
inc (iy+position+1)
2:
ld a,127
ld (iy+nleft),a
0:
ld h,(iy+iopointer+1)
ld l,(iy+iopointer)
ld a,(hl)
inc hl
ld (iy+iopointer),l
ld (iy+iopointer+1),h
pop hl
pop de
pop bc
ret
or a
putchar:
push hl
ld h,(iy+iopointer+1)
ld l,(iy+iopointer)
ld (hl),a
dec (iy+nleft)
jr z,0f
inc hl
ld (iy+iopointer+1),h
ld (iy+iopointer),l
pop hl
ret
0:
pop hl
flsbuf:
push hl
push de
push bc
push iy
pop hl
ld de,-128
add hl,de
ld (iy+iopointer+1),h
ld (iy+iopointer),l
ex de,hl
push iy
ld c,setdma
call bdos
pop de
#ifdef CPM1
ld c,seqwrite
#else
ld c,randomwrite
#endif
call bdos
or a
jr z,1f
pop bc
pop de
pop hl
scf
ret
1:
inc (iy+position)
jr nz,2f
inc (iy+position+1)
2:
ld a,128
ld (iy+nleft),a
ld b,a
push iy
pop hl
ld de,-128
add hl,de
ld a,26 ! ctrl z
1: ld (hl),a
inc hl
djnz 1b
pop bc
pop de
pop hl
or a
ret
parsename:
! parses file name pointed to by hl and fills in fcb
! of the file pointed to by iy.
! recognizes filenames as complicated as 'b:file.zot'
! and as simple as 'x'
push bc
push iy
pop de
xor a
push de
ld b,36 ! sizeof fcb
0: ld (de),a
inc de
djnz 0b
pop de
inc hl
ld a,(hl)
dec hl
cp ':' ! drive specified ?
jr nz,1f
ld a,(hl)
inc hl
inc hl
dec a
and 15
inc a ! now 1<= a <= 16
ld (de),a
1: inc de
ld b,8 ! filename maximum of 8 characters
1: ld a,(hl)
or a
jr nz,8f
dec hl
ld a,'.'
8:
inc hl
cp '.'
jr z,2f
and 0177 ! no parity
bit 6,a
jr z,9f
and 0337 ! UPPER case
9:
ld (de),a
inc de
djnz 1b
ld a,(hl)
inc hl
cp '.'
jr z,3f
ld a,' '
ld (de),a
inc de
ld (de),a
inc de
ld (de),a
pop bc
ret ! filenames longer than 8 are truncated
2: ld a,' ' ! fill with spaces
0: ld (de),a
inc de
djnz 0b
3: ld b,3 ! length of extension
1: ld a,(hl)
inc hl
or a
jr z,4f
cp 0100
jp m,2f
and 0137
2: ld (de),a
inc de
djnz 1b
pop bc
ret
4: ld a,' '
0: ld (de),a
inc de
djnz 0b
pop bc
ret
! various routines
ux_close:
pop hl
ld a,l
sub 3
jp m,1f
cp maxfiles
call m,closefil
1: ld hl,0
jr phl
closefil:
call findfile
ld a,(iy+openflag)
or a
jr z,3f
ld (iy+openflag),0
cp 1
jr z,2f
ld a,(iy+nleft)
cp 128
jr z,2f
call flsbuf
2:
push bc
push iy
pop de
ld c,close
call bdos
pop bc
3: ret
ux_ioctl:
pop hl
ld a,l
sub 3
jp p,1f
pop hl
ld a,h
cp 't'
jr nz,e.mon
ld a,l
cp 8
jr z,tiocgetp
cp 9
jr z,tiocsetp
jr e.mon
1: pop hl
pop hl
ld hl,-1
jr phl
tiocgetp:
pop de
ld hl,ttymode
2: push bc
ld bc,6
ldir
ld h,b
ld l,c
pop bc
jr phl
tiocsetp:
pop hl
ld de,ttymode
jr 2b
ux_time:
call time4
jr loop
ux_ftime:
pop hl
ld (retarea+6),hl
call time4
ld hl,(retarea+6)
pop de
ld (hl),e
inc hl
ld (hl),d
inc hl
pop de
ld (hl),e
inc hl
ld (hl),d
inc hl
xor a
ld (hl),a
inc hl
ld (hl),a
inc hl
ld (hl),a
inc hl
ld (hl),a
inc hl
ld (hl),a
inc hl
ld (hl),a
jr loop
time4:
pop hl
ld (retarea),bc
ld (retarea+2),ix
ld (retarea+4),hl
ld hl,(timebuf+2)
push hl
ld hl,(timebuf)
push hl
ld hl,0
push hl
ld hl,50
push hl
call .dvu4
ld bc,(retarea)
ld ix,(retarea+2)
ld hl,(retarea+4)
jp (hl)
ux_exit:
call uxfinish
ld c,reset
call bdos
! no return
ux_creat:
call emptyfile
jr c,openfailed
pop hl
call parsename
pop hl ! file mode, not used under CP/M
push bc
push iy
push iy
pop de
ld c,delete
call bdos
pop de
ld c,makefile
call bdos
pop bc
ld l,1
jr afteropen
ux_open:
call emptyfile
jr nc,1f
openfailed:
pop hl
pop hl ! remove params
ld hl,EMFILE
push hl
jr phl
1:
pop hl ! filename
call parsename
push bc
ld c,open
push iy
pop de
call bdos
pop bc
pop hl
afteropen:
inc a
jr nz,1f
ld hl,ENOENT
push hl
jr phl
1:
inc l
ld (iy+openflag),l
xor a
ld (iy+nleft),a
ld (iy+zcount),a
ld (iy+zsave),26
bit 1,l
jr z,2f
ld (iy+nleft),128
2:
ld (iy+position),a
ld (iy+position+1),a
push iy
pop hl
push bc
ld b,128
3: dec hl
ld (hl),26
djnz 3b
pop bc
ld (iy+iopointer+1),h
ld (iy+iopointer),l
ld h,a
ld l,(iy+fildes)
push hl
ld l,a
jr phl
ux_read:
pop hl
ld a,l
sub 3
jp p,readfile
ld a,(ttymode+4)
bit 5,a
jr z,1f ! not raw
push bc
#ifdef CPM1
!raw echo interface
ld c,consolein
call bdos
#else
!no echo interface
4:
ld c,diconio
ld e,0xff
call bdos
or a
jr z,4b
!end of no echo interface
#endif
pop bc
pop hl
ld (hl),a
pop hl
ld hl,1
push hl
ld hl,0
jr phl
1:
ld hl,sibuf+1 ! read from console assumed
dec (hl)
jp p,2f
dec hl ! go read console line
ld (hl),80 ! max line length
push bc
push hl
ld c,readconsole
ex de,hl
call bdos
ld c,writeconsole
ld e,'\n'
call bdos
pop hl
pop bc
inc hl
inc (hl)
ld (siptr),hl ! ready for transfer
push hl
ld e,(hl)
ld d,0
add hl,de
ld (hl),'\r'
inc hl
ld (hl),'\n'
pop hl
2:
push bc
pop iy
ld b,(hl)
inc b ! bytes remaining
pop hl ! copy to
pop de ! bytes wanted (probably 512)
push iy
ld iy,(siptr) ! copy from
xor a ! find out minimum of ramaining and wanted
or d
jr nz,3f ! more than 255 wanted (forget that)
ld a,b
cp e
jp m,3f ! not enough remaining
ld b,e
3:
ld c,b ! keep copy
0:
inc iy
ld a,(iy)
ld (hl),a
inc hl
djnz 0b
ld a,(sibuf+1)
sub c
inc a
ld (sibuf+1),a
ld (siptr),iy
pop hl
push bc
ld c,b
push bc ! load 0
ld b,h
ld c,l
jr loop
readfile:
call findfile
pop de
pop hl ! count
push bc
ld bc,0
0:
xor a
or l
jr z,1f
dec l
3:
! warning: this may not work if zcount overflows
ld a,(iy+zcount)
or a
jr nz,5f
ld a,(iy+zsave)
cp 26
jr z,4f
ld (iy+zsave),26
jr 8f
4:
call getchar
jr c,2f
ld (de),a
sub 26 ! CTRL-Z
jr z,7f
ld a,(iy+zcount)
or a
jr z,6f
ld a,(de)
ld (iy+zsave),a
5:
ld a,26
dec (iy+zcount)
8:
ld (de),a
6:
inc de
inc bc
jr 0b
1:
dec l
dec h
jp p,3b
2:
pop hl
push bc
ld b,h
ld c,l
ld hl,0
jr phl
7:
inc (iy+zcount)
jr 4b
ux_write:
pop hl
ld a,l
sub 3
jp p,writefile
pop hl ! buffer address
pop de ! count
push de
ld iy,0
push iy
push bc
ld b,e ! count now in 'db'
0:
ld a,b
or a
jr nz,1f
ld a,d
or a
jr nz,2f
pop bc
jr loop
2:
dec d
1:
dec b
ld e,(hl)
inc hl
push bc
push de
push hl
ld c,writeconsole
call bdos
pop hl
pop de
pop bc
jr 0b
writefile:
call findfile
pop de
pop hl ! count
push bc
ld bc,0
0:
xor a
or l
jr z,1f
dec l
3:
ld a,(de)
inc de
call putchar
jr c,4f
inc bc
jr 0b
1:
dec l
dec h
jp p,3b
ld iy,0
2:
pop hl
push bc
ld b,h
ld c,l
push iy
jr loop
4:
ld iy,ENOSPC
jr 2b
ux_unlink:
pop hl
ld iy,fcb
call parsename
push bc
ld c,delete
ld de,fcb
call bdos
pop bc
inc a
jr nz,1f
ld hl,ENOENT
jr phl
1:
ld hl,0
jr phl
ux_getpid:
ld hl,12345 ! nice number
jr phl
ux_exece:
ld iy,fcb
pop hl
call parsename
pop hl
ld b,h;ld c,l
pop iy
ld ix,0x82
ld (ix-1),' '
4: ld h,b;ld l,c
3: ld e,(hl)
inc hl
ld d,(hl)
inc hl
ld b,h;ld c,l
ex de,hl
ld a,h
or l
jr z,1f
2:
ld a,(hl)
inc hl
ld (ix),a
inc ix
or a
jr nz,2b
ld (ix-1),' '
jr 4b
1:
ld (ix),'X'
ld (ix+1),'\r'
ld (ix+2),'\n'
ld (ix+3),'$'
ld de,0x81
push ix
ld c,printstring
call bdos
pop hl
ld de,-129
add hl,de
ld a,l
ld (0x80),a
jr warmstart
dispat1: ! base for escaped opcodes
.data2 aar.l, aar.z, adf.l, adf.z, adi.l, adi.z, ads.l, ads.z
.data2 adu.l, adu.z, and.l, and.z, asp.l, ass.l, ass.z, bge.l
.data2 bgt.l, ble.l, blm.l, bls.l, bls.z, blt.l, bne.l, cai.z
.data2 cal.l, cfi.z, cfu.z, ciu.z, cmf.l, cmf.z, cmi.l, cmi.z
.data2 cms.l, cms.z, cmu.l, cmu.z, com.l, com.z, csa.l, csa.z
.data2 csb.l, csb.z, cuf.z, cui.z, cuu.z, dee.l, del.p, del.n
.data2 dup.l, dus.l, dus.z, dvf.l, dvf.z, dvi.l, dvi.z, dvu.l
.data2 dvu.z, fef.l, fef.z, fif.l, fif.z, inl.p, inl.n, inn.l
.data2 inn.z, ior.l, ior.z, lar.l, lar.z, ldc.l, ldf.l, ldl.p
.data2 ldl.n, lfr.l, lil.p, lil.n, lim.z, los.l, los.z, lor.s0
.data2 lpi.l, lxa.l, lxl.l, mlf.l, mlf.z, mli.l, mli.z, mlu.l
.data2 mlu.z, mon.z, ngf.l, ngf.z, ngi.l, ngi.z, nop.z, rck.l
.data2 rck.z, ret.l, rmi.l, rmi.z, rmu.l, rmu.z, rol.l, rol.z
.data2 ror.l, ror.z, rtt.z, sar.l, sar.z, sbf.l, sbf.z, sbi.l
.data2 sbi.z, sbs.l, sbs.z, sbu.l, sbu.z, sde.l, sdf.l, sdl.p
.data2 sdl.n, set.l, set.z, sig.z, sil.p, sil.n, sim.z, sli.l
.data2 sli.z, slu.l, slu.z, sri.l, sri.z, sru.l, sru.z, sti.l
.data2 sts.l, sts.z, str.s0, tge.z, tle.z, trp.z, xor.l, xor.z
.data2 zer.l, zer.z, zge.l, zgt.l, zle.l, zlt.l, zne.l, zrf.l
.data2 zrf.z, zrl.p, dch.z, exg.s0, exg.l, exg.z, lpb.z
dispat2: ! base for 4 byte offsets
.data2 ldc.f
ignmask: .data2 0 ! ignore mask (variable)
retarea: .data2 0 ! base of buffer for result values (max 8 bytes)
.data2 0
.data2 0
.data2 0
trapproc:
.data2 0
nextp: .data1 0
header:
ntext: .data2 0
ndata: .data2 0
nproc: .data2 0
entry: .data2 0
nline: .data2 0
hp: .data2 0
pb: .data2 0
pd: .data2 0