prog8/compiler/res/prog8lib/c64/graphics.p8

390 lines
12 KiB
Plaintext
Raw Normal View History

2020-03-28 12:33:16 +01:00
; bitmap pixel graphics module for the C64
2021-03-04 01:31:29 +01:00
; only black/white monochrome 320x200 for now
;
; NOTE: For sake of speed, NO BOUNDS CHECKING is performed in most routines!
; You'll have to make sure yourself that you're not writing outside of bitmap boundaries!
;
; Sets character matrix and bitmap screen memory at a higher memory location $5c00-$7fff
; so that the program itself can be larger without starting to overwrite the graphics memory.
2020-03-25 00:32:54 +01:00
2020-03-28 12:33:16 +01:00
graphics {
2023-12-29 22:21:44 +01:00
%option ignore_unused
2023-06-30 00:29:50 +02:00
2020-09-22 21:50:56 +02:00
const uword WIDTH = 320
const ubyte HEIGHT = 200
2020-03-25 00:32:54 +01:00
const uword BITMAP_ADDRESS = $6000 ; MUST BE IN REGULAR RAM if you are not messing with ROM/RAM banking. (and $2000-aligned)
; note: this constant is also used in a asm multiplication table below!
const uword CHARS_ADDRESS = $5c00 ; must be in same vic memory bank as the bitmap.
2020-03-28 12:33:16 +01:00
sub enable_bitmap_mode() {
2020-03-25 00:32:54 +01:00
; enable bitmap screen, erase it and set colors to black/white.
clear_screen(1, 0)
c64.SCROLY = %00111011 ; enable bitmap graphics mode
c64.SCROLX = %00001000 ; 40 column mode, no scrolling, multicolor bitmap off
c64.VMCSB = (lsb(CHARS_ADDRESS >> 6) & $F0) | (lsb((BITMAP_ADDRESS & $3fff) / $0800) << 1) ; set bitmap address
c64.CIA2DDRA |= %11
c64.CIA2PRA = lsb(BITMAP_ADDRESS >> 14) ^ 3 ; set VIC bank.
2020-08-22 19:00:03 +02:00
}
2020-11-22 18:17:43 +01:00
sub disable_bitmap_mode() {
; enables erase the text screen, text mode
sys.memset(CHARS_ADDRESS, 40*25, sc:' ')
c64.SCROLY = %00011011 ; disable bitmap graphics mode
c64.SCROLX = %00001000 ; 40 column mode, no scrolling
c64.VMCSB = %00010100 ; screen addresses back to defaults
c64.CIA2DDRA |= %11
c64.CIA2PRA = %11 ; back to VIC bank 0.
2020-11-22 18:17:43 +01:00
}
sub clear_screen(ubyte pixelcolor, ubyte bgcolor) {
sys.memset(BITMAP_ADDRESS, 320*200/8, 0)
sys.memset(CHARS_ADDRESS, 40*25, pixelcolor << 4 | bgcolor)
sys.memset(cbm.Colors, 40*25, 255)
2020-03-25 00:32:54 +01:00
}
sub line(uword @zp x1, ubyte @zp y1, uword @zp x2, ubyte @zp y2) {
; Bresenham algorithm.
2020-12-21 18:28:10 +01:00
; This code special-cases various quadrant loops to allow simple ++ and -- operations.
; TODO implement this as optimized assembly, for instance https://github.com/EgonOlsen71/bresenham/blob/main/src/asm/graphics.asm ??
; or from here https://retro64.altervista.org/blog/an-introduction-to-vector-based-graphics-the-commodore-64-rotating-simple-3d-objects/
if y1>y2 {
2020-12-21 18:28:10 +01:00
; make sure dy is always positive to have only 4 instead of 8 special cases
cx16.r0 = x1
x1 = x2
x2 = cx16.r0
cx16.r0L = y1
y1 = y2
y2 = cx16.r0L
}
word @zp dx = (x2 as word)-x1
word @zp dy = (y2 as word)-y1
2020-12-30 00:14:35 +01:00
if dx==0 {
vertical_line(x1, y1, abs(dy) as ubyte +1)
2020-12-30 00:14:35 +01:00
return
}
if dy==0 {
if x1>x2
x1=x2
horizontal_line(x1, y1, abs(dx) as uword +1)
2020-12-30 00:14:35 +01:00
return
}
word @zp d = 0
ubyte positive_ix = true
if dx < 0 {
dx = -dx
positive_ix = false
2020-03-25 00:32:54 +01:00
}
2021-03-09 20:44:06 +01:00
word @zp dx2 = dx*2
word @zp dy2 = dy*2
internal_plotx = x1
2020-03-26 23:33:55 +01:00
if dx >= dy {
if positive_ix {
repeat {
internal_plot(y1)
if internal_plotx==x2
return
internal_plotx++
2021-03-09 20:44:06 +01:00
d += dy2
if d > dx {
y1++
2021-03-09 20:44:06 +01:00
d -= dx2
2020-03-28 12:33:16 +01:00
}
}
} else {
repeat {
internal_plot(y1)
if internal_plotx==x2
return
internal_plotx--
2021-03-09 20:44:06 +01:00
d += dy2
if d > dx {
y1++
2021-03-09 20:44:06 +01:00
d -= dx2
2020-03-28 12:33:16 +01:00
}
2020-03-26 23:33:55 +01:00
}
}
}
else {
if positive_ix {
repeat {
internal_plot(y1)
if y1 == y2
return
y1++
2021-03-09 20:44:06 +01:00
d += dx2
if d > dy {
internal_plotx++
2021-03-09 20:44:06 +01:00
d -= dy2
2020-03-28 12:33:16 +01:00
}
}
} else {
repeat {
internal_plot(y1)
if y1 == y2
return
y1++
2021-03-09 20:44:06 +01:00
d += dx2
if d > dy {
internal_plotx--
2021-03-09 20:44:06 +01:00
d -= dy2
2020-03-28 12:33:16 +01:00
}
2020-03-26 23:33:55 +01:00
}
}
}
}
2020-03-25 00:32:54 +01:00
sub rect(uword xx, ubyte yy, uword width, ubyte height) {
2020-12-30 00:14:35 +01:00
if width==0 or height==0
return
horizontal_line(xx, yy, width)
2020-12-30 00:14:35 +01:00
if height==1
return
horizontal_line(xx, yy+height-1, width)
vertical_line(xx, yy+1, height-2)
2020-12-30 00:14:35 +01:00
if width==1
return
vertical_line(xx+width-1, yy+1, height-2)
2020-12-30 00:06:01 +01:00
}
sub fillrect(uword xx, ubyte yy, uword width, ubyte height) {
2020-12-30 00:14:35 +01:00
if width==0
return
repeat height {
horizontal_line(xx, yy, width)
yy++
2020-12-30 00:14:35 +01:00
}
2020-12-30 00:06:01 +01:00
}
sub horizontal_line(uword xx, ubyte yy, uword length) {
2020-12-30 18:02:46 +01:00
if length<8 {
internal_plotx=xx
2020-12-30 18:02:46 +01:00
repeat lsb(length) {
internal_plot(yy)
2020-12-30 18:02:46 +01:00
internal_plotx++
}
2020-12-30 00:06:01 +01:00
return
2020-12-30 18:02:46 +01:00
}
2020-12-30 00:06:01 +01:00
ubyte separate_pixels = lsb(xx) & 7
uword pixaddr = get_y_lookup(yy) + (xx&$fff8)
2020-12-30 18:02:46 +01:00
if separate_pixels {
%asm {{
2023-12-29 22:21:44 +01:00
lda p8v_pixaddr
2020-12-30 18:02:46 +01:00
sta P8ZP_SCRATCH_W1
2023-12-29 22:21:44 +01:00
lda p8v_pixaddr+1
2020-12-30 18:02:46 +01:00
sta P8ZP_SCRATCH_W1+1
2023-12-29 22:21:44 +01:00
ldy p8v_separate_pixels
lda hline_filled_right,y
2020-12-30 18:02:46 +01:00
eor #255
ldy #0
ora (P8ZP_SCRATCH_W1),y
sta (P8ZP_SCRATCH_W1),y
}}
pixaddr += 8
2020-12-30 18:02:46 +01:00
length += separate_pixels
length -= 8
}
if length {
%asm {{
2023-12-29 22:21:44 +01:00
lda p8v_length
2020-12-30 18:02:46 +01:00
and #7
2023-12-29 22:21:44 +01:00
sta p8v_separate_pixels
lsr p8v_length+1
ror p8v_length
lsr p8v_length+1
ror p8v_length
lsr p8v_length+1
ror p8v_length
lda p8v_pixaddr
2020-12-30 18:02:46 +01:00
sta _modified+1
2023-12-29 22:21:44 +01:00
lda p8v_pixaddr+1
2020-12-30 18:02:46 +01:00
sta _modified+2
2023-12-29 22:21:44 +01:00
lda p8v_length
ora p8v_length+1
2020-12-30 18:02:46 +01:00
beq _zero
2023-12-29 22:21:44 +01:00
ldy p8v_length
2020-12-30 18:02:46 +01:00
ldx #$ff
_modified stx $ffff ; modified
lda _modified+1
clc
adc #8
sta _modified+1
bcc +
inc _modified+2
+ dey
bne _modified
_zero
2023-12-29 22:21:44 +01:00
ldy p8v_separate_pixels
beq hline_zero2
2020-12-30 18:02:46 +01:00
lda _modified+1
sta P8ZP_SCRATCH_W1
lda _modified+2
sta P8ZP_SCRATCH_W1+1
lda hline_filled_right,y
2020-12-30 18:02:46 +01:00
ldy #0
ora (P8ZP_SCRATCH_W1),y
sta (P8ZP_SCRATCH_W1),y
jmp hline_zero2
hline_filled_right .byte 0, %10000000, %11000000, %11100000, %11110000, %11111000, %11111100, %11111110
hline_zero2
2020-12-30 18:02:46 +01:00
}}
2020-12-30 00:06:01 +01:00
}
}
sub vertical_line(uword xx, ubyte yy, ubyte height) {
internal_plotx = xx
2020-12-30 00:06:01 +01:00
repeat height {
internal_plot(yy)
yy++
2020-12-30 00:06:01 +01:00
}
}
2020-03-26 23:33:55 +01:00
sub circle(uword xcenter, ubyte ycenter, ubyte radius) {
; Warning: NO BOUNDS CHECKS. Make sure circle fits in the screen.
; Midpoint algorithm.
2020-12-27 17:34:25 +01:00
if radius==0
return
ubyte @zp ploty
ubyte @zp yy = 0
2020-12-30 00:06:01 +01:00
word @zp decisionOver2 = (1 as word)-radius
2020-03-27 23:45:01 +01:00
2020-12-30 00:06:01 +01:00
while radius>=yy {
internal_plotx = xcenter + radius
2020-03-27 23:45:01 +01:00
ploty = ycenter + yy
internal_plot(ploty)
2020-12-30 00:06:01 +01:00
internal_plotx = xcenter - radius
internal_plot(ploty)
2020-12-30 00:06:01 +01:00
internal_plotx = xcenter + radius
2020-03-27 23:45:01 +01:00
ploty = ycenter - yy
internal_plot(ploty)
2020-12-30 00:06:01 +01:00
internal_plotx = xcenter - radius
internal_plot(ploty)
internal_plotx = xcenter + yy
2020-12-30 00:06:01 +01:00
ploty = ycenter + radius
internal_plot(ploty)
internal_plotx = xcenter - yy
internal_plot(ploty)
internal_plotx = xcenter + yy
2020-12-30 00:06:01 +01:00
ploty = ycenter - radius
internal_plot(ploty)
internal_plotx = xcenter - yy
internal_plot(ploty)
2020-03-27 23:45:01 +01:00
yy++
if decisionOver2>=0 {
2020-12-30 00:06:01 +01:00
radius--
decisionOver2 -= radius*$0002
2020-03-25 00:32:54 +01:00
}
decisionOver2 += yy*$0002
decisionOver2++
2020-03-25 00:32:54 +01:00
}
}
sub disc(uword xcenter, ubyte ycenter, ubyte radius) {
; Warning: NO BOUNDS CHECKS. Make sure circle fits in the screen.
; Midpoint algorithm, filled
2020-12-27 17:34:25 +01:00
if radius==0
return
ubyte @zp yy = 0
2020-12-30 00:06:01 +01:00
word decisionOver2 = (1 as word)-radius
2020-03-27 23:45:01 +01:00
2020-12-30 00:06:01 +01:00
while radius>=yy {
2020-12-30 00:14:35 +01:00
horizontal_line(xcenter-radius, ycenter+yy, radius*2+1)
horizontal_line(xcenter-radius, ycenter-yy, radius*2+1)
horizontal_line(xcenter-yy, ycenter+radius, yy*2+1)
horizontal_line(xcenter-yy, ycenter-radius, yy*2+1)
2020-03-27 23:45:01 +01:00
yy++
if decisionOver2>=0 {
2020-12-30 00:06:01 +01:00
radius--
decisionOver2 -= radius*$0002
}
decisionOver2 += yy*$0002
decisionOver2++
}
}
2020-03-25 00:32:54 +01:00
2020-03-27 23:45:01 +01:00
2020-03-28 12:33:16 +01:00
; here is the non-asm code for the plot routine below:
; sub plot_nonasm(uword px, ubyte py) {
; ubyte[] ormask = [128, 64, 32, 16, 8, 4, 2, 1]
; uword pixaddr = BITMAP_ADDRESS + 320*(py>>3) + (py & 7) + (px & %0000000111111000)
; @(pixaddr) |= ormask[lsb(px) & 7]
2020-03-28 12:33:16 +01:00
; }
inline asmsub plot(uword plotx @XY, ubyte ploty @A) clobbers (A, X, Y) {
%asm {{
2023-12-29 22:21:44 +01:00
stx p8b_graphics.p8v_internal_plotx
sty p8b_graphics.p8v_internal_plotx+1
jsr p8b_graphics.p8s_internal_plot
}}
}
; for efficiency of internal algorithms here is the internal plot routine
; that takes the plotx coordinate in a separate variable instead of the XY register pair:
2022-08-04 18:28:33 +02:00
uword @zp internal_plotx ; 0..319 ; separate 'parameter' for internal_plot()
2020-03-27 23:45:01 +01:00
asmsub internal_plot(ubyte ploty @A) clobbers (A, X, Y) { ; internal_plotx is 16 bits 0 to 319... doesn't fit in a register
2020-03-27 23:45:01 +01:00
%asm {{
tay
2023-12-29 22:21:44 +01:00
lda p8v_internal_plotx+1
sta P8ZP_SCRATCH_W2+1
2020-03-27 23:45:01 +01:00
lsr a ; 0
sta P8ZP_SCRATCH_W2
2023-12-29 22:21:44 +01:00
lda p8v_internal_plotx
2020-03-27 23:45:01 +01:00
pha
and #7
tax
lda _y_lookup_lo,y
clc
adc P8ZP_SCRATCH_W2
sta P8ZP_SCRATCH_W2
2020-03-27 23:45:01 +01:00
lda _y_lookup_hi,y
adc P8ZP_SCRATCH_W2+1
sta P8ZP_SCRATCH_W2+1
2020-03-27 23:45:01 +01:00
pla ; internal_plotx
2020-03-27 23:45:01 +01:00
and #%11111000
tay
lda (P8ZP_SCRATCH_W2),y
2020-03-27 23:45:01 +01:00
ora _ormask,x
sta (P8ZP_SCRATCH_W2),y
2020-03-27 23:45:01 +01:00
rts
_ormask .byte 128, 64, 32, 16, 8, 4, 2, 1
; note: this can be even faster if we also have a 256 byte x-lookup table, but hey.
; see http://codebase64.org/doku.php?id=base:various_techniques_to_calculate_adresses_fast_common_screen_formats_for_pixel_graphics
2020-09-22 21:50:56 +02:00
; the y lookup tables encodes this formula: BITMAP_ADDRESS + 320*(py>>3) + (py & 7) (y from 0..199)
2020-08-31 21:38:56 +02:00
; We use the 64tass syntax for range expressions to calculate this table on assembly time.
2023-12-29 22:21:44 +01:00
_plot_y_values := p8c_BITMAP_ADDRESS + 320*(range(200)>>3) + (range(200) & 7)
2020-08-31 21:38:56 +02:00
_y_lookup_lo .byte <_plot_y_values
_y_lookup_hi .byte >_plot_y_values
2020-03-27 23:45:01 +01:00
}}
2020-03-25 00:32:54 +01:00
}
asmsub get_y_lookup(ubyte yy @Y) -> uword @AY {
2020-12-30 00:14:35 +01:00
%asm {{
2023-12-29 22:21:44 +01:00
lda p8s_internal_plot._y_lookup_lo,y
2020-12-30 00:14:35 +01:00
pha
2023-12-29 22:21:44 +01:00
lda p8s_internal_plot._y_lookup_hi,y
2020-12-30 00:14:35 +01:00
tay
pla
rts
}}
}
2020-03-25 00:32:54 +01:00
}