When an tile is rendered the writing to disk can be done in background

concurrently to the rendering of the next tile.
This commit is contained in:
Sascha L. Teichmann 2017-03-06 15:43:30 +01:00
parent 2cba483d32
commit 6a1356f0eb
5 changed files with 118 additions and 58 deletions

View File

@ -30,10 +30,29 @@ func createTiles(
jobs chan blockPos, jobs chan blockPos,
done *sync.WaitGroup) { done *sync.WaitGroup) {
defer done.Done() wFns := make(chan func() (bool, error))
defer btc.Close()
// Writing already rendered tiles to disk can be done in background.
go func() {
for wfn := range wFns {
if _, err := wfn(); err != nil {
log.Printf("WARN: writing file failed: %v.\n", err)
}
}
}()
defer func() {
close(wFns)
btc.Close()
done.Done()
}()
for job := range jobs { for job := range jobs {
btc.CreateTile(job.x-1, job.z-1, job.i, job.j) if err := btc.RenderArea(job.x-1, job.z-1); err != nil {
log.Printf("WARN: rendering failed: %v.\n", err)
continue
}
wFns <- btc.WriteFunc(job.i, job.j, nil)
} }
} }
@ -77,7 +96,7 @@ func createBaseLevel(
btc := common.NewBaseTileCreator( btc := common.NewBaseTileCreator(
client, colors, bg, client, colors, bg,
int16(yMin), int16(yMax), int16(yMin), int16(yMax),
transparent, baseDir, nil) transparent, baseDir)
go createTiles(btc, jobs, &done) go createTiles(btc, jobs, &done)
} }

View File

@ -213,9 +213,9 @@ func (tu *tileUpdater) doUpdates() {
btc := common.NewBaseTileCreator( btc := common.NewBaseTileCreator(
client, tu.colors, tu.bg, client, tu.colors, tu.bg,
tu.yMin, tu.yMax, tu.yMin, tu.yMax,
tu.transparent, baseDir, bth.Update) tu.transparent, baseDir)
done.Add(1) done.Add(1)
go tu.updateBaseTiles(jobs, btc, &done) go tu.updateBaseTiles(jobs, btc, &done, bth.Update)
} }
for i := range changes { for i := range changes {
@ -338,19 +338,42 @@ func (tu *tileUpdater) updatePyramidTile(scratch, resized *image.RGBA, level int
func (tu *tileUpdater) updateBaseTiles( func (tu *tileUpdater) updateBaseTiles(
jobs chan *xzc, jobs chan *xzc,
btc *common.BaseTileCreator, done *sync.WaitGroup) { btc *common.BaseTileCreator,
done *sync.WaitGroup,
update common.BaseTileUpdateFunc) {
type jobWriter struct {
job *xzc
wFn func() (bool, error)
}
jWs := make(chan jobWriter)
go func() {
for jw := range jWs {
updated, err := jw.wFn()
if err != nil {
log.Printf("WARN: writing tile failed: %v.\n", err)
}
if !updated {
jw.job.canceled = true
}
}
}()
defer func() {
close(jWs)
btc.Close()
done.Done()
}()
defer btc.Close()
defer done.Done()
for job := range jobs { for job := range jobs {
xz := job.dequantize() xz := job.dequantize()
//log.Printf("%d/%d %d/%d", x, z, job.X, job.Z) if err := btc.RenderArea(xz.X-1, xz.Z-1); err != nil {
updated, err := btc.CreateTile(xz.X-1, xz.Z-1, int(job.X), int(job.Z)) log.Printf("WARN: rendering tile failed: %v.\n", err)
if err != nil {
log.Printf("WARN: create tile failed: %s\n", err)
}
if !updated {
job.canceled = true job.canceled = true
continue
} }
jWs <- jobWriter{job, btc.WriteFunc(int(job.X), int(job.Z), update)}
} }
} }

View File

@ -52,14 +52,14 @@ var BackgroundColor = color.RGBA{R: 0xff, G: 0xff, B: 0xff, A: 0xff}
type BaseTileUpdateFunc func(x, y int, hash []byte) bool type BaseTileUpdateFunc func(x, y int, hash []byte) bool
type BaseTileCreator struct { type BaseTileCreator struct {
client *RedisClient client *RedisClient
colors *Colors colors *Colors
renderer *Renderer renderer *Renderer
yOrder *YOrder yOrder *YOrder
yMin int16 yMin int16
yMax int16 yMax int16
baseDir string baseDir string
update BaseTileUpdateFunc //update BaseTileUpdateFunc
emptyImage []byte emptyImage []byte
bg color.RGBA bg color.RGBA
} }
@ -70,8 +70,8 @@ func NewBaseTileCreator(
bg color.RGBA, bg color.RGBA,
yMin, yMax int16, yMin, yMax int16,
transparent bool, transparent bool,
baseDir string, baseDir string) *BaseTileCreator {
update BaseTileUpdateFunc) *BaseTileCreator { //update BaseTileUpdateFunc) *BaseTileCreator {
renderer := NewRenderer(tileWidth, tileHeight, transparent) renderer := NewRenderer(tileWidth, tileHeight, transparent)
yMin, yMax = Order16(yMin, yMax) yMin, yMax = Order16(yMin, yMax)
btc := &BaseTileCreator{ btc := &BaseTileCreator{
@ -82,7 +82,8 @@ func NewBaseTileCreator(
yMin: yMin, yMin: yMin,
yMax: yMax, yMax: yMax,
baseDir: baseDir, baseDir: baseDir,
update: update} }
//update: update}
btc.yOrder = NewYOrder(btc.renderBlock, yOrderCapacity) btc.yOrder = NewYOrder(btc.renderBlock, yOrderCapacity)
return btc return btc
} }
@ -105,7 +106,7 @@ func (btc *BaseTileCreator) blockLoaded(block *Block) *Block {
return block return block
} }
func (btc *BaseTileCreator) CreateTile(x, z int16, i, j int) (bool, error) { func (btc *BaseTileCreator) RenderArea(x, z int16) error {
btc.renderer.Reset() btc.renderer.Reset()
btc.renderer.SetPos(x, z) btc.renderer.SetPos(x, z)
btc.yOrder.Reset() btc.yOrder.Reset()
@ -136,7 +137,7 @@ func (btc *BaseTileCreator) CreateTile(x, z int16, i, j int) (bool, error) {
var count int var count int
var err error var err error
if count, err = btc.client.QueryCuboid(query, btc.blockLoaded); err != nil { if count, err = btc.client.QueryCuboid(query, btc.blockLoaded); err != nil {
return false, err return err
} }
if err = btc.yOrder.Drain(); err != nil { if err = btc.yOrder.Drain(); err != nil {
log.Printf("WARN: rendering block failed: %s\n", err) log.Printf("WARN: rendering block failed: %s\n", err)
@ -155,38 +156,50 @@ func (btc *BaseTileCreator) CreateTile(x, z int16, i, j int) (bool, error) {
} }
areas, nareas = nareas, areas[:0] areas, nareas = nareas, areas[:0]
} }
return nil
}
func (btc *BaseTileCreator) blankImage() []byte {
// To avoid redundant encoding cache the resulting empty image.
if btc.emptyImage == nil {
m := BackgroundImage((tileWidth-2)*16, (tileHeight-2)*16, btc.bg)
btc.emptyImage = EncodeToMem(m)
}
return btc.emptyImage
}
func (btc *BaseTileCreator) WriteFunc(i, j int, update BaseTileUpdateFunc) func() (bool, error) {
path := filepath.Join(btc.baseDir, strconv.Itoa(i), strconv.Itoa(j)+".png") path := filepath.Join(btc.baseDir, strconv.Itoa(i), strconv.Itoa(j)+".png")
// Empty images are likely to be produced during seeding. // Empty images are likely to be produced during seeding.
if btc.update == nil && btc.renderer.IsEmpty() { if update == nil && btc.renderer.IsEmpty() {
// To avoid redundant encoding cache the resulting empty image. return func() (bool, error) {
if btc.emptyImage == nil { //log.Printf("Writing empty (%d, %d) to file %s\n", x, z, path)
var err error return true, ioutil.WriteFile(path, btc.blankImage(), 0666)
m := BackgroundImage((tileWidth-2)*16, (tileHeight-2)*16, btc.bg)
if btc.emptyImage, err = EncodeToMem(m); err != nil {
return false, err
}
} }
//log.Printf("Writing empty (%d, %d) to file %s\n", x, z, path)
return true, ioutil.WriteFile(path, btc.emptyImage, 0666)
} }
image := btc.renderer.CreateShadedImage( image := btc.renderer.CreateShadedImage(
16, 16, (tileWidth-2)*16, (tileHeight-2)*16, 16, 16, (tileWidth-2)*16, (tileHeight-2)*16,
btc.colors, btc.bg) btc.colors, btc.bg)
if btc.update == nil { x, z := btc.renderer.GetPos()
log.Printf("Writing (%d, %d) to file %s.\n", x, z, path)
return true, SaveAsPNG(path, image) if update == nil {
return func() (bool, error) {
log.Printf("Writing (%d, %d) to file %s.\n", x, z, path)
return true, SaveAsPNG(path, image)
}
} }
if btc.update(i, j, HashImage(image)) { return func() (bool, error) {
log.Printf("Writing (%d, %d) to file %s.\n", x, z, path) if update(i, j, HashImage(image)) {
return true, SaveAsPNGAtomic(path, image) log.Printf("Writing (%d, %d) to file %s.\n", x, z, path)
return true, SaveAsPNGAtomic(path, image)
}
log.Printf("(%d, %d) is unchanged.\n", x, z)
return false, nil
} }
log.Printf("(%d, %d) is unchanged.\n", x, z)
return false, nil
} }

View File

@ -10,6 +10,7 @@ import (
"errors" "errors"
"image" "image"
"image/color" "image/color"
"image/draw"
"image/png" "image/png"
"log" "log"
"os" "os"
@ -44,13 +45,14 @@ func nextSuffix() string {
return strconv.Itoa(int(1e9 + r%1e9))[1:] return strconv.Itoa(int(1e9 + r%1e9))[1:]
} }
func EncodeToMem(img image.Image) ([]byte, error) { func EncodeToMem(img image.Image) []byte {
var buf bytes.Buffer var buf bytes.Buffer
enc := png.Encoder{CompressionLevel: png.BestCompression} enc := png.Encoder{CompressionLevel: png.BestCompression}
if err := enc.Encode(&buf, img); err != nil { if err := enc.Encode(&buf, img); err != nil {
return nil, err // This really should not happen.
panic(err)
} }
return buf.Bytes(), nil return buf.Bytes()
} }
func SaveAsPNG(path string, img image.Image) (err error) { func SaveAsPNG(path string, img image.Image) (err error) {
@ -126,3 +128,9 @@ func HashImage(img *image.RGBA) []byte {
return hash.Sum(nil) return hash.Sum(nil)
} }
func BackgroundImage(width, height int, bg color.RGBA) *image.RGBA {
m := image.NewRGBA(image.Rect(0, 0, width, height))
draw.Draw(m, m.Bounds(), &image.Uniform{bg}, image.ZP, draw.Src)
return m
}

View File

@ -7,7 +7,6 @@ package common
import ( import (
"image" "image"
"image/color" "image/color"
"image/draw"
"math" "math"
) )
@ -59,6 +58,10 @@ func (r *Renderer) SetPos(xOfs, zOfs int16) {
r.zOfs = zOfs r.zOfs = zOfs
} }
func (r *Renderer) GetPos() (int16, int16) {
return r.xOfs, r.zOfs
}
func (r *Renderer) initBuffers() { func (r *Renderer) initBuffers() {
yb := r.yBuffer yb := r.yBuffer
yb = yb[:len(yb)] yb = yb[:len(yb)]
@ -269,12 +272,6 @@ func safeColor(x int32) uint8 {
} }
} }
func BackgroundImage(width, height int, bg color.RGBA) *image.RGBA {
m := image.NewRGBA(image.Rect(0, 0, width, height))
draw.Draw(m, m.Bounds(), &image.Uniform{bg}, image.ZP, draw.Src)
return m
}
func (r *Renderer) CreateShadedImage( func (r *Renderer) CreateShadedImage(
xOfs, zOfs, width, height int, xOfs, zOfs, width, height int,
cols *Colors, background color.RGBA) *image.RGBA { cols *Colors, background color.RGBA) *image.RGBA {