prog8/examples/plasma.p8

118 lines
2.9 KiB
Plaintext
Raw Normal View History

%target c64
%import syslib
2020-11-22 18:17:43 +01:00
%import test_stack
%import textio
2020-08-21 16:53:56 +02:00
; converted from plasma test program for cc65.
; which is (w)2001 by groepaz/hitmen
;
; Cleanup and porting to C by Ullrich von Bassewitz.
; Converted to prog8 by Irmen de Jong.
2020-08-21 16:53:56 +02:00
2020-11-03 21:31:08 +01:00
2020-08-21 16:53:56 +02:00
main {
const uword SCREEN1 = $E000
const uword SCREEN2 = $E400
const uword CHARSET = $E800
const ubyte PAGE1 = ((SCREEN1 >> 6) & $F0) | ((CHARSET >> 10) & $0E)
const ubyte PAGE2 = ((SCREEN2 >> 6) & $F0) | ((CHARSET >> 10) & $0E)
sub start() {
txt.color(1)
2020-08-27 18:10:22 +02:00
txt.print("creating charset...\n")
2020-08-21 16:53:56 +02:00
makechar()
ubyte block = c64.CIA2PRA
2020-08-21 17:44:44 +02:00
; ubyte v = c64.VMCSB
2020-08-21 16:53:56 +02:00
c64.CIA2PRA = (block & $FC) | (lsb(SCREEN1 >> 14) ^ $03)
repeat {
2020-08-21 16:53:56 +02:00
doplasma(SCREEN1)
c64.VMCSB = PAGE1
doplasma(SCREEN2)
c64.VMCSB = PAGE2
}
2020-08-21 17:44:44 +02:00
; restore screen (if you want)
;c64.VMCSB = v
;c64.CIA2PRA = block
2020-08-27 18:10:22 +02:00
;txt.print("done!\n")
2020-11-22 18:17:43 +01:00
;test_stack.test()
;repeat {
;}
2020-08-21 16:53:56 +02:00
}
; several variables outside of doplasma to make them retain their value
ubyte c1A
ubyte c1B
ubyte c2A
ubyte c2B
sub doplasma(uword screen) {
ubyte[40] xbuf
ubyte[25] ybuf
ubyte c1a = c1A
ubyte c1b = c1B
ubyte c2a = c2A
ubyte c2b = c2B
2020-08-21 17:44:44 +02:00
ubyte @zp x
ubyte @zp y
2020-08-21 16:53:56 +02:00
2020-08-25 00:35:51 +02:00
for y in 24 downto 0 {
2020-08-21 17:44:44 +02:00
ybuf[y] = sin8u(c1a) + sin8u(c1b)
2020-08-21 16:53:56 +02:00
c1a += 4
c1b += 9
}
c1A += 3
c1B -= 5
2020-08-25 00:35:51 +02:00
for x in 39 downto 0 {
2020-08-21 17:44:44 +02:00
xbuf[x] = sin8u(c2a) + sin8u(c2b)
2020-08-21 16:53:56 +02:00
c2a += 3
c2b += 7
}
c2A += 2
c2B -= 3
2020-08-25 00:35:51 +02:00
for y in 24 downto 0 {
for x in 39 downto 0 {
; using a temp var here to enable expression optimization that can't be done on a 'problematic' ROM/RAM memory location
ubyte cc = xbuf[x] + ybuf[y]
@(screen) = cc
; this is the fastest way to do this inner part:
2020-08-25 00:59:02 +02:00
; %asm {{
; ldy i
2020-08-25 00:59:02 +02:00
; lda xbuf,y
; ldy ii
2020-08-25 00:59:02 +02:00
; clc
; adc ybuf,y
; ldy #0
; sta (screen),y
2020-08-25 00:59:02 +02:00
; }}
screen++
2020-08-25 00:59:02 +02:00
}
2020-08-21 16:53:56 +02:00
}
}
sub makechar() {
ubyte[8] bittab = [ $01, $02, $04, $08, $10, $20, $40, $80 ]
ubyte c
for c in 0 to 255 {
ubyte @zp s = sin8u(c)
ubyte i
for i in 0 to 7 {
ubyte b=0
ubyte @zp ii
for ii in 0 to 7 {
; use 16 bit rng for a bit more randomness instead of the 8-bit rng
if lsb(rndw()) > s {
b |= bittab[ii]
}
}
@(CHARSET + i + c*$0008) = b
2020-08-21 16:53:56 +02:00
}
}
}
}