Home | History | Annotate | Download | only in gif
      1 // Copyright 2013 The Go Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style
      3 // license that can be found in the LICENSE file.
      4 
      5 package gif
      6 
      7 import (
      8 	"bytes"
      9 	"image"
     10 	"image/color"
     11 	"image/color/palette"
     12 	_ "image/png"
     13 	"io/ioutil"
     14 	"math/rand"
     15 	"os"
     16 	"reflect"
     17 	"testing"
     18 )
     19 
     20 func readImg(filename string) (image.Image, error) {
     21 	f, err := os.Open(filename)
     22 	if err != nil {
     23 		return nil, err
     24 	}
     25 	defer f.Close()
     26 	m, _, err := image.Decode(f)
     27 	return m, err
     28 }
     29 
     30 func readGIF(filename string) (*GIF, error) {
     31 	f, err := os.Open(filename)
     32 	if err != nil {
     33 		return nil, err
     34 	}
     35 	defer f.Close()
     36 	return DecodeAll(f)
     37 }
     38 
     39 func delta(u0, u1 uint32) int64 {
     40 	d := int64(u0) - int64(u1)
     41 	if d < 0 {
     42 		return -d
     43 	}
     44 	return d
     45 }
     46 
     47 // averageDelta returns the average delta in RGB space. The two images must
     48 // have the same bounds.
     49 func averageDelta(m0, m1 image.Image) int64 {
     50 	b := m0.Bounds()
     51 	var sum, n int64
     52 	for y := b.Min.Y; y < b.Max.Y; y++ {
     53 		for x := b.Min.X; x < b.Max.X; x++ {
     54 			c0 := m0.At(x, y)
     55 			c1 := m1.At(x, y)
     56 			r0, g0, b0, _ := c0.RGBA()
     57 			r1, g1, b1, _ := c1.RGBA()
     58 			sum += delta(r0, r1)
     59 			sum += delta(g0, g1)
     60 			sum += delta(b0, b1)
     61 			n += 3
     62 		}
     63 	}
     64 	return sum / n
     65 }
     66 
     67 var testCase = []struct {
     68 	filename  string
     69 	tolerance int64
     70 }{
     71 	{"../testdata/video-001.png", 1 << 12},
     72 	{"../testdata/video-001.gif", 0},
     73 	{"../testdata/video-001.interlaced.gif", 0},
     74 }
     75 
     76 func TestWriter(t *testing.T) {
     77 	for _, tc := range testCase {
     78 		m0, err := readImg(tc.filename)
     79 		if err != nil {
     80 			t.Error(tc.filename, err)
     81 			continue
     82 		}
     83 		var buf bytes.Buffer
     84 		err = Encode(&buf, m0, nil)
     85 		if err != nil {
     86 			t.Error(tc.filename, err)
     87 			continue
     88 		}
     89 		m1, err := Decode(&buf)
     90 		if err != nil {
     91 			t.Error(tc.filename, err)
     92 			continue
     93 		}
     94 		if m0.Bounds() != m1.Bounds() {
     95 			t.Errorf("%s, bounds differ: %v and %v", tc.filename, m0.Bounds(), m1.Bounds())
     96 			continue
     97 		}
     98 		// Compare the average delta to the tolerance level.
     99 		avgDelta := averageDelta(m0, m1)
    100 		if avgDelta > tc.tolerance {
    101 			t.Errorf("%s: average delta is too high. expected: %d, got %d", tc.filename, tc.tolerance, avgDelta)
    102 			continue
    103 		}
    104 	}
    105 }
    106 
    107 func TestSubImage(t *testing.T) {
    108 	m0, err := readImg("../testdata/video-001.gif")
    109 	if err != nil {
    110 		t.Fatalf("readImg: %v", err)
    111 	}
    112 	m0 = m0.(*image.Paletted).SubImage(image.Rect(0, 0, 50, 30))
    113 	var buf bytes.Buffer
    114 	err = Encode(&buf, m0, nil)
    115 	if err != nil {
    116 		t.Fatalf("Encode: %v", err)
    117 	}
    118 	m1, err := Decode(&buf)
    119 	if err != nil {
    120 		t.Fatalf("Decode: %v", err)
    121 	}
    122 	if m0.Bounds() != m1.Bounds() {
    123 		t.Fatalf("bounds differ: %v and %v", m0.Bounds(), m1.Bounds())
    124 	}
    125 	if averageDelta(m0, m1) != 0 {
    126 		t.Fatalf("images differ")
    127 	}
    128 }
    129 
    130 // palettesEqual reports whether two color.Palette values are equal, ignoring
    131 // any trailing opaque-black palette entries.
    132 func palettesEqual(p, q color.Palette) bool {
    133 	n := len(p)
    134 	if n > len(q) {
    135 		n = len(q)
    136 	}
    137 	for i := 0; i < n; i++ {
    138 		if p[i] != q[i] {
    139 			return false
    140 		}
    141 	}
    142 	for i := n; i < len(p); i++ {
    143 		r, g, b, a := p[i].RGBA()
    144 		if r != 0 || g != 0 || b != 0 || a != 0xffff {
    145 			return false
    146 		}
    147 	}
    148 	for i := n; i < len(q); i++ {
    149 		r, g, b, a := q[i].RGBA()
    150 		if r != 0 || g != 0 || b != 0 || a != 0xffff {
    151 			return false
    152 		}
    153 	}
    154 	return true
    155 }
    156 
    157 var frames = []string{
    158 	"../testdata/video-001.gif",
    159 	"../testdata/video-005.gray.gif",
    160 }
    161 
    162 func testEncodeAll(t *testing.T, go1Dot5Fields bool, useGlobalColorModel bool) {
    163 	const width, height = 150, 103
    164 
    165 	g0 := &GIF{
    166 		Image:     make([]*image.Paletted, len(frames)),
    167 		Delay:     make([]int, len(frames)),
    168 		LoopCount: 5,
    169 	}
    170 	for i, f := range frames {
    171 		g, err := readGIF(f)
    172 		if err != nil {
    173 			t.Fatal(f, err)
    174 		}
    175 		m := g.Image[0]
    176 		if m.Bounds().Dx() != width || m.Bounds().Dy() != height {
    177 			t.Fatalf("frame %d had unexpected bounds: got %v, want width/height = %d/%d",
    178 				i, m.Bounds(), width, height)
    179 		}
    180 		g0.Image[i] = m
    181 	}
    182 	// The GIF.Disposal, GIF.Config and GIF.BackgroundIndex fields were added
    183 	// in Go 1.5. Valid Go 1.4 or earlier code should still produce valid GIFs.
    184 	//
    185 	// On the following line, color.Model is an interface type, and
    186 	// color.Palette is a concrete (slice) type.
    187 	globalColorModel, backgroundIndex := color.Model(color.Palette(nil)), uint8(0)
    188 	if useGlobalColorModel {
    189 		globalColorModel, backgroundIndex = color.Palette(palette.WebSafe), uint8(1)
    190 	}
    191 	if go1Dot5Fields {
    192 		g0.Disposal = make([]byte, len(g0.Image))
    193 		for i := range g0.Disposal {
    194 			g0.Disposal[i] = DisposalNone
    195 		}
    196 		g0.Config = image.Config{
    197 			ColorModel: globalColorModel,
    198 			Width:      width,
    199 			Height:     height,
    200 		}
    201 		g0.BackgroundIndex = backgroundIndex
    202 	}
    203 
    204 	var buf bytes.Buffer
    205 	if err := EncodeAll(&buf, g0); err != nil {
    206 		t.Fatal("EncodeAll:", err)
    207 	}
    208 	encoded := buf.Bytes()
    209 	config, err := DecodeConfig(bytes.NewReader(encoded))
    210 	if err != nil {
    211 		t.Fatal("DecodeConfig:", err)
    212 	}
    213 	g1, err := DecodeAll(bytes.NewReader(encoded))
    214 	if err != nil {
    215 		t.Fatal("DecodeAll:", err)
    216 	}
    217 
    218 	if !reflect.DeepEqual(config, g1.Config) {
    219 		t.Errorf("DecodeConfig inconsistent with DecodeAll")
    220 	}
    221 	if !palettesEqual(g1.Config.ColorModel.(color.Palette), globalColorModel.(color.Palette)) {
    222 		t.Errorf("unexpected global color model")
    223 	}
    224 	if w, h := g1.Config.Width, g1.Config.Height; w != width || h != height {
    225 		t.Errorf("got config width * height = %d * %d, want %d * %d", w, h, width, height)
    226 	}
    227 
    228 	if g0.LoopCount != g1.LoopCount {
    229 		t.Errorf("loop counts differ: %d and %d", g0.LoopCount, g1.LoopCount)
    230 	}
    231 	if backgroundIndex != g1.BackgroundIndex {
    232 		t.Errorf("background indexes differ: %d and %d", backgroundIndex, g1.BackgroundIndex)
    233 	}
    234 	if len(g0.Image) != len(g1.Image) {
    235 		t.Fatalf("image lengths differ: %d and %d", len(g0.Image), len(g1.Image))
    236 	}
    237 	if len(g1.Image) != len(g1.Delay) {
    238 		t.Fatalf("image and delay lengths differ: %d and %d", len(g1.Image), len(g1.Delay))
    239 	}
    240 	if len(g1.Image) != len(g1.Disposal) {
    241 		t.Fatalf("image and disposal lengths differ: %d and %d", len(g1.Image), len(g1.Disposal))
    242 	}
    243 
    244 	for i := range g0.Image {
    245 		m0, m1 := g0.Image[i], g1.Image[i]
    246 		if m0.Bounds() != m1.Bounds() {
    247 			t.Errorf("frame %d: bounds differ: %v and %v", i, m0.Bounds(), m1.Bounds())
    248 		}
    249 		d0, d1 := g0.Delay[i], g1.Delay[i]
    250 		if d0 != d1 {
    251 			t.Errorf("frame %d: delay values differ: %d and %d", i, d0, d1)
    252 		}
    253 		p0, p1 := uint8(0), g1.Disposal[i]
    254 		if go1Dot5Fields {
    255 			p0 = DisposalNone
    256 		}
    257 		if p0 != p1 {
    258 			t.Errorf("frame %d: disposal values differ: %d and %d", i, p0, p1)
    259 		}
    260 	}
    261 }
    262 
    263 func TestEncodeAllGo1Dot4(t *testing.T)                 { testEncodeAll(t, false, false) }
    264 func TestEncodeAllGo1Dot5(t *testing.T)                 { testEncodeAll(t, true, false) }
    265 func TestEncodeAllGo1Dot5GlobalColorModel(t *testing.T) { testEncodeAll(t, true, true) }
    266 
    267 func TestEncodeMismatchDelay(t *testing.T) {
    268 	images := make([]*image.Paletted, 2)
    269 	for i := range images {
    270 		images[i] = image.NewPaletted(image.Rect(0, 0, 5, 5), palette.Plan9)
    271 	}
    272 
    273 	g0 := &GIF{
    274 		Image: images,
    275 		Delay: make([]int, 1),
    276 	}
    277 	if err := EncodeAll(ioutil.Discard, g0); err == nil {
    278 		t.Error("expected error from mismatched delay and image slice lengths")
    279 	}
    280 
    281 	g1 := &GIF{
    282 		Image:    images,
    283 		Delay:    make([]int, len(images)),
    284 		Disposal: make([]byte, 1),
    285 	}
    286 	for i := range g1.Disposal {
    287 		g1.Disposal[i] = DisposalNone
    288 	}
    289 	if err := EncodeAll(ioutil.Discard, g1); err == nil {
    290 		t.Error("expected error from mismatched disposal and image slice lengths")
    291 	}
    292 }
    293 
    294 func TestEncodeZeroGIF(t *testing.T) {
    295 	if err := EncodeAll(ioutil.Discard, &GIF{}); err == nil {
    296 		t.Error("expected error from providing empty gif")
    297 	}
    298 }
    299 
    300 func TestEncodeAllFramesOutOfBounds(t *testing.T) {
    301 	images := []*image.Paletted{
    302 		image.NewPaletted(image.Rect(0, 0, 5, 5), palette.Plan9),
    303 		image.NewPaletted(image.Rect(2, 2, 8, 8), palette.Plan9),
    304 		image.NewPaletted(image.Rect(3, 3, 4, 4), palette.Plan9),
    305 	}
    306 	for _, upperBound := range []int{6, 10} {
    307 		g := &GIF{
    308 			Image:    images,
    309 			Delay:    make([]int, len(images)),
    310 			Disposal: make([]byte, len(images)),
    311 			Config: image.Config{
    312 				Width:  upperBound,
    313 				Height: upperBound,
    314 			},
    315 		}
    316 		err := EncodeAll(ioutil.Discard, g)
    317 		if upperBound >= 8 {
    318 			if err != nil {
    319 				t.Errorf("upperBound=%d: %v", upperBound, err)
    320 			}
    321 		} else {
    322 			if err == nil {
    323 				t.Errorf("upperBound=%d: got nil error, want non-nil", upperBound)
    324 			}
    325 		}
    326 	}
    327 }
    328 
    329 func TestEncodeNonZeroMinPoint(t *testing.T) {
    330 	points := []image.Point{
    331 		image.Point{-8, -9},
    332 		image.Point{-4, -4},
    333 		image.Point{-3, +3},
    334 		image.Point{+0, +0},
    335 		image.Point{+2, +2},
    336 	}
    337 	for _, p := range points {
    338 		src := image.NewPaletted(image.Rectangle{Min: p, Max: p.Add(image.Point{6, 6})}, palette.Plan9)
    339 		var buf bytes.Buffer
    340 		if err := Encode(&buf, src, nil); err != nil {
    341 			t.Errorf("p=%v: Encode: %v", p, err)
    342 			continue
    343 		}
    344 		m, err := Decode(&buf)
    345 		if err != nil {
    346 			t.Errorf("p=%v: Decode: %v", p, err)
    347 			continue
    348 		}
    349 		if got, want := m.Bounds(), image.Rect(0, 0, 6, 6); got != want {
    350 			t.Errorf("p=%v: got %v, want %v", p, got, want)
    351 		}
    352 	}
    353 }
    354 
    355 func TestEncodeImplicitConfigSize(t *testing.T) {
    356 	// For backwards compatibility for Go 1.4 and earlier code, the Config
    357 	// field is optional, and if zero, the width and height is implied by the
    358 	// first (and in this case only) frame's width and height.
    359 	//
    360 	// A Config only specifies a width and height (two integers) while an
    361 	// image.Image's Bounds method returns an image.Rectangle (four integers).
    362 	// For a gif.GIF, the overall bounds' top-left point is always implicitly
    363 	// (0, 0), and any frame whose bounds have a negative X or Y will be
    364 	// outside those overall bounds, so encoding should fail.
    365 	for _, lowerBound := range []int{-1, 0, 1} {
    366 		images := []*image.Paletted{
    367 			image.NewPaletted(image.Rect(lowerBound, lowerBound, 4, 4), palette.Plan9),
    368 		}
    369 		g := &GIF{
    370 			Image: images,
    371 			Delay: make([]int, len(images)),
    372 		}
    373 		err := EncodeAll(ioutil.Discard, g)
    374 		if lowerBound >= 0 {
    375 			if err != nil {
    376 				t.Errorf("lowerBound=%d: %v", lowerBound, err)
    377 			}
    378 		} else {
    379 			if err == nil {
    380 				t.Errorf("lowerBound=%d: got nil error, want non-nil", lowerBound)
    381 			}
    382 		}
    383 	}
    384 }
    385 
    386 func TestEncodePalettes(t *testing.T) {
    387 	const w, h = 5, 5
    388 	pals := []color.Palette{{
    389 		color.RGBA{0x00, 0x00, 0x00, 0xff},
    390 		color.RGBA{0x01, 0x00, 0x00, 0xff},
    391 		color.RGBA{0x02, 0x00, 0x00, 0xff},
    392 	}, {
    393 		color.RGBA{0x00, 0x00, 0x00, 0xff},
    394 		color.RGBA{0x00, 0x01, 0x00, 0xff},
    395 	}, {
    396 		color.RGBA{0x00, 0x00, 0x03, 0xff},
    397 		color.RGBA{0x00, 0x00, 0x02, 0xff},
    398 		color.RGBA{0x00, 0x00, 0x01, 0xff},
    399 		color.RGBA{0x00, 0x00, 0x00, 0xff},
    400 	}, {
    401 		color.RGBA{0x10, 0x07, 0xf0, 0xff},
    402 		color.RGBA{0x20, 0x07, 0xf0, 0xff},
    403 		color.RGBA{0x30, 0x07, 0xf0, 0xff},
    404 		color.RGBA{0x40, 0x07, 0xf0, 0xff},
    405 		color.RGBA{0x50, 0x07, 0xf0, 0xff},
    406 	}}
    407 	g0 := &GIF{
    408 		Image: []*image.Paletted{
    409 			image.NewPaletted(image.Rect(0, 0, w, h), pals[0]),
    410 			image.NewPaletted(image.Rect(0, 0, w, h), pals[1]),
    411 			image.NewPaletted(image.Rect(0, 0, w, h), pals[2]),
    412 			image.NewPaletted(image.Rect(0, 0, w, h), pals[3]),
    413 		},
    414 		Delay:    make([]int, len(pals)),
    415 		Disposal: make([]byte, len(pals)),
    416 		Config: image.Config{
    417 			ColorModel: pals[2],
    418 			Width:      w,
    419 			Height:     h,
    420 		},
    421 	}
    422 
    423 	var buf bytes.Buffer
    424 	if err := EncodeAll(&buf, g0); err != nil {
    425 		t.Fatalf("EncodeAll: %v", err)
    426 	}
    427 	g1, err := DecodeAll(&buf)
    428 	if err != nil {
    429 		t.Fatalf("DecodeAll: %v", err)
    430 	}
    431 	if len(g0.Image) != len(g1.Image) {
    432 		t.Fatalf("image lengths differ: %d and %d", len(g0.Image), len(g1.Image))
    433 	}
    434 	for i, m := range g1.Image {
    435 		if got, want := m.Palette, pals[i]; !palettesEqual(got, want) {
    436 			t.Errorf("frame %d:\ngot  %v\nwant %v", i, got, want)
    437 		}
    438 	}
    439 }
    440 
    441 func BenchmarkEncode(b *testing.B) {
    442 	b.StopTimer()
    443 
    444 	bo := image.Rect(0, 0, 640, 480)
    445 	rnd := rand.New(rand.NewSource(123))
    446 
    447 	// Restrict to a 256-color paletted image to avoid quantization path.
    448 	palette := make(color.Palette, 256)
    449 	for i := range palette {
    450 		palette[i] = color.RGBA{
    451 			uint8(rnd.Intn(256)),
    452 			uint8(rnd.Intn(256)),
    453 			uint8(rnd.Intn(256)),
    454 			255,
    455 		}
    456 	}
    457 	img := image.NewPaletted(image.Rect(0, 0, 640, 480), palette)
    458 	for y := bo.Min.Y; y < bo.Max.Y; y++ {
    459 		for x := bo.Min.X; x < bo.Max.X; x++ {
    460 			img.Set(x, y, palette[rnd.Intn(256)])
    461 		}
    462 	}
    463 
    464 	b.SetBytes(640 * 480 * 4)
    465 	b.StartTimer()
    466 	for i := 0; i < b.N; i++ {
    467 		Encode(ioutil.Discard, img, nil)
    468 	}
    469 }
    470 
    471 func BenchmarkQuantizedEncode(b *testing.B) {
    472 	b.StopTimer()
    473 	img := image.NewRGBA(image.Rect(0, 0, 640, 480))
    474 	bo := img.Bounds()
    475 	rnd := rand.New(rand.NewSource(123))
    476 	for y := bo.Min.Y; y < bo.Max.Y; y++ {
    477 		for x := bo.Min.X; x < bo.Max.X; x++ {
    478 			img.SetRGBA(x, y, color.RGBA{
    479 				uint8(rnd.Intn(256)),
    480 				uint8(rnd.Intn(256)),
    481 				uint8(rnd.Intn(256)),
    482 				255,
    483 			})
    484 		}
    485 	}
    486 	b.SetBytes(640 * 480 * 4)
    487 	b.StartTimer()
    488 	for i := 0; i < b.N; i++ {
    489 		Encode(ioutil.Discard, img, nil)
    490 	}
    491 }
    492