basic-system-cmd/online.asm
2015-09-23 22:12:32 -05:00

305 lines
5.5 KiB
NASM

;
; online.asm
;
; Copyright (c) 2015 Rob Greene. All rights reserved.
;
.PC02 ; Enable 65C02 instructions
; ASCII string with high-bit set, terminated by a zero
.macro asciizh string
.repeat .strlen(string),i
.byte .strat(string,i) | $80
.endrep
.byte 0
.endmacro
; ASCII string with high-bit set
.macro asciih string
.repeat .strlen(string),i
.byte .strat(string,i) | $80
.endrep
.endmacro
; Dextral (right-most) Character Inverted
.macro dci string
.repeat .strlen(string),i
.if .strlen(string) = i
.byte .strat(string,i) | $80
.else
.byte .strat(string,i) & $7f
.endif
.endrep
.endmacro
; BASIC.SYSTEM locations:
inbuf = $0200
extrncmd = $be06 ; External command JMP vector
xtrnaddr = $be50 ; Execution address of external command
xlen = $be52 ; Length of command string-1
xcnum = $be53 ; BASIC cmd number (external command = 0)
pbits = $be54 ; Parameter bits allowed (2 bytes)
fbits = $be56 ; Parameter bits found
vslot = $be61
vdriv = $be62
gosystem = $be70
sonline = $bec6 ; BASIC.SYSTEM ONLINE parameter table
sunitnum = $bec7
sbufadr = $bec8
getbufr = $bef5
; Parameter bits flags (note setup for two bytes and for lo/hi usage)
pfix = $0080 ; Prefix needs fetching, pathname optional
slot = $0040 ; No parameters to be processed
rrun = $0020 ; Command only valid during program
fnopt = $0010 ; Filename is optional
crflg = $0008 ; CREATE allowed
t = $0004 ; File type
fn2 = $0002 ; Filename #2 for RENAME
fn1 = $0001 ; Filename expected
ad = $8000 ; Address
b = $4000 ; Byte
e = $2000 ; End address
l = $1000 ; Length
line = $0800 ; '@' line number
sd = $0400 ; Slot and drive numbers
f = $0200 ; Field
r = $0100 ; Record
v = $0000 ; Volume number ignored
; MONITOR locations:
crout = $fd8e
prbyte = $fdda
cout = $fded
; Application stuff:
buffer = inbuf
.org $2000
install:
; Requires 65C02 or later:
sed
lda #$99
clc
adc #$01
cld
bmi @6502
; Move code to destination address:
; TODO: Get address from BASIC.SYSTEM and move there, relocate code
ldy #0
: lda _CodeStartAddress,y
sta _CodeBeginAddress,y
iny
cpy #(_CodeEndAddress-_CodeBeginAddress)
bne :-
; Setup BASIC.SYSTEM hooks:
; 1. Save EXTRNCMD
lda extrncmd+2
sta nextcmd+1
lda extrncmd+1
sta nextcmd
; 2. Place our hook into EXTRNCMD
lda #>entry
sta extrncmd+2
lda #<entry
sta extrncmd+1
; Notify user:
ldy #0
: lda msgInstalled,y
beq :+
jsr cout
iny
bne :-
: rts
@6502:
ldy #0
: lda err6502,y
beq :+
jsr cout
iny
bne :-
: rts
err6502:
asciizh "ERR: MUST HAVE ENHANCED //E, //C, OR IIGS"
msgInstalled:
asciizh "ONLINE COMMAND INSTALLED"
_CodeStartAddress:
.org $6000
_CodeBeginAddress:
entry:
ldx #0
: lda inbuf,x
cmp #$e0 ; Force input to UPPERCASE for comparison
bcc :+
and #$df
: cmp cmdtable,x
bne notOurCommand
inx
cpx #cmdlen
bne :--
; Setup for BASIC.SYSTEM to parse
opts = fnopt|sd ; Filename is optional (due to glitch) and slot and drive
lda #cmdlen-1
sta xlen
lda #<online
sta xtrnaddr
lda #>online
sta xtrnaddr+1
stz xcnum
lda #<opts
sta pbits
lda #>opts
sta pbits+1
clc
rts
notOurCommand:
sec
jmp (nextcmd)
;
; Perform ONLINE command
; Note we use the input buffer address
;
; Buffer format is:
; +000 DSSSLLLL D=Drive, SSS=Slot, LLLL=Length
; +001 CHAR1 Name... or error code
; ...
; +015 CHAR15
;
; Output:
; S7 D1 /HDD
; S6 D1 ERR=$28
; S5 D1 ERR=$57 (S7 D1)
;
online:
lda fbits+1
and #>sd
beq @1 ; Bit was NOT set; Acc = 0
lda vdriv ; 1 or 2, use 2nd bit to toggle drive (then drive 1 has bit off, drive 2 has bit on)
and #%00000010
asl
asl
ora vslot
asl
asl
asl
asl
@1: sta sunitnum
stz sbufadr
lda #>buffer
sta sbufadr+1
; Note: if we have a specific unit, this is not zero terminated -- fake it!
stz buffer+16
lda #$C5 ; ONLINE system command
jsr gosystem
@continue:
ldx #0
@loop:
ldy buffer,x
beq @exit
jsr printsd
jsr printspc
tya
and #$0f
beq @deverr
tay
lda #'/'|$80
: jsr cout
inx
lda buffer,x
ora #$80
dey
bpl :-
@adjust:
jsr crout
txa
and #$0f ; Check if we advanced past this buffer
beq @loop
txa
and #$f0
clc
adc #$10
tax
bne @loop
@exit:
jmp crout
; A device error message
@deverr:
ldy #0
: lda msgERR,y
beq :+
jsr cout
iny
bne :-
: inx
lda buffer,x
tay ; short-term save
jsr prbyte
tya
cmp #$57 ; duplicate volume error
bne @adjust
jsr printspc
lda #'('|$80
jsr cout
inx
ldy buffer,x
jsr printsd
lda #')'|$80
jsr cout
bra @adjust
printsd:
lda #'S'|$80
jsr cout
tya
and #$70
lsr
lsr
lsr
lsr
ora #'0'|$80
jsr cout
lda #','|$80
jsr cout
lda #'D'|$80
jsr cout
tya
and #$80
asl ; Drive 2 will set carry...
adc #'1'|$80 ; ... making the '1' a '2'
jmp cout
printspc:
lda #' '|$80
jmp cout
msgERR:
asciizh "ERR=$"
cmdtable:
asciih "ONLINE"
cmdlen = *-cmdtable
nextcmd: .word 0
_CodeEndAddress: