Source file
src/image/jpeg/writer_test.go
1
2
3
4
5 package jpeg
6
7 import (
8 "bytes"
9 "fmt"
10 "image"
11 "image/color"
12 "image/png"
13 "io"
14 "math/rand"
15 "os"
16 "strings"
17 "testing"
18 )
19
20
21
22
23 var zigzag = [blockSize]int{
24 0, 1, 5, 6, 14, 15, 27, 28,
25 2, 4, 7, 13, 16, 26, 29, 42,
26 3, 8, 12, 17, 25, 30, 41, 43,
27 9, 11, 18, 24, 31, 40, 44, 53,
28 10, 19, 23, 32, 39, 45, 52, 54,
29 20, 22, 33, 38, 46, 51, 55, 60,
30 21, 34, 37, 47, 50, 56, 59, 61,
31 35, 36, 48, 49, 57, 58, 62, 63,
32 }
33
34 func TestZigUnzig(t *testing.T) {
35 for i := 0; i < blockSize; i++ {
36 if unzig[zigzag[i]] != i {
37 t.Errorf("unzig[zigzag[%d]] == %d", i, unzig[zigzag[i]])
38 }
39 if zigzag[unzig[i]] != i {
40 t.Errorf("zigzag[unzig[%d]] == %d", i, zigzag[unzig[i]])
41 }
42 }
43 }
44
45
46
47 var unscaledQuantInNaturalOrder = [nQuantIndex][blockSize]byte{
48
49 {
50 16, 11, 10, 16, 24, 40, 51, 61,
51 12, 12, 14, 19, 26, 58, 60, 55,
52 14, 13, 16, 24, 40, 57, 69, 56,
53 14, 17, 22, 29, 51, 87, 80, 62,
54 18, 22, 37, 56, 68, 109, 103, 77,
55 24, 35, 55, 64, 81, 104, 113, 92,
56 49, 64, 78, 87, 103, 121, 120, 101,
57 72, 92, 95, 98, 112, 100, 103, 99,
58 },
59
60 {
61 17, 18, 24, 47, 99, 99, 99, 99,
62 18, 21, 26, 66, 99, 99, 99, 99,
63 24, 26, 56, 99, 99, 99, 99, 99,
64 47, 66, 99, 99, 99, 99, 99, 99,
65 99, 99, 99, 99, 99, 99, 99, 99,
66 99, 99, 99, 99, 99, 99, 99, 99,
67 99, 99, 99, 99, 99, 99, 99, 99,
68 99, 99, 99, 99, 99, 99, 99, 99,
69 },
70 }
71
72 func TestUnscaledQuant(t *testing.T) {
73 bad := false
74 for i := quantIndex(0); i < nQuantIndex; i++ {
75 for zig := 0; zig < blockSize; zig++ {
76 got := unscaledQuant[i][zig]
77 want := unscaledQuantInNaturalOrder[i][unzig[zig]]
78 if got != want {
79 t.Errorf("i=%d, zig=%d: got %d, want %d", i, zig, got, want)
80 bad = true
81 }
82 }
83 }
84 if bad {
85 names := [nQuantIndex]string{"Luminance", "Chrominance"}
86 buf := &strings.Builder{}
87 for i, name := range names {
88 fmt.Fprintf(buf, "// %s.\n{\n", name)
89 for zig := 0; zig < blockSize; zig++ {
90 fmt.Fprintf(buf, "%d, ", unscaledQuantInNaturalOrder[i][unzig[zig]])
91 if zig%8 == 7 {
92 buf.WriteString("\n")
93 }
94 }
95 buf.WriteString("},\n")
96 }
97 t.Logf("expected unscaledQuant values:\n%s", buf.String())
98 }
99 }
100
101 var testCase = []struct {
102 filename string
103 quality int
104 tolerance int64
105 }{
106 {"../testdata/video-001.png", 1, 24 << 8},
107 {"../testdata/video-001.png", 20, 12 << 8},
108 {"../testdata/video-001.png", 60, 8 << 8},
109 {"../testdata/video-001.png", 80, 6 << 8},
110 {"../testdata/video-001.png", 90, 4 << 8},
111 {"../testdata/video-001.png", 100, 2 << 8},
112 }
113
114 func delta(u0, u1 uint32) int64 {
115 d := int64(u0) - int64(u1)
116 if d < 0 {
117 return -d
118 }
119 return d
120 }
121
122 func readPng(filename string) (image.Image, error) {
123 f, err := os.Open(filename)
124 if err != nil {
125 return nil, err
126 }
127 defer f.Close()
128 return png.Decode(f)
129 }
130
131 func TestWriter(t *testing.T) {
132 for _, tc := range testCase {
133
134 m0, err := readPng(tc.filename)
135 if err != nil {
136 t.Error(tc.filename, err)
137 continue
138 }
139
140 var buf bytes.Buffer
141 err = Encode(&buf, m0, &Options{Quality: tc.quality})
142 if err != nil {
143 t.Error(tc.filename, err)
144 continue
145 }
146
147 m1, err := Decode(&buf)
148 if err != nil {
149 t.Error(tc.filename, err)
150 continue
151 }
152 if m0.Bounds() != m1.Bounds() {
153 t.Errorf("%s, bounds differ: %v and %v", tc.filename, m0.Bounds(), m1.Bounds())
154 continue
155 }
156
157 if averageDelta(m0, m1) > tc.tolerance {
158 t.Errorf("%s, quality=%d: average delta is too high", tc.filename, tc.quality)
159 continue
160 }
161 }
162 }
163
164
165
166 func TestWriteGrayscale(t *testing.T) {
167 m0 := image.NewGray(image.Rect(0, 0, 32, 32))
168 for i := range m0.Pix {
169 m0.Pix[i] = uint8(i)
170 }
171 var buf bytes.Buffer
172 if err := Encode(&buf, m0, nil); err != nil {
173 t.Fatal(err)
174 }
175 m1, err := Decode(&buf)
176 if err != nil {
177 t.Fatal(err)
178 }
179 if m0.Bounds() != m1.Bounds() {
180 t.Fatalf("bounds differ: %v and %v", m0.Bounds(), m1.Bounds())
181 }
182 if _, ok := m1.(*image.Gray); !ok {
183 t.Errorf("got %T, want *image.Gray", m1)
184 }
185
186 want := int64(2 << 8)
187 if got := averageDelta(m0, m1); got > want {
188 t.Errorf("average delta too high; got %d, want <= %d", got, want)
189 }
190 }
191
192
193
194 func averageDelta(m0, m1 image.Image) int64 {
195 b := m0.Bounds()
196 var sum, n int64
197 for y := b.Min.Y; y < b.Max.Y; y++ {
198 for x := b.Min.X; x < b.Max.X; x++ {
199 c0 := m0.At(x, y)
200 c1 := m1.At(x, y)
201 r0, g0, b0, _ := c0.RGBA()
202 r1, g1, b1, _ := c1.RGBA()
203 sum += delta(r0, r1)
204 sum += delta(g0, g1)
205 sum += delta(b0, b1)
206 n += 3
207 }
208 }
209 return sum / n
210 }
211
212 func TestEncodeYCbCr(t *testing.T) {
213 bo := image.Rect(0, 0, 640, 480)
214 imgRGBA := image.NewRGBA(bo)
215
216 imgYCbCr := image.NewYCbCr(bo, image.YCbCrSubsampleRatio444)
217 rnd := rand.New(rand.NewSource(123))
218
219 for y := bo.Min.Y; y < bo.Max.Y; y++ {
220 for x := bo.Min.X; x < bo.Max.X; x++ {
221 col := color.RGBA{
222 uint8(rnd.Intn(256)),
223 uint8(rnd.Intn(256)),
224 uint8(rnd.Intn(256)),
225 255,
226 }
227 imgRGBA.SetRGBA(x, y, col)
228 yo := imgYCbCr.YOffset(x, y)
229 co := imgYCbCr.COffset(x, y)
230 cy, ccr, ccb := color.RGBToYCbCr(col.R, col.G, col.B)
231 imgYCbCr.Y[yo] = cy
232 imgYCbCr.Cb[co] = ccr
233 imgYCbCr.Cr[co] = ccb
234 }
235 }
236
237
238 var bufRGBA, bufYCbCr bytes.Buffer
239 Encode(&bufRGBA, imgRGBA, nil)
240 Encode(&bufYCbCr, imgYCbCr, nil)
241 if !bytes.Equal(bufRGBA.Bytes(), bufYCbCr.Bytes()) {
242 t.Errorf("RGBA and YCbCr encoded bytes differ")
243 }
244 }
245
246 func BenchmarkEncodeRGBA(b *testing.B) {
247 img := image.NewRGBA(image.Rect(0, 0, 640, 480))
248 bo := img.Bounds()
249 rnd := rand.New(rand.NewSource(123))
250 for y := bo.Min.Y; y < bo.Max.Y; y++ {
251 for x := bo.Min.X; x < bo.Max.X; x++ {
252 img.SetRGBA(x, y, color.RGBA{
253 uint8(rnd.Intn(256)),
254 uint8(rnd.Intn(256)),
255 uint8(rnd.Intn(256)),
256 255,
257 })
258 }
259 }
260 b.SetBytes(640 * 480 * 4)
261 b.ReportAllocs()
262 b.ResetTimer()
263 options := &Options{Quality: 90}
264 for i := 0; i < b.N; i++ {
265 Encode(io.Discard, img, options)
266 }
267 }
268
269 func BenchmarkEncodeYCbCr(b *testing.B) {
270 img := image.NewYCbCr(image.Rect(0, 0, 640, 480), image.YCbCrSubsampleRatio420)
271 bo := img.Bounds()
272 rnd := rand.New(rand.NewSource(123))
273 for y := bo.Min.Y; y < bo.Max.Y; y++ {
274 for x := bo.Min.X; x < bo.Max.X; x++ {
275 cy := img.YOffset(x, y)
276 ci := img.COffset(x, y)
277 img.Y[cy] = uint8(rnd.Intn(256))
278 img.Cb[ci] = uint8(rnd.Intn(256))
279 img.Cr[ci] = uint8(rnd.Intn(256))
280 }
281 }
282 b.SetBytes(640 * 480 * 3)
283 b.ReportAllocs()
284 b.ResetTimer()
285 options := &Options{Quality: 90}
286 for i := 0; i < b.N; i++ {
287 Encode(io.Discard, img, options)
288 }
289 }
290
View as plain text