emailler/client/basic/kipperbas.s

1739 lines
32 KiB
ArmAsm
Raw Normal View History

.include "../inc/common.i"
.ifndef KPR_API_VERSION_NUMBER
.define EQU =
.include "../inc/kipper_constants.i"
.endif
VARTAB = $2D ;BASIC variable table storage
ARYTAB = $2F ;BASIC array table storage
FREETOP = $33 ;bottom of string text storage area
MEMSIZ = $37 ;highest address used by BASIC
VARNAM = $45 ;current BASIC variable name
VARPNT = $47 ; pointer to current BASIC variable value
SETNAM = $FFBD
SETLFS = $FFBA
OPEN = $FFC0
CHKIN = $FFC6
READST = $FFB7 ; read status byte
CHRIN = $FFCF ; get a byte from file
CLOSE = $FFC3
MEMTOP = $FE25
TXTPTR = $7A ;BASIC text pointer
IERROR = $0300
ICRUNCH = $0304 ;Crunch ASCII into token
IQPLOP = $0306 ;List
IGONE = $0308 ;Execute next BASIC token
IEVAL = $30A ; evaluate expression
CHRGET = $73
CHRGOT = $79
CHROUT = $FFD2
GETBYT = $B79E ;BASIC routine
GETPAR = $B7EB ;Get a 16,8 pair of numbers
CHKCOM = $AEFD
NEW = $A642
CLR = $A65E
NEWSTT = $A7AE
GETVAR = $B0E7 ;find or create a variable
FRMEVL = $AD9E ;evaluate expression
FRESTR = $B6A3 ;free temporary string
FRMNUM = $AD8A ;get a number
GETADR = $B7F7 ;convert number to 16 bit integer
INLIN = $A560 ; read a line from keyboard
VALTYP=$0D ;00=number, $FF=string
LINNUM = $14 ;Number returned by GETPAR
crunched_line = $0200 ;Input buffer
.import copymem
.importzp copy_src
.importzp copy_dest
.import dhcp_init
.import ip65_init
.import cfg_get_configuration_ptr
.import tcp_listen
.import tcp_callback
.import tcp_connect_ip
.import tcp_send
.import tcp_connect
.import tcp_close
.import tcp_send_data_len
.import tcp_inbound_data_ptr
.import tcp_inbound_data_length
.import dns_set_hostname
.import dns_resolve
.import dns_ip
.import ip65_process
.import ip65_error
.import cfg_ip
.import cfg_dns
.import cfg_gateway
.import cfg_netmask
.import cfg_tftp_server
.import icmp_ping
.import icmp_echo_ip
.import print_a
.import print_cr
.import dhcp_server
.import cfg_mac
.import cfg_mac_default
.import eth_driver_name
.importzp tftp_filename
.import tftp_ip
.import tftp_download
.import tftp_set_callback_vector
.import tftp_data_block_length
.import tftp_upload
.import get_key_if_available
.import tcp_send_keep_alive
.import timer_read
.import native_to_ascii
.import ascii_to_native
.import udp_send_dest
.import udp_send_dest_port
.import udp_send_src_port
.import udp_send_len
.import udp_send
.import udp_add_listener
.import udp_remove_listener
.import udp_callback
.zeropage
temp: .res 2
temp2: .res 2
pptr=temp
.segment "STARTUP" ;this is what gets put at the start of the file on the C64
.word basicstub ; load address
basicstub:
.word @nextline
.word 10 ;line number
.byte $9e ;SYS
.byte <(((relocate / 1000) .mod 10) + $30)
.byte <(((relocate / 100 ) .mod 10) + $30)
.byte <(((relocate / 10 ) .mod 10) + $30)
.byte <(((relocate ) .mod 10) + $30)
.byte ":"
.byte "D"
.byte $b2 ;=
.byte $c2 ;PEEK
.byte "(186):"
.byte $93 ;LOAD
.byte $22,"AUTOEXEC.BAS",$22,",D"
.byte 0
@nextline:
.word 0
relocate:
lda MEMSIZ+1
cmp #$A0 ;standard end of memory
beq ok_to_install
ldy #0
@loop:
lda not_installing,y
beq @done
jsr $ffd2
iny
bne @loop
@done:
rts
not_installing:
.byte "INSUFFICIENT FREE MEMORY",13,0
ok_to_install:
ldax #end_of_loader
stax copy_src
ldax #main_start
stax copy_dest
stax MEMSIZ
FS=$8000-main_start
ldax #FS
;copy memory
sta end
ldy #0
cpx #0
beq @tail
: lda (copy_src),y
sta (copy_dest),y
iny
bne :-
inc copy_src+1 ;next page
inc copy_dest+1 ;next page
dex
bne :-
@tail:
lda end
beq @done
: lda (copy_src),y
sta (copy_dest),y
iny
cpy end
bne :-
@done:
ldax #welcome_banner
jsr print
ldx #7 ;Copy CURRENT vectors
@copy_old_vectors_loop:
lda ICRUNCH,x
sta oldcrunch,x
dex
bpl @copy_old_vectors_loop
ldx #7 ;Copy CURRENT vectors
install_new_vectors_loop:
lda vectors,x
sta ICRUNCH,x
dex
bpl install_new_vectors_loop
;BASIC keywords installed, now bring up the ip65 stack
jsr ip65_init
bcc @init_ok
ldax #@no_nic
jsr print
@reboot:
jsr $e453 ;reset vectors
jsr $e3bf ;init BASIC
jsr $a644 ;NEW
jmp $e39d
@no_nic:
.byte "NO RR-NET FOUND - UNINSTALLING",0
@init_ok:
lda #0
sta ip65_error
sta connection_state
jsr set_error
@exit:
; jsr $A644 ;do a "NEW"
; jmp $A474 ;"READY" prompt
rts
welcome_banner:
.byte " *** KIPPER BASIC 1.22"
.include "timestamp.i"
.byte " ***"
.byte 13,0
end: .res 1
end_of_loader:
.segment "MAINSTART"
main_start:
safe_getvar: ;if GETVAR doesn't find the desired variable name in the VARTABLE, a routine at $B11D will create it
;however that routine checks if the low byte of the return address of the caller is $2A. if it is,
;it assumes the caller is the routine at $AF28 which just wants to get the variable value, and
;returns a pointer to a dummy 'zero' pointer.
;so if user code that is calling GETVAR happens to be compiled to an address $xx28, it will
;trigger this check, and not create a new variable, which (from painful experience) will create
;a really nasty condition to debug!
;so vector to GETVAR via here, so the return address seen by $B11D is here, and never $xx28
jsr GETVAR
rts
.code
; CRUNCH -- If this is one of our keywords, then tokenize it
;
crunch:
jsr jmp_crunch ;First crunch line normally
ldy #05 ;Offset for KERNAL
;Y will contain line length+5
@loop:
sty temp
jsr isword ;Are we at a keyword?
bcs @gotcha
@next:
jsr nextchar
bne @loop ;Null byte marks end
sta crunched_line-3,Y ;00 line number
lda #$FF ;'tis what A should be
rts ;Buh-bye
; Insert token and crunch line
@gotcha:
ldx temp ;If so, A contains opcode
sta crunched_line-5,X
@move:
inx
lda crunched_line-5,Y
sta crunched_line-5,X ;Move text backwards
beq @next
iny
bpl @move
; ISWORD -- Checks to see if word is
; in table. If a word is found, then
; C is set, Y is one past the last char
; and A contains opcode. Otherwise,
; carry is clear.
;
; On entry, TEMP must contain current
; character position.
;
isword:
ldx #00
@loop:
ldy temp
@loop2:
lda keywords,x
beq @notmine
cmp #$E0
bcs @done ;Tokens are >=$E0
cmp crunched_line-5,Y
bne @next
iny ;Success! Go to next char
inx
bne @loop2
@next:
inx
lda keywords,x ;Find next keyword
cmp #$E0
bcc @next
inx
bne @loop ;And check again
@notmine:
clc
@done:
rts
; NEXTCHAR finds the next char
; in the buffer, skipping
; spaces and quotes. On
; entry, TEMP contains the
; position of the last spot
; read. On exit, Y contains
; the index to the next char,
; A contains that char, and Z is set if at end of line.
nextchar:
ldy temp
@loop:
iny
lda crunched_line-5,Y
beq @done
cmp #$8F ;REM
bne @cont
lda #00
@skip:
sta temp2 ;Find matching character
@loop2:
iny
lda crunched_line-5,Y
beq @done
cmp temp2
bne @loop2 ;Skip to end of line
beq @loop
@cont:
cmp #$20 ;space
beq @loop
cmp #$22 ;quote
beq @skip
@done:
rts
;
; LIST -- patches the LIST routine
; to list our new tokens correctly.
;
list:
cmp #$E0
bcc @notmine ;Not my token
cmp #HITOKEN
bcs @notmine
bit $0F ;Check for quote mode
bmi @notmine
sec
sbc #$DF ;Find the corresponding text
tax
sty $49
ldy #00
@loop:
dex
beq @done
@loop2:
iny
lda keywords,y
cmp #$E0
bcc @loop2
iny
bne @loop
@done:
lda keywords,y
cmp #$91 ;is it "ON"?
bne @not_on
lda #'O'
jsr CHROUT
lda #'N'
bne @skip
@not_on:
cmp #$9B ;is it "LIST"?
bne @not_list
lda #'L'
jsr CHROUT
lda #'I'
jsr CHROUT
lda #'S'
jsr CHROUT
lda #'T'
bne @skip
@not_list:
lda keywords,y
bmi @out ;is it >=$80?
@skip:
jsr CHROUT
iny
bne @done
@out:
cmp #$E0 ;It might be BASIC token
bcs @cont
ldy $49
@notmine:
and #$FF
jmp (oldlist)
@cont:
ldy $49
jmp $A700 ;Normal exit
;
; EXECUTE -- if this is one of our new
; tokens, then execute it.
execute:
jsr CHRGET
execute_a:
php
cmp #':' ;is it a colon?
beq execute;if so, skip over and go to next token
cmp #$8B ;is it 'IF'?
bne @not_if
lda #$E0 ;our dummy IF token
@not_if:
cmp #$E0
bcc @notmine
cmp #HITOKEN
bcs @notmine
plp
jsr @disp
jmp NEWSTT
@disp:
eor #$E0
asl ;multiply by 2
tax
lda token_routines+1,x
pha
lda token_routines,x
pha
jmp CHRGET ;exit to routine (via RTS)
@notmine:
plp
cmp #0 ;reset flags
jmp $A7E7
;the standard BASIC IF routine calls the BASIC EXECUTE routine directly,
;without going through the vector. That means an extended keyword following THEN
;will lead to a syntax error. So we have to reimpliment IF here
;this is taken from TransBASIC - The Transactor, vol 5, Issue 04 (March 1985) page 34
if_keyword:
jsr FRMEVL ;evaluate expression
jsr CHRGOT
cmp #$89 ;is next token GOTO?
beq @ok
lda #$A7 ;is next token THEN
jsr $AEFF ;will generate SYNTAX ERROR if not
@ok:
jsr CHRGOT
ldx $61 ;result of expression : 0 means false
bne @expression_was_true
jmp $A93B ;go to REM implementation - skips rest of line
@expression_was_true:
bcs @not_numeric;CHRGOT clears carry flag if current char is a number
jmp $A8A0 ;do a GOTO
@not_numeric:
pla
pla ;pop the return address off the stack
jsr CHRGOT
jmp execute_a ;execute current token
find_var:
sta VARNAM
stx VARNAM+1
jsr safe_getvar
ldy #0
rts
set_connection_state:
lda #'C'+$80
ldx #'O'+$80
jsr find_var
tya
sta (VARPNT),y
iny
lda connection_state
sta (VARPNT),y
set_error:
lda #'E'+$80
ldx #'R'+$80
sta VARNAM+1
jsr find_var
lda #0
sta (VARPNT),y
iny
lda ip65_error
sta (VARPNT),y
rts
;emit the 4 bytes pointed at by AX as dotted decimals
emit_dotted_quad:
sta pptr
stx pptr + 1
ldy #0
lda (pptr),y
jsr emit_decimal
lda #'.'
jsr emit_a
ldy #1
lda (pptr),y
jsr emit_decimal
lda #'.'
jsr emit_a
ldy #2
lda (pptr),y
jsr emit_decimal
lda #'.'
jsr emit_a
ldy #3
lda (pptr),y
jsr emit_decimal
rts
emit_decimal: ;emit byte in A as a decimal number
pha
sta temp_bin ;save
sed ; Switch to decimal mode
lda #0 ; Ensure the result is clear
sta temp_bcd
sta temp_bcd+1
ldx #8 ; The number of source bits
:
asl temp_bin+0 ; Shift out one bit
lda temp_bcd+0 ; And add into result
adc temp_bcd+0
sta temp_bcd+0
lda temp_bcd+1 ; propagating any carry
adc temp_bcd+1
sta temp_bcd+1
dex ; And repeat for next bit
bne :-
cld ;back to binary
pla ;get back the original passed in number
bmi @emit_hundreds ; if N is set, the number is >=128 so emit all 3 digits
cmp #10
bmi @emit_units
cmp #100
bmi @emit_tens
@emit_hundreds:
lda temp_bcd+1 ;get the most significant digit
and #$0f
clc
adc #'0'
jsr emit_a
@emit_tens:
lda temp_bcd
lsr
lsr
lsr
lsr
clc
adc #'0'
jsr emit_a
@emit_units:
lda temp_bcd
and #$0f
clc
adc #'0'
jsr emit_a
rts
print:
sta pptr
stx pptr + 1
@print_loop:
ldy #0
lda (pptr),y
beq @done_print
jsr print_a
inc pptr
bne @print_loop
inc pptr+1
bne @print_loop ;if we ever get to $ffff, we've probably gone far enough ;-)
@done_print:
rts
extract_string:
jsr FRMEVL
jsr FRESTR ;if not string, will create type mismatch error
sta param_length
tay
lda #0
sta transfer_buffer,y
dey
@loop:
lda ($22),y
sta transfer_buffer,y
dey
bpl @loop
jmp FRESTR ;free up the temp string created by FRMEVL
;get a string value from BASIC command, turn into a 32 bit IP address,save it in the 4 bytes pointed at by AX
get_ip_parameter:
stax temp2
jsr extract_string
ldax #transfer_buffer
jsr dns_set_hostname
bcs @error
jsr dns_resolve
bcc @ok
@error:
ldax #address_resolution
jmp print_error
@ok:
ldax #dns_ip
ldx #4
@copy_dns_ip:
lda dns_ip,y
sta (temp2),y
iny
dex
bne @copy_dns_ip
rts
reset_string:
ldy #<string_buffer
sty current_output_ptr
ldy #>string_buffer
sty current_output_ptr+1
rts
print_dotted_quad:
jsr reset_string
jsr emit_dotted_quad
make_null_terminated_and_print:
lda #0
jsr emit_a
ldax #string_buffer
jmp print
print_mac:
jsr reset_string
jsr emit_mac
jmp make_null_terminated_and_print
;print 6 bytes printed at by AX as a MAC address
emit_mac:
stax pptr
ldy #0
@one_mac_digit:
tya ;just to set the Z flag
pha
beq @dont_print_colon
lda #':'
jsr emit_a
@dont_print_colon:
pla
tay
lda (pptr),y
jsr emit_hex
iny
cpy #06
bne @one_mac_digit
rts
emit_hex:
pha
pha
lsr
lsr
lsr
lsr
tax
lda hexdigits,x
jsr emit_a
pla
and #$0F
tax
lda hexdigits,x
jsr emit_a
pla
rts
print_hex:
jsr reset_string
jsr emit_hex
jmp make_null_terminated_and_print
print_error:
jsr print
ldax #error
jsr print
lda ip65_error
jsr print_hex
jsr print_cr
jsr set_error
sec
rts
get_optional_byte:
jsr CHRGOT
beq @no_param ;leave X as it was
jsr CHKCOM ;make sure next char is a comma (and skip it)
jsr CHRGOT
beq @eol
jsr GETBYT
@no_param:
rts
@eol:
jmp $AF08 ;SYNTAX ERROR
ipcfg_keyword:
ldax #interface_type
jsr print
ldax #eth_driver_name
jsr print
jsr print_cr
ldax #mac_address_msg
jsr print
ldax #cfg_mac
jsr print_mac
jsr print_cr
ldax #ip_address_msg
jsr print
ldax #cfg_ip
jsr print_dotted_quad
jsr print_cr
ldax #netmask_msg
jsr print
ldax #cfg_netmask
jsr print_dotted_quad
jsr print_cr
ldax #gateway_msg
jsr print
ldax #cfg_gateway
jsr print_dotted_quad
jsr print_cr
ldax #dns_server_msg
jsr print
ldax #cfg_dns
jsr print_dotted_quad
jsr print_cr
ldax #dhcp_server_msg
jsr print
ldax #dhcp_server
jsr print_dotted_quad
jsr print_cr
ldax #tftp_server_msg
jsr print
ldax #cfg_tftp_server
jsr print_dotted_quad
jsr print_cr
clear_error:
lda #0
sta ip65_error
jmp set_error
dhcp_keyword:
jsr dhcp_init
bcc @init_ok
jsr ip65_init ;if DHCP failed, then reinit the IP stack (which will reset IP address etc that DHCP messed with to default values)
@init_failed:
jsr set_error
ldax #dhcp
jmp print_error
@init_ok:
jmp clear_error
rts
mac_keyword:
jsr extract_string
ldy #2
:
lda transfer_buffer,y
sta cfg_mac_default+3,y
dey
bpl:-
jsr ip65_init
rts
ping_keyword:
ldax #icmp_echo_ip
jsr get_ip_parameter
bcc @no_error
rts
@no_error:
;is there an optional parameter?
ldx #3
jsr get_optional_byte
stx ping_counter
ldax #pinging
jsr print
ldax #dns_ip
jsr print_dotted_quad
jsr print_cr
@ping_loop:
jsr icmp_ping
bcs @error
lda #'.'
@print_and_loop:
jsr print_a
lda $cb ;current key pressed
cmp #$3F ;RUN/STOP?
beq @done
lda ping_counter
beq @ping_loop
dec ping_counter
cmp #1
bne @ping_loop
@done:
jsr print_cr
jmp set_error
@error:
lda #'!'
jmp @print_and_loop
myip_keyword:
ldax #cfg_ip
jmp get_ip_parameter
dns_keyword:
ldax #cfg_dns
jmp get_ip_parameter
gateway_keyword:
ldax #cfg_gateway
jmp get_ip_parameter
netmask_keyword:
ldax #cfg_netmask
jmp get_ip_parameter
tftp_keyword:
ldax #cfg_tftp_server
jmp get_ip_parameter
tf_param_setup:
jsr print
jsr extract_string
ldax #transfer_buffer
stax tftp_filename
jsr print
lda #' '
jsr print_a
lda #'('
jsr print_a
ldax #cfg_tftp_server
jsr print_dotted_quad
lda #')'
jsr print_a
jsr print_cr
ldx #$03
:
lda cfg_tftp_server,x
sta tftp_ip,x
dex
bpl :-
rts
tfget_keyword:
ldax #get_msg
jsr tf_param_setup
ldax #tftp_download_callback
jsr tftp_set_callback_vector
lda #0
sta file_opened
;make file output name
lda #'@'
sta string_buffer
lda #':'
sta string_buffer+1
ldy #$FF
@loop:
iny
lda transfer_buffer,y
sta string_buffer+2,y
bne @loop
iny
iny
lda #','
sta string_buffer,y
iny
lda #'P'
sta string_buffer,y
iny
lda #','
sta string_buffer,y
iny
lda #'W'
sta string_buffer,y
iny
sta string_length
jsr tftp_download
after_tftp_xfer:
bcc @no_error
ldax #tftp
@error_set:
jsr print_error
@no_error:
jsr close_file
jmp set_error
close_file:
lda #$02 ; filenumber 2
jsr $FFC3 ; call CLOSE
rts
open_file:
;A,X,Y set up ready for a call to SETNAM for file #2
jsr $FFBD ; call SETNAM
lda #$02 ; file number 2
ldx $BA ; last used drive
ldy #$02 ; secondary address 2
jsr $FFBA ; call SETLFS
jmp $FFC0 ; call OPEN
tftp_download_callback:
;buffer pointed at by AX now contains "tftp_data_block_length" bytes
stax temp
lda #'.'
jsr print_a
lda file_opened
bne @already_opened
lda string_length
ldx #<string_buffer
ldy #>string_buffer
jsr open_file
@already_opened:
ldx #$02 ; filenumber 2 = output file
jsr $FFC9 ; call CHKOUT
@copy_one_byte:
lda tftp_data_block_length
bne @not_done
lda tftp_data_block_length+1
beq @done
@not_done:
ldy #2 ;we want to skip the first 2 bytes in the buffer
lda (temp),y
jsr $ffd2 ;write byte
inc temp
bne :+
inc temp+1
:
lda tftp_data_block_length
dec tftp_data_block_length
cmp #0
bne @copy_one_byte
dec tftp_data_block_length+1
jmp @copy_one_byte
@done:
ldx #$00 ; filenumber 0 = console
jmp $FFC9 ; call CHKOUT
tfput_keyword:
ldax #put_msg
jsr tf_param_setup
lda param_length
ldx #<transfer_buffer
ldy #>transfer_buffer
jsr open_file
bcs @error
lda $90 ;get ST
beq @ok
@error:
ldx #4 ;"FILE NOT FOUND" error
jmp $A437 ;error
@ok:
ldax #tftp_upload_callback
jsr tftp_set_callback_vector
jsr tftp_upload
jmp after_tftp_xfer
tftp_upload_callback:
stax copy_dest
lda #'.'
jsr print_a
lda #0
sta bytes_read
sta bytes_read+1
ldx #$02 ; filenumber 2 = output file
jsr $FFC6 ; call CHKIN (file 2 now used as input)
@loop:
jsr $FFCF ; call CHRIN (get a byte from file)
ldy #0
sta (copy_dest),y
inc copy_dest
bne :+
inc copy_dest+1
:
inc bytes_read
bne :+
inc bytes_read+1
:
lda bytes_read+1
cmp #2
beq @done
jsr $FFB7 ; call READST (read status byte)
beq @loop ; nonzero mean either EOF or read error
@done:
ldx #$00 ; filenumber 0 = console
jsr $FFC6 ; call CHKIN (console now used as input)
ldax bytes_read
rts
skip_comma_get_integer:
jsr CHRGOT
jsr CHKCOM ;make sure next char is a comma (and skip it)
get_integer:
jsr CHRGOT
beq @eol
jsr FRMNUM
jsr GETADR
ldax LINNUM
@no_param:
rts
@eol:
jmp $AF08 ;SYNTAX ERROR
make_tcp_connection:
lda #0
sta connection_state
jsr set_connection_state
ldax #tcp_connect_ip
jsr get_ip_parameter
bcc @no_error
rts
@no_error:
jsr skip_comma_get_integer
jsr tcp_connect
bcc :+
@connect_error:
ldax #connect
jmp print_error
:
ldax #connected_msg
jsr print
lda #1
sta connection_state
lda #0
sta ip65_error
jsr set_connection_state
clc
rts
netcat_keyword:
lda $CC
sta cursor_state
lda #$0
sta $CC ;enable blinking cursor
ldax #netcat_callback
stax tcp_callback
jsr make_tcp_connection
bcs @exit
;is there an optional parameter?
ldx #0
jsr get_optional_byte
stx netcat_mode
@main_polling_loop:
jsr timer_read
txa
adc #$20 ;32 x 1/4 = ~ 8seconds
sta netcat_timeout
@wait_for_keypress:
jsr timer_read
cpx netcat_timeout
bne @no_timeout
jsr tcp_send_keep_alive
jmp @main_polling_loop
@no_timeout:
jsr ip65_process
lda connection_state
bne @not_disconnected
@disconnected:
ldax #disconnected
jsr print
@exit:
lda cursor_state
sta $CC
rts
@not_disconnected:
lda netcat_mode
beq @not_line_mode
lda #$00
sta string_length
;process inbound ip packets while waiting for a keypress
@read_line:
lda $cb ;current key pressed
cmp #$3F ;RUN/STOP?
beq @runstop
jsr ip65_process
lda connection_state
beq @disconnected
jsr $f142 ;not officially documented - where F13E (GETIN) falls through to if device # is 0 (KEYBD)
beq @read_line
cmp #$14 ;Delete
beq @delete
cmp #$0d ;Return
beq @input_done
;End reached?
ldy string_length
cpy #$FF
beq @read_line
jsr $ffd2 ;Print it
jsr native_to_ascii
sta transfer_buffer,y ;Add it to string
inc string_length
;Not yet.
jmp @read_line
@delete:
;First, check if we're at the beginning.
lda string_length
bne @delete_ok
jmp @read_line
;At least one character entered.
@delete_ok:
;Move pointer back.
dec string_length
;Print the delete char
lda #$14
jsr $ffd2
;Wait for next char
jmp @read_line
@input_done:
jsr reset_cursor
lda #$0d
jsr $ffd2 ;print a newline
ldy string_length
lda #$0d
sta transfer_buffer,y
iny
lda #$0a
sta transfer_buffer,y
iny
sty tcp_send_data_len
jmp @send_buffer
@not_line_mode:
;is there anything in the input buffer?
lda $c6 ;NDX - chars in keyboard buffer
bne :+
jmp @wait_for_keypress
:
lda #0
sta tcp_send_data_len
sta tcp_send_data_len+1
@get_next_char:
lda $cb ;current key pressed
cmp #$3F ;RUN/STOP?
bne @not_runstop
@runstop:
lda #0
sta $cb ;overwrite "current key pressed" else it's seen by the tcp stack and the close aborts
lda cursor_state
sta $CC
jmp tcp_close
@not_runstop:
jsr $ffe4 ;getkey - 0 means no input
tax
beq @no_more_input
txa
ldy tcp_send_data_len
sta transfer_buffer,y
inc tcp_send_data_len
jmp @get_next_char
@no_more_input:
@send_buffer:
ldax #transfer_buffer
jsr tcp_send
bcs @error_on_send
jmp @main_polling_loop
@error_on_send:
lda cursor_state
sta $CC
ldax #transmission
jmp print_error
reset_cursor:
lda $cf ;0 means last cursor blink set char to be reversed
beq @done
lda $ce ;original value of cursor char
ldx $287 ;original colour
ldy #$0 ;blink phase
sty $cf
jsr $ea13 ;restore char & colour
@done:
rts
netcat_callback:
jsr reset_cursor
lda tcp_inbound_data_length+1
cmp #$ff
bne @not_eof
lda #0
sta connection_state
rts
@not_eof:
ldax tcp_inbound_data_ptr
stax temp2
lda tcp_inbound_data_length
sta buffer_length
lda tcp_inbound_data_length+1
sta buffer_length+1
@next_byte:
lda $cb ;current key pressed
cmp #$3F ;RUN/STOP?
beq @finished
ldy #0
lda (temp2),y
ldx netcat_mode
beq @no_transform
jsr ascii_to_native
@no_transform:
jsr print_a
inc temp2
bne :+
inc temp2+1
:
lda buffer_length+1
beq @last_page
lda buffer_length
bne @not_end_of_page
dec buffer_length+1
@not_end_of_page:
dec buffer_length
jmp @next_byte
@last_page:
dec buffer_length
beq @finished
jmp @next_byte
@finished:
rts
tcpconnect_keyword:
ldax #tcpconnect_callback
stax tcp_callback
jmp make_tcp_connection
tcpconnect_callback:
ldax #transfer_buffer
stax copy_dest
ldax tcp_inbound_data_ptr
stax copy_src
lda tcp_inbound_data_length
ldx tcp_inbound_data_length+1
cpx #$ff
bne @not_end_packet
@end_packet:
lda #0
sta connection_state
rts
@not_end_packet:
stax inbound_data_length
set_packet_vars:
jsr copymem
ldx inbound_data_length+1
beq @short_packet
lda #$ff
@short_packet:
lda inbound_data_length
set_input_string:
pha
lda #'I'
ldx #'N'+$80
jsr find_var
ldy #0
pla
pha
sta (VARPNT),y
iny
lda #<transfer_buffer
sta (VARPNT),y
iny
lda #>transfer_buffer
sta (VARPNT),y
pla
beq :+
ldx #0
jsr copymem
:
lda #'I'+$80
ldx #'P'+$80
jsr find_var
lda #<transfer_buffer
sta (VARPNT),y
iny
lda #>transfer_buffer
sta (VARPNT),y
lda #'I'+$80
ldx #'L'+$80
jsr find_var
lda inbound_data_length+1
sta (VARPNT),y
iny
lda inbound_data_length
sta (VARPNT),y
rts
poll_keyword:
lda #0
sta tcp_inbound_data_length
sta tcp_inbound_data_length+1
jsr set_input_string
jsr set_connection_state
jsr ip65_process
lda ip65_error
beq @no_error
jmp set_error
@no_error:
jmp set_connection_state
tcplisten_keyword:
lda #0
sta connection_state
sta ip65_error
ldax #tcpconnect_callback
stax tcp_callback
jsr get_integer
jsr tcp_listen
bcs :+
inc connection_state
:
jmp set_connection_state
tcpsend_keyword:
jsr extract_string
ldy param_length
sty tcp_send_data_len
ldy #0
sty tcp_send_data_len+1
ldax #transfer_buffer
jsr tcp_send
jmp set_connection_state
tcpclose_keyword:
lda #0
sta connection_state
jsr tcp_close
jmp set_connection_state
tcpblat_keyword:
jsr extract_string
ldx #<transfer_buffer
ldy #>transfer_buffer
lda param_length
jsr SETNAM
lda #$02 ; file number 2
ldx $BA ; last used device number
bne @skip
ldx #$08 ; default to device 8
@skip:
ldy #$02 ; secondary address 2
jsr SETLFS
jsr OPEN
bcs @error ; if carry set, the file could not be opened
ldx #$02 ; filenumber 2
jsr CHKIN
ldy #$00
@loop:
jsr READST
bne @eof ; either EOF or read error
jsr CHRIN
sta transfer_buffer,y
iny
bne @loop
ldax #$100
stax tcp_send_data_len
ldax #transfer_buffer
jsr tcp_send
bcs @error_stored
ldy #0
jmp @loop
@eof:
and #$40 ; end of file?
beq @readerror
beq @readerror
lda #$00
sty tcp_send_data_len
sta tcp_send_data_len+1
ldax #transfer_buffer
jsr tcp_send
bcs @error_stored
@close:
lda #0
@store_error:
sta ip65_error
@error_stored:
lda #$02 ; filenumber 2
jsr CLOSE
ldx #$00 ; filenumber 0 = keyboard
jsr CHKIN ;keyboard now input device again
jmp set_error
@error:
lda #KPR_ERROR_DEVICE_FAILURE
jmp @store_error
@readerror:
lda #KPR_ERROR_FILE_ACCESS_FAILURE
jmp @store_error
;send udp packet
udpsend_keyword:
ldax #udp_send_dest
jsr get_ip_parameter
bcc @no_error
rts
@no_error:
jsr skip_comma_get_integer
stax udp_send_dest_port
jsr skip_comma_get_integer
stax udp_send_src_port
jsr CHRGOT
jsr CHKCOM ;make sure next char is a comma (and skip it)
jsr extract_string
inc $d020
ldy param_length
sty udp_send_len
ldy #0
sty udp_send_len+1
ldax #transfer_buffer
jsr udp_send
lda #0
sta ip65_error
clc
rts
udplisten_keyword:
ldax #udp_handler
stax udp_callback
jsr get_integer ;port number
stax port
jsr udp_remove_listener
ldax port
jsr udp_add_listener
bcc @ok
ldax #too_many_listeners
jmp print_error
@ok:
rts
udp_handler:
inc $d020
rts
evaluate:
lda #$00
sta $0D ;set string flag to not stringe
jsr CHRGET
cmp #$E3 ; PING keyword
bne @done
jsr CHRGET ;take PING command off stack
ldax #icmp_echo_ip
jsr get_ip_parameter
lda #$00
sta $0D ;set string flag to not string
bcs @error
jsr icmp_ping
bcc @no_error
@error:
lda #$ff
tax
@no_error:
tay
txa
jmp $b395 ;signed 16 bit number to floating point
rts
@done:
jsr CHRGOT
jmp $AE8D ;inside original EVAL routine
.rodata
vectors:
.word crunch
.word list
.word execute
.word evaluate
; Keyword list
; Keywords are stored as normal text,
; followed by the token number.
; All tokens are >$80,
; so they easily mark the end of the keyword
hexdigits:
.byte "0123456789ABCDEF"
pinging:
.byte"PINGING ",0
interface_type:
.byte "INTERFACE : ",0
mac_address_msg:
.byte "MAC ADDRESS : ", 0
ip_address_msg:
.byte "IP ADDRESS : ", 0
netmask_msg:
.byte "NETMASK : ", 0
gateway_msg:
.byte "GATEWAY : ", 0
dns_server_msg:
.byte "DNS SERVER : ", 0
dhcp_server_msg:
.byte "DHCP SERVER : ", 0
tftp_server_msg:
.byte "TFTP SERVER : ", 0
too_many_listeners:
.byte "TOO MANY LISTENERS",0
address_resolution:
.byte "ADDRESS RESOLUTION",0
get_msg:
.byte "GETTING ",0
put_msg:
.byte "PUTTING ",0
tftp:
.byte "TFTP",0
dhcp:
.byte "DHCP",0
connect:
.byte "CONNECT",0
transmission:
.byte "TRANSMISSION",0
error:
.byte " ERROR $",0
disconnected:
.byte 13,"DIS"
connected_msg:
.byte "CONNECTED",13,0
keywords:
.byte "IF",$E0 ;our dummy 'IF' entry takes $E0
.byte "IPCFG",$E1
.byte "DHCP",$E2
.byte "PING",$E3
.byte "MYIP",$E4
.byte "NETMASK",$E5
.byte "GATEWAY",$E6
.byte "DNS",$E7
.byte "TFTP",$E8
.byte "TF",$A1,$E9 ;TFGET - BASIC will replace GET with A1
.byte "TFPUT",$EA
.byte "NETCAT",$EB
.byte "TCPC",$91,"NECT",$EC ; TCPCONNECT - BASIC will replace ON with $91
.byte "POLL",$ED
.byte "TCP",$9B,"EN",$EE ;TCPLISTEN - BASIC will replace LIST with $9b
.byte "TCPS",$80,$EF ;TCPSEND - BASIC will replace END with $80
.byte "TCP",$A0,$F0 ;TCPLOSE - BASIC will replace CLOSE with $A0
.byte "TCPBLAT",$F1
.byte "MAC",$F2
.byte "UDPS",$80,$F3 ;UDPSEND - BASIC will replace END with $80
.byte "UDP",$9B,"EN",$F4 ;UDPLISTEN - BASIC will replace LIST with $9b
.byte $00 ;end of list
HITOKEN=$F5
;
; Table of token locations-1
;
token_routines:
E0: .word if_keyword-1
E1: .word ipcfg_keyword-1
E2: .word dhcp_keyword-1
E3: .word ping_keyword-1
E4: .word myip_keyword-1
E5: .word netmask_keyword-1
E6: .word gateway_keyword-1
E7: .word dns_keyword-1
E8: .word tftp_keyword-1
E9: .word tfget_keyword-1
EA: .word tfput_keyword-1
EB: .word netcat_keyword-1
EC: .word tcpconnect_keyword-1
ED: .word poll_keyword-1
EE: .word tcplisten_keyword-1
EF: .word tcpsend_keyword-1
FO: .word tcpclose_keyword-1
F1: .word tcpblat_keyword-1
F2: .word mac_keyword-1
F3: .word udpsend_keyword-1
F4: .word udplisten_keyword-1
.segment "SELF_MODIFIED_CODE"
jmp_crunch: .byte $4C ;JMP
oldcrunch: .res 2 ;Old CRUNCH vector
oldlist: .res 2
oldexec: .res 2
oldeval: .res 2
emit_a:
current_output_ptr=emit_a+1
sta $ffff
inc string_length
inc current_output_ptr
bne :+
inc current_output_ptr+1
:
rts
.bss
netcat_mode: .res 1
bytes_read: .res 2
string_length: .res 1
param_length: .res 1
ip_string: .res 15
netmask_string: .res 15
dns_string: .res 15
gateway_string: .res 15
temp_bin: .res 1
temp_bcd: .res 2
ping_counter: .res 1
string_buffer: .res 128
transfer_buffer: .res 1500
file_opened: .res 1
connection_state: .res 1
netcat_timeout: .res 1
buffer_length: .res 2
cursor_state: .res 1
port: .res 2
inbound_data_length: .res 2