mirror of
https://github.com/deater/dos33fsprogs.git
synced 2025-01-03 18:29:53 +00:00
407 lines
10 KiB
QBasic
407 lines
10 KiB
QBasic
' Applesoft BASIC Webserver
|
|
' by Vince Weaver <vince@deater.net>
|
|
'
|
|
1 REM *** Setup UTHERNET II - W5100
|
|
' SLOT0=$C080 49280 SLOT4=$C0C0 49344
|
|
' SLOT1=$C090 49296 SLOT5=$C0D0 49360
|
|
' SLOT2=$C0A0 49312 SLOT6=$C0E0 49376
|
|
' SLOT3=$C0B0 49328 SLOT7=$C0F0 49392
|
|
'
|
|
' Set up the memory addresses to use
|
|
'
|
|
2 REM *** OURS IS IN SLOT3 ($C0B0)
|
|
3 SLOT=49328: REM *** $C0B0
|
|
4 MR=SLOT+4: REM *** MODE REGISTER C0B4
|
|
5 HA=SLOT+5:LA=SLOT+6: REM *** HIGH/LOW ADDR $C0B5,$C0B6
|
|
7 DP=SLOT+7: REM *** DATA PORT $C0B7
|
|
'
|
|
' Init the W5100
|
|
'
|
|
10 REM *** Init W5100
|
|
12 POKE MR,128 : REM RESET W5100
|
|
14 POKE MR,3 : REM AUTOINCREMENT
|
|
20 REM *** Setup MAC Address 41:50:50:4c:45:32
|
|
22 POKE HA,0:POKE LA,9
|
|
23 POKE DP,65:POKE DP,80:POKE DP,80:POKE DP,76:POKE DP,69:POKE DP,50
|
|
30 REM *** Setup IP Address 192.168.8.15
|
|
32 POKE LA,15
|
|
33 POKE DP,192:POKE DP,168:POKE DP,8:POKE DP,15
|
|
40 PRINT "UTHERNET II READY: 192.168.8.15"
|
|
'
|
|
' Setup Machine Language Memcpy routine
|
|
' NOTE! This code assumes the Uthernet is in slot 3
|
|
' FIXME: patch on the fly once it works
|
|
' See Appendix 1 at the end of this for more details
|
|
'
|
|
50 FOR I=0 TO 72: READ X: POKE 768+I,X:NEXT I
|
|
51 DATA 169,0,133,6,169,64,133,7,162,11,240,36,160,0,177,6
|
|
52 DATA 141,183,192,213,9,208,15,217,8,0,208,10,169,64,141,181
|
|
53 DATA 192,169,0,141,182,192,200,208,229,230,7,202,208,224,162,10
|
|
54 DATA 177,6,141,183,192,217,8,0,208,10,169,64,141,181,192,169
|
|
55 DATA 0,141,182,192,200,202,208,232,96
|
|
'
|
|
' Setup Socket 0
|
|
'
|
|
100 REM *** Setup Socket 0
|
|
102 PRINT "** Setting up Socket 0"
|
|
105 POKE HA,0:POKE LA,26: REM RX MEMSIZE
|
|
110 POKE DP,3: REM 8kB RX buffer
|
|
115 POKE DP,3: REM 8kB TX buffer
|
|
200 REM *** Setup TCP MODE on SOCKET 0
|
|
205 POKE HA,4: POKE LA,0: REM *** 0x400 mode
|
|
210 POKE DP,65 : REM *** 0x41 MAC FILTER (non-promisc) TCP
|
|
300 REM ** Setup Source PORT
|
|
303 PRINT "** Setting up to use TCP port 80"
|
|
305 POKE HA,4: POKE LA,4: REM *** 0x404 port
|
|
310 POKE DP,0:POKE DP, 80: REM *** http port 80
|
|
'
|
|
' OPEN the socket
|
|
'
|
|
400 REM *** OPEN socket
|
|
404 PRINT "** OPENing socket"
|
|
405 POKE HA,4: POKE LA,1: REM *** 0x401 command register
|
|
410 POKE DP, 1: REM *** OPEN
|
|
'
|
|
' Check return value
|
|
'
|
|
500 REM *** Check if opened
|
|
505 POKE HA,4: POKE LA,3: REM *** 0x403 status register
|
|
510 RE=PEEK(DP)
|
|
515 PRINT "** STATUS IS ";RE;
|
|
520 IF RE=19 THEN PRINT " OPENED":GOTO 600
|
|
530 IF RE=0 THEN PRINT " CLOSED, ERROR": GOTO 5000
|
|
540 PRINT "UNKNOWN ERROR ";RE
|
|
550 GOTO 5000
|
|
'
|
|
' LISTEN on the socket
|
|
'
|
|
600 REM *** Connection opened, Listen
|
|
605 POKE HA,4: POKE LA,1: REM *** 0x401 command register
|
|
610 POKE DP, 2: REM *** LISTEN
|
|
'
|
|
' Check return value
|
|
'
|
|
620 REM *** Check if successful
|
|
625 POKE HA,4: POKE LA,3: REM *** 0x403 status register
|
|
630 RE=PEEK(DP)
|
|
635 PRINT "** STATUS IS ";RE;
|
|
640 IF RE=20 THEN PRINT " LISTENING":GOTO 700
|
|
650 IF RE=0 THEN PRINT " CLOSED, ERROR":GOTO 5000
|
|
655 PRINT "UNKNOWN ERROR ";RE
|
|
675 GOTO 5000
|
|
'
|
|
' Wait for incoming connection
|
|
'
|
|
700 REM *** Wait for incoming connection
|
|
705 POKE HA,4: POKE LA,1: REM *** 0x401 command register
|
|
710 POKE DP, 2: REM *** LISTEN
|
|
'
|
|
' Check for result
|
|
'
|
|
720 REM *** Check if successful
|
|
725 POKE HA,4: POKE LA,3: REM *** 0x403 status register
|
|
730 RE=PEEK(DP)
|
|
740 IF RE=23 THEN GOTO 800: REM ESTABLISHED
|
|
745 IF RE<>20 THEN PRINT "WAITING: UNEXPECTED STATUS=";RE
|
|
750 GOTO 700: REM *** Repeat until connected
|
|
'
|
|
' Established, repeat waiting for incoming data
|
|
'
|
|
800 PRINT "ESTABLISHED"
|
|
802 POKE HA,4: POKE LA,38: REM *** 0x426 Received Size
|
|
805 SH=PEEK(DP):SL=PEEK(DP)
|
|
810 SI=(SH*256)+SL
|
|
820 IF SI<>0 THEN GOTO 900
|
|
'
|
|
' Should we delay? busy polling seems wasteful
|
|
'
|
|
830 REM DELAY?
|
|
840 GOTO 802
|
|
'
|
|
' We have some data, let's read it
|
|
'
|
|
900 POKE HA,4: POKE LA,40: REM *** 0x428 Received ptr
|
|
905 OH=PEEK(DP):OL=PEEK(DP)
|
|
910 RF=(OH*256)+OL
|
|
920 REM *** MASK WITH 0x1ff
|
|
925 R%=RF/8192:RM=RF-(8192*R%)
|
|
930 RA=RM+24576:REM $6000
|
|
940 PRINT "READ OFFSET=";RM;" READ ADDRESS=";RA;" READ SIZE=";SI
|
|
'
|
|
' Check for buffer wraparound
|
|
'
|
|
942 BW=0
|
|
945 IF (SI+TA>=32768) THEN BW=1:BO=32768-TA:PRINT "RX BUFFER WRAPAROUND IN ";BO
|
|
'
|
|
' Print received packet
|
|
'
|
|
1000 REM *** PRINT PACKET
|
|
1001 FL=1:FL$=""
|
|
1003 R%=RA/256
|
|
1005 POKE HA,R%: POKE LA,RA-(R%*256)
|
|
1010 FOR I=1 TO SI
|
|
1020 C=PEEK(DP):C$=CHR$(C)
|
|
1025 IF FL=1 THEN FL$=FL$+C$
|
|
1027 IF C=10 THEN FL=0
|
|
1030 IF C<>10 THEN PRINT C$;
|
|
1032 IF BW=0 THEN GOTO 1040
|
|
1033 BO=BO-1: IF BO=0 THEN POKE HA,96:POKE LA,0:BW=0
|
|
1040 NEXT I
|
|
'
|
|
' Deal with first line
|
|
'
|
|
1050 PRINT "FIRST LINE=";FL$
|
|
1060 IF LEFT$(FL$,3)<>"GET" GOTO 7000
|
|
1065 N$=""
|
|
1070 FOR I=6 TO LEN(FL$)
|
|
1075 M$=MID$(FL$,I,1)
|
|
1080 IF M$=" " GOTO 1090
|
|
1085 N$=N$+M$
|
|
1087 NEXT I
|
|
1090 IF N$="" THEN N$="index.html"
|
|
1095 PRINT "SENDING FILE: ";N$
|
|
'
|
|
' TODO: handle wraparound of 8kb buffer
|
|
'
|
|
'
|
|
' Update read pointer
|
|
'
|
|
1100 REM *** Update read pointer
|
|
1110 POKE HA,4: POKE LA,40: REM *** 0x428 Received ptr
|
|
1120 RA=RF+SI
|
|
1130 R%=RA/256
|
|
1140 POKE DP,R%: POKE DP,RA-(R%*256)
|
|
1150 REM *** RECEIVE
|
|
1160 POKE HA,4: POKE LA,1: REM *** 0x401 command register
|
|
1170 POKE DP, 64: REM *** RECV
|
|
'
|
|
' Load file from disk
|
|
'
|
|
1200 REM *** LOAD FILE
|
|
1202 X$=RIGHT$(N$,3):M$="text/html"
|
|
1203 IF X$="txt" THEN M$="text/plain"
|
|
1204 IF X$="png" THEN M$="image/png"
|
|
1205 IF X$="jpg" THEN M$="image/jpg"
|
|
1206 IF X$="ico" THEN M$="image/x-icon"
|
|
1207 IF N$="teapot.html" GOTO 9000
|
|
1208 ONERR GOTO 8000
|
|
1209 PRINT "LOADING ";N$
|
|
1210 PRINT CHR$(4)+"BLOAD ";N$
|
|
1215 POKE 216,0: REM CANCEL ONERR
|
|
1220 FS=PEEK(43616)+256*PEEK(43617): REM FILESIZE
|
|
1225 PRINT "DONE LOADING"
|
|
' assume loaded at 0x4000, text page 2
|
|
' and that max size is 8kb
|
|
1240 A$="HTTP/1.1 200 OK"+CHR$(13)+CHR$(10)
|
|
1250 A$=A$+"Server: VMW-web"+CHR$(13)+CHR$(10)
|
|
1260 A$=A$+"Content-Length: "+STR$(FS)+CHR$(13)+CHR$(10)
|
|
1280 A$=A$+"Content-Type: "+M$+CHR$(13)+CHR$(10)+CHR$(13)+CHR$(10)
|
|
'
|
|
1380 PRINT "SENDING:":PRINT A$
|
|
1385 C=0
|
|
'
|
|
' read TX free size reg (0x420)
|
|
'
|
|
1700 SI=LEN(A$)+FS
|
|
1710 IF (SI>8192) THEN PRINT "FILE TOO BIG!": REM GOTO 403?
|
|
1800 POKE HA,4: POKE LA,32: REM *** 0x420 FREESIZE
|
|
1810 OH=PEEK(DP):OL=PEEK(DP)
|
|
1815 FR=(OH*256)+OL
|
|
1820 PRINT "FREE: ";FR
|
|
1830 IF SI>FR GOTO 1800: REM REPEAT UNTIL FREE
|
|
'
|
|
' Read tx offset
|
|
'
|
|
1900 POKE HA,4: POKE LA,36: REM *** 0x424 TX write ptr
|
|
1905 OH=PEEK(DP):OL=PEEK(DP)
|
|
1910 TF=(OH*256)+OL
|
|
1920 REM *** MASK WITH 0x1ff
|
|
1925 T%=TF/8192:TM=TF-(8192*T%)
|
|
1930 TA=TM+16384:REM $4000
|
|
1940 PRINT "OH/OL=";OH;"/";OL;" TX OFFSET=";TM;" TX ADDRESS=";TA;" TX SIZE=";SI
|
|
'
|
|
' Check for buffer wraparound
|
|
'
|
|
1942 BW=0:BO=0
|
|
1945 IF (SI+TA>=24576) THEN BW=1:BO=24576-TA:PRINT "TX BUFFER WRAPAROUND IN ";BO
|
|
'
|
|
' Write data to TX buffer
|
|
' First write header
|
|
'
|
|
2000 T%=TA/256
|
|
2005 POKE HA,T%: POKE LA,TA-(T%*256)
|
|
2010 FOR I=1 TO LEN(A$)
|
|
2015 POKE DP,ASC(MID$(A$,I,1))
|
|
2017 IF BW=0 THEN GOTO 2020
|
|
2018 BO=BO-1: IF BO=0 THEN POKE HA,64:POKE LA,0:BW=0
|
|
2020 NEXT I
|
|
'
|
|
' Write disk part
|
|
'
|
|
2025 FOR I=1 TO FS
|
|
2026 C=C+1: IF C=50 THEN PRINT ".";:C=0
|
|
2030 POKE DP,PEEK(16383+I)
|
|
2032 IF BW=0 THEN GOTO 2035
|
|
2033 BO=BO-1: IF BO=0 THEN POKE HA,64:POKE LA,0:BW=0
|
|
2035 NEXT I
|
|
2040 PRINT
|
|
'
|
|
' The above is slow
|
|
' Intead use our machine language routine
|
|
'
|
|
'2025 B%=BO/256:POKE 9,B%:POKE 8,BO-(B%*256)
|
|
'2027 B%=FS/256:POKE 11,B%:POKE 10,FS-(B%*256)
|
|
'2030 CALL 768
|
|
'
|
|
' Update TX write ptr
|
|
'
|
|
2050 REM ** UPDATE TX WRITE PTR
|
|
2060 POKE HA,4: POKE LA,36: REM *** 0x424 TX write ptr
|
|
2075 TA=TF+SI
|
|
2080 T%=TA/256
|
|
2085 POKE DP,T%: POKE DP,TA-(T%*256)
|
|
2090 PRINT "UPDATE TX TO ";T%;"/";TA-(T%*256)
|
|
'
|
|
' SEND packet
|
|
'
|
|
2100 REM *** SEND
|
|
2102 PRINT "SENDING"
|
|
2105 POKE HA,4: POKE LA,1: REM *** 0x401 command register
|
|
2110 POKE DP, 32: REM *** SEND
|
|
'
|
|
' Return to reading
|
|
'
|
|
4000 REM *** Check if successful
|
|
4010 POKE HA,4: POKE LA,3: REM *** 0x403 status register
|
|
4020 RE=PEEK(DP)
|
|
4030 PRINT "STATUS AFTER SEND ";RE
|
|
4035 IF RE=28 THEN GOTO 6000: REM CLOSE_WAIT
|
|
4040 IF RE=0 THEN GOTO 400: REM CLOSED
|
|
4060 REM *** RECEIVE
|
|
4075 POKE HA,4: POKE LA,1: REM *** 0x401 command register
|
|
4080 POKE DP, 64: REM *** RECV
|
|
4090 GOTO 800
|
|
'
|
|
' Close the socket
|
|
'
|
|
5000 REM *** CLOSE AND EXIT
|
|
5010 POKE HA,4: POKE LA,1: REM *** 0x401 command register
|
|
5020 POKE DP, 16: REM *** CLOSE
|
|
5030 END
|
|
6000 REM *** CLOSE AND RELISTEN
|
|
6010 POKE HA,4: POKE LA,1: REM *** 0x401 command register
|
|
6020 POKE DP, 16: REM *** CLOSE
|
|
' Check status?
|
|
6030 GOTO 400
|
|
'
|
|
'
|
|
' ERROR MESSAGES
|
|
'
|
|
'
|
|
7000 REM 400 BAD REQUEST
|
|
7005 S$="400 Bad Request"
|
|
7010 M$="<html><head><title>400 Bad Request</title></head><body><h1>Bad Request</h1><p>Your browser sent a request that this server could not understand.<br /></p></body></html>"+CHR$(13)+CHR$(10)
|
|
7020 GOTO 9100
|
|
8000 REM 404 NOT FOUND
|
|
8003 POKE 216,0: REM CANCEL ONERR
|
|
8004 PRINT "DISK ERROR: ";PEEK(222)
|
|
8005 S$="404 Not Found"
|
|
8010 M$="<html><head><title>404 Not Found</title></head><body><h1>Not Found</h1><p>File not found.<br /></p></body></html>"+CHR$(13)+CHR$(10)
|
|
8020 GOTO 9100
|
|
9000 REM 418 TEAPOT
|
|
9005 S$="418 I'm a Teapot"
|
|
9010 M$="<html><head><title>418 I'm a Teapot</title></head><body><h1>I'm a Teapot</h1><p>Short *and* stout.<br /></p></body></html>"+CHR$(13)+CHR$(10)
|
|
'
|
|
' Make header
|
|
'
|
|
9100 A$="HTTP/1.1 "+S$+CHR$(13)+CHR$(10)+"Server: VMW-web"+CHR$(13)+CHR$(10)
|
|
9105 A$=A$+"Content-Length: "+STR$(LEN(M$))+CHR$(13)+CHR$(10)
|
|
9110 A$=A$+"Connection: close"+CHR$(13)+CHR$(10)+"Content-Type: text/html; charset=iso-8859-1"+CHR$(13)+CHR$(10)+CHR$(13)+CHR$(10)
|
|
' Poke as if we had loaded from disk
|
|
9200 FS=LEN(M$)
|
|
9210 FOR I=1 TO FS
|
|
9220 POKE 16383+I,ASC(MID$(M$,I,1))
|
|
9300 NEXT I
|
|
9310 GOTO 1380
|
|
'
|
|
' STATUSES
|
|
' p28 of W5100 manual
|
|
' 0x0 0 SOCK_CLOSED
|
|
' 0x13 SOCK_INIT
|
|
' 0x14 SOCK_LISTEN
|
|
' 0x17 23 SOCK_ESTABLISHED
|
|
' 0x1C 28 SOCK_CLOSE_WAIT
|
|
' 0x22 SOCK_UDP
|
|
' 0x32 SOCK_IPRAW
|
|
' 0x42 SOCK_MACRAW
|
|
' 0x5f SOCK_PPOE
|
|
|
|
'
|
|
' Appendix 1: The memcpy machine code
|
|
'
|
|
'
|
|
'PTR EQU $06
|
|
'PTRH EQU $07
|
|
'
|
|
'WRAPL EQU $08
|
|
'WRAPH EQU $09
|
|
'
|
|
'SIZEL EQU $0A
|
|
'SIZEH EQU $0B
|
|
'
|
|
'tx_copy:
|
|
'
|
|
' lda #0 ; always copying from 0x4000
|
|
' sta PTR
|
|
' lda #$40
|
|
' sta PTR+1
|
|
'
|
|
' ldx #SIZEH ; number of 256-byte blocks
|
|
' beq copy_remainder ; if none, skip ahead
|
|
'
|
|
' ldy #0
|
|
'copy256:
|
|
' lda (PTR),y
|
|
' sta $C0B7 ; change based on uthernet slot
|
|
'
|
|
' cmp WRAPH,x
|
|
' bne nowrap256
|
|
'
|
|
' cmp WRAPL,y
|
|
' bne nowrap256
|
|
'
|
|
' lda #$40
|
|
' sta $C0B5
|
|
' lda #$00
|
|
' sta $C0B6 ; wrap tx buffer address to 0x4000
|
|
'
|
|
'nowrap256:
|
|
' iny
|
|
' bne copy256
|
|
'
|
|
' inc PTR+1 ; update 16-bit pointer
|
|
' dex ; finish a 256 byte block
|
|
' bne copy256
|
|
'
|
|
' ldx #SIZEL
|
|
'copy_remainder:
|
|
' lda (PTR),y
|
|
' sta $C0B7 ; change based on uthernet slot
|
|
'
|
|
' cmp WRAPL,y
|
|
' bne nowrap_r
|
|
'
|
|
' lda #$40
|
|
' sta $C0B5
|
|
' lda #$00
|
|
' sta $C0B6 ; wrap tx buffer address to 0x4000
|
|
'
|
|
'nowrap_r:
|
|
' iny
|
|
' dex
|
|
' bne copy_remainder
|
|
'
|
|
' rts
|
|
|
|
|