...

Source file src/image/jpeg/reader_test.go

Documentation: image/jpeg

		 1  // Copyright 2012 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 jpeg
		 6  
		 7  import (
		 8  	"bytes"
		 9  	"encoding/base64"
		10  	"fmt"
		11  	"image"
		12  	"image/color"
		13  	"io"
		14  	"math/rand"
		15  	"os"
		16  	"strings"
		17  	"testing"
		18  	"time"
		19  )
		20  
		21  // TestDecodeProgressive tests that decoding the baseline and progressive
		22  // versions of the same image result in exactly the same pixel data, in YCbCr
		23  // space for color images, and Y space for grayscale images.
		24  func TestDecodeProgressive(t *testing.T) {
		25  	testCases := []string{
		26  		"../testdata/video-001",
		27  		"../testdata/video-001.q50.410",
		28  		"../testdata/video-001.q50.411",
		29  		"../testdata/video-001.q50.420",
		30  		"../testdata/video-001.q50.422",
		31  		"../testdata/video-001.q50.440",
		32  		"../testdata/video-001.q50.444",
		33  		"../testdata/video-005.gray.q50",
		34  		"../testdata/video-005.gray.q50.2x2",
		35  		"../testdata/video-001.separate.dc.progression",
		36  	}
		37  	for _, tc := range testCases {
		38  		m0, err := decodeFile(tc + ".jpeg")
		39  		if err != nil {
		40  			t.Errorf("%s: %v", tc+".jpeg", err)
		41  			continue
		42  		}
		43  		m1, err := decodeFile(tc + ".progressive.jpeg")
		44  		if err != nil {
		45  			t.Errorf("%s: %v", tc+".progressive.jpeg", err)
		46  			continue
		47  		}
		48  		if m0.Bounds() != m1.Bounds() {
		49  			t.Errorf("%s: bounds differ: %v and %v", tc, m0.Bounds(), m1.Bounds())
		50  			continue
		51  		}
		52  		// All of the video-*.jpeg files are 150x103.
		53  		if m0.Bounds() != image.Rect(0, 0, 150, 103) {
		54  			t.Errorf("%s: bad bounds: %v", tc, m0.Bounds())
		55  			continue
		56  		}
		57  
		58  		switch m0 := m0.(type) {
		59  		case *image.YCbCr:
		60  			m1 := m1.(*image.YCbCr)
		61  			if err := check(m0.Bounds(), m0.Y, m1.Y, m0.YStride, m1.YStride); err != nil {
		62  				t.Errorf("%s (Y): %v", tc, err)
		63  				continue
		64  			}
		65  			if err := check(m0.Bounds(), m0.Cb, m1.Cb, m0.CStride, m1.CStride); err != nil {
		66  				t.Errorf("%s (Cb): %v", tc, err)
		67  				continue
		68  			}
		69  			if err := check(m0.Bounds(), m0.Cr, m1.Cr, m0.CStride, m1.CStride); err != nil {
		70  				t.Errorf("%s (Cr): %v", tc, err)
		71  				continue
		72  			}
		73  		case *image.Gray:
		74  			m1 := m1.(*image.Gray)
		75  			if err := check(m0.Bounds(), m0.Pix, m1.Pix, m0.Stride, m1.Stride); err != nil {
		76  				t.Errorf("%s: %v", tc, err)
		77  				continue
		78  			}
		79  		default:
		80  			t.Errorf("%s: unexpected image type %T", tc, m0)
		81  			continue
		82  		}
		83  	}
		84  }
		85  
		86  func decodeFile(filename string) (image.Image, error) {
		87  	f, err := os.Open(filename)
		88  	if err != nil {
		89  		return nil, err
		90  	}
		91  	defer f.Close()
		92  	return Decode(f)
		93  }
		94  
		95  type eofReader struct {
		96  	data		 []byte // deliver from Read without EOF
		97  	dataEOF	[]byte // then deliver from Read with EOF on last chunk
		98  	lenAtEOF int
		99  }
	 100  
	 101  func (r *eofReader) Read(b []byte) (n int, err error) {
	 102  	if len(r.data) > 0 {
	 103  		n = copy(b, r.data)
	 104  		r.data = r.data[n:]
	 105  	} else {
	 106  		n = copy(b, r.dataEOF)
	 107  		r.dataEOF = r.dataEOF[n:]
	 108  		if len(r.dataEOF) == 0 {
	 109  			err = io.EOF
	 110  			if r.lenAtEOF == -1 {
	 111  				r.lenAtEOF = n
	 112  			}
	 113  		}
	 114  	}
	 115  	return
	 116  }
	 117  
	 118  func TestDecodeEOF(t *testing.T) {
	 119  	// Check that if reader returns final data and EOF at same time, jpeg handles it.
	 120  	data, err := os.ReadFile("../testdata/video-001.jpeg")
	 121  	if err != nil {
	 122  		t.Fatal(err)
	 123  	}
	 124  
	 125  	n := len(data)
	 126  	for i := 0; i < n; {
	 127  		r := &eofReader{data[:n-i], data[n-i:], -1}
	 128  		_, err := Decode(r)
	 129  		if err != nil {
	 130  			t.Errorf("Decode with Read() = %d, EOF: %v", r.lenAtEOF, err)
	 131  		}
	 132  		if i == 0 {
	 133  			i = 1
	 134  		} else {
	 135  			i *= 2
	 136  		}
	 137  	}
	 138  }
	 139  
	 140  // check checks that the two pix data are equal, within the given bounds.
	 141  func check(bounds image.Rectangle, pix0, pix1 []byte, stride0, stride1 int) error {
	 142  	if stride0 <= 0 || stride0%8 != 0 {
	 143  		return fmt.Errorf("bad stride %d", stride0)
	 144  	}
	 145  	if stride1 <= 0 || stride1%8 != 0 {
	 146  		return fmt.Errorf("bad stride %d", stride1)
	 147  	}
	 148  	// Compare the two pix data, one 8x8 block at a time.
	 149  	for y := 0; y < len(pix0)/stride0 && y < len(pix1)/stride1; y += 8 {
	 150  		for x := 0; x < stride0 && x < stride1; x += 8 {
	 151  			if x >= bounds.Max.X || y >= bounds.Max.Y {
	 152  				// We don't care if the two pix data differ if the 8x8 block is
	 153  				// entirely outside of the image's bounds. For example, this can
	 154  				// occur with a 4:2:0 chroma subsampling and a 1x1 image. Baseline
	 155  				// decoding works on the one 16x16 MCU as a whole; progressive
	 156  				// decoding's first pass works on that 16x16 MCU as a whole but
	 157  				// refinement passes only process one 8x8 block within the MCU.
	 158  				continue
	 159  			}
	 160  
	 161  			for j := 0; j < 8; j++ {
	 162  				for i := 0; i < 8; i++ {
	 163  					index0 := (y+j)*stride0 + (x + i)
	 164  					index1 := (y+j)*stride1 + (x + i)
	 165  					if pix0[index0] != pix1[index1] {
	 166  						return fmt.Errorf("blocks at (%d, %d) differ:\n%sand\n%s", x, y,
	 167  							pixString(pix0, stride0, x, y),
	 168  							pixString(pix1, stride1, x, y),
	 169  						)
	 170  					}
	 171  				}
	 172  			}
	 173  		}
	 174  	}
	 175  	return nil
	 176  }
	 177  
	 178  func pixString(pix []byte, stride, x, y int) string {
	 179  	s := bytes.NewBuffer(nil)
	 180  	for j := 0; j < 8; j++ {
	 181  		fmt.Fprintf(s, "\t")
	 182  		for i := 0; i < 8; i++ {
	 183  			fmt.Fprintf(s, "%02x ", pix[(y+j)*stride+(x+i)])
	 184  		}
	 185  		fmt.Fprintf(s, "\n")
	 186  	}
	 187  	return s.String()
	 188  }
	 189  
	 190  func TestTruncatedSOSDataDoesntPanic(t *testing.T) {
	 191  	b, err := os.ReadFile("../testdata/video-005.gray.q50.jpeg")
	 192  	if err != nil {
	 193  		t.Fatal(err)
	 194  	}
	 195  	sosMarker := []byte{0xff, 0xda}
	 196  	i := bytes.Index(b, sosMarker)
	 197  	if i < 0 {
	 198  		t.Fatal("SOS marker not found")
	 199  	}
	 200  	i += len(sosMarker)
	 201  	j := i + 10
	 202  	if j > len(b) {
	 203  		j = len(b)
	 204  	}
	 205  	for ; i < j; i++ {
	 206  		Decode(bytes.NewReader(b[:i]))
	 207  	}
	 208  }
	 209  
	 210  func TestLargeImageWithShortData(t *testing.T) {
	 211  	// This input is an invalid JPEG image, based on the fuzzer-generated image
	 212  	// in issue 10413. It is only 504 bytes, and shouldn't take long for Decode
	 213  	// to return an error. The Start Of Frame marker gives the image dimensions
	 214  	// as 8192 wide and 8192 high, so even if an unreadByteStuffedByte bug
	 215  	// doesn't technically lead to an infinite loop, such a bug can still cause
	 216  	// an unreasonably long loop for such a short input.
	 217  	const input = "" +
	 218  		"\xff\xd8\xff\xe0\x00\x10\x4a\x46\x49\x46\x00\x01\x01\x00\x00\x01" +
	 219  		"\x00\x01\x00\x00\xff\xdb\x00\x43\x00\x10\x0b\x0c\x0e\x0c\x0a\x10" +
	 220  		"\x0e\x89\x0e\x12\x11\x10\x13\x18\xff\xd8\xff\xe0\x00\x10\x4a\x46" +
	 221  		"\x49\x46\x00\x01\x01\x00\x00\x01\x00\x01\x00\x00\xff\xdb\x00\x43" +
	 222  		"\x00\x10\x0b\x0c\x0e\x0c\x0a\x10\x0e\x0d\x0e\x12\x11\x10\x13\x18" +
	 223  		"\x28\x1a\x18\x16\x16\x18\x31\x23\x25\x1d\x28\x3a\x33\x3d\x3c\x39" +
	 224  		"\x33\x38\x37\x40\x48\x5c\x4e\x40\x44\x57\x45\x37\x38\x50\x6d\x51" +
	 225  		"\x57\x5f\x62\x67\x68\x67\x3e\x4d\x71\x79\x70\x64\x78\x5c\x65\x67" +
	 226  		"\x63\xff\xc0\x00\x0b\x08\x20\x00\x20\x00\x01\x01\x11\x00\xff\xc4" +
	 227  		"\x00\x1f\x00\x00\x01\x05\x01\x01\x01\x01\x01\x01\x00\x00\x00\x00" +
	 228  		"\x00\x00\x00\x00\x01\x02\x03\x04\x05\x06\x07\x08\x09\x0a\x0b\xff" +
	 229  		"\xc4\x00\xb5\x10\x00\x02\x01\x03\x03\x02\x04\x03\x05\x05\x04\x04" +
	 230  		"\x00\x00\x01\x7d\x01\x02\x03\x00\x04\x11\x05\x12\x21\x31\x01\x06" +
	 231  		"\x13\x51\x61\x07\x22\x71\x14\x32\x81\x91\xa1\x08\x23\xd8\xff\xdd" +
	 232  		"\x42\xb1\xc1\x15\x52\xd1\xf0\x24\x33\x62\x72\x82\x09\x0a\x16\x17" +
	 233  		"\x18\x19\x1a\x25\x26\x27\x28\x29\x2a\x34\x35\x36\x37\x38\x39\x3a" +
	 234  		"\x43\x44\x45\x46\x47\x48\x49\x4a\x53\x54\x55\x56\x57\x58\x59\x5a" +
	 235  		"\x00\x63\x64\x65\x66\x67\x68\x69\x6a\x73\x74\x75\x76\x77\x78\x79" +
	 236  		"\x7a\x83\x84\x85\x86\x87\x88\x89\x8a\x92\x93\x94\x95\x96\x97\x98" +
	 237  		"\x99\x9a\xa2\xa3\xa4\xa5\xa6\xa7\xa8\xa9\xaa\xb2\xb3\xb4\xb5\xb6" +
	 238  		"\xb7\xb8\xb9\xba\xc2\xc3\xc4\xc5\xc6\xc7\xff\xd8\xff\xe0\x00\x10" +
	 239  		"\x4a\x46\x49\x46\x00\x01\x01\x00\x00\x01\x00\x01\x00\x00\xff\xdb" +
	 240  		"\x00\x43\x00\x10\x0b\x0c\x0e\x0c\x0a\x10\x0e\x0d\x0e\x12\x11\x10" +
	 241  		"\x13\x18\x28\x1a\x18\x16\x16\x18\x31\x23\x25\x1d\xc8\xc9\xca\xd2" +
	 242  		"\xd3\xd4\xd5\xd6\xd7\xd8\xd9\xda\xe1\xe2\xe3\xe4\xe5\xe6\xe7\xe8" +
	 243  		"\xe9\xea\xf1\xf2\xf3\xf4\xf5\xf6\xf7\xf8\xf9\xfa\xff\xda\x00\x08" +
	 244  		"\x01\x01\x00\x00\x3f\x00\xb9\xeb\x50\xb0\xdb\xc8\xa8\xe4\x63\x80" +
	 245  		"\xdd\x31\xd6\x9d\xbb\xf2\xc5\x42\x1f\x6c\x6f\xf4\x34\xdd\x3c\xfc" +
	 246  		"\xac\xe7\x3d\x80\xa9\xcc\x87\x34\xb3\x37\xfa\x2b\x9f\x6a\xad\x63" +
	 247  		"\x20\x36\x9f\x78\x64\x75\xe6\xab\x7d\xb2\xde\x29\x70\xd3\x20\x27" +
	 248  		"\xde\xaf\xa4\xf0\xca\x9f\x24\xa8\xdf\x46\xa8\x24\x84\x96\xe3\x77" +
	 249  		"\xf9\x2e\xe0\x0a\x62\x7f\xdf\xd9"
	 250  	c := make(chan error, 1)
	 251  	go func() {
	 252  		_, err := Decode(strings.NewReader(input))
	 253  		c <- err
	 254  	}()
	 255  	select {
	 256  	case err := <-c:
	 257  		if err == nil {
	 258  			t.Fatalf("got nil error, want non-nil")
	 259  		}
	 260  	case <-time.After(3 * time.Second):
	 261  		t.Fatalf("timed out")
	 262  	}
	 263  }
	 264  
	 265  func TestPaddedRSTMarker(t *testing.T) {
	 266  	// This test image comes from golang.org/issue/28717
	 267  	const base64EncodedImage = `
	 268  /9j/4AAhQVZJMQABAQEAeAB4AAAAAAAAAAAAAAAAAAAAAAAAAP/bAEMABAIDAwMCBAMDAwQEBAQGCgYG
	 269  BQUGDAgJBwoODA8PDgwODxASFxMQERURDQ4UGhQVFxgZGhkPExweHBkeFxkZGP/bAEMBBAQEBgUGCwYG
	 270  CxgQDhAYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGP/EAaIA
	 271  AAEFAQEBAQEBAAAAAAAAAAABAgMEBQYHCAkKCxAAAgEDAwIEAwUFBAQAAAF9AQIDAAQRBRIhMUEGE1Fh
	 272  ByJxFDKBkaEII0KxwRVS0fAkM2JyggkKFhcYGRolJicoKSo0NTY3ODk6Q0RFRkdISUpTVFVWV1hZWmNk
	 273  ZWZnaGlqc3R1dnd4eXqDhIWGh4iJipKTlJWWl5iZmqKjpKWmp6ipqrKztLW2t7i5usLDxMXGx8jJytLT
	 274  1NXW19jZ2uHi4+Tl5ufo6erx8vP09fb3+Pn6AQADAQEBAQEBAQEBAAAAAAAAAQIDBAUGBwgJCgsRAAIB
	 275  AgQEAwQHBQQEAAECdwABAgMRBAUhMQYSQVEHYXETIjKBCBRCkaGxwQkjM1LwFWJy0QoWJDThJfEXGBka
	 276  JicoKSo1Njc4OTpDREVGR0hJSlNUVVZXWFlaY2RlZmdoaWpzdHV2d3h5eoKDhIWGh4iJipKTlJWWl5iZ
	 277  mqKjpKWmp6ipqrKztLW2t7i5usLDxMXGx8jJytLT1NXW19jZ2uLj5OXm5+jp6vLz9PX29/j5+v/dAAQA
	 278  Cv/gAAQAAP/AABEIALABQAMBIQACEQEDEQH/2gAMAwEAAhEDEQA/APnCFTk5BPPGKliAB718W7H2j3Ip
	 279  VUuwJxzTfKXacde9VBhYRUBAyO3pTmUAbSMU5WGmybywzHGAMdelPVFC+n1qXZCuyaJADxjj2qzbBMAP
	 280  xz1rKaVib6ltLcFvlIx2pLy0dwuAMMBnH1rFON9RNsszAZPFEYHldPzrOy3KewmBk9qUABugxjtTVmiW
	 281  xWRcjp+VJtHXgVL3K6AgBDdM9eRTNzAZViOe1VyxaJavuf/Q8aW4mUcSGpo764AyHz+FfnnJBvVH1UsN
	 282  CS1Q/wDte4Trip49ecA7g3FSqMW9zlqZandxJ4/EKADcSPqKni8QQMT865qOSUNjiqZdNbFiHWYXz84N
	 283  WE1KNsfMKj2zirHHPDSj0JFvo2H36d9pUjg1sqykYOm0KbgY60omXPXmr9pFkco3zBnrQzjGcnrRzp9S
	 284  bEbuOvao3fisZSXUpIYWGKGcbetTCSswsxnmACkYrtNSpJ2YNM//0fnK1BD7sDg9KmUHeOe/Svid3qfb
	 285  SdmQ3AHmnr1pGBC5z19a0hohNiJkensM1J0yCKmY0yZR82e+BT1BxnpmpepN9SRCR0NSpweOoPWs6isr
	 286  ijuWIZGBA/lVwzMVFY8ibuhXEfr+tOz8hIqUymhRnJGTSc5wBVRRDFPXBHJpB3qdmV0EX7vXmoyfl685
	 287  p2dxWR//0vFsHZ9TQv3T618Bqz7PSwwn1phPXpSWrQEUhIx0NVXc7j0rSNwViCS4dWYpJj3BpBqVzGy7
	 288  ZmHSq9kpblSpxkveQ+PX7uMf6wEDtU0fi24TAkX8jTeCjJaaHDUwFN7aFq28aL/GCMGrtr4xtHGGkA+t
	 289  YTy+a+E82eAa2LsXiWzI5mXPHercOsW8hwJB+dcUqVSCOKVFxdmiwl7E2MOPzp4nQ9GH51jzNbmUoOIC
	 290  TI4okOaUXoybDCevNBPHX8qIO4mf/9P52i4dix5zjp/n1qZFBCmviL6an2kt9CGcYnJznJpOBwegq4vQ
	 291  L9xIUytSkfu/bv70p7j6EnQgjHSpFGVqXclkkaHb1+makUHgdazm7IFuSKOasrnjis2+oDm9qnIHlgd/
	 292  es7gxqjkt1NLwH4xTTEhjkhutM3D15oGhkcnBGRTDKu3A7H1rS3cLn//1PEhJ8uM557UvmDaa/P7a3Ps
	 293  xpZcZ6mo5WG45pdUC2K8ko5JIzWfcTqu7HPHrW9OLbKWhSluVLNz3wKrS3I3KfcV1Rg9CrpXK8l0F7io
	 294  pLnLnJHGOldMYJGMpu5XNwuxjyRTBcAAjd1HWtfZmPORy3WAWWQDOM4PWtHRru6DFlmY88ZqKsVyXaKp
	 295  QjOoa7axe28G/cWqhoHjO/n1WeJwSkS9c981wUcFCopPayFj8JC8VFbs6e38VldvmHHFaMHimAoCzDB6
	 296  V5U8FKz5TlrZU4/CXYtdtnXIarEepW7jAcfnXGqEoKx5tfBzh0P/1fnqEAsc/wB6pI9owAD1618Qn3Ps
	 297  35EE4UzHrx79aXaMcdaqAMWIADvj271IMeXg59KUmNLQkUDfjb1FSLxzg0pWJRLGAQAeMVIoA+uaxlaw
	 298  0SoF/u1KowwwDUcwuo9wMjrUrY2ZPOKy0KY1T1NMdwG/CtBEFzMqnIPNUZ75FBJP5mtIQvYfoU21JFVs
	 299  N271AurRE/e611xw73Yj/9b50GsQhOXHWnpq8JX7w4PWvjPq76H2fzHjVYCud9Q3GrRAZDUvq75kNbMz
	 300  7vV0zjdjNZ82pqzMcj7tdlPDtIiVWKKct+AxwRxUbXi7VPJAIrZUdEZOsrsga8DFgelQtd98g5P6V0Qp
	 301  GE6qIUut2cZ470kd2FjYc4Oce1bSpJ3Rzxq21GNcDZhSeg710ujKRbKzAg5rkxceWnqd+XtOo7bD9cl8
	 302  qxLDPHasXwUvmyXU7Lgl8cegrnw2lGbZ14l3rU0bl3gMQCRgVU1y7WytUZQzMRwBXPRhzWRvVny3ZW8N
	 303  6xPdXBikiZc5IOa6GG6nDsd5xnAyfaliqEacrGOHarx5pI//1/nuL754HWngEkYx1r4VWsfaMjk4mP8A
	 304  OgnjPH1rRMLKwR4A2jH1FPA+TNRIa0ROvQcY4p4GF/pUskmi6+gqRACvPrWMnpca3JABjFSKCQOnFS2u
	 305  o7E3XBOKcR8ucdKzUkDGSHGemKpXchVuP0rSmDMfUrl1J5rn9TvnVCc9OtelhoJtDekW0Yb6pId3zdRw
	 306  RVT+0pAPvc57CvbhQVrHlTxD3P8A/9D4tbUpTH1I7cU/7ZdMnyqcdsV5vsErXPbWJbHLdXzYQDBY8c02
	 307  6udQjyGVuD1FHsqfMridepZ6ED3s4IDqeD0I68VEt7J5hy3GO9aKkuhPt2BumeRjnv3pJLlgwBYE8ZqH
	 308  T2GqujYLcuWYbhj0zTHm5B/vcGtVBLYzdRtEcUueoGB3FOjmBjcBhx2NNx1IhO+uwtqd93EgA5YcV32n
	 309  IqwrnAz2rzMx+FI9nKldy+RmeMpfLs8DGTxTvBUKw6Csjry2WPbrXHB2wzfdnbUu8SvJF4xh1LDAJNU9
	 310  UtVmDs3IiGB6CuenNx1R0yjfRGd4aRTqJdFG1ARXRgANg4/yK0xbvJehnhlaL9T/AP/R+e4x8xx609F+
	 311  YZ718L6n2ju2RzqTKcYpQMjsc1pHTQWlgjUjGVH0qbkr0BqJSKRMi+uBx3p8a5HYVD8yb32JY15FSKpx
	 312  nisp6RuNbj1BzUyrnkmo6FEqrz7U/advHOazvcRHIuSazNXDpbSSJjeqErnpmrp6CueeXusahO5zKi8c
	 313  7VrPuGklUiSQtkd6+po0YQs0edVrTd1cqeSoJOB0xUCxpnouAecV33e558rbH//S+KdmFHTk1btywUqc
	 314  YNcEnfc9SGl7DyAJ1AHIParx+YZ4/Guea2OmC1dhGjQn5kXNQtbQFiWiTlfSoTa2G0nuU5bG2aQ7V2jP
	 315  JU+1RSaXC2GjuApyOorX2slYz9lF3sV/7MmViFljaoJLG6SQbkyDXQqiZzyg1Yg8i4jBJjIBpsaPyXXB
	 316  Psea1TTMJJqysaeh2u/UUfP3QCBXdQJtTpivFzKV3FH0WURtCT8zmPHcrhkRSOWro7O28rRYIgOwGB3r
	 317  mnph4+bOxNvEy8kWFi+ULxwKzNRkMemSPj/WMT+FckNWdLv0KPhCMmGSZl6k1ulC3zY5x2+la4r+IyKH
	 318  wH//0/nuIHB9c9KevUAHk18La60PtHuRy/64+lOGBniqXcOlhUIxwB+NSrynSpndFImQc4A7d6lQccdR
	 319  WcyUyWMccDnPSpVAwM1nLYaeo5BjrUyjFTugJIwd2Kfgkc59qyTs7Axkigqao3qBkYdiMVpTeugHmF7b
	 320  hbhl2/dJB/OofJBTHfp9K+ppTbimeZOK5issYG5W7VWdBnHXB65rrjJs5JLof//U+LYtu7leM+lSpxIR
	 321  7VwO90eskrNkqZLo3PXHoausxI4wa557o2p6JitnOCoqvI3zEkdF5qIrUuW2ogO1iWHeibazIQncHA+l
	 322  DT0aCyaaZGNm8kA9PSl2qy9SB78Veq1ZCs9BkOGUrj86zdQGbllVMAe1bQdpGE1eBo+FoCbgtxkY966+
	 323  E4hOeo5rycxleR72VwcaRx+t/wCmeJ7WAdDIOPpzXbSpt/dkcRr+tZ4j3aVNLzNKOteo/QjuiY7Jm6Ej
	 324  ANYnitvL05YRxwOf8/WuXDK8l6nVUlaLZb8NQeXpijgZB/M1oIpyPzx74pV2nUbHTVoJH//V+fYhgnnv
	 325  Txy4GBXwse59m7kMyEzkj8qkQfKatdgewIo7nIqdQAnXms52RSehMoHHPapY1wMAgVnKzFtqSwjjg4qR
	 326  VJHXIzWc2rDiPVeeD+FSDqKh2sBKo54p+Pl61jbQG9RrDIPNU7teT6VpCztYDzfxGskWtXESdN5Pp15q
	 327  gN2GZpB0r6ig17OL8jz535miCSPdnaxHHpVV48D7xIB7iu2LOOS7H//W+MCoeIDcc5p4VhIMDkDvXnpq
	 328  +p6zu0SwZZlVm6HJFWyRg89MdawmkmrG1NtpiMcY5OevFQ7AWOT0FSkkU9UKUPmEh8jt+VMdGLDLYAIz
	 329  xUtrQfLo7Mj2SHjePrSspxgEk1rdGSTGJjymLEZArOjAd5GLHk9DW0NGznqa8qOj8IRHBbrnnmugu08u
	 330  1ZiSMn868LGz/eH1GAVqKOW8LR/bfG4c8rCCx46HpXZspk88jHzMf04pY7eEfIjDO7nLz/yKmqh/sjwR
	 331  EFwAemcVhamkmpTRxKpyCN2RWeFsveZ01FpbubsEaRWyqhAxnH5YpxIx8rf/AFuK5W3Jts2Wisf/1/n5
	 332  SSxHOM+lP7jGa+EVz7R2IpATN1IIpwB55NaJ2FuhYzx3PvU69OQaio7sEiZOvfpU0YwmMVnJ26DRJH2G
	 333  DipUyR361jN6FIeq8/0qUdBxWbkCRIg/D6U8j5e9Rza3ExrA8nmqt0Dkmri9BnnfjlSmvuwGQyhulYkr
	 334  yL86DANfTYRp0o3PPr3UnYbBOWU4zz7VHIGIJVjkGu1x5Tl5ro//0Pi3fhgMHJPXFWCeQwLe9ec+jPXj
	 335  1JIM7gw44qy+WPUjkcVjPdGsNmgdsNkjJFQ7mMhAB5FRHuXJ6WRIw+VwCc9KbtPy5JyCKgdmxhBDNj8s
	 336  U1Cyr0J/rWultDOzTuMuSiWjlT97r7HFZ1nkk4bIPXiuiD3uc00rqzOy8Lw+XBuJPQcGrXiGYJYMwJHB
	 337  xXz1d89U+sw8eWmkuxi/DGP/AEm+vHycYUE/jXXu6w2vzdcfiaMw1qpLsjnwi/dt+bKCn5nlw2W5Gacw
	 338  GD2wB+dcq2O4AnyADoM80QLukUsp9f0qb6XHuf/R+fkOWNSfwjnivhUz7Nq2hFJ/retPA4PWrWuwul2L
	 339  FjAA6VMMFeTms5PUpbEw6/hxUyfd4as5PUETRds9KlA+Xk96ym9FcaHJgGpOv1rPUpIkXg8mnNnGaz5r
	 340  aCaEPeqtx1OT0rSL2sC1OG+ISquoQuT1UjP4/wD16wGEZUYGPevo8E26UThrpc7G7ICDzg+1ROmF+91O
	 341  K7VKVrHNyxWx/9L4uuVAcH371JvKqScFPU1597pHqtWbZNZnc+QxI9atv8p4z9ayqPVI1pr3WyPBLDGf
	 342  qKYnExyeQKlFPQXH7zgdetSk8rgEnis29i0lqxijLEjjt1poU7iVHHpVX3uRbsZl+2IvLX+I56U/TUUA
	 343  KxGSfSulu0XY5oq80drpcZSzHvjpWd47fy9O2g8kjpXz0ZJ1kvM+skrUnbsSfDm1C+HlfJ/euXIPRq3b
	 344  lleQRYBCg5HrSxk+au/IxwkbUokRw0u0cBFyR70wEEbm6sc/gK5YuyZ1PzFVgVG4ZIzmpbTaJMt07+3F
	 345  Q9i7n//T+foic55yTipRkYBBxXwaPtXuRyZEg4pWII4qk7C6BFwAf51MhG31+lZ1Frca2LKHn8qlDY6L
	 346  UNgl1Jo+2akQ9BWVR9xpDgffrUq+wrO7tdDsSIPUUpPHvUK1xMM8HA61WmwWOB+NXENjiPiMhE1sw9WH
	 347  8q5vqRnjivosD/BXzOKv8QkKgZBA6ZpV27MkDOa7ObsYI//U+MdVGxlK9zninqd1sCQM45rzYaxR68vj
	 348  YtgT5h6jvV6Q5X0+lZ1n7yLofCxhOenfFMTI3cdRzWV9DWw5ARI3qPSnMSqKCOSRUy6FRurjFLjPp9KA
	 349  xx06n1q1qjO70Me6YtcOOcKcH1q9oqF5l75Oea6KtowOagnKol5neWcSJaocdgRzXGfEm53ERKfvEV89
	 350  gvfxCPqcXLlw8/Q6fwkph0aCEg4VB/KrsDbmcnA4PWoxFnVkxUVaml5IgR9sMj4+ZzSTuTjcOB0/CsUt
	 351  zo0VrhCQiF2GcAn/AD+dWLRlYZPQ8cVEk7aF+p//1fAIuvfOakxnr+NfBJ2SsfaN6jJRiUA9RSheCMfn
	 352  VXEtgjUk/wBTVgfdBwOfSs5stbE6g7unYVLGpwAazYvUmjHHanqDx061lLazKuh6DHBFSID27VEthkin
	 353  5cUHOPxqLvqJiEYziq8/FaQ8hHH/ABEVvIhYYyHNcsrSZG5RyOtfQYC3slfzOPEX5tAA+amHIjO31ruu
	 354  rHNa7P8A/9b411QMIwSDnNR2xYQNkjnnkV5sLctj15JqRLZjBzweeSKuycHkD8qyq6tF0rqLI2OTnK5p
	 355  sGWbBHQd6zWxo3qSdXLYxTpPvLnvjrWdr2LvuNYYUnj6Uxyu4/KMrVx6ky6GOGLSOwXIYmr9n58UQeFg
	 356  svbcCRXTVty2ZzYZOVRcpvDW721tv9LsBIpAHmQNn9K4zxPqSX2sK6hljDDO6uHBYWKre0i9PxPTzDFS
	 357  VDkmtX9x2mm65YG0REnTccAc1rx3EJgbbIpyvUGuDE0JxleSO+hXhUj7rGK/7uNcj1P6UmSU+fHPJrlS
	 358  5bnXe9mA/wBSQDxzkfh/+qp7YbIipbaOufwrN7WLR//X8DTO7I9e9SJnAIHNfBPY+ze4yQEPnGacoHof
	 359  amvIELCDnnqanXoBzUSaRSRMo5J744qRBxzUSelwRPEMingdsVk7pFLckA5p44OTUXuh2HqOKRskE4qV
	 360  sKQADkYqvcj8aqD0shWOV8fqDYqc4w4rj5D365Fe7lz/AHdjlxC11CNhyRnp0pqkYOc9a9G/c5bLof/Q
	 361  +OdUH7sDnOeKrREgEN6V5tN+6exUT5rlm1IC9ec9qssBg5xzjNYzdpJmkNtRhJzgflSRDqD6flU7FkqZ
	 362  55+holblfUVk90aLYYxJH3agvGCQPkjJ6DFaw7GMtrmdYpkcg1q6fsEwB6Ct8Q73M8DpKLZruu+3ZM43
	 363  ADNctfeHt120cEjO3U7scmuDCVvZNux7GLw31iNmypLoN5byByrYzzxVfzdTtXcxzSAD3r0oYinW0Z41
	 364  TB1cPrBlyy8S6nF9+QOMba1LbxepwsyFcYztrnrZdCabgb0MzlCyqGrZ+IbK4U7bgDg8E4rXtL+CQBfM
	 365  VgTn9BXi1sJOno0e5SxVOqrxZ//R8DiGScip0HOAOa+CfkfaDJV+fpRjBIxT6iHRjnIHFTAYXBIqZ7lK
	 366  1rEygZxUsYI68Gsm7gkSQ59PwqZRxx681lK/LYaHpjI7U8ge1ZK3Qdxw5XpijAOelGoNCDPOOaguOf8A
	 367  CnFCOa8cJu0uTI6EHH41xLou3gYBr3culaLXmc2JV7DIx8pXHFNAxknGc16b0OM//9L44v8A/V5wMiqy
	 368  /dbI6j1rzKb0PYq7li2C4HHHrVmT5Rx9ayqN8yLppKIwEj6+lPhAGfzNQ9rmqexKvcAYxTTjeM88isty
	 369  xhI3mqWrNwqADB5ranozCo9GQaeBjByR1q2GIzjj2rasryMaErRVieK/kjwHbIGOBVrSbtZNRmlfAB+U
	 370  ZP0/wrinQsnKPY9eljVJqM9zYgMTMAjoRnoRUV3ptrMrl4UY+o4rzIzcHc9Nx5kZc/hu1lUlBszWPqHh
	 371  Z1lYxHJHQZr0cPmElpI87E5dCptoZl1o9zADuVhwfaqqz39tIPLeVMDoPpXqUqsKyPHr0KuGfun/0/Bo
	 372  Rzlh+lToOBXwGltD7RjJQcjtSqODn9RVLyEEY96nVflzis6l3YpaEigdscCpl5XqeKiaVgJIhn/61Sgc
	 373  AdOaylqkyluPXv7U8gZx/Koih9R+ML1ppHynIoRLEAqKYZ6njFVHRXBnP+Lk3aXN0yFz+tcM6j15r2cv
	 374  ejOfEdLjMDb94ZHaoyvy9O/Y16mqOQ//1PjfUcBcnjJ9KqggRldo579682krxPWqO0i1AM7c59OlWJMA
	 375  /lWVTdGtJaNiLg89afCMkjOOO1ZtGu5IpCs+emf6Uxug6Y45rO3UruiFmKydvrVK8ZZLogAn+ldNOPU5
	 376  pvox1pH+7JzhumKlVfkOckmnNq5MI2SGScE5HTBqWDcBkMvJ6CpuraFWd+w3ULqW3gMsb4fOOO1VNP8A
	 377  FGoxsyy7JFA6kYNKGDp1Y+8aSzCrh5e7qjesPE9s8eJh5eRnIrShvbG7AMUqZ+teVVwU6TbWqPaw+OpV
	 378  0lezJWgV4yAVYc8GqV5pFpMMyW+O+V+lc1KtKD906q1KMlaR/9XwqPg9e9SqOQQetfnyfc+0Y2UfOKFy
	 379  B25q1K+grCx564FTL93rilMaRKvXtz7U9OnfrWbtbQdyaLoOtSr/AFrOSstCl2HLnPNP5z1qNEA9s4FM
	 380  7daSVhMAcZzTJlOD1prYDC8VoTpdzjP+rbH5V5w/nDkj9a9rK2mnc5sVeysQxmYMd4wDUkLM3LZFeu0u
	 381  hxan/9b44v8AG35cnnNVArKM+tefS+HU9WoveuizCQCBuOT61ZcnGcgmsZrVG1N3TGrk4x3qWBQGPUHF
	 382  RJspa7kjAFmwOp6Y9qSXPyDtkVgnsa9yFxhWYrj2rJUtJcFgB75rqpSvc5K0bWSRetV2Rg98c8d6UZCs
	 383  M8AZPNTLWTuVFWiiIgjcBnDY5NToCsZAyDnqKUrtJDitWUfETbYI1Ixz7ZNYkXyzMAMevHSuvDfCceK3
	 384  NKGIG0Ppj73pVdi0F2MMQDj7pog7uSYTXKotGxYajeQwA/aHOAeG5zV6HxJLCdk8KuvTK8dvSvNq4OFV
	 385  6aM9ajmM6SSlqj//1/DIOpHGaljG4cYH4V+e62PtWEi45pvzcjIwKuLsS9RYwM9s1Jj5Md88molqUnYl
	 386  Xlue1SryKlpodiWLGB6GpBWU9FcEPQd6eBgjNRYpCn7opDyM0Ru9RMQdxxzSODVR12F6mTryFrWVcfeU
	 387  j9K82ctvxxXrZY7XRjiF7qEjUHgkYFQuuCeB19K9dO5xSSsf/9D44uwSmM556VXkXMJHGenFeZTvoexN
	 388  Ilg4bBOTjrip5uQcY4x2qJ3ckVDSLQ6MdCPpT4uGIHGOaze1jRJbkjFgDgYpjEgqxI7dqyWhpuQ3rkQS
	 389  HjpWfaR9/wCf1rqp6RZy1Peki8MGMfNx7d6jc7Yyc4ye1JptjvorjBneDw2QMVPEHBIOFwR3qbXF5ox/
	 390  EDl71U4ODiqDrjJABIrsoaRSRx19ZNmrbD/RgScAjsag1CF9pcBTxWMXaTNZRbirDrF3CbSQTg9f5U+Z
	 391  QJxwFBA/lRZc1yrvlsf/0fD4fvE1MnYk/nX5/wBD7R7iP1yMdaQLzRF9LCa7hGvOf0qXHy9qUtxkgXnt
	 392  UqDk0pK7sCfYkUcYA6VKnIFYzWha7jhT1Hes42TsC1FIyOlNYEU4oT8hB3pGzt68U1tYVzP1IZQ56Yrz
	 393  e6UC5lUrwGOB+Nepl3xNGddXiV0Y8jGPw6U1slSCec9a9m6OCx//0vji/wAiMY45qux3Rbh/KvNprRHr
	 394  1N2iW3z5gHY81ZnA3AnipqfErF01aDuPhwXAPHSpME7jyOK53pKxt9kT5vmB4IPHFMkwcEnkkVPkPpqU
	 395  75zt8sDBJ+tNtVCt8x59q6rWi7HKneSuTxAhCAQOcc0yQEA4br2FRfVqxfK7LUYw4weCu3n1qbdiNmOM
	 396  EgY9KV9CXo2YF+VMyNx8zE5qMgsrhccdq7obK5xT3aRdtTi2VmPGMdc+lWJgr5XAGMfrWElq7G8XsmUx
	 397  C8ZJjOQQcgdakVXZxv4PbjvijmTJlBr7z//T8RjIGPc1KhXqPXivzzdH2bWoSD5sg4INJwRzz9aEw2BM
	 398  dM5qTjb60SdykSA88enepEOOKU31EkTREdqkH54NZT1Ra0Hd/SnHip0Q/IVjwPWmsePehPqiWC9T1pHH
	 399  B5oTaFYoaiOOn415tqm4X842jiRh+tenlvxMzrfCV05JLDHTgD3ppYFCB6/Svb1aOHbc/wD/1PjnUv8A
	 400  Vct3qq53IMeleZDZM9ipu0SRYDAkkewqxI24+w6VM90EL2aJYCN5AbgY/lU0R+9z+FZS0N076DZSNzc5
	 401  yc0xyoUfh25qG9mU7XZnXLZuTgnaKsW4zjqRnriuiStHQ5Y/GOxgEVFuwpBA9+MVKsytUOjILcZ7daNQ
	 402  ZEt3O7k8fjU3ldDbVmc/cPm4QYA/rS8gMD2zz6139EcF7tli2lVIAhVsYPap45lZiFG08DnrWLT1Ztde
	 403  6kidMGMZHGD3pkqjcOcZ447cVjF6o0nZxbP/1fElB7HOD1qVQOOtfnSVj7VsRwNx60Ywp4PFOCVwvoIg
	 404  HB9e1SAAjGcUpa7AiQDkf0qRRxgZGaUrWuwJYwOByalXGB6A1nOzRS3FAI70/GSOtTtuMVunemHGKUe1
	 405  iRRgk0hwKqPkDZS1DkYFed+JEEWs3Q5GHz9M816GX6VPkZ1V7pQibOck8jrUbgDIA4r3epwJ6an/1vjf
	 406  U9vl43YyelUwR5RGTxx0rzaauj1qrSlckiA3D5j071aYjccnHSlJXaHDZsntzkY3Z96lQjnnt9K55I3T
	 407  GsRyDnI9ajfKKOemDxio7Iq+7M3JeVieSecYq7AMKF3Fj6V0zVlaxzQfvXGDpy2McdajjydwIO2h2Hd6
	 408  ND41G8nAGMdD1qLV2zAFORl+R1qEtVcTdkzBl5vSCMgNwKlVG3MFJ4967r6I4bO7JY1BjJJ7Y6dDVqyQ
	 409  CYkqTjvWEn7rN1unYtYQLntg9+tV2+aQBlbPTHrxWNPzNqux/9fxWMDP41IoUYr87WzPs3uEgBJ9uKAB
	 410  tINCSbDoJGAMcdKkAG04/KiSXQpPQkXGcY7VINp56VLBbksYBxUgAx0rOadrDT7C45pxA4wKVkUhX7ZF
	 411  NODzRG1iGAA5NIwGeB1FNLuBTvEBI7V5/wCNFP8Awkl1tIABUdP9kV35cl7VehFbWBlxbNuD8x9aY20Y
	 412  xjOa96xwWstD/9D451XAhBOOTxVRlURdOa8ymrI9epu2SW23cMgDHb1qw4BcrgY9KJ25kOn8LLFsArYG
	 413  McVIu3GdvOOTWLV2axeliOYgM+FzmoL1isAwME4GKVloDvqVoI9zEEAMMGrbIAmFUdcZrSbS0MqcWQBg
	 414  0WMg4PalhCFCdpHWiXUFZtaDtqhywBwMY9Ko6sR5oXPTn2p09WianVIyEYebwMEnPNTL5eXBA4Hautp6
	 415  WONSWt0TQMhUgjKj1q5bhQWKMDwBxWE20mbws3ElXy9pUEDqMd6hfaHUgE7B2HtWUO7NKiVrI//R8Vj4
	 416  br37VMuT9TX55HY+ze+gMM5PSkzwR1AoWu4IRRg9akPC8HpSlfRFLXYeCN3TkipV60ulmBJGeR71IvTP
	 417  Ws5K6Ghwp3FQkMH6elNJ461didOoq9SaRjxSjqwuVbgZkH1rzrxWwk8QXj9hIR+XFejl1nVt5EV/gM2I
	 418  qCTjPTH50x+eRxya961tzzt9j//S+O9WJ+zjgDmqJYGDJYZNedSSaPXqu0tew+L7wPTvmrBO5x7Y5qZf
	 419  EnYUPhZdhUZyO3pTgcOxGOR0rnbu7M6fhK9yVEvPy59O9QX75VBjnIoS+ETa95DLbdu6nnp7VYGSMk59
	 420  vxrWemxlTZCvMe4BTg/nSRKHVsjGfpSe90C1VmTsG2H2x3rF1J1eVxgcHr+FVQWtyK7srFGAgOrHqDnp
	 421  7VPGNzO/O3kkCut7pnGno0idCCNyt6Crca5ZiDjpgVzXsnc6X71rDyBgcbtucn8KhjBMpOfrz7VlTdtT
	 422  Sqm7JH//2Q==
	 423  `
	 424  
	 425  	data, err := base64.StdEncoding.DecodeString(base64EncodedImage)
	 426  	if err != nil {
	 427  		t.Fatalf("base64 DecodeString: %v", err)
	 428  	}
	 429  	if _, err = Decode(bytes.NewReader(data)); err != nil {
	 430  		t.Fatalf("Decode: %v", err)
	 431  	}
	 432  }
	 433  
	 434  func TestExtraneousData(t *testing.T) {
	 435  	// Encode a 1x1 red image.
	 436  	src := image.NewRGBA(image.Rect(0, 0, 1, 1))
	 437  	src.Set(0, 0, color.RGBA{0xff, 0x00, 0x00, 0xff})
	 438  	buf := new(bytes.Buffer)
	 439  	if err := Encode(buf, src, nil); err != nil {
	 440  		t.Fatalf("encode: %v", err)
	 441  	}
	 442  	enc := buf.String()
	 443  	// Sanity check that the encoded JPEG is long enough, that it ends in a
	 444  	// "\xff\xd9" EOI marker, and that it contains a "\xff\xda" SOS marker
	 445  	// somewhere in the final 64 bytes.
	 446  	if len(enc) < 64 {
	 447  		t.Fatalf("encoded JPEG is too short: %d bytes", len(enc))
	 448  	}
	 449  	if got, want := enc[len(enc)-2:], "\xff\xd9"; got != want {
	 450  		t.Fatalf("encoded JPEG ends with %q, want %q", got, want)
	 451  	}
	 452  	if s := enc[len(enc)-64:]; !strings.Contains(s, "\xff\xda") {
	 453  		t.Fatalf("encoded JPEG does not contain a SOS marker (ff da) near the end: % x", s)
	 454  	}
	 455  	// Test that adding some random junk between the SOS marker and the
	 456  	// EOI marker does not affect the decoding.
	 457  	rnd := rand.New(rand.NewSource(1))
	 458  	for i, nerr := 0, 0; i < 1000 && nerr < 10; i++ {
	 459  		buf.Reset()
	 460  		// Write all but the trailing "\xff\xd9" EOI marker.
	 461  		buf.WriteString(enc[:len(enc)-2])
	 462  		// Write some random extraneous data.
	 463  		for n := rnd.Intn(10); n > 0; n-- {
	 464  			if x := byte(rnd.Intn(256)); x != 0xff {
	 465  				buf.WriteByte(x)
	 466  			} else {
	 467  				// The JPEG format escapes a SOS 0xff data byte as "\xff\x00".
	 468  				buf.WriteString("\xff\x00")
	 469  			}
	 470  		}
	 471  		// Write the "\xff\xd9" EOI marker.
	 472  		buf.WriteString("\xff\xd9")
	 473  
	 474  		// Check that we can still decode the resultant image.
	 475  		got, err := Decode(buf)
	 476  		if err != nil {
	 477  			t.Errorf("could not decode image #%d: %v", i, err)
	 478  			nerr++
	 479  			continue
	 480  		}
	 481  		if got.Bounds() != src.Bounds() {
	 482  			t.Errorf("image #%d, bounds differ: %v and %v", i, got.Bounds(), src.Bounds())
	 483  			nerr++
	 484  			continue
	 485  		}
	 486  		if averageDelta(got, src) > 2<<8 {
	 487  			t.Errorf("image #%d changed too much after a round trip", i)
	 488  			nerr++
	 489  			continue
	 490  		}
	 491  	}
	 492  }
	 493  
	 494  func benchmarkDecode(b *testing.B, filename string) {
	 495  	data, err := os.ReadFile(filename)
	 496  	if err != nil {
	 497  		b.Fatal(err)
	 498  	}
	 499  	cfg, err := DecodeConfig(bytes.NewReader(data))
	 500  	if err != nil {
	 501  		b.Fatal(err)
	 502  	}
	 503  	b.SetBytes(int64(cfg.Width * cfg.Height * 4))
	 504  	b.ReportAllocs()
	 505  	b.ResetTimer()
	 506  	for i := 0; i < b.N; i++ {
	 507  		Decode(bytes.NewReader(data))
	 508  	}
	 509  }
	 510  
	 511  func BenchmarkDecodeBaseline(b *testing.B) {
	 512  	benchmarkDecode(b, "../testdata/video-001.jpeg")
	 513  }
	 514  
	 515  func BenchmarkDecodeProgressive(b *testing.B) {
	 516  	benchmarkDecode(b, "../testdata/video-001.progressive.jpeg")
	 517  }
	 518  

View as plain text