avframes/main.go

132 lines
2.5 KiB
Go
Raw Normal View History

2021-01-25 08:03:24 +00:00
package main
import (
"bufio"
2021-01-25 18:35:18 +00:00
"errors"
2021-01-25 08:03:24 +00:00
"fmt"
"io"
"log"
"os"
)
2021-01-26 16:08:40 +00:00
const fname = "/home/rob/home-office/example.mp3"
type (
AudioVersionId int
LayerIndex int
header uint32
)
const (
AudioVersionMPEG25 AudioVersionId = iota
AudioVersionReserved
AudioVersionMPEG2
AudioVersionMPEG1
)
2021-01-25 08:03:24 +00:00
2021-01-25 18:35:18 +00:00
const (
LayerIndexReserved LayerIndex = iota
LayerIndexIII
LayerIndexII
LayerIndexI
2021-01-25 18:35:18 +00:00
)
2021-01-26 16:08:40 +00:00
var sampleRates = [5][16]uint32{
{0, 32, 64, 96, 128, 160, 192, 224, 256, 288, 320, 352, 384, 416, 448, 0},
{0, 32, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 256, 320, 384, 0},
{0, 32, 40, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 256, 320, 0},
{0, 32, 48, 56, 64, 80, 96, 112, 128, 144, 160, 176, 192, 224, 256, 0},
{0, 8, 16, 24, 32, 40, 48, 56, 64, 80, 96, 112, 128, 144, 160, 0},
}
func (h header) AudioVersionId() AudioVersionId {
2021-01-26 16:08:40 +00:00
return AudioVersionId(((h >> 8) & 0x18) >> 3)
}
func (h header) LayerIndex() LayerIndex {
2021-01-26 16:08:40 +00:00
return LayerIndex(((h >> 8) & 0x6) >> 1)
}
func (h header) IsPrivate() bool {
return (h>>8)&1 == 1
}
func (h header) SampleRate() uint32 {
var j int
switch h.AudioVersionId() {
case AudioVersionMPEG1:
switch h.LayerIndex() {
case LayerIndexIII:
j = 2
case LayerIndexII:
j = 1
case LayerIndexI:
j = 0
default:
return 0
}
case AudioVersionReserved:
return 0
case AudioVersionMPEG2, AudioVersionMPEG25:
switch h.LayerIndex() {
case LayerIndexIII, LayerIndexII:
j = 4
case LayerIndexI:
j = 3
default:
return 0
}
}
2021-01-26 16:08:40 +00:00
k := ((h >> 20) & 0xf)
return sampleRates[j][k]
}
2021-01-25 08:03:24 +00:00
func main() {
f, err := os.Open(fname)
if err != nil {
log.Fatal(err)
}
bufreader := bufio.NewReader(f)
buf := make([]byte, 4)
2021-01-25 08:03:24 +00:00
var i int
for {
_, err := io.ReadFull(bufreader, buf)
if err != nil {
2021-01-25 18:35:18 +00:00
if err == io.EOF {
break
}
2021-01-25 08:03:24 +00:00
log.Fatal(err)
}
header, err := newHeaderFromBytes(buf)
2021-01-25 18:35:18 +00:00
if err == nil {
2021-01-26 16:08:40 +00:00
fmt.Printf("Got header: %08X, audio version: %d, layer index: %d\n", header, header.AudioVersionId(), header.LayerIndex())
2021-01-25 18:35:18 +00:00
}
2021-01-25 08:03:24 +00:00
i++
2021-01-25 18:35:18 +00:00
bufreader.Discard(3)
2021-01-25 08:03:24 +00:00
}
}
func newHeaderFromBytes(buf []byte) (header, error) {
2021-01-25 18:35:18 +00:00
if len(buf) < 4 {
return 0, errors.New("insufficient bytes")
}
if buf[0] != 0xFF || buf[1]&0xE0 != 0xE0 {
return 0, errors.New("no header")
}
// always little-endian for now
n := uint32(buf[0]) | uint32(buf[1])<<8 | uint32(buf[2])<<16 | uint32(buf[3])<<24
return newHeader(n)
}
2021-01-25 18:35:18 +00:00
// TODO use unsafe here? https://codereview.stackexchange.com/a/60161
func newHeader(n uint32) (header, error) {
2021-01-25 18:35:18 +00:00
return header(n), nil
2021-01-25 08:03:24 +00:00
}