2017-06-12 19:49:15 +02:00
|
|
|
// 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"
|
2017-06-16 17:34:54 +02:00
|
|
|
"io"
|
2017-06-12 19:49:15 +02:00
|
|
|
)
|
|
|
|
|
2017-06-17 11:20:43 +02:00
|
|
|
var g_mpeg1_bitrates = map[mpeg1Layer][15]int{
|
|
|
|
mpeg1Layer1: {
|
2017-06-13 19:22:11 +02:00
|
|
|
0, 32000, 64000, 96000, 128000, 160000, 192000, 224000,
|
|
|
|
256000, 288000, 320000, 352000, 384000, 416000, 448000,
|
|
|
|
},
|
2017-06-17 11:20:43 +02:00
|
|
|
mpeg1Layer2: {
|
2017-06-13 19:22:11 +02:00
|
|
|
0, 32000, 48000, 56000, 64000, 80000, 96000, 112000,
|
|
|
|
128000, 160000, 192000, 224000, 256000, 320000, 384000,
|
|
|
|
},
|
2017-06-17 11:20:43 +02:00
|
|
|
mpeg1Layer3: {
|
2017-06-13 19:22:11 +02:00
|
|
|
0, 32000, 40000, 48000, 56000, 64000, 80000, 96000,
|
|
|
|
112000, 128000, 160000, 192000, 224000, 256000, 320000,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
var g_sampling_frequency = [3]int{44100, 48000, 32000}
|
|
|
|
|
2017-06-17 11:42:43 +02:00
|
|
|
func (f *frame) readAudioL3() error {
|
2017-06-17 16:33:29 +02:00
|
|
|
nch := f.numberOfChannels()
|
2017-06-13 19:22:11 +02:00
|
|
|
/* Calculate header audio data size */
|
|
|
|
framesize := (144*
|
2017-06-17 11:42:43 +02:00
|
|
|
g_mpeg1_bitrates[f.header.layer][f.header.bitrate_index])/
|
|
|
|
g_sampling_frequency[f.header.sampling_frequency] +
|
|
|
|
int(f.header.padding_bit)
|
2017-06-13 19:22:11 +02:00
|
|
|
if framesize > 2000 {
|
2017-06-16 20:01:39 +02:00
|
|
|
return fmt.Errorf("mp3: framesize = %d\n", framesize)
|
2017-06-13 19:22:11 +02:00
|
|
|
}
|
|
|
|
/* 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 */
|
2017-06-17 11:42:43 +02:00
|
|
|
if f.header.protection_bit == 0 {
|
2017-06-13 19:22:11 +02:00
|
|
|
main_data_size -= 2
|
|
|
|
}
|
|
|
|
/* Read sideinfo from bitstream into buffer used by getSideBits() */
|
2017-06-16 20:01:39 +02:00
|
|
|
if err := getSideinfo(sideinfo_size); err != nil {
|
|
|
|
return err
|
2017-06-13 19:22:11 +02:00
|
|
|
}
|
|
|
|
/* Parse audio data */
|
|
|
|
/* Pointer to where we should start reading main data */
|
2017-06-17 11:42:43 +02:00
|
|
|
f.sideInfo.main_data_begin = getSideBits(9)
|
2017-06-13 19:22:11 +02:00
|
|
|
/* Get private bits. Not used for anything. */
|
2017-06-17 11:42:43 +02:00
|
|
|
if f.header.mode == mpeg1ModeSingleChannel {
|
|
|
|
f.sideInfo.private_bits = getSideBits(5)
|
2017-06-13 19:22:11 +02:00
|
|
|
} else {
|
2017-06-17 11:42:43 +02:00
|
|
|
f.sideInfo.private_bits = getSideBits(3)
|
2017-06-13 19:22:11 +02:00
|
|
|
}
|
|
|
|
/* Get scale factor selection information */
|
|
|
|
for ch := 0; ch < nch; ch++ {
|
|
|
|
for scfsi_band := 0; scfsi_band < 4; scfsi_band++ {
|
2017-06-17 11:42:43 +02:00
|
|
|
f.sideInfo.scfsi[ch][scfsi_band] = getSideBits(1)
|
2017-06-13 19:22:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Get the rest of the side information */
|
|
|
|
for gr := 0; gr < 2; gr++ {
|
|
|
|
for ch := 0; ch < nch; ch++ {
|
2017-06-17 11:42:43 +02:00
|
|
|
f.sideInfo.part2_3_length[gr][ch] = getSideBits(12)
|
|
|
|
f.sideInfo.big_values[gr][ch] = getSideBits(9)
|
|
|
|
f.sideInfo.global_gain[gr][ch] = getSideBits(8)
|
|
|
|
f.sideInfo.scalefac_compress[gr][ch] = getSideBits(4)
|
|
|
|
f.sideInfo.win_switch_flag[gr][ch] = getSideBits(1)
|
|
|
|
if f.sideInfo.win_switch_flag[gr][ch] == 1 {
|
|
|
|
f.sideInfo.block_type[gr][ch] = getSideBits(2)
|
|
|
|
f.sideInfo.mixed_block_flag[gr][ch] = getSideBits(1)
|
2017-06-13 19:22:11 +02:00
|
|
|
for region := 0; region < 2; region++ {
|
2017-06-17 11:42:43 +02:00
|
|
|
f.sideInfo.table_select[gr][ch][region] = getSideBits(5)
|
2017-06-13 19:22:11 +02:00
|
|
|
}
|
|
|
|
for window := 0; window < 3; window++ {
|
2017-06-17 11:42:43 +02:00
|
|
|
f.sideInfo.subblock_gain[gr][ch][window] = getSideBits(3)
|
2017-06-13 19:22:11 +02:00
|
|
|
}
|
2017-06-17 11:42:43 +02:00
|
|
|
if (f.sideInfo.block_type[gr][ch] == 2) && (f.sideInfo.mixed_block_flag[gr][ch] == 0) {
|
|
|
|
f.sideInfo.region0_count[gr][ch] = 8 /* Implicit */
|
2017-06-13 19:22:11 +02:00
|
|
|
} else {
|
2017-06-17 11:42:43 +02:00
|
|
|
f.sideInfo.region0_count[gr][ch] = 7 /* Implicit */
|
2017-06-13 19:22:11 +02:00
|
|
|
}
|
|
|
|
/* The standard is wrong on this!!! */ /* Implicit */
|
2017-06-17 11:42:43 +02:00
|
|
|
f.sideInfo.region1_count[gr][ch] = 20 - f.sideInfo.region0_count[gr][ch]
|
2017-06-13 19:22:11 +02:00
|
|
|
} else {
|
|
|
|
for region := 0; region < 3; region++ {
|
2017-06-17 11:42:43 +02:00
|
|
|
f.sideInfo.table_select[gr][ch][region] = getSideBits(5)
|
2017-06-13 19:22:11 +02:00
|
|
|
}
|
2017-06-17 11:42:43 +02:00
|
|
|
f.sideInfo.region0_count[gr][ch] = getSideBits(4)
|
|
|
|
f.sideInfo.region1_count[gr][ch] = getSideBits(3)
|
|
|
|
f.sideInfo.block_type[gr][ch] = 0 /* Implicit */
|
2017-06-13 19:22:11 +02:00
|
|
|
}
|
2017-06-17 11:42:43 +02:00
|
|
|
f.sideInfo.preflag[gr][ch] = getSideBits(1)
|
|
|
|
f.sideInfo.scalefac_scale[gr][ch] = getSideBits(1)
|
|
|
|
f.sideInfo.count1table_select[gr][ch] = getSideBits(1)
|
2017-06-13 19:22:11 +02:00
|
|
|
}
|
|
|
|
}
|
2017-06-16 20:01:39 +02:00
|
|
|
return nil
|
2017-06-13 19:22:11 +02:00
|
|
|
}
|
|
|
|
|
2017-06-12 19:49:15 +02:00
|
|
|
// A sideInfo is a bit reservoir for side info
|
|
|
|
type sideInfo struct {
|
|
|
|
vec []int
|
|
|
|
idx int // Index into the current byte(0-7)
|
|
|
|
}
|
|
|
|
|
|
|
|
var theSideInfo sideInfo
|
|
|
|
|
2017-06-16 20:01:39 +02:00
|
|
|
func getSideinfo(size int) error {
|
2017-06-16 17:34:54 +02:00
|
|
|
buf := make([]int, size)
|
2017-06-16 20:01:39 +02:00
|
|
|
n := 0
|
|
|
|
var err error
|
|
|
|
for n < size && err == nil {
|
|
|
|
nn, err2 := getBytes(buf[n:])
|
|
|
|
n += nn
|
|
|
|
err = err2
|
|
|
|
}
|
|
|
|
if n < size {
|
|
|
|
if err == io.EOF {
|
|
|
|
return fmt.Errorf("mp3: unexpected EOF at getSideinfo")
|
|
|
|
}
|
|
|
|
return fmt.Errorf("mp3: couldn't read sideinfo %d bytes at pos %d: %v",
|
2017-06-16 23:48:27 +02:00
|
|
|
size, getFilepos(), err)
|
2017-06-12 19:49:15 +02:00
|
|
|
}
|
2017-06-16 17:34:54 +02:00
|
|
|
theSideInfo.vec = buf[:n]
|
2017-06-12 19:49:15 +02:00
|
|
|
theSideInfo.idx = 0
|
2017-06-16 20:01:39 +02:00
|
|
|
return nil
|
2017-06-12 19:49:15 +02:00
|
|
|
}
|
|
|
|
|
2017-06-13 19:22:11 +02:00
|
|
|
func getSideBits(num int) int {
|
2017-06-12 19:49:15 +02:00
|
|
|
// Form a word of the next four bytes
|
|
|
|
// TODO: endianness?
|
|
|
|
b := make([]int, 4)
|
|
|
|
for i := range b {
|
|
|
|
if len(theSideInfo.vec) > i {
|
|
|
|
b[i] = theSideInfo.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(theSideInfo.idx)
|
|
|
|
// Remove bits after the desired bits
|
2017-06-13 19:22:11 +02:00
|
|
|
tmp = tmp >> (32 - uint(num))
|
2017-06-12 19:49:15 +02:00
|
|
|
// Update pointers
|
|
|
|
theSideInfo.vec = theSideInfo.vec[(theSideInfo.idx+int(num))>>3:]
|
|
|
|
theSideInfo.idx = (theSideInfo.idx + int(num)) & 0x07
|
2017-06-13 19:22:11 +02:00
|
|
|
return int(tmp)
|
2017-06-12 19:49:15 +02:00
|
|
|
}
|