| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245 | // Copyright 2011 The Go Authors. All rights reserved.// Use of this source code is governed by a BSD-style// license that can be found in the LICENSE file.// Package bmp implements a BMP image decoder and encoder.//// The BMP specification is at http://www.digicamsoft.com/bmp/bmp.html.package bmp // import "golang.org/x/image/bmp"import (	"errors"	"image"	"image/color"	"io")// ErrUnsupported means that the input BMP image uses a valid but unsupported// feature.var ErrUnsupported = errors.New("bmp: unsupported BMP image")func readUint16(b []byte) uint16 {	return uint16(b[0]) | uint16(b[1])<<8}func readUint32(b []byte) uint32 {	return uint32(b[0]) | uint32(b[1])<<8 | uint32(b[2])<<16 | uint32(b[3])<<24}// decodePaletted reads an 8 bit-per-pixel BMP image from r.// If topDown is false, the image rows will be read bottom-up.func decodePaletted(r io.Reader, c image.Config, topDown bool) (image.Image, error) {	paletted := image.NewPaletted(image.Rect(0, 0, c.Width, c.Height), c.ColorModel.(color.Palette))	if c.Width == 0 || c.Height == 0 {		return paletted, nil	}	var tmp [4]byte	y0, y1, yDelta := c.Height-1, -1, -1	if topDown {		y0, y1, yDelta = 0, c.Height, +1	}	for y := y0; y != y1; y += yDelta {		p := paletted.Pix[y*paletted.Stride : y*paletted.Stride+c.Width]		if _, err := io.ReadFull(r, p); err != nil {			return nil, err		}		// Each row is 4-byte aligned.		if c.Width%4 != 0 {			_, err := io.ReadFull(r, tmp[:4-c.Width%4])			if err != nil {				return nil, err			}		}	}	return paletted, nil}// decodeRGB reads a 24 bit-per-pixel BMP image from r.// If topDown is false, the image rows will be read bottom-up.func decodeRGB(r io.Reader, c image.Config, topDown bool) (image.Image, error) {	rgba := image.NewRGBA(image.Rect(0, 0, c.Width, c.Height))	if c.Width == 0 || c.Height == 0 {		return rgba, nil	}	// There are 3 bytes per pixel, and each row is 4-byte aligned.	b := make([]byte, (3*c.Width+3)&^3)	y0, y1, yDelta := c.Height-1, -1, -1	if topDown {		y0, y1, yDelta = 0, c.Height, +1	}	for y := y0; y != y1; y += yDelta {		if _, err := io.ReadFull(r, b); err != nil {			return nil, err		}		p := rgba.Pix[y*rgba.Stride : y*rgba.Stride+c.Width*4]		for i, j := 0, 0; i < len(p); i, j = i+4, j+3 {			// BMP images are stored in BGR order rather than RGB order.			p[i+0] = b[j+2]			p[i+1] = b[j+1]			p[i+2] = b[j+0]			p[i+3] = 0xFF		}	}	return rgba, nil}// decodeNRGBA reads a 32 bit-per-pixel BMP image from r.// If topDown is false, the image rows will be read bottom-up.func decodeNRGBA(r io.Reader, c image.Config, topDown, allowAlpha bool) (image.Image, error) {	rgba := image.NewNRGBA(image.Rect(0, 0, c.Width, c.Height))	if c.Width == 0 || c.Height == 0 {		return rgba, nil	}	y0, y1, yDelta := c.Height-1, -1, -1	if topDown {		y0, y1, yDelta = 0, c.Height, +1	}	for y := y0; y != y1; y += yDelta {		p := rgba.Pix[y*rgba.Stride : y*rgba.Stride+c.Width*4]		if _, err := io.ReadFull(r, p); err != nil {			return nil, err		}		for i := 0; i < len(p); i += 4 {			// BMP images are stored in BGRA order rather than RGBA order.			p[i+0], p[i+2] = p[i+2], p[i+0]			if !allowAlpha {				p[i+3] = 0xFF			}		}	}	return rgba, nil}// Decode reads a BMP image from r and returns it as an image.Image.// Limitation: The file must be 8, 24 or 32 bits per pixel.func Decode(r io.Reader) (image.Image, error) {	c, bpp, topDown, allowAlpha, err := decodeConfig(r)	if err != nil {		return nil, err	}	switch bpp {	case 8:		return decodePaletted(r, c, topDown)	case 24:		return decodeRGB(r, c, topDown)	case 32:		return decodeNRGBA(r, c, topDown, allowAlpha)	}	panic("unreachable")}// DecodeConfig returns the color model and dimensions of a BMP image without// decoding the entire image.// Limitation: The file must be 8, 24 or 32 bits per pixel.func DecodeConfig(r io.Reader) (image.Config, error) {	config, _, _, _, err := decodeConfig(r)	return config, err}func decodeConfig(r io.Reader) (config image.Config, bitsPerPixel int, topDown bool, allowAlpha bool, err error) {	// We only support those BMP images with one of the following DIB headers:	// - BITMAPINFOHEADER (40 bytes)	// - BITMAPV4HEADER (108 bytes)	// - BITMAPV5HEADER (124 bytes)	const (		fileHeaderLen   = 14		infoHeaderLen   = 40		v4InfoHeaderLen = 108		v5InfoHeaderLen = 124	)	var b [1024]byte	if _, err := io.ReadFull(r, b[:fileHeaderLen+4]); err != nil {		if err == io.EOF {			err = io.ErrUnexpectedEOF		}		return image.Config{}, 0, false, false, err	}	if string(b[:2]) != "BM" {		return image.Config{}, 0, false, false, errors.New("bmp: invalid format")	}	offset := readUint32(b[10:14])	infoLen := readUint32(b[14:18])	if infoLen != infoHeaderLen && infoLen != v4InfoHeaderLen && infoLen != v5InfoHeaderLen {		return image.Config{}, 0, false, false, ErrUnsupported	}	if _, err := io.ReadFull(r, b[fileHeaderLen+4:fileHeaderLen+infoLen]); err != nil {		if err == io.EOF {			err = io.ErrUnexpectedEOF		}		return image.Config{}, 0, false, false, err	}	width := int(int32(readUint32(b[18:22])))	height := int(int32(readUint32(b[22:26])))	if height < 0 {		height, topDown = -height, true	}	if width < 0 || height < 0 {		return image.Config{}, 0, false, false, ErrUnsupported	}	// We only support 1 plane and 8, 24 or 32 bits per pixel and no	// compression.	planes, bpp, compression := readUint16(b[26:28]), readUint16(b[28:30]), readUint32(b[30:34])	// if compression is set to BI_BITFIELDS, but the bitmask is set to the default bitmask	// that would be used if compression was set to 0, we can continue as if compression was 0	if compression == 3 && infoLen > infoHeaderLen &&		readUint32(b[54:58]) == 0xff0000 && readUint32(b[58:62]) == 0xff00 &&		readUint32(b[62:66]) == 0xff && readUint32(b[66:70]) == 0xff000000 {		compression = 0	}	if planes != 1 || compression != 0 {		return image.Config{}, 0, false, false, ErrUnsupported	}	switch bpp {	case 8:		if offset != fileHeaderLen+infoLen+256*4 {			return image.Config{}, 0, false, false, ErrUnsupported		}		_, err = io.ReadFull(r, b[:256*4])		if err != nil {			return image.Config{}, 0, false, false, err		}		pcm := make(color.Palette, 256)		for i := range pcm {			// BMP images are stored in BGR order rather than RGB order.			// Every 4th byte is padding.			pcm[i] = color.RGBA{b[4*i+2], b[4*i+1], b[4*i+0], 0xFF}		}		return image.Config{ColorModel: pcm, Width: width, Height: height}, 8, topDown, false, nil	case 24:		if offset != fileHeaderLen+infoLen {			return image.Config{}, 0, false, false, ErrUnsupported		}		return image.Config{ColorModel: color.RGBAModel, Width: width, Height: height}, 24, topDown, false, nil	case 32:		if offset != fileHeaderLen+infoLen {			return image.Config{}, 0, false, false, ErrUnsupported		}		// 32 bits per pixel is possibly RGBX (X is padding) or RGBA (A is		// alpha transparency). However, for BMP images, "Alpha is a		// poorly-documented and inconsistently-used feature" says		// https://source.chromium.org/chromium/chromium/src/+/bc0a792d7ebc587190d1a62ccddba10abeea274b:third_party/blink/renderer/platform/image-decoders/bmp/bmp_image_reader.cc;l=621		//		// That goes on to say "BITMAPV3HEADER+ have an alpha bitmask in the		// info header... so we respect it at all times... [For earlier		// (smaller) headers we] ignore alpha in Windows V3 BMPs except inside		// ICO files".		//		// "Ignore" means to always set alpha to 0xFF (fully opaque):		// https://source.chromium.org/chromium/chromium/src/+/bc0a792d7ebc587190d1a62ccddba10abeea274b:third_party/blink/renderer/platform/image-decoders/bmp/bmp_image_reader.h;l=272		//		// Confusingly, "Windows V3" does not correspond to BITMAPV3HEADER, but		// instead corresponds to the earlier (smaller) BITMAPINFOHEADER:		// https://source.chromium.org/chromium/chromium/src/+/bc0a792d7ebc587190d1a62ccddba10abeea274b:third_party/blink/renderer/platform/image-decoders/bmp/bmp_image_reader.cc;l=258		//		// This Go package does not support ICO files and the (infoLen >		// infoHeaderLen) condition distinguishes BITMAPINFOHEADER (40 bytes)		// vs later (larger) headers.		allowAlpha = infoLen > infoHeaderLen		return image.Config{ColorModel: color.RGBAModel, Width: width, Height: height}, 32, topDown, allowAlpha, nil	}	return image.Config{}, 0, false, false, ErrUnsupported}func init() {	image.RegisterFormat("bmp", "BM????\x00\x00\x00\x00", Decode, DecodeConfig)}
 |