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

250 lines
7.3 KiB
Plaintext
Raw Normal View History

2020-09-09 22:49:32 +02:00
%target c64
%import textio
2020-03-25 00:32:54 +01:00
2020-03-28 12:33:16 +01:00
; bitmap pixel graphics module for the C64
; only black/white monchrome 320x200 for now
2020-08-31 21:38:56 +02:00
; assumes bitmap screen memory is $2000-$3fff
2020-03-25 00:32:54 +01:00
2020-03-28 12:33:16 +01:00
graphics {
2020-09-22 21:50:56 +02:00
const uword BITMAP_ADDRESS = $2000
const uword WIDTH = 320
const ubyte HEIGHT = 200
2020-03-25 00:32:54 +01:00
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.
c64.SCROLY = %00111011
2020-11-22 18:17:43 +01:00
c64.SCROLX = %00001000
2020-03-25 00:32:54 +01:00
c64.VMCSB = (c64.VMCSB & %11110000) | %00001000 ; $2000-$3fff
clear_screen(1, 0)
2020-08-22 19:00:03 +02:00
}
2020-11-22 18:17:43 +01:00
sub disable_bitmap_mode() {
; enables text mode, erase the text screen, color white
c64.SCROLY = %00011011
2020-11-22 18:17:43 +01:00
c64.SCROLX = %00001000
c64.VMCSB = (c64.VMCSB & %11110000) | %00000100 ; $1000-$2fff
txt.fill_screen(' ', 1)
}
sub clear_screen(ubyte pixelcolor, ubyte bgcolor) {
2020-09-22 21:50:56 +02:00
memset(BITMAP_ADDRESS, 320*200/8, 0)
2020-08-30 19:31:20 +02:00
txt.fill_screen(pixelcolor << 4 | bgcolor, 0)
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 rewrite this in optimized assembly
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
swap(x1, x2)
swap(y1, y2)
}
word @zp d = 0
ubyte positive_ix = true
word @zp dx = x2-x1 as word
word @zp dy = y2-y1
if dx < 0 {
dx = -dx
positive_ix = false
2020-03-25 00:32:54 +01:00
}
dx *= 2
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++
d += dy
if d > dx {
y1++
d -= dx
2020-03-28 12:33:16 +01:00
}
}
} else {
repeat {
internal_plot(y1)
if internal_plotx==x2
return
internal_plotx--
d += dy
if d > dx {
y1++
d -= dx
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++
d += dx
if d > dy {
internal_plotx++
d -= dy
2020-03-28 12:33:16 +01:00
}
}
} else {
repeat {
internal_plot(y1)
if y1 == y2
return
y1++
d += dx
if d > dy {
internal_plotx--
d -= dy
2020-03-28 12:33:16 +01:00
}
2020-03-26 23:33:55 +01:00
}
}
}
}
2020-03-25 00:32:54 +01:00
2020-03-26 23:33:55 +01:00
sub circle(uword xcenter, ubyte ycenter, ubyte radius) {
2020-03-25 00:32:54 +01:00
; Midpoint algorithm
ubyte @zp ploty
ubyte @zp xx = radius
ubyte @zp yy = 0
byte @zp decisionOver2 = 1-xx as byte
2020-03-27 23:45:01 +01:00
while xx>=yy {
internal_plotx = xcenter + xx
2020-03-27 23:45:01 +01:00
ploty = ycenter + yy
internal_plot(ploty)
internal_plotx = xcenter - xx
internal_plot(ploty)
internal_plotx = xcenter + xx
2020-03-27 23:45:01 +01:00
ploty = ycenter - yy
internal_plot(ploty)
internal_plotx = xcenter - xx
internal_plot(ploty)
internal_plotx = xcenter + yy
2020-03-27 23:45:01 +01:00
ploty = ycenter + xx
internal_plot(ploty)
internal_plotx = xcenter - yy
internal_plot(ploty)
internal_plotx = xcenter + yy
2020-03-27 23:45:01 +01:00
ploty = ycenter - xx
internal_plot(ploty)
internal_plotx = xcenter - yy
internal_plot(ploty)
2020-03-27 23:45:01 +01:00
yy++
2020-03-25 00:32:54 +01:00
if decisionOver2<=0
2020-03-27 23:45:01 +01:00
decisionOver2 += 2*yy+1
2020-03-25 00:32:54 +01:00
else {
2020-03-27 23:45:01 +01:00
xx--
decisionOver2 += 2*(yy-xx)+1
2020-03-25 00:32:54 +01:00
}
}
}
sub disc(uword xcenter, ubyte ycenter, ubyte radius) {
; Midpoint algorithm, filled
2020-03-27 23:45:01 +01:00
ubyte xx = radius
ubyte yy = 0
byte decisionOver2 = 1-xx as byte
2020-03-27 23:45:01 +01:00
while xx>=yy {
ubyte ycenter_plus_yy = ycenter + yy
ubyte ycenter_min_yy = ycenter - yy
ubyte ycenter_plus_xx = ycenter + xx
ubyte ycenter_min_xx = ycenter - xx
2020-04-08 03:38:22 +02:00
internal_plotx = xcenter-xx
repeat xx*2+1 {
internal_plot(ycenter_plus_yy)
internal_plot(ycenter_min_yy)
internal_plotx++
}
internal_plotx = xcenter-yy
repeat yy*2+1 {
internal_plot(ycenter_plus_xx)
internal_plot(ycenter_min_xx)
internal_plotx++
}
2020-03-27 23:45:01 +01:00
yy++
if decisionOver2<=0
2020-03-27 23:45:01 +01:00
decisionOver2 += 2*yy+1
else {
2020-03-27 23:45:01 +01:00
xx--
decisionOver2 += 2*(yy-xx)+1
}
}
}
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]
2020-09-22 21:50:56 +02:00
; uword addr = BITMAP_ADDRESS + 320*(py>>3) + (py & 7) + (px & %0000000111111000)
2020-03-28 12:33:16 +01:00
; @(addr) |= ormask[lsb(px) & 7]
; }
asmsub plot(uword plotx @XY, ubyte ploty @A) clobbers (A, X, Y) {
%asm {{
stx internal_plotx
sty internal_plotx+1
jmp 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:
uword 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
lda internal_plotx+1
sta P8ZP_SCRATCH_W2+1
2020-03-27 23:45:01 +01:00
lsr a ; 0
sta P8ZP_SCRATCH_W2
lda 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.
_plot_y_values := $2000 + 320*(range(200)>>3) + (range(200) & 7)
_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
}
}