2022-07-09 15:40:56 +02:00
|
|
|
%import textio
|
2022-10-22 16:35:57 +02:00
|
|
|
%import math
|
2022-07-09 15:40:56 +02:00
|
|
|
|
|
|
|
; Even though prog8 only has support for extremely limited recursion,
|
|
|
|
; you can write recursive algorithms with a bit of extra work by building your own explicit stack structure.
|
|
|
|
; This program shows a depth-first maze generation algorithm (1 possible path from start to finish),
|
|
|
|
; and a depth-first maze solver algorithm, both using a stack to store the path taken.
|
|
|
|
|
2023-06-03 19:14:45 +02:00
|
|
|
; Note: this program can be compiled for multiple target systems.
|
2022-07-09 15:40:56 +02:00
|
|
|
|
|
|
|
main {
|
|
|
|
sub start() {
|
|
|
|
repeat {
|
|
|
|
maze.initialize()
|
|
|
|
maze.drawStartFinish()
|
|
|
|
maze.generate()
|
2023-08-29 11:50:35 +02:00
|
|
|
maze.openpassages()
|
2022-07-09 15:40:56 +02:00
|
|
|
maze.drawStartFinish()
|
|
|
|
maze.solve()
|
|
|
|
maze.drawStartFinish()
|
|
|
|
|
|
|
|
txt.print(" enter=new maze")
|
2023-04-28 23:13:03 +02:00
|
|
|
void cbm.CHRIN()
|
2022-07-09 15:40:56 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
maze {
|
|
|
|
const uword screenwidth = txt.DEFAULT_WIDTH
|
|
|
|
const uword screenheight = txt.DEFAULT_HEIGHT
|
|
|
|
|
|
|
|
const ubyte numCellsHoriz = (screenwidth-1) / 2
|
|
|
|
const ubyte numCellsVert = (screenheight-1) / 2
|
|
|
|
|
|
|
|
; maze start and finish cells
|
|
|
|
const ubyte startCx = 0
|
|
|
|
const ubyte startCy = 0
|
|
|
|
const ubyte finishCx = numCellsHoriz-1
|
|
|
|
const ubyte finishCy = numCellsVert-1
|
|
|
|
|
|
|
|
; cell properties
|
|
|
|
const ubyte STONE = 128
|
|
|
|
const ubyte WALKED = 64
|
|
|
|
const ubyte BACKTRACKED = 32
|
|
|
|
const ubyte UP = 1
|
|
|
|
const ubyte RIGHT = 2
|
|
|
|
const ubyte DOWN = 4
|
|
|
|
const ubyte LEFT = 8
|
|
|
|
const ubyte WALLCOLOR = 12
|
|
|
|
const ubyte EMPTYCOLOR = 0
|
|
|
|
|
|
|
|
; unfortunately on larger screens (cx16), the number of cells exceeds 256 and doesn't fit in a regular array anymore.
|
|
|
|
uword cells = memory("cells", numCellsHoriz*numCellsVert, 0)
|
|
|
|
|
|
|
|
ubyte[256] cx_stack
|
|
|
|
ubyte[256] cy_stack
|
|
|
|
ubyte stackptr
|
|
|
|
|
|
|
|
sub draw() {
|
|
|
|
ubyte cx
|
|
|
|
ubyte cy
|
|
|
|
for cx in 0 to numCellsHoriz-1 {
|
|
|
|
for cy in 0 to numCellsVert-1 {
|
|
|
|
drawCell(cx, cy)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-29 11:50:35 +02:00
|
|
|
ubyte[4] directionflags = [LEFT,RIGHT,UP,DOWN]
|
|
|
|
|
2022-07-09 15:40:56 +02:00
|
|
|
sub generate() {
|
|
|
|
ubyte cx = startCx
|
|
|
|
ubyte cy = startCy
|
|
|
|
|
|
|
|
stackptr = 0
|
|
|
|
@(celladdr(cx,cy)) &= ~STONE
|
|
|
|
drawCell(cx, cy)
|
|
|
|
uword cells_to_carve = numCellsHoriz * numCellsVert - 1
|
|
|
|
|
|
|
|
repeat {
|
|
|
|
carve_restart_after_repath:
|
|
|
|
ubyte direction = choose_uncarved_direction()
|
|
|
|
if direction==0 {
|
|
|
|
;backtrack
|
|
|
|
stackptr--
|
|
|
|
if stackptr==255 {
|
|
|
|
; stack empty.
|
|
|
|
; repath if we are not done yet. (this is a workaround for the prog8 256 array lenght limit)
|
2024-02-04 23:18:11 +01:00
|
|
|
if cells_to_carve!=0 {
|
2022-07-09 15:40:56 +02:00
|
|
|
if repath()
|
|
|
|
goto carve_restart_after_repath
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
cx = cx_stack[stackptr]
|
|
|
|
cy = cy_stack[stackptr]
|
|
|
|
} else {
|
|
|
|
cx_stack[stackptr] = cx
|
|
|
|
cy_stack[stackptr] = cy
|
|
|
|
stackptr++
|
|
|
|
if stackptr==0 {
|
|
|
|
; stack overflow, we can't track our path any longer.
|
|
|
|
; repath if we are not done yet. (this is a workaround for the prog8 256 array lenght limit)
|
2024-02-04 23:18:11 +01:00
|
|
|
if cells_to_carve!=0 {
|
2022-07-09 15:40:56 +02:00
|
|
|
if repath()
|
|
|
|
goto carve_restart_after_repath
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
@(celladdr(cx,cy)) |= direction
|
|
|
|
when direction {
|
|
|
|
UP -> {
|
|
|
|
cy--
|
|
|
|
@(celladdr(cx,cy)) |= DOWN
|
|
|
|
}
|
|
|
|
RIGHT -> {
|
|
|
|
cx++
|
|
|
|
@(celladdr(cx,cy)) |= LEFT
|
|
|
|
}
|
|
|
|
DOWN -> {
|
|
|
|
cy++
|
|
|
|
@(celladdr(cx,cy)) |= UP
|
|
|
|
}
|
|
|
|
LEFT -> {
|
|
|
|
cx--
|
|
|
|
@(celladdr(cx,cy)) |= RIGHT
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@(celladdr(cx,cy)) &= ~STONE
|
|
|
|
cells_to_carve--
|
|
|
|
drawCell(cx, cy)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-09 22:13:30 +02:00
|
|
|
sub repath() -> bool {
|
2022-07-09 15:40:56 +02:00
|
|
|
; repath: try to find a new start cell with possible directions.
|
|
|
|
; we limit our number of searches so that the algorith doesn't get stuck
|
|
|
|
; for too long on bad rng... just accept a few unused cells in that case.
|
|
|
|
repeat 255 {
|
|
|
|
do {
|
2022-10-22 16:35:57 +02:00
|
|
|
cx = math.rnd() % numCellsHoriz
|
|
|
|
cy = math.rnd() % numCellsVert
|
2024-02-04 23:18:11 +01:00
|
|
|
} until @(celladdr(cx, cy)) & STONE ==0
|
|
|
|
if available_uncarved()!=0
|
2022-07-09 15:40:56 +02:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
sub available_uncarved() -> ubyte {
|
|
|
|
ubyte candidates = 0
|
2024-02-04 23:18:11 +01:00
|
|
|
if cx>0 and @(celladdr(cx-1, cy)) & STONE !=0
|
2022-07-09 15:40:56 +02:00
|
|
|
candidates |= LEFT
|
2024-02-04 23:18:11 +01:00
|
|
|
if cx<numCellsHoriz-1 and @(celladdr(cx+1, cy)) & STONE !=0
|
2022-07-09 15:40:56 +02:00
|
|
|
candidates |= RIGHT
|
2024-02-04 23:18:11 +01:00
|
|
|
if cy>0 and @(celladdr(cx, cy-1)) & STONE !=0
|
2022-07-09 15:40:56 +02:00
|
|
|
candidates |= UP
|
2024-02-04 23:18:11 +01:00
|
|
|
if cy<numCellsVert-1 and @(celladdr(cx, cy+1)) & STONE !=0
|
2022-07-09 15:40:56 +02:00
|
|
|
candidates |= DOWN
|
|
|
|
return candidates
|
|
|
|
}
|
|
|
|
|
|
|
|
sub choose_uncarved_direction() -> ubyte {
|
|
|
|
ubyte candidates = available_uncarved()
|
2024-02-04 23:18:11 +01:00
|
|
|
if candidates==0
|
2022-07-09 15:40:56 +02:00
|
|
|
return 0
|
|
|
|
|
|
|
|
repeat {
|
2023-08-29 11:50:35 +02:00
|
|
|
ubyte choice = candidates & directionflags[math.rnd() & 3]
|
2024-02-04 23:18:11 +01:00
|
|
|
if choice!=0
|
2022-07-09 15:40:56 +02:00
|
|
|
return choice
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-29 11:50:35 +02:00
|
|
|
sub openpassages() {
|
|
|
|
; open just a few extra passages, so that multiple routes are possible in theory.
|
|
|
|
ubyte numpassages
|
|
|
|
ubyte cx
|
|
|
|
ubyte cy
|
|
|
|
do {
|
|
|
|
do {
|
|
|
|
cx = math.rnd() % (numCellsHoriz-2) + 1
|
|
|
|
cy = math.rnd() % (numCellsVert-2) + 1
|
2024-02-04 23:18:11 +01:00
|
|
|
} until @(celladdr(cx, cy)) & STONE ==0
|
2023-08-29 11:50:35 +02:00
|
|
|
ubyte direction = directionflags[math.rnd() & 3]
|
2024-02-04 23:18:11 +01:00
|
|
|
if @(celladdr(cx, cy)) & direction == 0 {
|
2023-08-29 11:50:35 +02:00
|
|
|
when direction {
|
|
|
|
LEFT -> {
|
2024-02-04 23:18:11 +01:00
|
|
|
if @(celladdr(cx-1,cy)) & STONE == 0 {
|
2023-08-29 11:50:35 +02:00
|
|
|
@(celladdr(cx,cy)) |= LEFT
|
|
|
|
drawCell(cx,cy)
|
|
|
|
numpassages++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
RIGHT -> {
|
2024-02-04 23:18:11 +01:00
|
|
|
if @(celladdr(cx+1,cy)) & STONE == 0 {
|
2023-08-29 11:50:35 +02:00
|
|
|
@(celladdr(cx,cy)) |= RIGHT
|
|
|
|
drawCell(cx,cy)
|
|
|
|
numpassages++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
UP -> {
|
2024-02-04 23:18:11 +01:00
|
|
|
if @(celladdr(cx,cy-1)) & STONE == 0 {
|
2023-08-29 11:50:35 +02:00
|
|
|
@(celladdr(cx,cy)) |= UP
|
|
|
|
drawCell(cx,cy)
|
|
|
|
numpassages++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DOWN -> {
|
2024-02-04 23:18:11 +01:00
|
|
|
if @(celladdr(cx,cy+1)) & STONE == 0 {
|
2023-08-29 11:50:35 +02:00
|
|
|
@(celladdr(cx,cy)) |= DOWN
|
|
|
|
drawCell(cx,cy)
|
|
|
|
numpassages++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} until numpassages==10
|
|
|
|
}
|
|
|
|
|
2022-07-09 15:40:56 +02:00
|
|
|
sub solve() {
|
|
|
|
ubyte cx = startCx
|
|
|
|
ubyte cy = startCy
|
|
|
|
const uword max_path_length = 1024
|
|
|
|
|
|
|
|
; the path through the maze can be longer than 256 so doesn't fit in a regular array.... :(
|
|
|
|
uword pathstack = memory("pathstack", max_path_length, 0)
|
|
|
|
uword pathstackptr = 0
|
|
|
|
|
|
|
|
@(celladdr(cx,cy)) |= WALKED
|
|
|
|
txt.setcc(cx*2+1, cy*2+1, 81, 1)
|
|
|
|
|
|
|
|
repeat {
|
|
|
|
solve_loop:
|
|
|
|
sys.waitvsync()
|
|
|
|
if cx==finishCx and cy==finishCy {
|
|
|
|
txt.home()
|
|
|
|
txt.print("found! path length: ")
|
|
|
|
txt.print_uw(pathstackptr)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ubyte cell = @(celladdr(cx,cy))
|
2024-02-04 23:18:11 +01:00
|
|
|
if cell & UP!=0 and @(celladdr(cx,cy-1)) & (WALKED|BACKTRACKED) ==0 {
|
2022-07-09 15:40:56 +02:00
|
|
|
@(pathstack + pathstackptr) = UP
|
|
|
|
txt.setcc(cx*2+1, cy*2, 81, 3)
|
|
|
|
cy--
|
|
|
|
}
|
2024-02-04 23:18:11 +01:00
|
|
|
else if cell & DOWN !=0 and @(celladdr(cx,cy+1)) & (WALKED|BACKTRACKED) ==0 {
|
2022-07-09 15:40:56 +02:00
|
|
|
@(pathstack + pathstackptr) = DOWN
|
|
|
|
txt.setcc(cx*2+1, cy*2+2, 81, 3)
|
|
|
|
cy++
|
|
|
|
}
|
2024-02-04 23:18:11 +01:00
|
|
|
else if cell & LEFT !=0 and @(celladdr(cx-1,cy)) & (WALKED|BACKTRACKED) ==0 {
|
2022-07-09 15:40:56 +02:00
|
|
|
@(pathstack + pathstackptr) = LEFT
|
|
|
|
txt.setcc(cx*2, cy*2+1, 81, 3)
|
|
|
|
cx--
|
|
|
|
}
|
2024-02-04 23:18:11 +01:00
|
|
|
else if cell & RIGHT !=0 and @(celladdr(cx+1,cy)) & (WALKED|BACKTRACKED) ==0 {
|
2022-07-09 15:40:56 +02:00
|
|
|
@(pathstack + pathstackptr) = RIGHT
|
|
|
|
txt.setcc(cx*2+2, cy*2+1, 81, 3)
|
|
|
|
cx++
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
; dead end, pop stack
|
|
|
|
pathstackptr--
|
2024-02-20 22:53:15 +01:00
|
|
|
if pathstackptr==65535 {
|
2022-07-09 15:40:56 +02:00
|
|
|
txt.print("no solution?!")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
@(celladdr(cx,cy)) |= BACKTRACKED
|
|
|
|
txt.setcc(cx*2+1, cy*2+1, 81, 2)
|
|
|
|
when @(pathstack + pathstackptr) {
|
|
|
|
UP -> {
|
|
|
|
txt.setcc(cx*2+1, cy*2+2, 81, 9)
|
|
|
|
cy++
|
|
|
|
}
|
|
|
|
DOWN -> {
|
|
|
|
txt.setcc(cx*2+1, cy*2, 81, 9)
|
|
|
|
cy--
|
|
|
|
}
|
|
|
|
LEFT -> {
|
|
|
|
txt.setcc(cx*2+2, cy*2+1, 81, 9)
|
|
|
|
cx++
|
|
|
|
}
|
|
|
|
RIGHT -> {
|
|
|
|
txt.setcc(cx*2, cy*2+1, 81, 9)
|
|
|
|
cx--
|
|
|
|
}
|
|
|
|
}
|
|
|
|
goto solve_loop
|
|
|
|
}
|
|
|
|
pathstackptr++
|
|
|
|
if pathstackptr==max_path_length {
|
|
|
|
txt.print("stack overflow, path too long")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
@(celladdr(cx,cy)) |= WALKED
|
|
|
|
txt.setcc(cx*2+1, cy*2+1, 81, 1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub celladdr(ubyte cx, ubyte cy) -> uword {
|
|
|
|
return cells+(numCellsHoriz as uword)*cy+cx
|
|
|
|
}
|
|
|
|
|
|
|
|
sub drawCell(ubyte cx, ubyte cy) {
|
|
|
|
ubyte x = cx * 2 + 1
|
|
|
|
ubyte y = cy * 2 + 1
|
|
|
|
ubyte doors = @(celladdr(cx,cy))
|
2024-02-04 23:18:11 +01:00
|
|
|
if doors & UP !=0
|
2022-07-09 15:40:56 +02:00
|
|
|
txt.setcc(x, y-1, ' ', EMPTYCOLOR)
|
2024-02-04 23:18:11 +01:00
|
|
|
if doors & RIGHT !=0
|
2022-07-09 15:40:56 +02:00
|
|
|
txt.setcc(x+1, y, ' ', EMPTYCOLOR)
|
2024-02-04 23:18:11 +01:00
|
|
|
if doors & DOWN !=0
|
2022-07-09 15:40:56 +02:00
|
|
|
txt.setcc(x, y+1, ' ', EMPTYCOLOR)
|
2024-02-04 23:18:11 +01:00
|
|
|
if doors & LEFT !=0
|
2022-07-09 15:40:56 +02:00
|
|
|
txt.setcc(x-1, y, ' ', EMPTYCOLOR)
|
2024-02-04 23:18:11 +01:00
|
|
|
if doors & STONE !=0
|
2022-07-09 15:40:56 +02:00
|
|
|
txt.setcc(x, y, 160, WALLCOLOR)
|
|
|
|
else
|
|
|
|
txt.setcc(x, y, 32, EMPTYCOLOR)
|
|
|
|
|
2024-02-04 23:18:11 +01:00
|
|
|
if doors & WALKED !=0
|
2022-07-09 15:40:56 +02:00
|
|
|
txt.setcc(x, y, 81, 1)
|
2024-02-04 23:18:11 +01:00
|
|
|
if doors & BACKTRACKED !=0
|
2022-07-09 15:40:56 +02:00
|
|
|
txt.setcc(x, y, 81, 2)
|
|
|
|
}
|
|
|
|
|
|
|
|
sub initialize() {
|
|
|
|
sys.memset(cells, numCellsHoriz*numCellsVert, STONE)
|
|
|
|
txt.fill_screen(160, WALLCOLOR)
|
|
|
|
drawStartFinish()
|
|
|
|
}
|
|
|
|
|
|
|
|
sub drawStartFinish() {
|
|
|
|
txt.setcc(startCx*2+1,startCy*2+1,sc:'s',5)
|
|
|
|
txt.setcc(finishCx*2+1, finishCy*2+1, sc:'f', 13)
|
|
|
|
}
|
|
|
|
}
|