diff options
Diffstat (limited to 'src/pkg/image/jpeg/idct.go')
| -rw-r--r-- | src/pkg/image/jpeg/idct.go | 124 | 
1 files changed, 69 insertions, 55 deletions
| diff --git a/src/pkg/image/jpeg/idct.go b/src/pkg/image/jpeg/idct.go index e5a2f40f5..b387dfdff 100644 --- a/src/pkg/image/jpeg/idct.go +++ b/src/pkg/image/jpeg/idct.go @@ -2,6 +2,8 @@  // Use of this source code is governed by a BSD-style  // license that can be found in the LICENSE file. +package jpeg +  // This is a Go translation of idct.c from  //  // http://standards.iso.org/ittf/PubliclyAvailableStandards/ISO_IEC_13818-4_2004_Conformance_Testing/Video/verifier/mpeg2decode_960109.tar.gz @@ -35,8 +37,6 @@   *   */ -package jpeg -  const (  	w1 = 2841 // 2048*sqrt(2)*cos(1*pi/16)  	w2 = 2676 // 2048*sqrt(2)*cos(2*pi/16) @@ -55,41 +55,45 @@ const (  	r2 = 181 // 256/sqrt(2)  ) -// 2-D Inverse Discrete Cosine Transformation, followed by a +128 level shift. +// idct performs a 2-D Inverse Discrete Cosine Transformation, followed by a +// +128 level shift and a clip to [0, 255], writing the results to dst. +// stride is the number of elements between successive rows of dst.  // -// The input coefficients should already have been multiplied by the appropriate quantization table. -// We use fixed-point computation, with the number of bits for the fractional component varying over the -// intermediate stages. The final values are expected to range within [0, 255], after a +128 level shift. +// The input coefficients should already have been multiplied by the +// appropriate quantization table. We use fixed-point computation, with the +// number of bits for the fractional component varying over the intermediate +// stages.  // -// For more on the actual algorithm, see Z. Wang, "Fast algorithms for the discrete W transform and -// for the discrete Fourier transform", IEEE Trans. on ASSP, Vol. ASSP- 32, pp. 803-816, Aug. 1984. -func idct(b *block) { +// For more on the actual algorithm, see Z. Wang, "Fast algorithms for the +// discrete W transform and for the discrete Fourier transform", IEEE Trans. on +// ASSP, Vol. ASSP- 32, pp. 803-816, Aug. 1984. +func idct(dst []byte, stride int, src *block) {  	// Horizontal 1-D IDCT.  	for y := 0; y < 8; y++ {  		// If all the AC components are zero, then the IDCT is trivial. -		if b[y*8+1] == 0 && b[y*8+2] == 0 && b[y*8+3] == 0 && -			b[y*8+4] == 0 && b[y*8+5] == 0 && b[y*8+6] == 0 && b[y*8+7] == 0 { -			dc := b[y*8+0] << 3 -			b[y*8+0] = dc -			b[y*8+1] = dc -			b[y*8+2] = dc -			b[y*8+3] = dc -			b[y*8+4] = dc -			b[y*8+5] = dc -			b[y*8+6] = dc -			b[y*8+7] = dc +		if src[y*8+1] == 0 && src[y*8+2] == 0 && src[y*8+3] == 0 && +			src[y*8+4] == 0 && src[y*8+5] == 0 && src[y*8+6] == 0 && src[y*8+7] == 0 { +			dc := src[y*8+0] << 3 +			src[y*8+0] = dc +			src[y*8+1] = dc +			src[y*8+2] = dc +			src[y*8+3] = dc +			src[y*8+4] = dc +			src[y*8+5] = dc +			src[y*8+6] = dc +			src[y*8+7] = dc  			continue  		}  		// Prescale. -		x0 := (b[y*8+0] << 11) + 128 -		x1 := b[y*8+4] << 11 -		x2 := b[y*8+6] -		x3 := b[y*8+2] -		x4 := b[y*8+1] -		x5 := b[y*8+7] -		x6 := b[y*8+5] -		x7 := b[y*8+3] +		x0 := (src[y*8+0] << 11) + 128 +		x1 := src[y*8+4] << 11 +		x2 := src[y*8+6] +		x3 := src[y*8+2] +		x4 := src[y*8+1] +		x5 := src[y*8+7] +		x6 := src[y*8+5] +		x7 := src[y*8+3]  		// Stage 1.  		x8 := w7 * (x4 + x5) @@ -119,14 +123,14 @@ func idct(b *block) {  		x4 = (r2*(x4-x5) + 128) >> 8  		// Stage 4. -		b[8*y+0] = (x7 + x1) >> 8 -		b[8*y+1] = (x3 + x2) >> 8 -		b[8*y+2] = (x0 + x4) >> 8 -		b[8*y+3] = (x8 + x6) >> 8 -		b[8*y+4] = (x8 - x6) >> 8 -		b[8*y+5] = (x0 - x4) >> 8 -		b[8*y+6] = (x3 - x2) >> 8 -		b[8*y+7] = (x7 - x1) >> 8 +		src[8*y+0] = (x7 + x1) >> 8 +		src[8*y+1] = (x3 + x2) >> 8 +		src[8*y+2] = (x0 + x4) >> 8 +		src[8*y+3] = (x8 + x6) >> 8 +		src[8*y+4] = (x8 - x6) >> 8 +		src[8*y+5] = (x0 - x4) >> 8 +		src[8*y+6] = (x3 - x2) >> 8 +		src[8*y+7] = (x7 - x1) >> 8  	}  	// Vertical 1-D IDCT. @@ -136,14 +140,14 @@ func idct(b *block) {  		// we do not bother to check for the all-zero case.  		// Prescale. -		y0 := (b[8*0+x] << 8) + 8192 -		y1 := b[8*4+x] << 8 -		y2 := b[8*6+x] -		y3 := b[8*2+x] -		y4 := b[8*1+x] -		y5 := b[8*7+x] -		y6 := b[8*5+x] -		y7 := b[8*3+x] +		y0 := (src[8*0+x] << 8) + 8192 +		y1 := src[8*4+x] << 8 +		y2 := src[8*6+x] +		y3 := src[8*2+x] +		y4 := src[8*1+x] +		y5 := src[8*7+x] +		y6 := src[8*5+x] +		y7 := src[8*3+x]  		// Stage 1.  		y8 := w7*(y4+y5) + 4 @@ -173,18 +177,28 @@ func idct(b *block) {  		y4 = (r2*(y4-y5) + 128) >> 8  		// Stage 4. -		b[8*0+x] = (y7 + y1) >> 14 -		b[8*1+x] = (y3 + y2) >> 14 -		b[8*2+x] = (y0 + y4) >> 14 -		b[8*3+x] = (y8 + y6) >> 14 -		b[8*4+x] = (y8 - y6) >> 14 -		b[8*5+x] = (y0 - y4) >> 14 -		b[8*6+x] = (y3 - y2) >> 14 -		b[8*7+x] = (y7 - y1) >> 14 +		src[8*0+x] = (y7 + y1) >> 14 +		src[8*1+x] = (y3 + y2) >> 14 +		src[8*2+x] = (y0 + y4) >> 14 +		src[8*3+x] = (y8 + y6) >> 14 +		src[8*4+x] = (y8 - y6) >> 14 +		src[8*5+x] = (y0 - y4) >> 14 +		src[8*6+x] = (y3 - y2) >> 14 +		src[8*7+x] = (y7 - y1) >> 14  	} -	// Level shift. -	for i := range *b { -		b[i] += 128 +	// Level shift by +128, clip to [0, 255], and write to dst. +	for y := 0; y < 8; y++ { +		for x := 0; x < 8; x++ { +			c := src[y*8+x] +			if c < -128 { +				c = 0 +			} else if c > 127 { +				c = 255 +			} else { +				c += 128 +			} +			dst[y*stride+x] = uint8(c) +		}  	}  } | 
