1
1
mirror of https://github.com/wader/fq.git synced 2024-12-23 13:22:58 +03:00
fq/format/mp3/mp3.go

140 lines
3.5 KiB
Go
Raw Normal View History

2020-06-08 03:29:51 +03:00
package mp3
// TODO: vbri
import (
"fmt"
"github.com/wader/fq/format"
"github.com/wader/fq/pkg/decode"
"github.com/wader/fq/pkg/interp"
2020-06-08 03:29:51 +03:00
)
var headerFormat decode.Group
var footerFormat decode.Group
var mp3Frame decode.Group
2020-06-08 03:29:51 +03:00
func init() {
interp.RegisterFormat(decode.Format{
2020-06-08 03:29:51 +03:00
Name: format.MP3,
ProbeOrder: format.ProbeOrderBinFuzzy, // after most others (silent samples and jpeg header can look like mp3 sync)
2020-06-08 03:29:51 +03:00
Description: "MP3 file",
Groups: []string{format.PROBE},
DecodeFn: mp3Decode,
DefaultInArg: format.Mp3In{
MaxUniqueHeaderConfigs: 5,
MaxUnknown: 50,
MaxSyncSeek: 4 * 1024 * 8,
},
2020-06-08 03:29:51 +03:00
Dependencies: []decode.Dependency{
{Names: []string{format.ID3V2}, Group: &headerFormat},
2021-11-19 18:44:06 +03:00
{
Names: []string{
format.ID3V1,
format.ID3V11,
format.APEV2,
},
Group: &footerFormat,
},
{Names: []string{format.MP3_FRAME}, Group: &mp3Frame},
2020-06-08 03:29:51 +03:00
},
})
}
func mp3Decode(d *decode.D) any {
var mi format.Mp3In
d.ArgAs(&mi)
// things in a mp3 stream usually have few unique combinations of.
// does not include bitrate on purpose
type headerConfig struct {
MPEGVersion int
ProtectionAbsent bool
SampleRate int
ChannelsIndex int
ChannelModeIndex int
}
uniqueHeaderConfigs := map[headerConfig]struct{}{}
knownSize := int64(0)
2020-06-08 03:29:51 +03:00
// there are mp3s files in the wild with multiple headers, two id3v2 tags etc
d.FieldArray("headers", func(d *decode.D) {
2020-06-08 03:29:51 +03:00
for d.NotEnd() {
headerStart := d.Pos()
if dv, _, _ := d.TryFieldFormat("header", headerFormat, nil); dv == nil {
2020-06-08 03:29:51 +03:00
return
}
knownSize += d.Pos() - headerStart
2020-06-08 03:29:51 +03:00
}
})
2021-09-14 13:54:57 +03:00
lastValidEnd := int64(0)
2020-06-08 03:29:51 +03:00
validFrames := 0
2021-09-14 13:54:57 +03:00
decodeFailures := 0
d.FieldArray("frames", func(d *decode.D) {
2020-06-08 03:29:51 +03:00
for d.NotEnd() {
syncLen, _, err := d.TryPeekFind(16, 8, int64(mi.MaxSyncSeek), func(v uint64) bool {
return (v&0b1111_1111_1110_0000 == 0b1111_1111_1110_0000 && // sync header
v&0b0000_0000_0001_1000 != 0b0000_0000_0000_1000 && // not reserved mpeg version
v&0b0000_0000_0000_0110 == 0b0000_0000_0000_0010) // layer 3
})
if err != nil || syncLen < 0 {
2020-06-08 03:29:51 +03:00
break
}
if syncLen > 0 {
d.SeekRel(syncLen)
}
frameStart := d.Pos()
dv, v, _ := d.TryFieldFormat("frame", mp3Frame, nil)
if dv == nil {
2021-09-14 13:54:57 +03:00
decodeFailures++
d.SeekRel(8)
continue
2020-06-08 03:29:51 +03:00
}
mfo, ok := v.(format.MP3FrameOut)
if !ok {
panic(fmt.Sprintf("expected MP3FrameOut got %#+v", v))
}
knownSize += d.Pos() - frameStart
uniqueHeaderConfigs[headerConfig{
MPEGVersion: mfo.MPEGVersion,
ProtectionAbsent: mfo.ProtectionAbsent,
SampleRate: mfo.SampleRate,
ChannelsIndex: mfo.ChannelsIndex,
ChannelModeIndex: mfo.ChannelModeIndex,
}] = struct{}{}
2021-09-14 13:54:57 +03:00
lastValidEnd = d.Pos()
2020-06-08 03:29:51 +03:00
validFrames++
if len(uniqueHeaderConfigs) >= mi.MaxUniqueHeaderConfigs {
d.Errorf("too many unique header configurations")
}
2020-06-08 03:29:51 +03:00
}
})
2021-09-14 13:54:57 +03:00
if validFrames == 0 || (validFrames < 2 && decodeFailures > 0) {
d.Errorf("no frames found")
2020-06-08 03:29:51 +03:00
}
2021-09-14 13:54:57 +03:00
d.SeekAbs(lastValidEnd)
d.FieldArray("footers", func(d *decode.D) {
2020-06-08 03:29:51 +03:00
for d.NotEnd() {
footerStart := d.Pos()
if dv, _, _ := d.TryFieldFormat("footer", footerFormat, nil); dv == nil {
2020-06-08 03:29:51 +03:00
return
}
knownSize += d.Pos() - footerStart
2020-06-08 03:29:51 +03:00
}
})
unknownPercent := int(float64((d.Len() - knownSize)) / float64(d.Len()) * 100.0)
if unknownPercent > mi.MaxUnknown {
d.Errorf(fmt.Sprintf("exceeds max precent unknown bits, %d > %d", unknownPercent, mi.MaxUnknown))
}
2020-06-08 03:29:51 +03:00
return nil
}