restorable: Reland: Refactoring

This is reland of 7917b423f4be9494125c8de76a8a3def15561594

Fixed a bug that an image passed to Apply might not be initialized,
and there was no correct way to treat such an image. Now Apply
accepts only an initialized image.
This commit is contained in:
Hajime Hoshi 2019-07-21 10:52:29 +09:00
parent a3b35e5fba
commit 434802af6e

View File

@ -28,6 +28,7 @@ type Pixels struct {
rectToPixels *rectToPixels rectToPixels *rectToPixels
} }
// Apply applies the Pixels state to the given image especially for restoring.
func (p *Pixels) Apply(img *graphicscommand.Image) { func (p *Pixels) Apply(img *graphicscommand.Image) {
if p.rectToPixels == nil { if p.rectToPixels == nil {
clearImage(img) clearImage(img)
@ -77,14 +78,13 @@ type drawTrianglesHistoryItem struct {
type Image struct { type Image struct {
image *graphicscommand.Image image *graphicscommand.Image
basePixels *Pixels basePixels Pixels
// drawTrianglesHistory is a set of draw-image commands. // drawTrianglesHistory is a set of draw-image commands.
// TODO: This should be merged with the similar command queue in package graphics (#433). // TODO: This should be merged with the similar command queue in package graphics (#433).
drawTrianglesHistory []*drawTrianglesHistoryItem drawTrianglesHistory []*drawTrianglesHistoryItem
// stale indicates whether the image needs to be synced with GPU as soon as possible. // stale indicates whether the image needs to be synced with GPU as soon as possible.
// TODO: Instead of this boolean value, can we represent the stale state with basePixels's state?
stale bool stale bool
// volatile indicates whether the image is cleared whenever a frame starts. // volatile indicates whether the image is cleared whenever a frame starts.
@ -125,7 +125,7 @@ func NewImage(width, height int) *Image {
i := &Image{ i := &Image{
image: graphicscommand.NewImage(width, height), image: graphicscommand.NewImage(width, height),
} }
i.clearForInitialization() i.clear()
theImages.add(i) theImages.add(i)
return i return i
} }
@ -154,7 +154,6 @@ func (i *Image) Extend(width, height int) *Image {
} }
newImg := NewImage(width, height) newImg := NewImage(width, height)
i.basePixels.Apply(newImg.image) i.basePixels.Apply(newImg.image)
newImg.basePixels = i.basePixels newImg.basePixels = i.basePixels
@ -178,7 +177,7 @@ func NewScreenFramebufferImage(width, height int) *Image {
image: graphicscommand.NewScreenFramebufferImage(width, height), image: graphicscommand.NewScreenFramebufferImage(width, height),
screen: true, screen: true,
} }
i.clearForInitialization() i.clear()
theImages.add(i) theImages.add(i)
return i return i
} }
@ -188,12 +187,6 @@ func (i *Image) Clear() {
i.clear() i.clear()
} }
// clearForInitialization clears the underlying image for initialization.
func (i *Image) clearForInitialization() {
// As this is for initialization, drawing history doesn't have to be adjusted.
i.clear()
}
// clearImage clears a graphicscommand.Image. // clearImage clears a graphicscommand.Image.
// This does nothing to do with a restorable.Image's rendering state. // This does nothing to do with a restorable.Image's rendering state.
func clearImage(img *graphicscommand.Image) { func clearImage(img *graphicscommand.Image) {
@ -232,7 +225,7 @@ func (i *Image) clear() {
// //
// After ResetRestoringState, the image is assumed to be cleared. // After ResetRestoringState, the image is assumed to be cleared.
func (i *Image) ResetRestoringState() { func (i *Image) ResetRestoringState() {
i.basePixels = &Pixels{} i.basePixels = Pixels{}
i.drawTrianglesHistory = nil i.drawTrianglesHistory = nil
i.stale = false i.stale = false
} }
@ -243,7 +236,7 @@ func (i *Image) IsVolatile() bool {
// BasePixelsForTesting returns the image's basePixels for testing. // BasePixelsForTesting returns the image's basePixels for testing.
func (i *Image) BasePixelsForTesting() *Pixels { func (i *Image) BasePixelsForTesting() *Pixels {
return i.basePixels return &i.basePixels
} }
// Size returns the image's size. // Size returns the image's size.
@ -281,7 +274,7 @@ func (i *Image) PutVertex(vs []float32, dx, dy, sx, sy float32, bx0, by0, bx1, b
// makeStale makes the image stale. // makeStale makes the image stale.
func (i *Image) makeStale() { func (i *Image) makeStale() {
i.basePixels = nil i.basePixels = Pixels{}
i.drawTrianglesHistory = nil i.drawTrianglesHistory = nil
i.stale = true i.stale = true
@ -322,9 +315,6 @@ func (i *Image) ReplacePixels(pixels []byte, x, y, width, height int) {
} }
if x == 0 && y == 0 && width == w && height == h { if x == 0 && y == 0 && width == w && height == h {
if i.basePixels == nil {
i.basePixels = &Pixels{}
}
if pixels != nil { if pixels != nil {
i.basePixels.AddOrReplace(pixels, 0, 0, w, h) i.basePixels.AddOrReplace(pixels, 0, 0, w, h)
} else { } else {
@ -347,9 +337,6 @@ func (i *Image) ReplacePixels(pixels []byte, x, y, width, height int) {
return return
} }
if i.basePixels == nil {
i.basePixels = &Pixels{}
}
if pixels != nil { if pixels != nil {
i.basePixels.AddOrReplace(pixels, x, y, width, height) i.basePixels.AddOrReplace(pixels, x, y, width, height)
} else { } else {
@ -401,7 +388,7 @@ func (i *Image) appendDrawTrianglesHistory(image *Image, vertices []float32, ind
} }
func (i *Image) readPixelsFromGPUIfNeeded() { func (i *Image) readPixelsFromGPUIfNeeded() {
if i.basePixels == nil || len(i.drawTrianglesHistory) > 0 || i.stale { if len(i.drawTrianglesHistory) > 0 || i.stale {
graphicscommand.FlushCommands() graphicscommand.FlushCommands()
i.readPixelsFromGPU() i.readPixelsFromGPU()
i.drawTrianglesHistory = nil i.drawTrianglesHistory = nil
@ -420,11 +407,6 @@ func (i *Image) At(x, y int) (byte, byte, byte, byte) {
i.readPixelsFromGPUIfNeeded() i.readPixelsFromGPUIfNeeded()
// Even after readPixelsFromGPU, basePixels might be nil when OpenGL error happens.
if i.basePixels == nil {
return 0, 0, 0, 0
}
return i.basePixels.At(x, y) return i.basePixels.At(x, y)
} }
@ -441,7 +423,7 @@ func (i *Image) makeStaleIfDependingOn(target *Image) {
// readPixelsFromGPU reads the pixels from GPU and resolves the image's 'stale' state. // readPixelsFromGPU reads the pixels from GPU and resolves the image's 'stale' state.
func (i *Image) readPixelsFromGPU() { func (i *Image) readPixelsFromGPU() {
w, h := i.Size() w, h := i.Size()
i.basePixels = &Pixels{} i.basePixels = Pixels{}
i.basePixels.AddOrReplace(i.image.Pixels(), 0, 0, w, h) i.basePixels.AddOrReplace(i.image.Pixels(), 0, 0, w, h)
i.drawTrianglesHistory = nil i.drawTrianglesHistory = nil
i.stale = false i.stale = false
@ -499,14 +481,14 @@ func (i *Image) restore() error {
// The screen image should also be recreated because framebuffer might // The screen image should also be recreated because framebuffer might
// be changed. // be changed.
i.image = graphicscommand.NewScreenFramebufferImage(w, h) i.image = graphicscommand.NewScreenFramebufferImage(w, h)
i.basePixels = nil i.basePixels = Pixels{}
i.drawTrianglesHistory = nil i.drawTrianglesHistory = nil
i.stale = false i.stale = false
return nil return nil
} }
if i.volatile { if i.volatile {
i.image = graphicscommand.NewImage(w, h) i.image = graphicscommand.NewImage(w, h)
i.clearForInitialization() i.clear()
return nil return nil
} }
if i.stale { if i.stale {
@ -517,9 +499,7 @@ func (i *Image) restore() error {
gimg := graphicscommand.NewImage(w, h) gimg := graphicscommand.NewImage(w, h)
// Clear the image explicitly. // Clear the image explicitly.
clearImage(gimg) clearImage(gimg)
if i.basePixels != nil { i.basePixels.Apply(gimg)
i.basePixels.Apply(gimg)
}
for _, c := range i.drawTrianglesHistory { for _, c := range i.drawTrianglesHistory {
if c.image.hasDependency() { if c.image.hasDependency() {
@ -529,7 +509,7 @@ func (i *Image) restore() error {
} }
if len(i.drawTrianglesHistory) > 0 { if len(i.drawTrianglesHistory) > 0 {
i.basePixels = &Pixels{} i.basePixels = Pixels{}
i.basePixels.AddOrReplace(gimg.Pixels(), 0, 0, w, h) i.basePixels.AddOrReplace(gimg.Pixels(), 0, 0, w, h)
} }
@ -547,7 +527,7 @@ func (i *Image) Dispose() {
i.image.Dispose() i.image.Dispose()
i.image = nil i.image = nil
i.basePixels = nil i.basePixels = Pixels{}
i.drawTrianglesHistory = nil i.drawTrianglesHistory = nil
i.stale = false i.stale = false
} }