2020-10-14 01:17:18 +02:00
|
|
|
%import textio
|
|
|
|
%import syslib
|
|
|
|
|
|
|
|
; Note: this code is compatible with C64 and CX16.
|
|
|
|
|
|
|
|
diskio {
|
|
|
|
|
2020-12-07 23:29:34 +01:00
|
|
|
sub directory(ubyte drivenumber) -> ubyte {
|
2020-12-10 00:02:21 +01:00
|
|
|
; -- Prints the directory contents of disk drive 8-11 to the screen. Returns success.
|
2020-10-14 01:17:18 +02:00
|
|
|
|
|
|
|
c64.SETNAM(1, "$")
|
2020-12-11 22:36:14 +01:00
|
|
|
c64.SETLFS(13, drivenumber, 0)
|
|
|
|
void c64.OPEN() ; open 13,8,0,"$"
|
2020-10-14 01:17:18 +02:00
|
|
|
if_cs
|
|
|
|
goto io_error
|
2020-12-11 22:36:14 +01:00
|
|
|
void c64.CHKIN(13) ; use #13 as input channel
|
2020-10-14 01:17:18 +02:00
|
|
|
if_cs
|
|
|
|
goto io_error
|
|
|
|
|
|
|
|
repeat 4 {
|
|
|
|
void c64.CHRIN() ; skip the 4 prologue bytes
|
|
|
|
}
|
|
|
|
|
|
|
|
; while not key pressed / EOF encountered, read data.
|
|
|
|
ubyte status = c64.READST()
|
|
|
|
while not status {
|
2020-12-08 01:02:38 +01:00
|
|
|
ubyte low = c64.CHRIN()
|
|
|
|
ubyte high = c64.CHRIN()
|
|
|
|
txt.print_uw(mkword(high, low))
|
2020-10-14 01:17:18 +02:00
|
|
|
txt.chrout(' ')
|
|
|
|
ubyte @zp char
|
2020-12-22 03:35:00 +01:00
|
|
|
repeat {
|
2020-10-14 01:17:18 +02:00
|
|
|
char = c64.CHRIN()
|
2020-12-22 05:54:28 +01:00
|
|
|
if char==0
|
2020-12-22 03:35:00 +01:00
|
|
|
break
|
2020-10-14 01:17:18 +02:00
|
|
|
txt.chrout(char)
|
2020-12-22 03:35:00 +01:00
|
|
|
}
|
2020-10-14 01:17:18 +02:00
|
|
|
txt.chrout('\n')
|
|
|
|
void c64.CHRIN() ; skip 2 bytes
|
|
|
|
void c64.CHRIN()
|
|
|
|
status = c64.READST()
|
2020-12-22 03:35:00 +01:00
|
|
|
if c64.STOP2()
|
2020-10-14 01:17:18 +02:00
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
io_error:
|
|
|
|
status = c64.READST()
|
|
|
|
c64.CLRCHN() ; restore default i/o devices
|
2020-12-11 22:36:14 +01:00
|
|
|
c64.CLOSE(13)
|
2020-10-14 01:17:18 +02:00
|
|
|
|
|
|
|
if status and status != 64 { ; 64=end of file
|
|
|
|
txt.print("\ni/o error, status: ")
|
|
|
|
txt.print_ub(status)
|
|
|
|
txt.chrout('\n')
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-12-10 00:58:59 +01:00
|
|
|
; internal variables for the iterative file lister / loader
|
2020-12-08 03:28:29 +01:00
|
|
|
ubyte list_suffixmatch
|
|
|
|
ubyte list_pattern_size
|
|
|
|
ubyte list_skip_disk_name
|
|
|
|
uword list_pattern
|
|
|
|
uword list_blocks
|
2020-12-10 00:02:21 +01:00
|
|
|
ubyte iteration_in_progress = false
|
2020-12-11 22:36:14 +01:00
|
|
|
str list_filename = "?" * 32
|
|
|
|
|
|
|
|
|
|
|
|
; ----- get a list of files (uses iteration functions internally -----
|
|
|
|
|
|
|
|
sub list_files(ubyte drivenumber, uword pattern, ubyte suffixmatch, uword name_ptrs, ubyte max_names) -> ubyte {
|
|
|
|
; -- fill the array 'name_ptrs' with (pointers to) the names of the files requested.
|
|
|
|
ubyte[256] names_buffer
|
2020-12-14 14:29:42 +01:00
|
|
|
ubyte[256] names_buffer1 ; to store a bit more names
|
2020-12-11 22:36:14 +01:00
|
|
|
uword buf_ptr = &names_buffer
|
|
|
|
ubyte files_found = 0
|
|
|
|
if lf_start_list(drivenumber, pattern, suffixmatch) {
|
|
|
|
while lf_next_entry() {
|
|
|
|
@(name_ptrs) = lsb(buf_ptr)
|
|
|
|
name_ptrs++
|
|
|
|
@(name_ptrs) = msb(buf_ptr)
|
|
|
|
name_ptrs++
|
2020-12-14 17:20:16 +01:00
|
|
|
buf_ptr += strcopy(diskio.list_filename, buf_ptr) + 1
|
2020-12-11 22:36:14 +01:00
|
|
|
files_found++
|
2020-12-14 14:29:42 +01:00
|
|
|
if buf_ptr - &names_buffer > (len(names_buffer) + len(names_buffer1) - 18)
|
2020-12-11 22:36:14 +01:00
|
|
|
break
|
|
|
|
if files_found == max_names
|
|
|
|
break
|
|
|
|
}
|
|
|
|
lf_end_list()
|
|
|
|
}
|
|
|
|
return files_found
|
|
|
|
}
|
2020-12-10 00:58:59 +01:00
|
|
|
|
|
|
|
; ----- iterative file lister functions -----
|
|
|
|
|
2020-12-08 03:28:29 +01:00
|
|
|
sub lf_start_list(ubyte drivenumber, uword pattern, ubyte suffixmatch) -> ubyte {
|
|
|
|
; -- start an iterative file listing with optional prefix or suffix name matching.
|
2020-12-11 22:36:14 +01:00
|
|
|
; note: only a single iteration loop can be active at a time!
|
2020-12-08 03:28:29 +01:00
|
|
|
lf_end_list()
|
|
|
|
list_pattern = pattern
|
|
|
|
list_suffixmatch = suffixmatch
|
|
|
|
list_skip_disk_name = true
|
2020-12-10 00:02:21 +01:00
|
|
|
iteration_in_progress = true
|
2020-12-08 03:28:29 +01:00
|
|
|
if pattern==0
|
|
|
|
list_pattern_size = 0
|
|
|
|
else
|
|
|
|
list_pattern_size = strlen(pattern)
|
2020-12-07 23:29:34 +01:00
|
|
|
|
|
|
|
c64.SETNAM(1, "$")
|
2020-12-11 22:36:14 +01:00
|
|
|
c64.SETLFS(12, drivenumber, 0)
|
|
|
|
void c64.OPEN() ; open 12,8,0,"$"
|
2020-12-07 23:29:34 +01:00
|
|
|
if_cs
|
2020-12-08 03:28:29 +01:00
|
|
|
goto io_error
|
2020-12-11 22:36:14 +01:00
|
|
|
void c64.CHKIN(12) ; use #12 as input channel
|
2020-12-07 23:29:34 +01:00
|
|
|
if_cs
|
2020-12-08 03:28:29 +01:00
|
|
|
goto io_error
|
2020-12-07 23:29:34 +01:00
|
|
|
|
|
|
|
repeat 4 {
|
|
|
|
void c64.CHRIN() ; skip the 4 prologue bytes
|
|
|
|
}
|
|
|
|
|
2020-12-08 03:28:29 +01:00
|
|
|
if c64.READST()==0
|
|
|
|
return true
|
|
|
|
|
|
|
|
io_error:
|
|
|
|
lf_end_list()
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
sub lf_next_entry() -> ubyte {
|
|
|
|
; -- retrieve the next entry from an iterative file listing session.
|
|
|
|
; results will be found in list_blocks and list_filename.
|
|
|
|
; if it returns false though, there are no more entries (or an error occurred).
|
|
|
|
|
2020-12-10 00:02:21 +01:00
|
|
|
if not iteration_in_progress
|
2020-12-08 03:28:29 +01:00
|
|
|
return false
|
2020-12-08 01:02:38 +01:00
|
|
|
|
2020-12-11 22:36:14 +01:00
|
|
|
repeat {
|
|
|
|
void c64.CHKIN(12) ; use #12 as input channel again
|
|
|
|
|
2020-12-08 03:28:29 +01:00
|
|
|
uword nameptr = &list_filename
|
|
|
|
ubyte blocks_lsb = c64.CHRIN()
|
|
|
|
ubyte blocks_msb = c64.CHRIN()
|
2020-12-11 22:36:14 +01:00
|
|
|
|
|
|
|
if c64.READST()
|
|
|
|
goto close_end
|
|
|
|
|
2020-12-08 03:28:29 +01:00
|
|
|
list_blocks = mkword(blocks_msb, blocks_lsb)
|
2020-12-07 23:29:34 +01:00
|
|
|
|
|
|
|
; read until the filename starts after the first "
|
|
|
|
while c64.CHRIN()!='\"' {
|
|
|
|
if c64.READST()
|
2020-12-08 01:34:08 +01:00
|
|
|
goto close_end
|
2020-12-07 23:29:34 +01:00
|
|
|
}
|
|
|
|
|
2020-12-08 03:28:29 +01:00
|
|
|
; read the filename
|
2020-12-08 01:02:38 +01:00
|
|
|
repeat {
|
|
|
|
ubyte char = c64.CHRIN()
|
2020-12-22 03:35:00 +01:00
|
|
|
if char==0
|
2020-12-08 01:34:08 +01:00
|
|
|
break
|
2020-12-08 01:02:38 +01:00
|
|
|
if char=='\"'
|
|
|
|
break
|
2020-12-08 03:28:29 +01:00
|
|
|
@(nameptr) = char
|
|
|
|
nameptr++
|
2020-12-08 01:02:38 +01:00
|
|
|
}
|
|
|
|
|
2020-12-08 03:28:29 +01:00
|
|
|
@(nameptr) = 0
|
2020-12-08 01:02:38 +01:00
|
|
|
|
2020-12-08 22:10:12 +01:00
|
|
|
while c64.CHRIN() {
|
|
|
|
; read the rest of the entry until the end
|
|
|
|
}
|
2020-12-08 01:02:38 +01:00
|
|
|
|
2020-12-07 23:29:34 +01:00
|
|
|
void c64.CHRIN() ; skip 2 bytes
|
|
|
|
void c64.CHRIN()
|
2020-12-08 03:28:29 +01:00
|
|
|
|
|
|
|
if not list_skip_disk_name {
|
|
|
|
if list_pattern_size {
|
|
|
|
; do filename matching
|
|
|
|
if list_suffixmatch
|
|
|
|
rightstr(list_filename, filename, list_pattern_size)
|
|
|
|
else
|
|
|
|
leftstr(list_filename, filename, list_pattern_size)
|
2020-12-08 03:39:45 +01:00
|
|
|
if strcmp(filename, list_pattern)==0
|
|
|
|
return true
|
|
|
|
} else
|
2020-12-08 03:28:29 +01:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
list_skip_disk_name = false
|
2020-12-07 23:29:34 +01:00
|
|
|
}
|
|
|
|
|
2020-12-08 01:34:08 +01:00
|
|
|
close_end:
|
2020-12-08 03:28:29 +01:00
|
|
|
lf_end_list()
|
|
|
|
return false
|
2020-12-07 23:29:34 +01:00
|
|
|
}
|
|
|
|
|
2020-12-08 03:28:29 +01:00
|
|
|
sub lf_end_list() {
|
|
|
|
; -- end an iterative file listing session (close channels).
|
2020-12-10 00:02:21 +01:00
|
|
|
if iteration_in_progress {
|
2020-12-08 03:28:29 +01:00
|
|
|
c64.CLRCHN()
|
2020-12-11 22:36:14 +01:00
|
|
|
c64.CLOSE(12)
|
2020-12-10 00:02:21 +01:00
|
|
|
iteration_in_progress = false
|
2020-12-08 03:28:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-12-10 00:58:59 +01:00
|
|
|
; ----- iterative file loader functions -----
|
|
|
|
|
|
|
|
sub f_open(ubyte drivenumber, uword filenameptr) -> ubyte {
|
2020-12-11 22:36:14 +01:00
|
|
|
; -- open a file for iterative reading with f_read
|
|
|
|
; note: only a single iteration loop can be active at a time!
|
2020-12-10 00:58:59 +01:00
|
|
|
f_close()
|
|
|
|
|
|
|
|
c64.SETNAM(strlen(filenameptr), filenameptr)
|
|
|
|
c64.SETLFS(11, drivenumber, 0)
|
|
|
|
void c64.OPEN() ; open 11,8,0,"filename"
|
|
|
|
if_cc {
|
|
|
|
iteration_in_progress = true
|
2020-12-11 22:36:14 +01:00
|
|
|
void c64.CHKIN(11) ; use #11 as input channel
|
2020-12-10 00:58:59 +01:00
|
|
|
if_cc
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
f_close()
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2020-12-24 06:24:52 +01:00
|
|
|
sub f_read(uword bufferpointer, uword num_bytes) -> uword {
|
|
|
|
; -- read from the currently open file, up to the given number of bytes.
|
|
|
|
; returns the actual number of bytes read. (checks for End-of-file and error conditions)
|
2020-12-10 00:58:59 +01:00
|
|
|
if not iteration_in_progress
|
|
|
|
return 0
|
|
|
|
|
|
|
|
uword actual = 0
|
2020-12-11 22:36:14 +01:00
|
|
|
void c64.CHKIN(11) ; use #11 as input channel again
|
2020-12-24 06:24:52 +01:00
|
|
|
repeat num_bytes {
|
2020-12-10 00:58:59 +01:00
|
|
|
ubyte data = c64.CHRIN()
|
|
|
|
@(bufferpointer) = data
|
|
|
|
bufferpointer++
|
|
|
|
actual++
|
|
|
|
ubyte status = c64.READST()
|
|
|
|
if status==64
|
|
|
|
f_close() ; end of file, close it
|
|
|
|
if status
|
|
|
|
return actual
|
|
|
|
}
|
|
|
|
return actual
|
|
|
|
}
|
|
|
|
|
2020-12-24 06:24:52 +01:00
|
|
|
sub f_read_exact(uword bufferpointer, uword num_bytes) {
|
|
|
|
; -- read from the currently open file, the given number of bytes. File must contain enough data!
|
|
|
|
; doesn't check for error conditions or end of file, to make the read as fast as possible.
|
|
|
|
if not iteration_in_progress
|
|
|
|
return
|
|
|
|
|
|
|
|
void c64.CHKIN(11) ; use #11 as input channel again
|
|
|
|
repeat num_bytes {
|
|
|
|
@(bufferpointer) = c64.CHRIN()
|
|
|
|
bufferpointer++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-10 00:58:59 +01:00
|
|
|
sub f_close() {
|
|
|
|
; -- end an iterative file loading session (close channels).
|
|
|
|
if iteration_in_progress {
|
|
|
|
c64.CLRCHN()
|
|
|
|
c64.CLOSE(11)
|
|
|
|
iteration_in_progress = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-12-10 00:02:21 +01:00
|
|
|
sub status(ubyte drivenumber) -> uword {
|
|
|
|
; -- retrieve the disk drive's current status message
|
|
|
|
uword messageptr = &filename
|
2020-12-08 01:02:38 +01:00
|
|
|
c64.SETNAM(0, filename)
|
2020-10-14 01:17:18 +02:00
|
|
|
c64.SETLFS(15, drivenumber, 15)
|
|
|
|
void c64.OPEN() ; open 15,8,15
|
|
|
|
if_cs
|
|
|
|
goto io_error
|
|
|
|
void c64.CHKIN(15) ; use #15 as input channel
|
|
|
|
if_cs
|
|
|
|
goto io_error
|
|
|
|
|
2020-12-10 00:02:21 +01:00
|
|
|
while not c64.READST() {
|
|
|
|
@(messageptr) = c64.CHRIN()
|
|
|
|
messageptr++
|
|
|
|
}
|
2020-10-14 01:17:18 +02:00
|
|
|
|
|
|
|
io_error:
|
2020-12-10 00:02:21 +01:00
|
|
|
@(messageptr) = 0
|
2020-10-14 01:17:18 +02:00
|
|
|
c64.CLRCHN() ; restore default i/o devices
|
|
|
|
c64.CLOSE(15)
|
2020-12-10 00:02:21 +01:00
|
|
|
return filename
|
2020-10-14 01:17:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-12-07 23:29:34 +01:00
|
|
|
sub save(ubyte drivenumber, uword filenameptr, uword address, uword size) -> ubyte {
|
2020-10-14 22:33:49 +02:00
|
|
|
c64.SETNAM(strlen(filenameptr), filenameptr)
|
|
|
|
c64.SETLFS(1, drivenumber, 0)
|
|
|
|
uword end_address = address + size
|
|
|
|
|
|
|
|
%asm {{
|
|
|
|
lda address
|
|
|
|
sta P8ZP_SCRATCH_W1
|
|
|
|
lda address+1
|
|
|
|
sta P8ZP_SCRATCH_W1+1
|
|
|
|
stx P8ZP_SCRATCH_REG
|
|
|
|
lda #<P8ZP_SCRATCH_W1
|
|
|
|
ldx end_address
|
|
|
|
ldy end_address+1
|
|
|
|
jsr c64.SAVE
|
|
|
|
php
|
|
|
|
ldx P8ZP_SCRATCH_REG
|
|
|
|
plp
|
|
|
|
}}
|
|
|
|
|
2020-12-03 16:01:58 +01:00
|
|
|
ubyte result=0
|
2020-10-14 22:33:49 +02:00
|
|
|
if_cc
|
2020-12-03 16:01:58 +01:00
|
|
|
result = c64.READST()==0
|
2020-10-14 22:33:49 +02:00
|
|
|
|
2020-12-03 16:01:58 +01:00
|
|
|
c64.CLRCHN()
|
|
|
|
c64.CLOSE(1)
|
|
|
|
|
|
|
|
return result
|
2020-10-14 01:17:18 +02:00
|
|
|
}
|
|
|
|
|
2020-10-14 22:33:49 +02:00
|
|
|
sub load(ubyte drivenumber, uword filenameptr, uword address_override) -> uword {
|
|
|
|
c64.SETNAM(strlen(filenameptr), filenameptr)
|
|
|
|
ubyte secondary = 1
|
|
|
|
uword end_of_load = 0
|
|
|
|
if address_override
|
|
|
|
secondary = 0
|
|
|
|
c64.SETLFS(1, drivenumber, secondary)
|
|
|
|
%asm {{
|
|
|
|
stx P8ZP_SCRATCH_REG
|
|
|
|
lda #0
|
|
|
|
ldx address_override
|
|
|
|
ldy address_override+1
|
|
|
|
jsr c64.LOAD
|
|
|
|
bcs +
|
|
|
|
stx end_of_load
|
|
|
|
sty end_of_load+1
|
|
|
|
+ ldx P8ZP_SCRATCH_REG
|
|
|
|
}}
|
|
|
|
|
2020-12-03 16:01:58 +01:00
|
|
|
c64.CLRCHN()
|
|
|
|
c64.CLOSE(1)
|
|
|
|
|
2020-10-14 22:33:49 +02:00
|
|
|
if end_of_load
|
|
|
|
return end_of_load - address_override
|
|
|
|
|
|
|
|
return 0
|
|
|
|
}
|
2020-10-14 01:17:18 +02:00
|
|
|
|
|
|
|
|
2020-10-14 22:33:49 +02:00
|
|
|
str filename = "0:??????????????????????????????????????"
|
2020-10-14 01:17:18 +02:00
|
|
|
|
|
|
|
sub delete(ubyte drivenumber, uword filenameptr) {
|
|
|
|
; -- delete a file on the drive
|
|
|
|
ubyte flen = strlen(filenameptr)
|
|
|
|
filename[0] = 's'
|
|
|
|
filename[1] = ':'
|
|
|
|
memcopy(filenameptr, &filename+2, flen+1)
|
|
|
|
c64.SETNAM(flen+2, filename)
|
|
|
|
c64.SETLFS(1, drivenumber, 15)
|
|
|
|
void c64.OPEN()
|
|
|
|
c64.CLRCHN()
|
|
|
|
c64.CLOSE(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
sub rename(ubyte drivenumber, uword oldfileptr, uword newfileptr) {
|
|
|
|
; -- rename a file on the drive
|
|
|
|
ubyte flen_old = strlen(oldfileptr)
|
|
|
|
ubyte flen_new = strlen(newfileptr)
|
|
|
|
filename[0] = 'r'
|
|
|
|
filename[1] = ':'
|
|
|
|
memcopy(newfileptr, &filename+2, flen_new)
|
2020-10-22 23:41:16 +02:00
|
|
|
filename[flen_new+2] = '='
|
2020-10-14 01:17:18 +02:00
|
|
|
memcopy(oldfileptr, &filename+3+flen_new, flen_old+1)
|
|
|
|
c64.SETNAM(3+flen_new+flen_old, filename)
|
|
|
|
c64.SETLFS(1, drivenumber, 15)
|
|
|
|
void c64.OPEN()
|
|
|
|
c64.CLRCHN()
|
|
|
|
c64.CLOSE(1)
|
|
|
|
}
|
|
|
|
}
|