prog8/examples/cx16/imageviewer/iff_module.p8

362 lines
13 KiB
Plaintext
Raw Normal View History

2020-12-13 03:59:02 +01:00
%target cx16
%import graphics
%import textio
%import diskio
2020-12-14 14:30:18 +01:00
iff_module {
ubyte[256] cmap
ubyte[256] cmap1
ubyte[256] cmap2
uword num_colors
uword[16] cycle_rates
uword[16] cycle_rate_ticks
ubyte[16] cycle_reverseflags
ubyte[16] cycle_lows
ubyte[16] cycle_highs
ubyte num_cycles
2020-12-14 15:34:52 +01:00
sub show_image(uword filenameptr) -> ubyte {
2020-12-13 03:59:02 +01:00
ubyte load_ok = false
uword size
ubyte[32] buffer
uword camg = 0
str chunk_id = "????"
uword chunk_size_hi
uword chunk_size_lo
uword scanline_data_ptr = progend()
uword width
uword height
ubyte num_planes
ubyte compression
ubyte have_cmap = false
ubyte cycle_crng = false
ubyte cycle_ccrt = false
2020-12-13 03:59:02 +01:00
cmap[0] = 0
cmap1[0] = 0
cmap2[0] = 0
num_cycles = 0
2020-12-13 03:59:02 +01:00
if diskio.f_open(8, filenameptr) {
size = diskio.f_read(buffer, 12)
if size==12 {
if buffer[0]=='f' and buffer[1]=='o' and buffer[2]=='r' and buffer[3]=='m'
and buffer[8]=='i' and buffer[9]=='l' and buffer[10]=='b' and buffer[11]=='m'{
while read_chunk_header() {
if chunk_id == "bmhd" {
2020-12-24 06:24:52 +01:00
diskio.f_read_exact(buffer, chunk_size_lo)
2020-12-13 03:59:02 +01:00
width = mkword(buffer[0], buffer[1])
height = mkword(buffer[2], buffer[3])
num_planes = buffer[8]
num_colors = 2 ** num_planes
compression = buffer[10]
}
else if chunk_id == "camg" {
2020-12-24 06:24:52 +01:00
diskio.f_read_exact(buffer, chunk_size_lo)
2020-12-13 03:59:02 +01:00
camg = mkword(buffer[2], buffer[3])
if camg & $0800 {
txt.print("ham mode not supported!\n")
break
}
}
else if chunk_id == "cmap" {
have_cmap = true
2020-12-24 06:24:52 +01:00
diskio.f_read_exact(&cmap, chunk_size_lo)
2020-12-13 03:59:02 +01:00
}
else if chunk_id == "crng" {
; DeluxePaint color cycle range
if not cycle_ccrt {
cycle_crng = true
2020-12-24 06:24:52 +01:00
diskio.f_read_exact(buffer, chunk_size_lo)
cycle_rates[num_cycles] = mkword(buffer[2], buffer[3])
cycle_rate_ticks[num_cycles] = 1
cycle_lows[num_cycles] = buffer[6]
cycle_highs[num_cycles] = buffer[7]
cycle_reverseflags[num_cycles] = (buffer[5] & 2)!=0
num_cycles++
} else
skip_chunk()
}
else if chunk_id == "ccrt" {
; Graphicraft color cycle range
if not cycle_crng {
cycle_ccrt = true
2020-12-24 06:24:52 +01:00
diskio.f_read_exact(buffer, chunk_size_lo)
ubyte direction = buffer[1]
if direction {
; delay_sec = buffer[4] * 256 * 256 * 256 + buffer[5] * 256 * 256 + buffer[6] * 256 + buffer[7]
; delay_micro = buffer[8] * 256 * 256 * 256 + buffer[9] * 256 * 256 + buffer[10] * 256 + buffer[11]
; We're ignoring the delay_sec field for now. Not many images will have this slow of a color cycle anyway (>1 sec per cycle)
; rate = int(16384 // (60*delay_micro/1e6))
; float rate = (65*16384.0) / (mkword(buffer[9], buffer[10]) as float) ; fairly good approximation using float arithmetic
cycle_rates[num_cycles] = 33280 / (mkword(buffer[9], buffer[10]) >> 5) ; reasonable approximation using only 16-bit integer arithmetic
cycle_rate_ticks[num_cycles] = 1
cycle_lows[num_cycles] = buffer[2]
cycle_highs[num_cycles] = buffer[3]
cycle_reverseflags[num_cycles] = direction == 1 ; TODO weird, the spec say that -1 = reversed but several example images that I have downloaded are the opposite
num_cycles++
}
} else
skip_chunk()
}
2020-12-13 03:59:02 +01:00
else if chunk_id == "body" {
graphics.clear_screen(1, 0)
if camg & $0004
height /= 2 ; interlaced: just skip every odd scanline later
if camg & $0080 and have_cmap
make_ehb_palette()
2020-12-14 14:30:18 +01:00
palette.set_rgb8(&cmap, num_colors)
2020-12-13 03:59:02 +01:00
if compression
decode_rle()
else
decode_raw()
load_ok = true
break ; done after body
}
else {
skip_chunk()
}
}
} else
txt.print("not an iff ilbm file!\n")
}
diskio.f_close()
}
2020-12-14 15:34:52 +01:00
return load_ok
2020-12-13 03:59:02 +01:00
sub read_chunk_header() -> ubyte {
size = diskio.f_read(buffer, 8)
if size==8 {
chunk_id[0] = buffer[0]
chunk_id[1] = buffer[1]
chunk_id[2] = buffer[2]
chunk_id[3] = buffer[3]
chunk_size_hi = mkword(buffer[4], buffer[5])
chunk_size_lo = mkword(buffer[6], buffer[7])
return true
}
return false
}
sub skip_chunk() {
repeat lsb(chunk_size_hi)*8 + (chunk_size_lo >> 13)
2020-12-24 06:24:52 +01:00
diskio.f_read_exact(scanline_data_ptr, $2000)
2020-12-13 03:59:02 +01:00
2020-12-24 06:24:52 +01:00
diskio.f_read_exact(scanline_data_ptr, chunk_size_lo & $1fff)
2020-12-13 03:59:02 +01:00
}
sub make_ehb_palette() {
; generate 32 additional Extra-Halfbrite colors in the cmap
uword palletteptr = &cmap
uword ehbptr = palletteptr + 32*3
repeat 32 {
@(ehbptr) = @(palletteptr)>>1
ehbptr++
palletteptr++
@(ehbptr) = @(palletteptr)>>1
ehbptr++
palletteptr++
@(ehbptr) = @(palletteptr)>>1
ehbptr++
palletteptr++
}
}
2020-12-23 19:48:44 +01:00
ubyte bitplane_stride
2020-12-13 03:59:02 +01:00
uword interleave_stride
uword offsetx
uword offsety
sub start_plot() {
2020-12-23 19:48:44 +01:00
bitplane_stride = lsb(width>>3)
interleave_stride = (bitplane_stride as uword) * num_planes
2020-12-13 03:59:02 +01:00
offsetx = 0
offsety = 0
if width < graphics.WIDTH
offsetx = (graphics.WIDTH - width - 1) / 2
if height < graphics.HEIGHT
offsety = (graphics.HEIGHT - height - 1) / 2
if width > graphics.WIDTH
width = graphics.WIDTH
if height > graphics.HEIGHT-1
height = graphics.HEIGHT-1
}
sub decode_raw() {
start_plot()
ubyte interlaced = (camg & $0004) != 0
uword y
for y in 0 to height-1 {
2020-12-24 06:24:52 +01:00
diskio.f_read_exact(scanline_data_ptr, interleave_stride)
2020-12-13 03:59:02 +01:00
if interlaced
2020-12-24 06:24:52 +01:00
diskio.f_read_exact(scanline_data_ptr, interleave_stride)
2020-12-21 23:45:26 +01:00
cx16.FB_cursor_position(offsetx, offsety+y)
2020-12-13 03:59:02 +01:00
planar_to_chunky_scanline()
}
}
sub decode_rle() {
start_plot()
ubyte interlaced = (camg & $0004) != 0
uword y
for y in 0 to height-1 {
decode_rle_scanline()
if interlaced
decode_rle_scanline()
2020-12-21 23:45:26 +01:00
cx16.FB_cursor_position(offsetx, offsety+y)
2020-12-13 03:59:02 +01:00
planar_to_chunky_scanline()
}
}
sub decode_rle_scanline() {
uword x = interleave_stride
uword plane_ptr = scanline_data_ptr
while x {
ubyte b = c64.CHRIN()
if b > 128 {
ubyte b2 = c64.CHRIN()
repeat 2+(b^255) {
@(plane_ptr) = b2
plane_ptr++
x--
}
} else if b < 128 {
repeat b+1 {
@(plane_ptr) = c64.CHRIN()
plane_ptr++
x--
}
} else
break
}
}
sub planar_to_chunky_scanline() {
2020-12-23 19:48:44 +01:00
; ubyte[8] masks = [128,64,32,16,8,4,2,1]
2020-12-13 03:59:02 +01:00
uword x
for x in 0 to width-1 {
2020-12-23 19:48:44 +01:00
; ubyte mask = masks[lsb(x) & 7]
2020-12-13 03:59:02 +01:00
uword pixptr = x/8 + scanline_data_ptr
2020-12-23 19:48:44 +01:00
ubyte bits = 0
%asm {{
bra +
_masks .byte 128, 64, 32, 16, 8, 4, 2, 1
+ lda pixptr
sta P8ZP_SCRATCH_W1
lda pixptr+1
sta P8ZP_SCRATCH_W1+1
lda x
2020-12-23 19:48:44 +01:00
and #7
tay
lda _masks,y
sta P8ZP_SCRATCH_B1 ; mask
phx
ldx num_planes
ldy #0
- lda (P8ZP_SCRATCH_W1),y
2020-12-23 19:48:44 +01:00
clc
and P8ZP_SCRATCH_B1
beq +
sec
+ ror bits ; shift planar bit into chunky byte
lda P8ZP_SCRATCH_W1
2020-12-23 19:48:44 +01:00
; clc
adc bitplane_stride
sta P8ZP_SCRATCH_W1
2020-12-23 19:48:44 +01:00
bcc +
inc P8ZP_SCRATCH_W1+1
2020-12-23 19:48:44 +01:00
+ dex
bne -
plx
lda #8
sec
sbc num_planes
beq +
tay
- lsr bits
dey
bne -
+
}}
; the assembly above is the optimized version of this:
; repeat num_planes {
; clear_carry()
; if @(pixptr) & mask
; set_carry()
; ror(bits) ; shift planar bit into chunky byte
; pixptr += bitplane_stride
; }
; bits >>= 8-num_planes
2020-12-13 03:59:02 +01:00
cx16.FB_set_pixel(bits)
}
}
}
sub cycle_colors_each_jiffy() {
if num_cycles==0
return
ubyte changed = false
ubyte ci
for ci in 0 to num_cycles-1 {
cycle_rate_ticks[ci]--
if cycle_rate_ticks[ci]==0 {
changed = true
cycle_rate_ticks[ci] = 16384 / cycle_rates[ci]
do_cycle(cycle_lows[ci], cycle_highs[ci], cycle_reverseflags[ci])
}
}
if changed
palette.set_rgb8(&cmap, num_colors) ; set the new palette
sub do_cycle(uword low, uword high, ubyte reversed) {
low *= 3
high *= 3
uword bytecount = high-low
uword cptr
ubyte red
ubyte green
ubyte blue
if reversed {
cptr = &cmap + low
red = @(cptr)
green = @(cptr+1)
blue = @(cptr+2)
repeat bytecount {
@(cptr) = @(cptr+3)
cptr++
}
@(cptr) = red
cptr++
@(cptr) = green
cptr++
@(cptr) = blue
} else {
cptr = &cmap + high
red = @(cptr)
cptr++
green = @(cptr)
cptr++
blue = @(cptr)
repeat bytecount {
@(cptr) = @(cptr-3)
cptr--
}
@(cptr) = blue
cptr--
@(cptr) = green
cptr--
@(cptr) = red
}
}
}
2020-12-13 03:59:02 +01:00
}