continue development
This commit is contained in:
parent
85c66d74c2
commit
f8888ef614
66
main.go
66
main.go
|
@ -9,7 +9,7 @@ import (
|
||||||
"os"
|
"os"
|
||||||
)
|
)
|
||||||
|
|
||||||
const fname = "/home/rob/share/home-office/example.mp3"
|
const fname = "/home/rob/home-office/example.mp3"
|
||||||
|
|
||||||
type (
|
type (
|
||||||
AudioVersionId int
|
AudioVersionId int
|
||||||
|
@ -32,32 +32,55 @@ const (
|
||||||
LayerIndexI
|
LayerIndexI
|
||||||
)
|
)
|
||||||
|
|
||||||
|
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 {
|
func (h header) AudioVersionId() AudioVersionId {
|
||||||
b := h >> 8
|
return AudioVersionId(((h >> 8) & 0x18) >> 3)
|
||||||
if b&0x18 == 0x18 {
|
|
||||||
return AudioVersionMPEG1
|
|
||||||
}
|
|
||||||
if b&0x10 == 0x10 {
|
|
||||||
return AudioVersionMPEG2
|
|
||||||
}
|
|
||||||
if b&0x8 == 0x8 {
|
|
||||||
return AudioVersionReserved
|
|
||||||
}
|
|
||||||
return AudioVersionMPEG25
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h header) LayerIndex() LayerIndex {
|
func (h header) LayerIndex() LayerIndex {
|
||||||
b := h >> 8
|
return LayerIndex(((h >> 8) & 0x6) >> 1)
|
||||||
if b&0x6 == 0x6 {
|
|
||||||
return LayerIndexI
|
|
||||||
}
|
}
|
||||||
if b&0x4 == 0x4 {
|
|
||||||
return LayerIndexII
|
func (h header) IsPrivate() bool {
|
||||||
|
return (h>>8)&1 == 1
|
||||||
}
|
}
|
||||||
if b&0x2 == 0x2 {
|
|
||||||
return LayerIndexIII
|
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
|
||||||
}
|
}
|
||||||
return LayerIndexReserved
|
case AudioVersionReserved:
|
||||||
|
return 0
|
||||||
|
case AudioVersionMPEG2, AudioVersionMPEG25:
|
||||||
|
switch h.LayerIndex() {
|
||||||
|
case LayerIndexIII, LayerIndexII:
|
||||||
|
j = 4
|
||||||
|
case LayerIndexI:
|
||||||
|
j = 3
|
||||||
|
default:
|
||||||
|
return 0
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
k := ((h >> 20) & 0xf)
|
||||||
|
return sampleRates[j][k]
|
||||||
}
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
|
@ -80,9 +103,8 @@ func main() {
|
||||||
}
|
}
|
||||||
|
|
||||||
header, err := newHeaderFromBytes(buf)
|
header, err := newHeaderFromBytes(buf)
|
||||||
|
|
||||||
if err == nil {
|
if err == nil {
|
||||||
fmt.Printf("Got header: %08X, audio version: %d\n", header, header.AudioVersionId())
|
fmt.Printf("Got header: %08X, audio version: %d, layer index: %d\n", header, header.AudioVersionId(), header.LayerIndex())
|
||||||
}
|
}
|
||||||
i++
|
i++
|
||||||
|
|
||||||
|
|
79
main_test.go
79
main_test.go
|
@ -46,3 +46,82 @@ func TestLayerIndex(t *testing.T) {
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
assert.Equal(t, LayerIndexI, h.LayerIndex())
|
assert.Equal(t, LayerIndexI, h.LayerIndex())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestIsPrivate(t *testing.T) {
|
||||||
|
bytes := []byte{0xff, 0xe1, 0, 0}
|
||||||
|
h, err := newHeader(binary.LittleEndian.Uint32(bytes))
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.True(t, h.IsPrivate())
|
||||||
|
|
||||||
|
bytes = []byte{0xff, 0xe0, 0, 0}
|
||||||
|
h, err = newHeader(binary.LittleEndian.Uint32(bytes))
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.False(t, h.IsPrivate())
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSampleRate(t *testing.T) {
|
||||||
|
testCases := []struct {
|
||||||
|
name string
|
||||||
|
bytes []byte
|
||||||
|
expSampleRate uint32
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
"empty header",
|
||||||
|
[]byte{0, 0, 0, 0},
|
||||||
|
0,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"MPEG1 LayerI 32kbps",
|
||||||
|
[]byte{0xff, 0xfe, 0x10, 0},
|
||||||
|
32,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"MPEG1 LayerI 448kbps",
|
||||||
|
[]byte{0xff, 0xfe, 0xe0, 0},
|
||||||
|
448,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"MPEG1 LayerII 32kbps",
|
||||||
|
[]byte{0xff, 0xfc, 0x10, 0},
|
||||||
|
32,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"MPEG1 LayerII 448kbps",
|
||||||
|
[]byte{0xff, 0xfc, 0xe0, 0},
|
||||||
|
384,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"MPEG1 LayerIII 32kbps",
|
||||||
|
[]byte{0xff, 0xfa, 0x10, 0},
|
||||||
|
32,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"MPEG1 LayerIII 128kbps",
|
||||||
|
[]byte{0xff, 0xfa, 0x90, 0},
|
||||||
|
128,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"MPEG1 LayerIII 160kbps",
|
||||||
|
[]byte{0xff, 0xfa, 0xa0, 0},
|
||||||
|
160,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"MPEG1 LayerIII 320kbps",
|
||||||
|
[]byte{0xff, 0xfa, 0xe0, 0x0},
|
||||||
|
320,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"MPEG1 LayerIII reserved",
|
||||||
|
[]byte{0xff, 0xfa, 0xf0, 0x0},
|
||||||
|
0,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range testCases {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
h, err := newHeader(binary.LittleEndian.Uint32(tc.bytes))
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, tc.expSampleRate, h.SampleRate())
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue