mirror of
https://github.com/a2geek/basic-system-cmd.git
synced 2025-01-16 05:30:23 +00:00
305 lines
5.5 KiB
NASM
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:
|
|
|