mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-12 12:08:58 +01:00
internal/restorable: bug fix: ReplacePixels on a sub-image might panic on Android
If regions by ReplacePixel are overlapped, this can panics. This can happen only on Android, where a context lost can happen. Thus, a sub-image cannot call a direct ReplacePixels. internal/buffer has to care this.
This commit is contained in:
parent
696bbc088f
commit
54b4e87506
11
image.go
11
image.go
@ -745,7 +745,7 @@ func (i *Image) Set(x, y int, clr color.Color) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
r, g, b, a := clr.RGBA()
|
r, g, b, a := clr.RGBA()
|
||||||
i.image.Set(byte(r>>8), byte(g>>8), byte(b>>8), byte(a>>8), x, y)
|
i.image.ReplacePartialPixels([]byte{byte(r >> 8), byte(g >> 8), byte(b >> 8), byte(a >> 8)}, x, y, 1, 1)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Dispose disposes the image data.
|
// Dispose disposes the image data.
|
||||||
@ -786,12 +786,17 @@ func (i *Image) ReplacePixels(pixels []byte) {
|
|||||||
if i.isDisposed() {
|
if i.isDisposed() {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
r := i.Bounds()
|
|
||||||
|
|
||||||
|
if !i.isSubImage() {
|
||||||
|
i.image.ReplacePixels(pixels)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
r := i.Bounds()
|
||||||
// Do not need to copy pixels here.
|
// Do not need to copy pixels here.
|
||||||
// * In internal/mipmap, pixels are copied when necessary.
|
// * In internal/mipmap, pixels are copied when necessary.
|
||||||
// * In internal/atlas, pixels are copied to make its paddings.
|
// * In internal/atlas, pixels are copied to make its paddings.
|
||||||
i.image.ReplacePixels(pixels, r.Min.X, r.Min.Y, r.Dx(), r.Dy())
|
i.image.ReplacePartialPixels(pixels, r.Min.X, r.Min.Y, r.Dx(), r.Dy())
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewImage returns an empty image.
|
// NewImage returns an empty image.
|
||||||
|
@ -46,23 +46,24 @@ func EndFrame(graphicsDriver graphicsdriver.Graphics) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func NewImage(width, height int) *Image {
|
func NewImage(width, height int) *Image {
|
||||||
i := &Image{}
|
i := &Image{
|
||||||
i.initialize(width, height)
|
width: width,
|
||||||
|
height: height,
|
||||||
|
}
|
||||||
|
i.initialize()
|
||||||
return i
|
return i
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Image) initialize(width, height int) {
|
func (i *Image) initialize() {
|
||||||
if maybeCanAddDelayedCommand() {
|
if maybeCanAddDelayedCommand() {
|
||||||
if tryAddDelayedCommand(func() error {
|
if tryAddDelayedCommand(func() error {
|
||||||
i.initialize(width, height)
|
i.initialize()
|
||||||
return nil
|
return nil
|
||||||
}) {
|
}) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
i.img = atlas.NewImage(width, height)
|
i.img = atlas.NewImage(i.width, i.height)
|
||||||
i.width = width
|
|
||||||
i.height = height
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Image) SetIndependent(independent bool) {
|
func (i *Image) SetIndependent(independent bool) {
|
||||||
@ -168,8 +169,8 @@ func (i *Image) DumpScreenshot(graphicsDriver graphicsdriver.Graphics, name stri
|
|||||||
|
|
||||||
// ReplacePixels replaces the pixels at the specified region.
|
// ReplacePixels replaces the pixels at the specified region.
|
||||||
// This call is not accumulated and send one draw call to replace pixels.
|
// This call is not accumulated and send one draw call to replace pixels.
|
||||||
func (i *Image) ReplacePixels(pix []byte, x, y, width, height int) {
|
func (i *Image) ReplacePixels(pix []byte) {
|
||||||
if l := 4 * width * height; len(pix) != l {
|
if l := 4 * i.width * i.height; len(pix) != l {
|
||||||
panic(fmt.Sprintf("buffered: len(pix) was %d but must be %d", len(pix), l))
|
panic(fmt.Sprintf("buffered: len(pix) was %d but must be %d", len(pix), l))
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -177,7 +178,7 @@ func (i *Image) ReplacePixels(pix []byte, x, y, width, height int) {
|
|||||||
copied := make([]byte, len(pix))
|
copied := make([]byte, len(pix))
|
||||||
copy(copied, pix)
|
copy(copied, pix)
|
||||||
if tryAddDelayedCommand(func() error {
|
if tryAddDelayedCommand(func() error {
|
||||||
i.ReplacePixels(copied, x, y, width, height)
|
i.ReplacePixels(copied)
|
||||||
return nil
|
return nil
|
||||||
}) {
|
}) {
|
||||||
return
|
return
|
||||||
@ -186,31 +187,27 @@ func (i *Image) ReplacePixels(pix []byte, x, y, width, height int) {
|
|||||||
|
|
||||||
i.invalidatePendingPixels()
|
i.invalidatePendingPixels()
|
||||||
|
|
||||||
i.img.ReplacePixels(pix, x, y, width, height)
|
i.img.ReplacePixels(pix, 0, 0, i.width, i.height)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set replaces the pixel at the specified position.
|
// ReplacePartial replaces the pixel at the specified partial region.
|
||||||
// This call might be accumulated and send one draw call to replace pixels for the accumulated calls.
|
// This call might be accumulated and send one draw call to replace pixels for the accumulated calls.
|
||||||
func (i *Image) Set(graphicsDriver graphicsdriver.Graphics, r, g, b, a byte, x, y int) error {
|
func (i *Image) ReplacePartialPixels(graphicsDriver graphicsdriver.Graphics, pix []byte, x, y, width, height int) error {
|
||||||
|
if l := 4 * width * height; len(pix) != l {
|
||||||
|
panic(fmt.Sprintf("buffered: len(pix) was %d but must be %d", len(pix), l))
|
||||||
|
}
|
||||||
|
|
||||||
if maybeCanAddDelayedCommand() {
|
if maybeCanAddDelayedCommand() {
|
||||||
if tryAddDelayedCommand(func() error {
|
if tryAddDelayedCommand(func() error {
|
||||||
i.Set(graphicsDriver, r, g, b, a, x, y)
|
copied := make([]byte, len(pix))
|
||||||
|
copy(copied, pix)
|
||||||
|
i.ReplacePartialPixels(graphicsDriver, copied, x, y, width, height)
|
||||||
return nil
|
return nil
|
||||||
}) {
|
}) {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if i.width == 1 && i.height == 1 {
|
|
||||||
i.invalidatePendingPixels()
|
|
||||||
|
|
||||||
// Call ReplacePixels immediately. Do not buffer the command.
|
|
||||||
// If a lot of new images are created but they are used at different timings,
|
|
||||||
// pixels are sent to GPU at different timings, which is very inefficient.
|
|
||||||
i.img.ReplacePixels([]byte{r, g, b, a}, x, y, 1, 1)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if i.pixels == nil {
|
if i.pixels == nil {
|
||||||
pix, err := i.img.Pixels(graphicsDriver, 0, 0, i.width, i.height)
|
pix, err := i.img.Pixels(graphicsDriver, 0, 0, i.width, i.height)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -218,16 +215,14 @@ func (i *Image) Set(graphicsDriver graphicsdriver.Graphics, r, g, b, a byte, x,
|
|||||||
}
|
}
|
||||||
i.pixels = pix
|
i.pixels = pix
|
||||||
}
|
}
|
||||||
i.setPendingPixel(r, g, b, a, x, y)
|
i.replacePendingPixels(pix, x, y, width, height)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Image) setPendingPixel(r, g, b, a byte, x, y int) {
|
func (i *Image) replacePendingPixels(pix []byte, x, y, width, height int) {
|
||||||
idx := 4 * (y*i.width + x)
|
for j := 0; j < height; j++ {
|
||||||
i.pixels[idx] = r
|
copy(i.pixels[4*((j+y)*i.width+x):], pix[4*j*width:4*(j+1)*width])
|
||||||
i.pixels[idx+1] = g
|
}
|
||||||
i.pixels[idx+2] = b
|
|
||||||
i.pixels[idx+3] = a
|
|
||||||
i.needsToResolvePixels = true
|
i.needsToResolvePixels = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,13 +71,13 @@ func (m *Mipmap) DumpScreenshot(graphicsDriver graphicsdriver.Graphics, name str
|
|||||||
return m.orig.DumpScreenshot(graphicsDriver, name, blackbg)
|
return m.orig.DumpScreenshot(graphicsDriver, name, blackbg)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *Mipmap) ReplacePixels(pix []byte, x, y, width, height int) {
|
func (m *Mipmap) ReplacePixels(pix []byte) {
|
||||||
m.orig.ReplacePixels(pix, x, y, width, height)
|
m.orig.ReplacePixels(pix)
|
||||||
m.disposeMipmaps()
|
m.disposeMipmaps()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *Mipmap) Set(graphicsDriver graphicsdriver.Graphics, r, g, b, a byte, x, y int) error {
|
func (m *Mipmap) ReplacePartialPixels(graphicsDriver graphicsdriver.Graphics, pix []byte, x, y, width, height int) error {
|
||||||
if err := m.orig.Set(graphicsDriver, r, g, b, a, x, y); err != nil {
|
if err := m.orig.ReplacePartialPixels(graphicsDriver, pix, x, y, width, height); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
m.disposeMipmaps()
|
m.disposeMipmaps()
|
||||||
|
@ -289,7 +289,7 @@ func (i *Image) ClearPixels(x, y, width, height int) {
|
|||||||
|
|
||||||
// ReplacePixels replaces the image pixels with the given pixels slice.
|
// ReplacePixels replaces the image pixels with the given pixels slice.
|
||||||
//
|
//
|
||||||
// ReplacePixels for a part is forbidden if the image is rendered with DrawTriangles or Fill.
|
// The specified region must not be overlapped with other regions by ReplacePixels.
|
||||||
func (i *Image) ReplacePixels(pixels []byte, x, y, width, height int) {
|
func (i *Image) ReplacePixels(pixels []byte, x, y, width, height int) {
|
||||||
if width <= 0 || height <= 0 {
|
if width <= 0 || height <= 0 {
|
||||||
panic("restorable: width/height must be positive")
|
panic("restorable: width/height must be positive")
|
||||||
|
@ -68,15 +68,15 @@ func (i *Image) DrawTriangles(srcs [graphics.ShaderImageNum]*Image, vertices []f
|
|||||||
i.mipmap.DrawTriangles(srcMipmaps, vertices, indices, colorm, mode, filter, address, dstRegion, srcRegion, subimageOffsets, s, uniforms, evenOdd, canSkipMipmap)
|
i.mipmap.DrawTriangles(srcMipmaps, vertices, indices, colorm, mode, filter, address, dstRegion, srcRegion, subimageOffsets, s, uniforms, evenOdd, canSkipMipmap)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Image) ReplacePixels(pix []byte, x, y, width, height int) {
|
func (i *Image) ReplacePixels(pix []byte) {
|
||||||
i.mipmap.ReplacePixels(pix, x, y, width, height)
|
i.mipmap.ReplacePixels(pix)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Image) Set(r, g, b, a byte, x, y int) {
|
func (i *Image) ReplacePartialPixels(pix []byte, x, y, width, height int) {
|
||||||
if theGlobalState.error() != nil {
|
if theGlobalState.error() != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if err := i.mipmap.Set(graphicsDriver(), r, g, b, a, x, y); err != nil {
|
if err := i.mipmap.ReplacePartialPixels(graphicsDriver(), pix, x, y, width, height); err != nil {
|
||||||
theGlobalState.setError(err)
|
theGlobalState.setError(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user