mirror of
https://github.com/tebl/RC6502-Apple-1-Replica.git
synced 2024-11-24 11:30:55 +00:00
255 lines
11 KiB
NASM
255 lines
11 KiB
NASM
.CR 6502
|
|
.TF wozmon.hex,HEX,8
|
|
.LF wozmon.list
|
|
;-------------------------------------------------------------------------
|
|
;
|
|
; The WOZ Monitor for the Apple 1
|
|
; Written by Steve Wozniak 1976
|
|
;
|
|
;-------------------------------------------------------------------------
|
|
|
|
|
|
;-------------------------------------------------------------------------
|
|
; Memory declaration
|
|
;-------------------------------------------------------------------------
|
|
|
|
XAML .EQ $24 Last "opened" location Low
|
|
XAMH .EQ $25 Last "opened" location High
|
|
STL .EQ $26 Store address Low
|
|
STH .EQ $27 Store address High
|
|
L .EQ $28 Hex value parsing Low
|
|
H .EQ $29 Hex value parsing High
|
|
YSAV .EQ $2A Used to see if hex value is given
|
|
MODE .EQ $2B $00=XAM, $7F=STOR, $AE=BLOCK XAM
|
|
|
|
IN .EQ $0200,$027F Input buffer
|
|
|
|
KBD .EQ $D010 PIA.A keyboard input
|
|
KBDCR .EQ $D011 PIA.A keyboard control register
|
|
DSP .EQ $D012 PIA.B display output register
|
|
DSPCR .EQ $D013 PIA.B display control register
|
|
|
|
; KBD b7..b0 are inputs, b6..b0 is ASCII input, b7 is constant high
|
|
; Programmed to respond to low to high KBD strobe
|
|
; DSP b6..b0 are outputs, b7 is input
|
|
; CB2 goes low when data is written, returns high when CB1 goes high
|
|
; Interrupts are enabled, though not used. KBD can be jumpered to IRQ,
|
|
; whereas DSP can be jumpered to NMI.
|
|
|
|
;-------------------------------------------------------------------------
|
|
; Constants
|
|
;-------------------------------------------------------------------------
|
|
|
|
BS .EQ $DF Backspace key, arrow left key
|
|
CR .EQ $8D Carriage Return
|
|
ESC .EQ $9B ESC key
|
|
PROMPT .EQ "\" Prompt character
|
|
|
|
;-------------------------------------------------------------------------
|
|
; Let's get started
|
|
;
|
|
; Remark the RESET routine is only to be entered by asserting the RESET
|
|
; line of the system. This ensures that the data direction registers
|
|
; are selected.
|
|
;-------------------------------------------------------------------------
|
|
.OR $FF00
|
|
RESET CLD Clear decimal arithmetic mode
|
|
CLI
|
|
LDY #%0111.1111 Mask for DSP data direction reg
|
|
STY DSP (DDR mode is assumed after reset)
|
|
LDA #%1010.0111 KBD and DSP control register mask
|
|
STA KBDCR Enable interrupts, set CA1, CB1 for
|
|
STA DSPCR positive edge sense/output mode.
|
|
|
|
; Program falls through to the GETLINE routine to save some program bytes
|
|
; Please note that Y still holds $7F, which will cause an automatic Escape
|
|
|
|
;-------------------------------------------------------------------------
|
|
; The GETLINE process
|
|
;-------------------------------------------------------------------------
|
|
|
|
NOTCR CMP #BS Backspace key?
|
|
BEQ BACKSPACE Yes
|
|
CMP #ESC ESC?
|
|
BEQ ESCAPE Yes
|
|
INY Advance text index
|
|
BPL NEXTCHAR Auto ESC if line longer than 127
|
|
|
|
ESCAPE LDA #PROMPT Print prompt character
|
|
JSR ECHO Output it.
|
|
|
|
GETLINE LDA #CR Send CR
|
|
JSR ECHO
|
|
|
|
LDY #0+1 Start a new input line
|
|
BACKSPACE DEY Backup text index
|
|
BMI GETLINE Oops, line's empty, reinitialize
|
|
|
|
NEXTCHAR LDA KBDCR Wait for key press
|
|
BPL NEXTCHAR No key yet!
|
|
LDA KBD Load character. B7 should be '1'
|
|
STA IN,Y Add to text buffer
|
|
JSR ECHO Display character
|
|
CMP #CR
|
|
BNE NOTCR It's not CR!
|
|
|
|
; Line received, now let's parse it
|
|
|
|
LDY #-1 Reset text index
|
|
LDA #0 Default mode is XAM
|
|
TAX X=0
|
|
|
|
SETSTOR ASL Leaves $7B if setting STOR mode
|
|
|
|
SETMODE STA MODE Set mode flags
|
|
|
|
BLSKIP INY Advance text index
|
|
|
|
NEXTITEM LDA IN,Y Get character
|
|
CMP #CR
|
|
BEQ GETLINE We're done if it's CR!
|
|
CMP #"."
|
|
BCC BLSKIP Ignore everything below "."!
|
|
BEQ SETMODE Set BLOCK XAM mode ("." = $AE)
|
|
CMP #":"
|
|
BEQ SETSTOR Set STOR mode! $BA will become $7B
|
|
CMP #"R"
|
|
BEQ RUN Run the program! Forget the rest
|
|
STX L Clear input value (X=0)
|
|
STX H
|
|
STY YSAV Save Y for comparison
|
|
|
|
; Here we're trying to parse a new hex value
|
|
|
|
NEXTHEX LDA IN,Y Get character for hex test
|
|
EOR #$B0 Map digits to 0-9
|
|
CMP #9+1 Is it a decimal digit?
|
|
BCC DIG Yes!
|
|
ADC #$88 Map letter "A"-"F" to $FA-FF
|
|
CMP #$FA Hex letter?
|
|
BCC NOTHEX No! Character not hex
|
|
|
|
DIG ASL
|
|
ASL Hex digit to MSD of A
|
|
ASL
|
|
ASL
|
|
|
|
LDX #4 Shift count
|
|
HEXSHIFT ASL Hex digit left, MSB to carry
|
|
ROL L Rotate into LSD
|
|
ROL H Rotate into MSD's
|
|
DEX Done 4 shifts?
|
|
BNE HEXSHIFT No, loop
|
|
INY Advance text index
|
|
BNE NEXTHEX Always taken
|
|
|
|
NOTHEX CPY YSAV Was at least 1 hex digit given?
|
|
BEQ ESCAPE No! Ignore all, start from scratch
|
|
|
|
BIT MODE Test MODE byte
|
|
BVC NOTSTOR B6=0 is STOR, 1 is XAM or BLOCK XAM
|
|
|
|
; STOR mode, save LSD of new hex byte
|
|
|
|
LDA L LSD's of hex data
|
|
STA (STL,X) Store current 'store index'(X=0)
|
|
INC STL Increment store index.
|
|
BNE NEXTITEM No carry!
|
|
INC STH Add carry to 'store index' high
|
|
TONEXTITEM JMP NEXTITEM Get next command item.
|
|
|
|
;-------------------------------------------------------------------------
|
|
; RUN user's program from last opened location
|
|
;-------------------------------------------------------------------------
|
|
|
|
RUN JMP (XAML) Run user's program
|
|
|
|
;-------------------------------------------------------------------------
|
|
; We're not in Store mode
|
|
;-------------------------------------------------------------------------
|
|
|
|
NOTSTOR BMI XAMNEXT B7 = 0 for XAM, 1 for BLOCK XAM
|
|
|
|
; We're in XAM mode now
|
|
|
|
LDX #2 Copy 2 bytes
|
|
SETADR LDA L-1,X Copy hex data to
|
|
STA STL-1,X 'store index'
|
|
STA XAML-1,X and to 'XAM index'
|
|
DEX Next of 2 bytes
|
|
BNE SETADR Loop unless X = 0
|
|
|
|
; Print address and data from this address, fall through next BNE.
|
|
|
|
NXTPRNT BNE PRDATA NE means no address to print
|
|
LDA #CR Print CR first
|
|
JSR ECHO
|
|
LDA XAMH Output high-order byte of address
|
|
JSR PRBYTE
|
|
LDA XAML Output low-order byte of address
|
|
JSR PRBYTE
|
|
LDA #":" Print colon
|
|
JSR ECHO
|
|
|
|
PRDATA LDA #" " Print space
|
|
JSR ECHO
|
|
LDA (XAML,X) Get data from address (X=0)
|
|
JSR PRBYTE Output it in hex format
|
|
XAMNEXT STX MODE 0 -> MODE (XAM mode).
|
|
LDA XAML See if there's more to print
|
|
CMP L
|
|
LDA XAMH
|
|
SBC H
|
|
BCS TONEXTITEM Not less! No more data to output
|
|
|
|
INC XAML Increment 'examine index'
|
|
BNE MOD8CHK No carry!
|
|
INC XAMH
|
|
|
|
MOD8CHK LDA XAML If address MOD 8 = 0 start new line
|
|
AND #%0000.0111
|
|
BPL NXTPRNT Always taken.
|
|
|
|
;-------------------------------------------------------------------------
|
|
; Subroutine to print a byte in A in hex form (destructive)
|
|
;-------------------------------------------------------------------------
|
|
|
|
PRBYTE PHA Save A for LSD
|
|
LSR
|
|
LSR
|
|
LSR MSD to LSD position
|
|
LSR
|
|
JSR PRHEX Output hex digit
|
|
PLA Restore A
|
|
|
|
; Fall through to print hex routine
|
|
|
|
;-------------------------------------------------------------------------
|
|
; Subroutine to print a hexadecimal digit
|
|
;-------------------------------------------------------------------------
|
|
|
|
PRHEX AND #%0000.1111 Mask LSD for hex print
|
|
ORA #"0" Add "0"
|
|
CMP #"9"+1 Is it a decimal digit?
|
|
BCC ECHO Yes! output it
|
|
ADC #6 Add offset for letter A-F
|
|
|
|
; Fall through to print routine
|
|
|
|
;-------------------------------------------------------------------------
|
|
; Subroutine to print a character to the terminal
|
|
;-------------------------------------------------------------------------
|
|
|
|
ECHO BIT DSP DA bit (B7) cleared yet?
|
|
BMI ECHO No! Wait for display ready
|
|
STA DSP Output character. Sets DA
|
|
RTS
|
|
|
|
;-------------------------------------------------------------------------
|
|
; Vector area
|
|
;-------------------------------------------------------------------------
|
|
|
|
.DA $0000 Unused, what a pity
|
|
NMI_VEC .DA $0F00 NMI vector
|
|
RESET_VEC .DA RESET RESET vector
|
|
IRQ_VEC .DA $0000 IRQ vector |