ebiten/audio/mp3/sideinfo.go
2017-06-18 18:34:58 +09:00

151 lines
4.4 KiB
Go

// Copyright 2017 The Ebiten Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// +build !js
package mp3
import (
"fmt"
"io"
)
func (src *source) readSideInfo(header *mpeg1FrameHeader) (*mpeg1SideInfo, error) {
nch := header.numberOfChannels()
// Calculate header audio data size
framesize := header.frameSize()
if framesize > 2000 {
return nil, fmt.Errorf("mp3: framesize = %d\n", framesize)
}
// Sideinfo is 17 bytes for one channel and 32 bytes for two
sideinfo_size := 32
if nch == 1 {
sideinfo_size = 17
}
// Main data size is the rest of the frame,including ancillary data
main_data_size := framesize - sideinfo_size - 4 // sync+header
// CRC is 2 bytes
if header.protection_bit == 0 {
main_data_size -= 2
}
// Read sideinfo from bitstream into buffer used by getSideBits()
s, err := src.getSideinfo(sideinfo_size)
if err != nil {
return nil, err
}
// Parse audio data
// Pointer to where we should start reading main data
si := &mpeg1SideInfo{}
si.main_data_begin = s.getSideBits(9)
// Get private bits. Not used for anything.
if header.mode == mpeg1ModeSingleChannel {
si.private_bits = s.getSideBits(5)
} else {
si.private_bits = s.getSideBits(3)
}
// Get scale factor selection information
for ch := 0; ch < nch; ch++ {
for scfsi_band := 0; scfsi_band < 4; scfsi_band++ {
si.scfsi[ch][scfsi_band] = s.getSideBits(1)
}
}
// Get the rest of the side information
for gr := 0; gr < 2; gr++ {
for ch := 0; ch < nch; ch++ {
si.part2_3_length[gr][ch] = s.getSideBits(12)
si.big_values[gr][ch] = s.getSideBits(9)
si.global_gain[gr][ch] = s.getSideBits(8)
si.scalefac_compress[gr][ch] = s.getSideBits(4)
si.win_switch_flag[gr][ch] = s.getSideBits(1)
if si.win_switch_flag[gr][ch] == 1 {
si.block_type[gr][ch] = s.getSideBits(2)
si.mixed_block_flag[gr][ch] = s.getSideBits(1)
for region := 0; region < 2; region++ {
si.table_select[gr][ch][region] = s.getSideBits(5)
}
for window := 0; window < 3; window++ {
si.subblock_gain[gr][ch][window] = s.getSideBits(3)
}
if (si.block_type[gr][ch] == 2) && (si.mixed_block_flag[gr][ch] == 0) {
si.region0_count[gr][ch] = 8 // Implicit
} else {
si.region0_count[gr][ch] = 7 // Implicit
}
// The standard is wrong on this!!!
// Implicit
si.region1_count[gr][ch] = 20 - si.region0_count[gr][ch]
} else {
for region := 0; region < 3; region++ {
si.table_select[gr][ch][region] = s.getSideBits(5)
}
si.region0_count[gr][ch] = s.getSideBits(4)
si.region1_count[gr][ch] = s.getSideBits(3)
si.block_type[gr][ch] = 0 // Implicit
}
si.preflag[gr][ch] = s.getSideBits(1)
si.scalefac_scale[gr][ch] = s.getSideBits(1)
si.count1table_select[gr][ch] = s.getSideBits(1)
}
}
return si, nil
}
// A sideInfoBytes is a bit reservoir for side info
type sideInfoBytes struct {
vec []int
idx int // Index into the current byte(0-7)
}
func (src *source) getSideinfo(size int) (*sideInfoBytes, error) {
buf := make([]int, size)
n := 0
var err error
for n < size && err == nil {
nn, err2 := src.getBytes(buf[n:])
n += nn
err = err2
}
if n < size {
if err == io.EOF {
return nil, fmt.Errorf("mp3: unexpected EOF at getSideinfo")
}
return nil, fmt.Errorf("mp3: couldn't read sideinfo %d bytes at pos %d: %v",
size, src.getFilepos(), err)
}
s := &sideInfoBytes{
vec: buf[:n],
}
return s, nil
}
func (s *sideInfoBytes) getSideBits(num int) int {
// Form a word of the next four bytes
// TODO: endianness?
b := make([]int, 4)
for i := range b {
if len(s.vec) > i {
b[i] = s.vec[i]
}
}
tmp := (uint32(b[0]) << 24) | (uint32(b[1]) << 16) | (uint32(b[2]) << 8) | (uint32(b[3]) << 0)
// Remove bits already used
tmp = tmp << uint(s.idx)
// Remove bits after the desired bits
tmp = tmp >> (32 - uint(num))
// Update pointers
s.vec = s.vec[(s.idx+int(num))>>3:]
s.idx = (s.idx + int(num)) & 0x07
return int(tmp)
}