1
1
mirror of https://github.com/wader/fq.git synced 2024-11-29 12:14:17 +03:00
fq/format/riff/webp.go

114 lines
3.1 KiB
Go

package riff
// https://developers.google.com/speed/webp/docs/riff_container
import (
"github.com/wader/fq/format"
"github.com/wader/fq/pkg/decode"
"github.com/wader/fq/pkg/interp"
"github.com/wader/fq/pkg/scalar"
)
var exifGroup decode.Group
var iccpGroup decode.Group
var vp8FrameGroup decode.Group
var xmlGroup decode.Group
func init() {
interp.RegisterFormat(
format.WebP,
&decode.Format{
Description: "WebP image",
Groups: []*decode.Group{format.Probe, format.Image},
DecodeFn: webpDecode,
Dependencies: []decode.Dependency{
{Groups: []*decode.Group{format.Exif}, Out: &exifGroup},
{Groups: []*decode.Group{format.VP8_Frame}, Out: &vp8FrameGroup},
{Groups: []*decode.Group{format.ICC_Profile}, Out: &iccpGroup},
{Groups: []*decode.Group{format.XML}, Out: &xmlGroup},
},
})
}
const webpRiffType = "WEBP"
func webpDecode(d *decode.D) any {
d.Endian = decode.LittleEndian
var riffType string
riffDecode(
d,
nil,
func(d *decode.D, path path) (string, int64) {
id := d.FieldUTF8("id", 4, scalar.ActualTrimSpace, chunkIDDescriptions)
size := d.FieldU32("size")
return id, int64(size)
},
func(d *decode.D, id string, path path) (bool, any) {
switch id {
case "RIFF":
riffType = d.FieldUTF8("format", 4, d.StrAssert(webpRiffType))
return true, nil
case "VP8":
d.Format(&vp8FrameGroup, nil)
return false, nil
case "VP8L":
d.FieldU8("signature", d.UintAssert(0x2f), scalar.UintHex)
n := d.FieldU32("width_height_flags")
// TODO: replace with "bit endian" decoding
b0 := (n >> 24) & 0xff
b1 := (n >> 16) & 0xff
b2 := (n >> 8) & 0xff
b3 := (n >> 0) & 0xf
width := b3 | (b2&0b0011_111)<<8
width += 1
height := (b2&0b1100_0000)>>6 | b1<<8 | (b0&0b0000_1111)<<16
height += 1
alphaIsUsed := b3&0b0001_0000 != 0
versionNumber := (b3 & 0b1110_0000) >> 5
d.FieldValueUint("width", width)
d.FieldValueUint("height", height)
d.FieldValueBool("alpha_is_used", alphaIsUsed)
d.FieldValueUint("version_number", versionNumber)
d.FieldRawLen("data", d.BitsLeft())
return false, nil
case "VP8X":
d.FieldU2("reserved0")
d.FieldBool("icc_profile")
d.FieldBool("alpha")
d.FieldBool("exif_metadata")
d.FieldBool("xml_metadata")
d.FieldBool("animation")
d.FieldBool("reserved1")
d.FieldU24("reserved2")
d.FieldU24("width", scalar.UintActualAdd(1))
d.FieldU24("height", scalar.UintActualAdd(1))
return false, nil
case "EXIF":
// TODO: where is this documented? both exif in jpeg and webp has this prefix sometimes
var exifPrefix = []byte("Exif\x00\x00")
hasExifPrefix := d.TryHasBytes(exifPrefix)
if hasExifPrefix {
d.FieldUTF8("exif_prefix", len(exifPrefix))
}
d.Format(&exifGroup, nil)
return false, nil
case "ICCP":
d.Format(&iccpGroup, nil)
return false, nil
case "XMP":
d.FieldFormatOrRawLen("data", d.BitsLeft(), &xmlGroup, nil)
return false, nil
default:
d.FieldRawLen("data", d.BitsLeft())
return false, nil
}
})
if riffType != webpRiffType {
d.Errorf("wrong or no WEBP riff type found (%s)", riffType)
}
return nil
}