2009-08-11 11:32:06 +00:00
|
|
|
;routine for parsing a URL
|
|
|
|
|
|
|
|
|
|
|
|
.include "../inc/common.i"
|
|
|
|
|
|
|
|
.ifndef NB65_API_VERSION_NUMBER
|
|
|
|
.define EQU =
|
|
|
|
.include "../inc/nb65_constants.i"
|
|
|
|
.endif
|
|
|
|
|
|
|
|
.import output_buffer
|
|
|
|
.importzp copy_src
|
|
|
|
.importzp copy_dest
|
2009-08-17 04:23:05 +00:00
|
|
|
.import copymem
|
|
|
|
|
2009-08-16 04:15:43 +00:00
|
|
|
.import ip65_error
|
2009-08-17 04:23:05 +00:00
|
|
|
.import ip65_process
|
2009-08-11 11:32:06 +00:00
|
|
|
.import parser_init
|
|
|
|
.import parser_skip_next
|
|
|
|
.import dns_set_hostname
|
|
|
|
.import dns_resolve
|
|
|
|
.import parse_integer
|
|
|
|
.import dns_ip
|
2009-08-17 04:23:05 +00:00
|
|
|
.import tcp_connect
|
|
|
|
.import tcp_send_string
|
|
|
|
.import tcp_send_data_len
|
|
|
|
.import tcp_callback
|
|
|
|
.import tcp_close
|
|
|
|
.import tcp_connect_ip
|
|
|
|
.import tcp_inbound_data_length
|
|
|
|
.import tcp_inbound_data_ptr
|
|
|
|
|
|
|
|
|
2009-08-11 11:32:06 +00:00
|
|
|
.export url_ip
|
|
|
|
.export url_port
|
|
|
|
.export url_selector
|
2009-08-12 10:34:23 +00:00
|
|
|
.export url_resource_type
|
2009-08-16 04:15:43 +00:00
|
|
|
.export url_parse
|
2009-08-17 04:23:05 +00:00
|
|
|
.export url_download
|
|
|
|
.export url_download_buffer
|
|
|
|
.export url_download_buffer_length
|
2009-08-11 11:32:06 +00:00
|
|
|
|
|
|
|
target_string=copy_src
|
|
|
|
search_string=copy_dest
|
|
|
|
selector_buffer=output_buffer
|
|
|
|
|
|
|
|
.bss
|
|
|
|
url_string: .res 2
|
|
|
|
url_ip: .res 4 ;will be set with ip address of host in url
|
|
|
|
url_port: .res 2 ;will be set with port number of url
|
|
|
|
url_selector: .res 2 ;will be set with address of selector part of URL
|
|
|
|
url_type: .res 1
|
2009-08-12 10:34:23 +00:00
|
|
|
url_resource_type: .res 1
|
2009-08-11 11:32:06 +00:00
|
|
|
url_type_unknown=0
|
|
|
|
url_type_gopher=1
|
|
|
|
url_type_http=2
|
|
|
|
|
|
|
|
src_ptr: .res 1
|
|
|
|
dest_ptr: .res 1
|
2009-08-17 04:23:05 +00:00
|
|
|
|
|
|
|
url_download_buffer: .res 2 ; points to a buffer that url will be downloaded into
|
|
|
|
url_download_buffer_length: .res 2 ;length of buffer that url will be downloaded into
|
|
|
|
|
|
|
|
temp_buffer: .res 2
|
|
|
|
temp_buffer_length: .res 2
|
|
|
|
|
|
|
|
download_flag: .res 1
|
|
|
|
|
|
|
|
|
2009-08-11 11:32:06 +00:00
|
|
|
.code
|
|
|
|
|
|
|
|
|
|
|
|
;parses a URL into a form that makes it easy to retrieve the specified resource
|
|
|
|
;inputs:
|
|
|
|
;AX = address of URL string
|
|
|
|
;outputs:
|
|
|
|
; sec if a malformed url, otherwise:
|
|
|
|
; url_ip = ip address of host in url
|
|
|
|
; url_port = port number of url
|
|
|
|
; url_selector= address of selector part of URL
|
|
|
|
url_parse:
|
|
|
|
stax url_string
|
|
|
|
ldy #0
|
|
|
|
sty url_type
|
|
|
|
sty url_port
|
|
|
|
sty url_port+1
|
2009-08-12 10:34:23 +00:00
|
|
|
sty url_resource_type
|
2009-08-11 11:32:06 +00:00
|
|
|
|
|
|
|
jsr skip_to_hostname
|
|
|
|
bcc :+
|
|
|
|
ldax url_string
|
|
|
|
jmp @no_protocol_specifier
|
|
|
|
:
|
|
|
|
ldax url_string
|
|
|
|
stax search_string
|
|
|
|
|
|
|
|
lda (search_string),y
|
|
|
|
cmp #'g'
|
|
|
|
beq @gopher
|
|
|
|
cmp #'G'
|
|
|
|
beq @gopher
|
|
|
|
cmp #'h'
|
|
|
|
beq @http
|
|
|
|
cmp #'H'
|
|
|
|
beq @http
|
|
|
|
@exit_with_error:
|
|
|
|
lda #NB65_MALFORMED_URL
|
|
|
|
sta ip65_error
|
|
|
|
@exit_with_sec:
|
|
|
|
sec
|
|
|
|
rts
|
|
|
|
@http:
|
|
|
|
lda #url_type_http
|
|
|
|
sta url_type
|
|
|
|
lda #80
|
|
|
|
sta url_port
|
|
|
|
jmp @protocol_set
|
|
|
|
@gopher:
|
|
|
|
lda #url_type_gopher
|
|
|
|
sta url_type
|
|
|
|
lda #70
|
|
|
|
sta url_port
|
|
|
|
@protocol_set:
|
|
|
|
jsr skip_to_hostname
|
|
|
|
;now pointing at hostname
|
|
|
|
bcs @exit_with_error
|
|
|
|
@no_protocol_specifier:
|
|
|
|
jsr dns_set_hostname
|
|
|
|
bcs @exit_with_sec
|
|
|
|
jsr dns_resolve
|
2009-08-17 04:23:05 +00:00
|
|
|
bcc :+
|
|
|
|
lda #NB65_DNS_LOOKUP_FAILED
|
|
|
|
sta ip65_error
|
|
|
|
jmp @exit_with_sec
|
|
|
|
:
|
2009-08-11 11:32:06 +00:00
|
|
|
;copy IP address
|
|
|
|
ldx #3
|
|
|
|
:
|
|
|
|
lda dns_ip,x
|
|
|
|
sta url_ip,x
|
|
|
|
dex
|
|
|
|
bpl :-
|
|
|
|
|
|
|
|
jsr skip_to_hostname
|
|
|
|
|
|
|
|
;skip over next colon
|
|
|
|
ldax #colon
|
|
|
|
jsr parser_skip_next
|
|
|
|
bcs @no_port_in_url
|
|
|
|
;AX now point at first thing past a colon - should be a number:
|
|
|
|
jsr parse_integer
|
|
|
|
stax url_port
|
|
|
|
@no_port_in_url:
|
|
|
|
;skip over next slash
|
|
|
|
ldax #slash
|
|
|
|
jsr parser_skip_next
|
|
|
|
;AX now pointing at selector
|
|
|
|
stax copy_src
|
|
|
|
ldax #selector_buffer
|
|
|
|
stax copy_dest
|
|
|
|
lda #0
|
|
|
|
sta src_ptr
|
|
|
|
sta dest_ptr
|
|
|
|
lda url_type
|
2009-08-12 10:34:23 +00:00
|
|
|
|
|
|
|
cmp #url_type_gopher
|
|
|
|
bne @not_gopher
|
|
|
|
;first byte after / in a gopher url is the resource type
|
|
|
|
ldy src_ptr
|
|
|
|
lda (copy_src),y
|
2009-08-17 04:23:05 +00:00
|
|
|
beq @start_of_selector
|
2009-08-12 10:34:23 +00:00
|
|
|
sta url_resource_type
|
|
|
|
inc src_ptr
|
|
|
|
jmp @start_of_selector
|
|
|
|
@not_gopher:
|
2009-08-11 11:32:06 +00:00
|
|
|
cmp #url_type_http
|
2009-08-17 04:23:05 +00:00
|
|
|
beq @build_http_request
|
|
|
|
jmp @done ; if it's not gopher or http, we don't know how to build a selector
|
|
|
|
@build_http_request:
|
|
|
|
ldy #get_length-1
|
2009-08-11 11:32:06 +00:00
|
|
|
sty dest_ptr
|
|
|
|
:
|
|
|
|
lda get,y
|
|
|
|
sta (copy_dest),y
|
|
|
|
dey
|
2009-08-12 10:34:23 +00:00
|
|
|
bpl :-
|
2009-08-17 04:23:05 +00:00
|
|
|
|
2009-08-12 10:34:23 +00:00
|
|
|
@start_of_selector:
|
2009-08-11 11:32:06 +00:00
|
|
|
lda #'/'
|
2009-08-12 10:34:23 +00:00
|
|
|
inc dest_ptr
|
2009-08-11 11:32:06 +00:00
|
|
|
jmp @save_first_byte_of_selector
|
|
|
|
@copy_one_byte:
|
|
|
|
ldy src_ptr
|
|
|
|
lda (copy_src),y
|
|
|
|
beq @end_of_selector
|
|
|
|
inc src_ptr
|
|
|
|
@save_first_byte_of_selector:
|
|
|
|
ldy dest_ptr
|
2009-08-12 10:34:23 +00:00
|
|
|
sta (copy_dest),y
|
2009-08-11 11:32:06 +00:00
|
|
|
inc dest_ptr
|
|
|
|
bne @copy_one_byte
|
|
|
|
@end_of_selector:
|
2009-08-17 04:23:05 +00:00
|
|
|
|
|
|
|
|
|
|
|
ldx #1 ;number of CRLF at end of gopher request
|
|
|
|
lda url_type
|
|
|
|
|
|
|
|
cmp #url_type_http
|
|
|
|
bne @final_crlf
|
|
|
|
|
|
|
|
;now the HTTP version number & Host: field
|
|
|
|
ldx #0
|
|
|
|
:
|
|
|
|
lda http_version_and_host,x
|
|
|
|
beq :+
|
|
|
|
ldy dest_ptr
|
|
|
|
inc dest_ptr
|
|
|
|
sta (copy_dest),y
|
|
|
|
inx
|
|
|
|
bne :-
|
|
|
|
:
|
|
|
|
|
|
|
|
|
|
|
|
;now copy the host field
|
|
|
|
jsr skip_to_hostname
|
|
|
|
;AX now pointing at hostname
|
|
|
|
stax copy_src
|
|
|
|
ldax #selector_buffer
|
|
|
|
stax copy_dest
|
|
|
|
|
|
|
|
lda #0
|
|
|
|
sta src_ptr
|
|
|
|
|
|
|
|
@copy_one_byte_of_hostname:
|
|
|
|
ldy src_ptr
|
|
|
|
lda (copy_src),y
|
|
|
|
beq @end_of_hostname
|
|
|
|
cmp #':'
|
|
|
|
beq @end_of_hostname
|
|
|
|
cmp #'/'
|
|
|
|
beq @end_of_hostname
|
|
|
|
inc src_ptr
|
|
|
|
ldy dest_ptr
|
|
|
|
sta (copy_dest),y
|
|
|
|
inc dest_ptr
|
|
|
|
bne @copy_one_byte_of_hostname
|
|
|
|
@end_of_hostname:
|
|
|
|
|
|
|
|
ldx #2 ;number of CRLF at end of HTTP request
|
|
|
|
|
|
|
|
@final_crlf:
|
2009-08-11 11:32:06 +00:00
|
|
|
ldy dest_ptr
|
|
|
|
lda #$0d
|
|
|
|
sta (copy_dest),y
|
|
|
|
iny
|
|
|
|
lda #$0a
|
|
|
|
sta (copy_dest),y
|
|
|
|
iny
|
2009-08-17 04:23:05 +00:00
|
|
|
dex
|
|
|
|
bne @final_crlf
|
|
|
|
|
2009-08-11 11:32:06 +00:00
|
|
|
@done:
|
|
|
|
lda #$00
|
|
|
|
sta (copy_dest),y
|
|
|
|
ldax #selector_buffer
|
2009-08-17 04:23:05 +00:00
|
|
|
stax url_selector
|
2009-08-11 11:32:06 +00:00
|
|
|
clc
|
2009-08-17 04:23:05 +00:00
|
|
|
|
2009-08-11 11:32:06 +00:00
|
|
|
rts
|
|
|
|
|
|
|
|
skip_to_hostname:
|
|
|
|
ldax url_string
|
|
|
|
jsr parser_init
|
|
|
|
ldax #colon_slash_slash
|
|
|
|
jmp parser_skip_next
|
|
|
|
|
2009-08-17 04:23:05 +00:00
|
|
|
.code
|
|
|
|
|
|
|
|
|
|
|
|
;download a resource specified by an URL
|
|
|
|
;inputs:
|
|
|
|
;AX = address of URL string
|
|
|
|
; url_download_buffer - points to a buffer that url will be downloaded into
|
|
|
|
; url_download_buffer_length - length of buffer
|
|
|
|
;outputs:
|
|
|
|
; sec if an error occured, else buffer pointed at by url_download_buffer is filled with contents
|
|
|
|
; of specified resource (with an extra 2 null bytes at the end),
|
|
|
|
; AX = length of resource downloaded.
|
|
|
|
url_download:
|
|
|
|
jsr url_parse
|
|
|
|
bcc @url_parsed_ok
|
|
|
|
rts
|
|
|
|
@url_parsed_ok:
|
|
|
|
ldax url_download_buffer
|
|
|
|
stax temp_buffer
|
|
|
|
ldax url_download_buffer_length
|
|
|
|
stax temp_buffer_length
|
|
|
|
|
|
|
|
ldx #3 ; save IP address just retrieved
|
|
|
|
: lda url_ip,x
|
|
|
|
sta tcp_connect_ip,x
|
|
|
|
dex
|
|
|
|
bpl :-
|
|
|
|
ldax #url_download_callback
|
|
|
|
stax tcp_callback
|
|
|
|
|
|
|
|
ldax url_port
|
|
|
|
jsr tcp_connect
|
|
|
|
bcs @error
|
|
|
|
|
|
|
|
;connected, now send the selector
|
|
|
|
ldx #0
|
|
|
|
stx download_flag
|
|
|
|
ldax url_selector
|
|
|
|
|
|
|
|
jsr tcp_send_string
|
|
|
|
;now loop until we're done
|
|
|
|
@download_loop:
|
|
|
|
jsr ip65_process
|
|
|
|
lda download_flag
|
|
|
|
beq @download_loop
|
|
|
|
jsr tcp_close
|
|
|
|
clc
|
|
|
|
@error:
|
|
|
|
rts
|
|
|
|
|
|
|
|
|
|
|
|
lda #NB65_ERROR_FILE_ACCESS_FAILURE
|
|
|
|
sta ip65_error
|
|
|
|
sec
|
|
|
|
rts
|
|
|
|
|
|
|
|
|
|
|
|
url_download_callback:
|
|
|
|
|
|
|
|
lda tcp_inbound_data_length+1
|
|
|
|
cmp #$ff
|
|
|
|
bne @not_end_of_file
|
|
|
|
@end_of_file:
|
|
|
|
lda #1
|
|
|
|
sta download_flag
|
|
|
|
|
|
|
|
;put a zero byte at the end of the file (in case it was a text file)
|
|
|
|
ldax temp_buffer
|
|
|
|
stax copy_dest
|
|
|
|
lda #0
|
|
|
|
tay
|
|
|
|
sta (copy_dest),y
|
|
|
|
rts
|
|
|
|
@not_end_of_file:
|
|
|
|
|
|
|
|
;copy this chunk to our input buffer
|
|
|
|
ldax temp_buffer
|
|
|
|
stax copy_dest
|
|
|
|
ldax tcp_inbound_data_ptr
|
|
|
|
stax copy_src
|
|
|
|
sec
|
|
|
|
lda temp_buffer_length
|
|
|
|
sbc tcp_inbound_data_length
|
|
|
|
pha
|
|
|
|
lda temp_buffer_length+1
|
|
|
|
sbc tcp_inbound_data_length+1
|
|
|
|
bcc @would_overflow_buffer
|
|
|
|
sta temp_buffer_length+1
|
|
|
|
pla
|
|
|
|
sta temp_buffer_length
|
|
|
|
ldax tcp_inbound_data_length
|
|
|
|
jsr copymem
|
|
|
|
;increment the pointer into the input buffer
|
|
|
|
clc
|
|
|
|
lda temp_buffer
|
|
|
|
adc tcp_inbound_data_length
|
|
|
|
sta temp_buffer
|
|
|
|
lda temp_buffer+1
|
|
|
|
adc tcp_inbound_data_length+1
|
|
|
|
sta temp_buffer+1
|
|
|
|
; lda #'*'
|
|
|
|
; jsr print_a
|
|
|
|
|
|
|
|
rts
|
|
|
|
@would_overflow_buffer:
|
|
|
|
pla ;clean up the stack
|
|
|
|
ldax temp_buffer_length
|
|
|
|
jsr copymem
|
|
|
|
lda temp_buffer
|
|
|
|
adc temp_buffer_length
|
|
|
|
sta temp_buffer
|
|
|
|
lda temp_buffer+1
|
|
|
|
adc temp_buffer_length+1
|
|
|
|
sta temp_buffer+1
|
|
|
|
lda #0
|
|
|
|
sta temp_buffer_length
|
|
|
|
sta temp_buffer_length+1
|
|
|
|
rts
|
|
|
|
|
|
|
|
.rodata
|
2009-08-11 11:32:06 +00:00
|
|
|
get: .byte "GET "
|
2009-08-17 04:23:05 +00:00
|
|
|
get_length=4
|
|
|
|
http_version_and_host: .byte " HTTP/1.1",$0d,$0a, "Host: ",0
|
|
|
|
; http_trailer: .byte " HTTP/1.1",$0a,$0a
|
|
|
|
; http_trailer_end:
|
|
|
|
; http_trailer_length=http_trailer_end-http_trailer
|
|
|
|
|
2009-08-11 11:32:06 +00:00
|
|
|
colon_slash_slash: .byte ":/"
|
|
|
|
slash: .byte "/",0
|
2009-08-17 04:23:05 +00:00
|
|
|
colon: .byte ":",0
|
|
|
|
|
|
|
|
|