added caching for much better performance

This commit is contained in:
2024-05-28 00:07:31 +02:00
committed by T.v.Dein
parent ba247d0606
commit 8cb5585456
2 changed files with 32 additions and 52 deletions

40
main.go
View File

@@ -5,7 +5,6 @@ import (
"log" "log"
"os" "os"
"runtime/pprof" "runtime/pprof"
"time"
_ "net/http/pprof" _ "net/http/pprof"
@@ -26,7 +25,8 @@ func main() {
game := NewGame(config, Play) game := NewGame(config, Play)
if config.ProfileFile != "" { if config.ProfileFile != "" {
// enable cpu profiling and use fake game loop // enable cpu profiling. Do NOT use q to stop the game but
// close the window to get a profile
fd, err := os.Create(config.ProfileFile) fd, err := os.Create(config.ProfileFile)
if err != nil { if err != nil {
log.Fatal(err) log.Fatal(err)
@@ -35,13 +35,6 @@ func main() {
pprof.StartCPUProfile(fd) pprof.StartCPUProfile(fd)
defer pprof.StopCPUProfile() defer pprof.StopCPUProfile()
Ebitfake(game)
pprof.StopCPUProfile()
fd.Close()
os.Exit(0)
} }
// main loop // main loop
@@ -49,32 +42,3 @@ func main() {
log.Fatal(err) log.Fatal(err)
} }
} }
// fake game loop, required to be able to profile the program using
// pprof. Otherwise any kind of program exit leads to an empty profile
// file.
func Ebitfake(game *Game) {
screen := ebiten.NewImage(game.ScreenWidth, game.ScreenHeight)
var loops int64
for {
err := game.Update()
if err != nil {
log.Fatal(err)
}
if game.Config.ProfileDraw {
game.Draw(screen)
}
fmt.Print(".")
time.Sleep(16 * time.Millisecond) // around 60 TPS
if loops >= game.Config.ProfileMaxLoops {
break
}
loops++
}
}

View File

@@ -35,7 +35,7 @@ type ScenePlay struct {
TicksElapsed int // tick counter for game speed TicksElapsed int // tick counter for game speed
Tiles Images // pre-computed tiles for dead and alife cells Tiles Images // pre-computed tiles for dead and alife cells
Camera Camera // for zoom+move Camera Camera // for zoom+move
World *ebiten.Image // actual image we render to World, Cache *ebiten.Image // actual image we render to
WheelTurned bool // when user turns wheel multiple times, zoom faster WheelTurned bool // when user turns wheel multiple times, zoom faster
Dragging bool // middle mouse is pressed, move canvas Dragging bool // middle mouse is pressed, move canvas
LastCursorPos []int // used to check if the user is dragging LastCursorPos []int // used to check if the user is dragging
@@ -411,11 +411,8 @@ func (scene *ScenePlay) Draw(screen *ebiten.Image) {
// a nice grey grid with grid lines // a nice grey grid with grid lines
op := &ebiten.DrawImageOptions{} op := &ebiten.DrawImageOptions{}
if scene.Config.NoGrid { op.GeoM.Translate(0, 0)
scene.World.Fill(scene.White) scene.World.DrawImage(scene.Cache, op)
} else {
scene.World.Fill(scene.Grey)
}
for y := 0; y < scene.Config.Height; y++ { for y := 0; y < scene.Config.Height; y++ {
for x := 0; x < scene.Config.Width; x++ { for x := 0; x < scene.Config.Width; x++ {
@@ -443,8 +440,6 @@ func (scene *ScenePlay) Draw(screen *ebiten.Image) {
default: default:
scene.World.DrawImage(scene.Tiles.Age4, op) scene.World.DrawImage(scene.Tiles.Age4, op)
} }
} else {
scene.World.DrawImage(scene.Tiles.White, op)
} }
} }
} }
@@ -475,11 +470,11 @@ func (scene *ScenePlay) Draw(screen *ebiten.Image) {
paused = "-- paused --" paused = "-- paused --"
} }
ebitenutil.DebugPrint( debug := fmt.Sprintf("FPS: %0.2f, TPG: %d, Mem: %0.2f MB, Generations: %d %s",
screen, ebiten.ActualTPS(), scene.TPG, GetMem(), scene.Generations, paused)
fmt.Sprintf("FPS: %0.2f, TPG: %d, Mem: %0.2f MB, Generations: %d %s",
ebiten.ActualTPS(), scene.TPG, GetMem(), scene.Generations, paused), ebitenutil.DebugPrint(screen, debug)
) fmt.Println(debug)
} }
} }
@@ -505,6 +500,25 @@ func (scene *ScenePlay) InitPattern() {
} }
} }
func (scene *ScenePlay) InitCache() {
op := &ebiten.DrawImageOptions{}
if scene.Config.NoGrid {
scene.Cache.Fill(scene.White)
} else {
scene.Cache.Fill(scene.Grey)
}
for y := 0; y < scene.Config.Height; y++ {
for x := 0; x < scene.Config.Width; x++ {
op.GeoM.Reset()
op.GeoM.Translate(float64(x*scene.Config.Cellsize), float64(y*scene.Config.Cellsize))
scene.Cache.DrawImage(scene.Tiles.White, op)
}
}
}
func (scene *ScenePlay) InitGrid(grid *Grid) { func (scene *ScenePlay) InitGrid(grid *Grid) {
if grid != nil { if grid != nil {
// use pre-loaded grid // use pre-loaded grid
@@ -599,10 +613,12 @@ func (scene *ScenePlay) Init() {
} }
scene.World = ebiten.NewImage(scene.Config.ScreenWidth, scene.Config.ScreenHeight) scene.World = ebiten.NewImage(scene.Config.ScreenWidth, scene.Config.ScreenHeight)
scene.Cache = ebiten.NewImage(scene.Config.ScreenWidth, scene.Config.ScreenHeight)
scene.InitTiles()
scene.InitCache()
scene.InitGrid(grid) scene.InitGrid(grid)
scene.InitPattern() scene.InitPattern()
scene.InitTiles()
scene.Index = 0 scene.Index = 0
scene.TicksElapsed = 0 scene.TicksElapsed = 0