mirror of
https://github.com/irmen/prog8.git
synced 2025-01-17 07:30:15 +00:00
207 lines
7.0 KiB
Lua
207 lines
7.0 KiB
Lua
%import syslib
|
|
%import string
|
|
|
|
; Streaming routine for MCF files (multipurpose chunk format):
|
|
; 1. call open()
|
|
; 2. set callbacks if needed; set_callbacks()
|
|
; 3. set bonk ram (cartridge ram) load buffer, if needed; set_bonkbuffer()
|
|
; 4. call stream() in a loop
|
|
; 5. call close() if you want to cleanup halfway through for some reason
|
|
|
|
|
|
mcf {
|
|
uword loadlist_buf = memory("loadlist", 256, 0)
|
|
uword @zp loadlist_ptr
|
|
uword bonkbuffer
|
|
bool needs_loadlist
|
|
ubyte file_channel
|
|
|
|
sub open(str filename, ubyte drive, ubyte channel) -> bool {
|
|
file_channel = channel
|
|
cbm.SETNAM(string.length(filename), filename)
|
|
cbm.SETLFS(channel, drive, 2)
|
|
void cbm.OPEN()
|
|
if_cc {
|
|
if cbm.READST()==0 {
|
|
void cbm.CHKIN(channel)
|
|
loadlist_ptr = loadlist_buf
|
|
needs_loadlist = true
|
|
return true
|
|
}
|
|
}
|
|
close()
|
|
return false
|
|
}
|
|
|
|
sub close() {
|
|
cbm.CLRCHN()
|
|
cbm.CLOSE(file_channel)
|
|
}
|
|
|
|
asmsub set_callbacks(uword getbuffer_routine @R0, uword processchunk_routine @R1) {
|
|
%asm {{
|
|
lda cx16.r0
|
|
ldy cx16.r0+1
|
|
sta p8b_mcf.p8s_stream.getbuffer_call+1
|
|
sty p8b_mcf.p8s_stream.getbuffer_call+2
|
|
lda cx16.r1
|
|
ldy cx16.r1+1
|
|
sta p8b_mcf.p8s_stream.processchunk_call+1
|
|
sty p8b_mcf.p8s_stream.processchunk_call+2
|
|
rts
|
|
}}
|
|
}
|
|
|
|
sub set_bonkbuffer(uword buffer) {
|
|
; needs to be a buffer of 256 bytes (1 page)
|
|
bonkbuffer = buffer
|
|
}
|
|
|
|
sub stream() {
|
|
repeat {
|
|
if needs_loadlist {
|
|
if not read_loadlist() {
|
|
sys.set_carry()
|
|
return
|
|
}
|
|
needs_loadlist = false
|
|
}
|
|
|
|
while (loadlist_ptr-loadlist_buf)<256 {
|
|
when @(loadlist_ptr) {
|
|
255 -> {
|
|
; simply ignore this byte
|
|
loadlist_ptr++
|
|
}
|
|
254 -> {
|
|
; End of File
|
|
close()
|
|
sys.set_carry()
|
|
return
|
|
}
|
|
253 -> {
|
|
; pause streaming
|
|
cx16.r0 = peekw(loadlist_ptr+1)
|
|
loadlist_ptr+=6
|
|
sys.clear_carry()
|
|
return
|
|
}
|
|
252 -> {
|
|
; load into vram
|
|
blockload_vram(peekw(loadlist_ptr+1), peek(loadlist_ptr+3), peekw(loadlist_ptr+4))
|
|
loadlist_ptr+=6
|
|
}
|
|
251 -> {
|
|
; load into system ram
|
|
blockload_ram(peekw(loadlist_ptr+1), peek(loadlist_ptr+3), peekw(loadlist_ptr+4))
|
|
loadlist_ptr+=6
|
|
}
|
|
250 -> {
|
|
; bonk ram (cartridge ram)
|
|
; This cannot use MACPTR (because the kernal rom isn't banked in)
|
|
; so we have to load it into a buffer and copy it manually.
|
|
; Because this will be rarely used, the buffer is not allocated here to save memory, and instead
|
|
; the user has to set it with the config routine when the program wants to use this chunk type.
|
|
blockload_bonkram(peekw(loadlist_ptr+1), peek(loadlist_ptr+3), peekw(loadlist_ptr+4))
|
|
loadlist_ptr+=6
|
|
}
|
|
249 -> {
|
|
; dummy chunk
|
|
blockload_dummy(peekw(loadlist_ptr+1))
|
|
loadlist_ptr+=6
|
|
}
|
|
else -> {
|
|
; custom chunk
|
|
uword @shared chunksize = peekw(loadlist_ptr+1)
|
|
%asm {{
|
|
lda (p8b_mcf.p8v_loadlist_ptr)
|
|
ldx p8v_chunksize
|
|
ldy p8v_chunksize+1
|
|
getbuffer_call jsr $ffff ; modified
|
|
bcc +
|
|
rts ; fail - exit as if EOF
|
|
+ sta cx16.r1L
|
|
stx cx16.r0
|
|
sty cx16.r0+1
|
|
}}
|
|
blockload_ram(chunksize, cx16.r1L, cx16.r0)
|
|
loadlist_ptr += 6
|
|
%asm {{
|
|
processchunk_call jsr $ffff ; modified
|
|
bcc +
|
|
rts
|
|
+
|
|
}}
|
|
}
|
|
}
|
|
}
|
|
needs_loadlist = true
|
|
}
|
|
}
|
|
|
|
sub read_loadlist() -> bool {
|
|
blockload_ram(256, cx16.getrambank(), loadlist_buf)
|
|
if loadlist_buf[0]!=sc:'L' or loadlist_buf[1]!=sc:'L' or loadlist_buf[2]!=1
|
|
return false ; header error
|
|
loadlist_ptr = loadlist_buf+4
|
|
}
|
|
|
|
sub blockload_vram(uword size, ubyte bank, uword address) {
|
|
cx16.VERA_CTRL = 0
|
|
cx16.VERA_ADDR_L = lsb(address)
|
|
cx16.VERA_ADDR_M = msb(address)
|
|
cx16.VERA_ADDR_H = bank | %00010000 ; enable vera auto increment
|
|
while size!=0 {
|
|
size -= readblock(size, &cx16.VERA_DATA0, true)
|
|
}
|
|
}
|
|
|
|
sub blockload_dummy(uword size) {
|
|
ubyte buffer
|
|
while size!=0 {
|
|
size -= readblock(size, &buffer, true)
|
|
}
|
|
}
|
|
|
|
sub blockload_ram(uword size, ubyte bank, uword address) {
|
|
ubyte orig_ram_bank = cx16.getrambank()
|
|
cx16.rambank(bank)
|
|
cx16.r3 = address
|
|
while size!=0 {
|
|
cx16.r2 = readblock(size, cx16.r3, false)
|
|
size -= cx16.r2
|
|
cx16.r3 += cx16.r2
|
|
}
|
|
cx16.rambank(orig_ram_bank)
|
|
}
|
|
|
|
sub blockload_bonkram(uword size, ubyte bonk, uword address) {
|
|
ubyte orig_rom_bank = cx16.getrombank()
|
|
cx16.r3 = address
|
|
while size!=0 {
|
|
ubyte readsize = 255
|
|
if msb(size)==0
|
|
readsize = lsb(size)
|
|
void, cx16.r2 = cx16.MACPTR(readsize, bonkbuffer, false) ; can't MACPTR directly to bonk ram
|
|
cx16.rombank(bonk)
|
|
sys.memcopy(bonkbuffer, cx16.r3, cx16.r2) ; copy to bonk ram
|
|
cx16.rombank(orig_rom_bank)
|
|
size -= cx16.r2
|
|
cx16.r3 += cx16.r2
|
|
}
|
|
}
|
|
|
|
sub readblock(uword size, uword address, bool dontAdvance) -> uword {
|
|
if msb(size)>=2 {
|
|
void, cx16.r0 = cx16.MACPTR(0, address, dontAdvance) ; read 512 bytes
|
|
return cx16.r0
|
|
}
|
|
if msb(size)!=0 {
|
|
void, cx16.r0 = cx16.MACPTR(255, address, dontAdvance) ; read 255 bytes
|
|
return cx16.r0
|
|
}
|
|
void, cx16.r0 = cx16.MACPTR(lsb(size), address, dontAdvance) ; read remaining number of bytes
|
|
return cx16.r0
|
|
}
|
|
}
|