format.mx raw

   1  // Copyright 2010 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 image
   6  
   7  import (
   8  	"bufio"
   9  	"errors"
  10  	"io"
  11  	"sync"
  12  	"sync/atomic"
  13  )
  14  
  15  // ErrFormat indicates that decoding encountered an unknown format.
  16  var ErrFormat = errors.New("image: unknown format")
  17  
  18  // A format holds an image format's name, magic header and how to decode it.
  19  type format struct {
  20  	name, magic  []byte
  21  	decode       func(io.Reader) (Image, error)
  22  	decodeConfig func(io.Reader) (Config, error)
  23  }
  24  
  25  // Formats is the list of registered formats.
  26  var (
  27  	formatsMu     sync.Mutex
  28  	atomicFormats atomic.Value
  29  )
  30  
  31  // RegisterFormat registers an image format for use by [Decode].
  32  // Name is the name of the format, like "jpeg" or "png".
  33  // Magic is the magic prefix that identifies the format's encoding. The magic
  34  // string can contain "?" wildcards that each match any one byte.
  35  // [Decode] is the function that decodes the encoded image.
  36  // [DecodeConfig] is the function that decodes just its configuration.
  37  func RegisterFormat(name, magic []byte, decode func(io.Reader) (Image, error), decodeConfig func(io.Reader) (Config, error)) {
  38  	formatsMu.Lock()
  39  	formats, _ := atomicFormats.Load().([]format)
  40  	atomicFormats.Store(append(formats, format{name, magic, decode, decodeConfig}))
  41  	formatsMu.Unlock()
  42  }
  43  
  44  // A reader is an io.Reader that can also peek ahead.
  45  type reader interface {
  46  	io.Reader
  47  	Peek(int) ([]byte, error)
  48  }
  49  
  50  // asReader converts an io.Reader to a reader.
  51  func asReader(r io.Reader) reader {
  52  	if rr, ok := r.(reader); ok {
  53  		return rr
  54  	}
  55  	return bufio.NewReader(r)
  56  }
  57  
  58  // match reports whether magic matches b. Magic may contain "?" wildcards.
  59  func match(magic []byte, b []byte) bool {
  60  	if len(magic) != len(b) {
  61  		return false
  62  	}
  63  	for i, c := range b {
  64  		if magic[i] != c && magic[i] != '?' {
  65  			return false
  66  		}
  67  	}
  68  	return true
  69  }
  70  
  71  // sniff determines the format of r's data.
  72  func sniff(r reader) format {
  73  	formats, _ := atomicFormats.Load().([]format)
  74  	for _, f := range formats {
  75  		b, err := r.Peek(len(f.magic))
  76  		if err == nil && match(f.magic, b) {
  77  			return f
  78  		}
  79  	}
  80  	return format{}
  81  }
  82  
  83  // Decode decodes an image that has been encoded in a registered format.
  84  // The string returned is the format name used during format registration.
  85  // Format registration is typically done by an init function in the codec-
  86  // specific package.
  87  func Decode(r io.Reader) (Image, []byte, error) {
  88  	rr := asReader(r)
  89  	f := sniff(rr)
  90  	if f.decode == nil {
  91  		return nil, "", ErrFormat
  92  	}
  93  	m, err := f.decode(rr)
  94  	return m, f.name, err
  95  }
  96  
  97  // DecodeConfig decodes the color model and dimensions of an image that has
  98  // been encoded in a registered format. The string returned is the format name
  99  // used during format registration. Format registration is typically done by
 100  // an init function in the codec-specific package.
 101  func DecodeConfig(r io.Reader) (Config, []byte, error) {
 102  	rr := asReader(r)
 103  	f := sniff(rr)
 104  	if f.decodeConfig == nil {
 105  		return Config{}, "", ErrFormat
 106  	}
 107  	c, err := f.decodeConfig(rr)
 108  	return c, f.name, err
 109  }
 110