2017-05-31 18:27:56 +02:00
|
|
|
// Copyright 2017 The Ebiten Authors
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package restorable_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
2017-06-01 19:13:18 +02:00
|
|
|
"image"
|
2017-05-31 18:27:56 +02:00
|
|
|
"image/color"
|
|
|
|
"os"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/hajimehoshi/ebiten"
|
2018-10-28 12:25:52 +01:00
|
|
|
"github.com/hajimehoshi/ebiten/internal/graphics"
|
2017-05-31 18:27:56 +02:00
|
|
|
. "github.com/hajimehoshi/ebiten/internal/restorable"
|
2018-04-05 20:54:10 +02:00
|
|
|
"github.com/hajimehoshi/ebiten/internal/testflock"
|
2017-05-31 18:27:56 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestMain(m *testing.M) {
|
2018-04-05 20:54:10 +02:00
|
|
|
testflock.Lock()
|
|
|
|
defer testflock.Unlock()
|
|
|
|
|
2017-07-02 08:58:00 +02:00
|
|
|
EnableRestoringForTesting()
|
2017-05-31 18:27:56 +02:00
|
|
|
code := 0
|
|
|
|
regularTermination := errors.New("regular termination")
|
|
|
|
f := func(screen *ebiten.Image) error {
|
|
|
|
code = m.Run()
|
|
|
|
return regularTermination
|
|
|
|
}
|
|
|
|
if err := ebiten.Run(f, 320, 240, 1, "Test"); err != nil && err != regularTermination {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
os.Exit(code)
|
|
|
|
}
|
|
|
|
|
2019-02-12 15:31:52 +01:00
|
|
|
func pixelsToColor(p *Pixels, index int) color.RGBA {
|
2017-06-01 19:13:18 +02:00
|
|
|
i := index * 4
|
2019-02-12 15:31:52 +01:00
|
|
|
return color.RGBA{p.At(i), p.At(i + 1), p.At(i + 2), p.At(i + 3)}
|
2017-05-31 18:27:56 +02:00
|
|
|
}
|
|
|
|
|
2018-02-03 08:43:18 +01:00
|
|
|
func abs(x int) int {
|
|
|
|
if x < 0 {
|
|
|
|
return -x
|
|
|
|
}
|
|
|
|
return x
|
|
|
|
}
|
|
|
|
|
|
|
|
// sameColors compares c1 and c2 and returns a boolean value indicating
|
|
|
|
// if the two colors are (almost) same.
|
|
|
|
//
|
|
|
|
// Pixels read from GPU might include errors (#492), and
|
|
|
|
// sameColors considers such errors as delta.
|
|
|
|
func sameColors(c1, c2 color.RGBA, delta int) bool {
|
|
|
|
return abs(int(c1.R)-int(c2.R)) <= delta &&
|
|
|
|
abs(int(c1.G)-int(c2.G)) <= delta &&
|
|
|
|
abs(int(c1.B)-int(c2.B)) <= delta &&
|
|
|
|
abs(int(c1.A)-int(c2.A)) <= delta
|
|
|
|
}
|
|
|
|
|
2017-05-31 18:27:56 +02:00
|
|
|
func TestRestore(t *testing.T) {
|
2019-02-12 07:05:05 +01:00
|
|
|
img0 := NewImage(1, 1)
|
2018-04-05 19:15:49 +02:00
|
|
|
defer img0.Dispose()
|
|
|
|
|
2017-05-31 18:27:56 +02:00
|
|
|
clr0 := color.RGBA{0x00, 0x00, 0x00, 0xff}
|
2019-02-12 15:31:52 +01:00
|
|
|
img0.Fill(clr0.R, clr0.G, clr0.B, clr0.A)
|
2018-07-11 19:11:18 +02:00
|
|
|
ResolveStaleImages()
|
2019-05-26 14:10:25 +02:00
|
|
|
if err := RestoreIfNeeded(); err != nil {
|
2017-05-31 18:27:56 +02:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
want := clr0
|
2019-02-12 15:31:52 +01:00
|
|
|
got := pixelsToColor(img0.BasePixelsForTesting(), 0)
|
2018-02-03 08:43:18 +01:00
|
|
|
if !sameColors(got, want, 1) {
|
2017-05-31 18:27:56 +02:00
|
|
|
t.Errorf("got %v, want %v", got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-30 03:08:38 +01:00
|
|
|
func TestRestoreWithoutDraw(t *testing.T) {
|
2019-02-12 07:05:05 +01:00
|
|
|
img0 := NewImage(1024, 1024)
|
2018-10-30 03:08:38 +01:00
|
|
|
defer img0.Dispose()
|
|
|
|
|
|
|
|
// If there is no drawing command on img0, img0 is cleared when restored.
|
|
|
|
|
|
|
|
ResolveStaleImages()
|
2019-05-26 14:10:25 +02:00
|
|
|
if err := RestoreIfNeeded(); err != nil {
|
2018-10-30 03:08:38 +01:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < 1024*1024; i++ {
|
|
|
|
want := color.RGBA{0x00, 0x00, 0x00, 0x00}
|
2019-02-12 15:31:52 +01:00
|
|
|
got := pixelsToColor(img0.BasePixelsForTesting(), i)
|
2018-10-30 03:08:38 +01:00
|
|
|
if !sameColors(got, want, 0) {
|
|
|
|
t.Errorf("got %v, want %v", got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-14 15:44:02 +01:00
|
|
|
func quadVertices(src *Image, sw, sh, x, y int) []float32 {
|
2019-06-20 18:53:05 +02:00
|
|
|
vs := graphics.VertexSlice(4)
|
2019-06-21 04:09:48 +02:00
|
|
|
graphics.PutQuadVertices(vs, src, 0, 0, sw, sh,
|
2019-02-14 13:04:34 +01:00
|
|
|
1, 0, 0, 1, float32(x), float32(y),
|
|
|
|
1, 1, 1, 1)
|
2019-06-20 18:53:05 +02:00
|
|
|
return vs
|
2019-02-14 13:04:34 +01:00
|
|
|
}
|
|
|
|
|
2017-06-01 03:44:28 +02:00
|
|
|
func TestRestoreChain(t *testing.T) {
|
|
|
|
const num = 10
|
|
|
|
imgs := []*Image{}
|
|
|
|
for i := 0; i < num; i++ {
|
2019-02-12 07:05:05 +01:00
|
|
|
img := NewImage(1, 1)
|
2017-06-01 19:13:18 +02:00
|
|
|
imgs = append(imgs, img)
|
2017-06-01 03:44:28 +02:00
|
|
|
}
|
|
|
|
defer func() {
|
|
|
|
for _, img := range imgs {
|
|
|
|
img.Dispose()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
clr := color.RGBA{0x00, 0x00, 0x00, 0xff}
|
2019-02-12 15:31:52 +01:00
|
|
|
imgs[0].Fill(clr.R, clr.G, clr.B, clr.A)
|
2017-06-01 03:44:28 +02:00
|
|
|
for i := 0; i < num-1; i++ {
|
2019-02-14 15:44:02 +01:00
|
|
|
vs := quadVertices(imgs[i], 1, 1, 0, 0)
|
2018-10-28 15:03:06 +01:00
|
|
|
is := graphics.QuadIndices()
|
2019-04-22 16:12:03 +02:00
|
|
|
imgs[i+1].DrawTriangles(imgs[i], vs, is, nil, graphics.CompositeModeCopy, graphics.FilterNearest, graphics.AddressClampToZero)
|
2017-06-01 03:44:28 +02:00
|
|
|
}
|
2018-07-11 19:11:18 +02:00
|
|
|
ResolveStaleImages()
|
2019-05-26 14:10:25 +02:00
|
|
|
if err := RestoreIfNeeded(); err != nil {
|
2017-06-01 03:44:28 +02:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
want := clr
|
|
|
|
for i, img := range imgs {
|
2019-02-12 15:31:52 +01:00
|
|
|
got := pixelsToColor(img.BasePixelsForTesting(), 0)
|
2018-02-03 08:43:18 +01:00
|
|
|
if !sameColors(got, want, 1) {
|
2017-06-01 03:44:28 +02:00
|
|
|
t.Errorf("%d: got %v, want %v", i, got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-25 12:42:44 +02:00
|
|
|
func TestRestoreChain2(t *testing.T) {
|
2018-06-10 15:59:00 +02:00
|
|
|
const (
|
|
|
|
num = 10
|
|
|
|
w = 1
|
|
|
|
h = 1
|
|
|
|
)
|
2018-03-25 12:42:44 +02:00
|
|
|
imgs := []*Image{}
|
|
|
|
for i := 0; i < num; i++ {
|
2019-02-12 07:05:05 +01:00
|
|
|
img := NewImage(w, h)
|
2018-03-25 12:42:44 +02:00
|
|
|
imgs = append(imgs, img)
|
|
|
|
}
|
|
|
|
defer func() {
|
|
|
|
for _, img := range imgs {
|
|
|
|
img.Dispose()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
clr0 := color.RGBA{0xff, 0x00, 0x00, 0xff}
|
2019-02-12 15:31:52 +01:00
|
|
|
imgs[0].Fill(clr0.R, clr0.G, clr0.B, clr0.A)
|
2018-03-25 12:42:44 +02:00
|
|
|
clr7 := color.RGBA{0x00, 0xff, 0x00, 0xff}
|
2019-02-12 15:31:52 +01:00
|
|
|
imgs[7].Fill(clr7.R, clr7.G, clr7.B, clr7.A)
|
2018-03-25 12:42:44 +02:00
|
|
|
clr8 := color.RGBA{0x00, 0x00, 0xff, 0xff}
|
2019-02-12 15:31:52 +01:00
|
|
|
imgs[8].Fill(clr8.R, clr8.G, clr8.B, clr8.A)
|
2018-03-25 12:42:44 +02:00
|
|
|
|
2018-10-28 15:03:06 +01:00
|
|
|
is := graphics.QuadIndices()
|
2019-04-22 16:12:03 +02:00
|
|
|
imgs[8].DrawTriangles(imgs[7], quadVertices(imgs[7], w, h, 0, 0), is, nil, graphics.CompositeModeCopy, graphics.FilterNearest, graphics.AddressClampToZero)
|
|
|
|
imgs[9].DrawTriangles(imgs[8], quadVertices(imgs[8], w, h, 0, 0), is, nil, graphics.CompositeModeCopy, graphics.FilterNearest, graphics.AddressClampToZero)
|
2018-03-25 12:42:44 +02:00
|
|
|
for i := 0; i < 7; i++ {
|
2019-04-22 16:12:03 +02:00
|
|
|
imgs[i+1].DrawTriangles(imgs[i], quadVertices(imgs[i], w, h, 0, 0), is, nil, graphics.CompositeModeCopy, graphics.FilterNearest, graphics.AddressClampToZero)
|
2018-03-25 12:42:44 +02:00
|
|
|
}
|
|
|
|
|
2018-07-11 19:11:18 +02:00
|
|
|
ResolveStaleImages()
|
2019-05-26 14:10:25 +02:00
|
|
|
if err := RestoreIfNeeded(); err != nil {
|
2018-03-25 12:42:44 +02:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
for i, img := range imgs {
|
|
|
|
want := clr0
|
|
|
|
if i == 8 || i == 9 {
|
|
|
|
want = clr7
|
|
|
|
}
|
2019-02-12 15:31:52 +01:00
|
|
|
got := pixelsToColor(img.BasePixelsForTesting(), 0)
|
2018-03-25 12:42:44 +02:00
|
|
|
if !sameColors(got, want, 1) {
|
|
|
|
t.Errorf("%d: got %v, want %v", i, got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-31 19:37:21 +02:00
|
|
|
func TestRestoreOverrideSource(t *testing.T) {
|
2018-06-10 15:59:00 +02:00
|
|
|
const (
|
|
|
|
w = 1
|
|
|
|
h = 1
|
|
|
|
)
|
2019-02-12 07:05:05 +01:00
|
|
|
img0 := NewImage(w, h)
|
|
|
|
img1 := NewImage(w, h)
|
|
|
|
img2 := NewImage(w, h)
|
|
|
|
img3 := NewImage(w, h)
|
2017-05-31 18:27:56 +02:00
|
|
|
defer func() {
|
|
|
|
img3.Dispose()
|
|
|
|
img2.Dispose()
|
|
|
|
img1.Dispose()
|
|
|
|
img0.Dispose()
|
|
|
|
}()
|
|
|
|
clr0 := color.RGBA{0x00, 0x00, 0x00, 0xff}
|
|
|
|
clr1 := color.RGBA{0x00, 0x00, 0x01, 0xff}
|
2019-02-12 15:31:52 +01:00
|
|
|
img1.Fill(clr0.R, clr0.G, clr0.B, clr0.A)
|
2018-10-28 15:03:06 +01:00
|
|
|
is := graphics.QuadIndices()
|
2019-04-22 16:12:03 +02:00
|
|
|
img2.DrawTriangles(img1, quadVertices(img1, w, h, 0, 0), is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
|
|
|
img3.DrawTriangles(img2, quadVertices(img2, w, h, 0, 0), is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
2019-02-12 15:31:52 +01:00
|
|
|
img0.Fill(clr1.R, clr1.G, clr1.B, clr1.A)
|
2019-04-22 16:12:03 +02:00
|
|
|
img1.DrawTriangles(img0, quadVertices(img0, w, h, 0, 0), is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
2018-07-11 19:11:18 +02:00
|
|
|
ResolveStaleImages()
|
2019-05-26 14:10:25 +02:00
|
|
|
if err := RestoreIfNeeded(); err != nil {
|
2017-05-31 18:27:56 +02:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
testCases := []struct {
|
|
|
|
name string
|
|
|
|
want color.RGBA
|
|
|
|
got color.RGBA
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
"0",
|
|
|
|
clr1,
|
2019-02-12 15:31:52 +01:00
|
|
|
pixelsToColor(img0.BasePixelsForTesting(), 0),
|
2017-05-31 18:27:56 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"1",
|
|
|
|
clr1,
|
2019-02-12 15:31:52 +01:00
|
|
|
pixelsToColor(img1.BasePixelsForTesting(), 0),
|
2017-05-31 18:27:56 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"2",
|
|
|
|
clr0,
|
2019-02-12 15:31:52 +01:00
|
|
|
pixelsToColor(img2.BasePixelsForTesting(), 0),
|
2017-05-31 18:27:56 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"3",
|
|
|
|
clr0,
|
2019-02-12 15:31:52 +01:00
|
|
|
pixelsToColor(img3.BasePixelsForTesting(), 0),
|
2017-05-31 18:27:56 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, c := range testCases {
|
2018-02-03 08:43:18 +01:00
|
|
|
if !sameColors(c.got, c.want, 1) {
|
2017-05-31 18:27:56 +02:00
|
|
|
t.Errorf("%s: got %v, want %v", c.name, c.got, c.want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-05-31 19:37:21 +02:00
|
|
|
|
2017-06-02 14:58:26 +02:00
|
|
|
func TestRestoreComplexGraph(t *testing.T) {
|
2018-06-10 15:59:00 +02:00
|
|
|
const (
|
|
|
|
w = 4
|
|
|
|
h = 1
|
|
|
|
)
|
2017-06-02 14:58:26 +02:00
|
|
|
// 0 -> 3
|
|
|
|
// 1 -> 3
|
|
|
|
// 1 -> 4
|
|
|
|
// 2 -> 4
|
|
|
|
// 2 -> 7
|
|
|
|
// 3 -> 5
|
|
|
|
// 3 -> 6
|
|
|
|
// 3 -> 7
|
|
|
|
// 4 -> 6
|
2018-06-10 15:59:00 +02:00
|
|
|
base := image.NewRGBA(image.Rect(0, 0, w, h))
|
2017-06-01 19:13:18 +02:00
|
|
|
base.Pix[0] = 0xff
|
|
|
|
base.Pix[1] = 0xff
|
|
|
|
base.Pix[2] = 0xff
|
|
|
|
base.Pix[3] = 0xff
|
2018-02-25 15:13:06 +01:00
|
|
|
img0 := newImageFromImage(base)
|
|
|
|
img1 := newImageFromImage(base)
|
|
|
|
img2 := newImageFromImage(base)
|
2019-02-12 07:05:05 +01:00
|
|
|
img3 := NewImage(w, h)
|
|
|
|
img4 := NewImage(w, h)
|
|
|
|
img5 := NewImage(w, h)
|
|
|
|
img6 := NewImage(w, h)
|
|
|
|
img7 := NewImage(w, h)
|
2017-05-31 19:37:21 +02:00
|
|
|
defer func() {
|
2017-06-02 14:58:26 +02:00
|
|
|
img7.Dispose()
|
|
|
|
img6.Dispose()
|
|
|
|
img5.Dispose()
|
|
|
|
img4.Dispose()
|
|
|
|
img3.Dispose()
|
|
|
|
img2.Dispose()
|
2017-05-31 19:37:21 +02:00
|
|
|
img1.Dispose()
|
|
|
|
img0.Dispose()
|
|
|
|
}()
|
2019-02-14 15:44:02 +01:00
|
|
|
vs := quadVertices(img0, w, h, 0, 0)
|
2018-10-28 15:03:06 +01:00
|
|
|
is := graphics.QuadIndices()
|
2019-04-22 16:12:03 +02:00
|
|
|
img3.DrawTriangles(img0, vs, is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
2019-02-14 15:44:02 +01:00
|
|
|
vs = quadVertices(img1, w, h, 1, 0)
|
2019-04-22 16:12:03 +02:00
|
|
|
img3.DrawTriangles(img1, vs, is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
2019-02-14 15:44:02 +01:00
|
|
|
vs = quadVertices(img1, w, h, 1, 0)
|
2019-04-22 16:12:03 +02:00
|
|
|
img4.DrawTriangles(img1, vs, is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
2019-02-14 15:44:02 +01:00
|
|
|
vs = quadVertices(img2, w, h, 2, 0)
|
2019-04-22 16:12:03 +02:00
|
|
|
img4.DrawTriangles(img2, vs, is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
2019-02-14 15:44:02 +01:00
|
|
|
vs = quadVertices(img3, w, h, 0, 0)
|
2019-04-22 16:12:03 +02:00
|
|
|
img5.DrawTriangles(img3, vs, is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
2019-02-14 15:44:02 +01:00
|
|
|
vs = quadVertices(img3, w, h, 0, 0)
|
2019-04-22 16:12:03 +02:00
|
|
|
img6.DrawTriangles(img3, vs, is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
2019-02-14 15:44:02 +01:00
|
|
|
vs = quadVertices(img4, w, h, 1, 0)
|
2019-04-22 16:12:03 +02:00
|
|
|
img6.DrawTriangles(img4, vs, is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
2019-02-14 15:44:02 +01:00
|
|
|
vs = quadVertices(img2, w, h, 0, 0)
|
2019-04-22 16:12:03 +02:00
|
|
|
img7.DrawTriangles(img2, vs, is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
2019-02-14 15:44:02 +01:00
|
|
|
vs = quadVertices(img3, w, h, 2, 0)
|
2019-04-22 16:12:03 +02:00
|
|
|
img7.DrawTriangles(img3, vs, is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
2018-07-11 19:11:18 +02:00
|
|
|
ResolveStaleImages()
|
2019-05-26 14:10:25 +02:00
|
|
|
if err := RestoreIfNeeded(); err != nil {
|
2017-05-31 19:37:21 +02:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
testCases := []struct {
|
2017-06-02 14:58:26 +02:00
|
|
|
name string
|
|
|
|
out string
|
|
|
|
image *Image
|
2017-05-31 19:37:21 +02:00
|
|
|
}{
|
|
|
|
{
|
2017-06-02 14:58:26 +02:00
|
|
|
"0",
|
|
|
|
"*---",
|
|
|
|
img0,
|
2017-05-31 19:37:21 +02:00
|
|
|
},
|
|
|
|
{
|
2017-06-02 14:58:26 +02:00
|
|
|
"1",
|
|
|
|
"*---",
|
|
|
|
img1,
|
2017-06-01 19:13:18 +02:00
|
|
|
},
|
|
|
|
{
|
2017-06-02 15:00:02 +02:00
|
|
|
"2",
|
2017-06-02 14:58:26 +02:00
|
|
|
"*---",
|
|
|
|
img2,
|
2017-06-01 19:13:18 +02:00
|
|
|
},
|
|
|
|
{
|
2017-06-02 14:58:26 +02:00
|
|
|
"3",
|
|
|
|
"**--",
|
|
|
|
img3,
|
2017-06-01 19:13:18 +02:00
|
|
|
},
|
|
|
|
{
|
2017-06-02 14:58:26 +02:00
|
|
|
"4",
|
|
|
|
"-**-",
|
|
|
|
img4,
|
2017-06-01 19:13:18 +02:00
|
|
|
},
|
|
|
|
{
|
2017-06-02 14:58:26 +02:00
|
|
|
"5",
|
|
|
|
"**--",
|
|
|
|
img5,
|
2017-06-01 19:13:18 +02:00
|
|
|
},
|
|
|
|
{
|
2017-06-02 14:58:26 +02:00
|
|
|
"6",
|
|
|
|
"****",
|
|
|
|
img6,
|
2017-06-01 19:13:18 +02:00
|
|
|
},
|
|
|
|
{
|
2017-06-02 14:58:26 +02:00
|
|
|
"7",
|
|
|
|
"*-**",
|
|
|
|
img7,
|
2017-05-31 19:37:21 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, c := range testCases {
|
2017-06-02 14:58:26 +02:00
|
|
|
for i := 0; i < 4; i++ {
|
|
|
|
want := color.RGBA{}
|
|
|
|
if c.out[i] == '*' {
|
|
|
|
want = color.RGBA{0xff, 0xff, 0xff, 0xff}
|
|
|
|
}
|
2019-02-12 15:31:52 +01:00
|
|
|
got := pixelsToColor(c.image.BasePixelsForTesting(), i)
|
2018-02-03 08:43:18 +01:00
|
|
|
if !sameColors(got, want, 1) {
|
2017-06-02 14:58:26 +02:00
|
|
|
t.Errorf("%s[%d]: got %v, want %v", c.name, i, got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-25 15:13:06 +01:00
|
|
|
func newImageFromImage(rgba *image.RGBA) *Image {
|
|
|
|
s := rgba.Bounds().Size()
|
2019-02-12 07:05:05 +01:00
|
|
|
img := NewImage(s.X, s.Y)
|
2018-02-28 16:27:55 +01:00
|
|
|
img.ReplacePixels(rgba.Pix, 0, 0, s.X, s.Y)
|
2018-02-25 15:13:06 +01:00
|
|
|
return img
|
|
|
|
}
|
|
|
|
|
2017-06-02 14:58:26 +02:00
|
|
|
func TestRestoreRecursive(t *testing.T) {
|
2018-06-10 15:59:00 +02:00
|
|
|
const (
|
|
|
|
w = 4
|
|
|
|
h = 1
|
|
|
|
)
|
|
|
|
base := image.NewRGBA(image.Rect(0, 0, w, h))
|
2017-06-02 14:58:26 +02:00
|
|
|
base.Pix[0] = 0xff
|
|
|
|
base.Pix[1] = 0xff
|
|
|
|
base.Pix[2] = 0xff
|
|
|
|
base.Pix[3] = 0xff
|
2018-03-19 16:37:02 +01:00
|
|
|
|
2018-02-25 15:13:06 +01:00
|
|
|
img0 := newImageFromImage(base)
|
2019-02-12 07:05:05 +01:00
|
|
|
img1 := NewImage(w, h)
|
2017-06-02 14:58:26 +02:00
|
|
|
defer func() {
|
|
|
|
img1.Dispose()
|
|
|
|
img0.Dispose()
|
|
|
|
}()
|
2018-10-28 15:03:06 +01:00
|
|
|
is := graphics.QuadIndices()
|
2019-04-22 16:12:03 +02:00
|
|
|
img1.DrawTriangles(img0, quadVertices(img0, w, h, 1, 0), is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
|
|
|
img0.DrawTriangles(img1, quadVertices(img1, w, h, 1, 0), is, nil, graphics.CompositeModeSourceOver, graphics.FilterNearest, graphics.AddressClampToZero)
|
2018-07-11 19:11:18 +02:00
|
|
|
ResolveStaleImages()
|
2019-05-26 14:10:25 +02:00
|
|
|
if err := RestoreIfNeeded(); err != nil {
|
2017-06-02 14:58:26 +02:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
testCases := []struct {
|
|
|
|
name string
|
|
|
|
out string
|
|
|
|
image *Image
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
"0",
|
|
|
|
"*-*-",
|
|
|
|
img0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"1",
|
|
|
|
"-*--",
|
|
|
|
img1,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, c := range testCases {
|
|
|
|
for i := 0; i < 4; i++ {
|
|
|
|
want := color.RGBA{}
|
|
|
|
if c.out[i] == '*' {
|
|
|
|
want = color.RGBA{0xff, 0xff, 0xff, 0xff}
|
|
|
|
}
|
2019-02-12 15:31:52 +01:00
|
|
|
got := pixelsToColor(c.image.BasePixelsForTesting(), i)
|
2018-02-03 08:43:18 +01:00
|
|
|
if !sameColors(got, want, 1) {
|
2017-06-02 14:58:26 +02:00
|
|
|
t.Errorf("%s[%d]: got %v, want %v", c.name, i, got, want)
|
|
|
|
}
|
2017-05-31 19:37:21 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-28 16:27:55 +01:00
|
|
|
func TestReplacePixels(t *testing.T) {
|
2019-02-12 07:05:05 +01:00
|
|
|
img := NewImage(17, 31)
|
2018-03-19 16:37:02 +01:00
|
|
|
defer img.Dispose()
|
|
|
|
|
2018-02-28 16:27:55 +01:00
|
|
|
pix := make([]byte, 4*4*4)
|
|
|
|
for i := range pix {
|
|
|
|
pix[i] = 0xff
|
|
|
|
}
|
|
|
|
img.ReplacePixels(pix, 5, 7, 4, 4)
|
2018-03-19 16:37:02 +01:00
|
|
|
// Check the region (5, 7)-(9, 11). Outside state is indeterministic.
|
|
|
|
for j := 7; j < 11; j++ {
|
|
|
|
for i := 5; i < 9; i++ {
|
2019-01-13 20:25:39 +01:00
|
|
|
r, g, b, a := img.At(i, j)
|
|
|
|
got := color.RGBA{r, g, b, a}
|
2018-03-19 16:37:02 +01:00
|
|
|
want := color.RGBA{0xff, 0xff, 0xff, 0xff}
|
2018-02-28 16:27:55 +01:00
|
|
|
if got != want {
|
|
|
|
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-07-11 19:11:18 +02:00
|
|
|
ResolveStaleImages()
|
2019-05-26 14:10:25 +02:00
|
|
|
if err := RestoreIfNeeded(); err != nil {
|
2018-02-28 18:43:59 +01:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2018-03-19 16:37:02 +01:00
|
|
|
for j := 7; j < 11; j++ {
|
|
|
|
for i := 5; i < 9; i++ {
|
2019-01-13 20:25:39 +01:00
|
|
|
r, g, b, a := img.At(i, j)
|
|
|
|
got := color.RGBA{r, g, b, a}
|
2018-03-19 16:37:02 +01:00
|
|
|
want := color.RGBA{0xff, 0xff, 0xff, 0xff}
|
2018-02-28 18:43:59 +01:00
|
|
|
if got != want {
|
|
|
|
t.Errorf("img.At(%d, %d): got: %v, want: %v", i, j, got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-02-28 16:27:55 +01:00
|
|
|
}
|
|
|
|
|
2019-04-22 16:12:03 +02:00
|
|
|
func TestDrawTrianglesAndReplacePixels(t *testing.T) {
|
2018-03-19 16:37:02 +01:00
|
|
|
base := image.NewRGBA(image.Rect(0, 0, 1, 1))
|
|
|
|
base.Pix[0] = 0xff
|
2019-01-05 19:15:32 +01:00
|
|
|
base.Pix[1] = 0
|
|
|
|
base.Pix[2] = 0
|
2018-03-19 16:37:02 +01:00
|
|
|
base.Pix[3] = 0xff
|
|
|
|
img0 := newImageFromImage(base)
|
|
|
|
defer img0.Dispose()
|
2019-02-12 07:05:05 +01:00
|
|
|
img1 := NewImage(2, 1)
|
2018-03-19 16:37:02 +01:00
|
|
|
defer img1.Dispose()
|
2018-06-10 15:59:00 +02:00
|
|
|
|
2019-02-14 15:44:02 +01:00
|
|
|
vs := quadVertices(img0, 1, 1, 0, 0)
|
2018-10-28 15:03:06 +01:00
|
|
|
is := graphics.QuadIndices()
|
2019-04-22 16:12:03 +02:00
|
|
|
img1.DrawTriangles(img0, vs, is, nil, graphics.CompositeModeCopy, graphics.FilterNearest, graphics.AddressClampToZero)
|
2019-01-05 19:15:32 +01:00
|
|
|
img1.ReplacePixels([]byte{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}, 0, 0, 2, 1)
|
2018-03-19 16:37:02 +01:00
|
|
|
|
2018-07-11 19:11:18 +02:00
|
|
|
ResolveStaleImages()
|
2019-05-26 14:10:25 +02:00
|
|
|
if err := RestoreIfNeeded(); err != nil {
|
2018-03-19 16:37:02 +01:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-01-13 20:25:39 +01:00
|
|
|
r, g, b, a := img1.At(0, 0)
|
|
|
|
got := color.RGBA{r, g, b, a}
|
2018-03-19 16:37:02 +01:00
|
|
|
want := color.RGBA{0xff, 0xff, 0xff, 0xff}
|
|
|
|
if !sameColors(got, want, 1) {
|
|
|
|
t.Errorf("got: %v, want: %v", got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-24 16:50:11 +01:00
|
|
|
func TestDispose(t *testing.T) {
|
|
|
|
base0 := image.NewRGBA(image.Rect(0, 0, 1, 1))
|
|
|
|
img0 := newImageFromImage(base0)
|
|
|
|
defer img0.Dispose()
|
|
|
|
|
|
|
|
base1 := image.NewRGBA(image.Rect(0, 0, 1, 1))
|
|
|
|
img1 := newImageFromImage(base1)
|
|
|
|
|
|
|
|
base2 := image.NewRGBA(image.Rect(0, 0, 1, 1))
|
|
|
|
base2.Pix[0] = 0xff
|
|
|
|
base2.Pix[1] = 0xff
|
|
|
|
base2.Pix[2] = 0xff
|
|
|
|
base2.Pix[3] = 0xff
|
|
|
|
img2 := newImageFromImage(base2)
|
|
|
|
defer img2.Dispose()
|
|
|
|
|
2018-10-28 15:03:06 +01:00
|
|
|
is := graphics.QuadIndices()
|
2019-04-22 16:12:03 +02:00
|
|
|
img1.DrawTriangles(img2, quadVertices(img2, 1, 1, 0, 0), is, nil, graphics.CompositeModeCopy, graphics.FilterNearest, graphics.AddressClampToZero)
|
|
|
|
img0.DrawTriangles(img1, quadVertices(img1, 1, 1, 0, 0), is, nil, graphics.CompositeModeCopy, graphics.FilterNearest, graphics.AddressClampToZero)
|
2018-03-24 16:50:11 +01:00
|
|
|
img1.Dispose()
|
|
|
|
|
2018-07-11 19:11:18 +02:00
|
|
|
ResolveStaleImages()
|
2019-05-26 14:10:25 +02:00
|
|
|
if err := RestoreIfNeeded(); err != nil {
|
2018-03-24 16:50:11 +01:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-01-13 20:25:39 +01:00
|
|
|
r, g, b, a := img0.At(0, 0)
|
|
|
|
got := color.RGBA{r, g, b, a}
|
2018-03-24 16:50:11 +01:00
|
|
|
want := color.RGBA{0xff, 0xff, 0xff, 0xff}
|
|
|
|
if !sameColors(got, want, 1) {
|
|
|
|
t.Errorf("got: %v, want: %v", got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-28 15:50:00 +02:00
|
|
|
func TestClear(t *testing.T) {
|
|
|
|
pix := make([]uint8, 4*4*4)
|
|
|
|
for i := range pix {
|
|
|
|
pix[i] = 0xff
|
|
|
|
}
|
|
|
|
|
2019-02-12 07:05:05 +01:00
|
|
|
img := NewImage(4, 4)
|
2018-04-28 15:50:00 +02:00
|
|
|
img.ReplacePixels(pix, 0, 0, 4, 4)
|
|
|
|
// This doesn't make the image stale. Its base pixels are available.
|
|
|
|
img.ReplacePixels(nil, 1, 1, 2, 2)
|
|
|
|
|
|
|
|
cases := []struct {
|
|
|
|
Index int
|
|
|
|
Want color.RGBA
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
Index: 0,
|
|
|
|
Want: color.RGBA{0xff, 0xff, 0xff, 0xff},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Index: 3,
|
|
|
|
Want: color.RGBA{0xff, 0xff, 0xff, 0xff},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Index: 4,
|
|
|
|
Want: color.RGBA{0xff, 0xff, 0xff, 0xff},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Index: 5,
|
|
|
|
Want: color.RGBA{0, 0, 0, 0},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Index: 7,
|
|
|
|
Want: color.RGBA{0xff, 0xff, 0xff, 0xff},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Index: 8,
|
|
|
|
Want: color.RGBA{0xff, 0xff, 0xff, 0xff},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Index: 10,
|
|
|
|
Want: color.RGBA{0, 0, 0, 0},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Index: 11,
|
|
|
|
Want: color.RGBA{0xff, 0xff, 0xff, 0xff},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Index: 12,
|
|
|
|
Want: color.RGBA{0xff, 0xff, 0xff, 0xff},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Index: 15,
|
|
|
|
Want: color.RGBA{0xff, 0xff, 0xff, 0xff},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, c := range cases {
|
2019-02-12 15:31:52 +01:00
|
|
|
got := pixelsToColor(img.BasePixelsForTesting(), c.Index)
|
2018-04-28 15:50:00 +02:00
|
|
|
want := c.Want
|
|
|
|
if got != want {
|
|
|
|
t.Errorf("base pixel [%d]: got %v, want %v", c.Index, got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-26 12:10:12 +01:00
|
|
|
func TestReplacePixelsOnly(t *testing.T) {
|
|
|
|
const w, h = 128, 128
|
2019-02-12 07:05:05 +01:00
|
|
|
img0 := NewImage(w, h)
|
2018-12-26 13:14:29 +01:00
|
|
|
defer img0.Dispose()
|
2019-02-12 07:05:05 +01:00
|
|
|
img1 := NewImage(1, 1)
|
2018-12-26 13:14:29 +01:00
|
|
|
defer img1.Dispose()
|
2018-12-26 12:10:12 +01:00
|
|
|
|
|
|
|
for i := 0; i < w*h; i += 5 {
|
2018-12-26 13:14:29 +01:00
|
|
|
img0.ReplacePixels([]byte{1, 2, 3, 4}, i%w, i/w, 1, 1)
|
2018-12-26 12:10:12 +01:00
|
|
|
}
|
|
|
|
|
2019-02-14 15:44:02 +01:00
|
|
|
vs := quadVertices(img0, 1, 1, 0, 0)
|
2018-12-26 13:14:29 +01:00
|
|
|
is := graphics.QuadIndices()
|
2019-04-22 16:12:03 +02:00
|
|
|
img1.DrawTriangles(img0, vs, is, nil, graphics.CompositeModeCopy, graphics.FilterNearest, graphics.AddressClampToZero)
|
2018-12-26 13:14:29 +01:00
|
|
|
img0.ReplacePixels([]byte{5, 6, 7, 8}, 0, 0, 1, 1)
|
|
|
|
|
2018-12-26 12:10:12 +01:00
|
|
|
// BasePixelsForTesting is available without GPU accessing.
|
|
|
|
for i := 0; i < w*h; i++ {
|
|
|
|
var want color.RGBA
|
2018-12-26 13:14:29 +01:00
|
|
|
switch {
|
|
|
|
case i == 0:
|
|
|
|
want = color.RGBA{5, 6, 7, 8}
|
|
|
|
case i%5 == 0:
|
2018-12-26 12:10:12 +01:00
|
|
|
want = color.RGBA{1, 2, 3, 4}
|
|
|
|
}
|
2019-02-12 15:31:52 +01:00
|
|
|
got := pixelsToColor(img0.BasePixelsForTesting(), i)
|
2018-12-26 12:10:12 +01:00
|
|
|
if !sameColors(got, want, 0) {
|
|
|
|
t.Errorf("got %v, want %v", got, want)
|
|
|
|
}
|
|
|
|
}
|
2018-12-26 13:14:29 +01:00
|
|
|
|
|
|
|
ResolveStaleImages()
|
2019-05-26 14:10:25 +02:00
|
|
|
if err := RestoreIfNeeded(); err != nil {
|
2018-12-26 13:14:29 +01:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
want := color.RGBA{1, 2, 3, 4}
|
2019-02-12 15:31:52 +01:00
|
|
|
got := pixelsToColor(img1.BasePixelsForTesting(), 0)
|
2018-12-26 13:14:29 +01:00
|
|
|
if !sameColors(got, want, 0) {
|
|
|
|
t.Errorf("got %v, want %v", got, want)
|
|
|
|
}
|
2018-12-26 12:10:12 +01:00
|
|
|
}
|
|
|
|
|
2017-05-31 19:37:21 +02:00
|
|
|
// TODO: How about volatile/screen images?
|
2019-01-21 19:35:48 +01:00
|
|
|
|
|
|
|
// Issue #793
|
|
|
|
func TestReadPixelsFromVolatileImage(t *testing.T) {
|
|
|
|
const w, h = 16, 16
|
2019-02-12 07:05:05 +01:00
|
|
|
dst := NewImage(w, h)
|
|
|
|
dst.MakeVolatile()
|
|
|
|
src := NewImage(w, h)
|
2019-01-21 19:35:48 +01:00
|
|
|
|
|
|
|
// First, make sure that dst has pixels
|
|
|
|
dst.ReplacePixels(make([]byte, 4*w*h), 0, 0, w, h)
|
|
|
|
|
2019-01-22 15:33:56 +01:00
|
|
|
// Second, draw src to dst. If the implementation is correct, dst becomes stale.
|
2019-02-12 15:31:52 +01:00
|
|
|
src.Fill(0xff, 0xff, 0xff, 0xff)
|
2019-02-14 15:44:02 +01:00
|
|
|
vs := quadVertices(src, 1, 1, 0, 0)
|
2019-01-21 19:35:48 +01:00
|
|
|
is := graphics.QuadIndices()
|
2019-04-22 16:12:03 +02:00
|
|
|
dst.DrawTriangles(src, vs, is, nil, graphics.CompositeModeCopy, graphics.FilterNearest, graphics.AddressClampToZero)
|
2019-01-21 19:35:48 +01:00
|
|
|
|
2019-01-22 15:33:56 +01:00
|
|
|
// Read the pixels. If the implementation is correct, dst tries to read its pixels from GPU due to being
|
|
|
|
// stale.
|
2019-01-21 19:35:48 +01:00
|
|
|
want := byte(0xff)
|
|
|
|
got, _, _, _ := dst.At(0, 0)
|
|
|
|
if got != want {
|
|
|
|
t.Errorf("got: %v, want: %v", got, want)
|
|
|
|
}
|
|
|
|
}
|