ebiten/audio/mp3/decode_notjs.go

151 lines
2.9 KiB
Go
Raw Normal View History

2017-06-11 11:12:12 +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
// #include "pdmp3.h"
2017-06-16 23:28:25 +02:00
//
2017-06-16 23:59:27 +02:00
// extern t_mpeg1_header g_frame_header;
2017-06-11 11:12:12 +02:00
import "C"
import (
"io"
)
var (
reader io.Reader
readerCache []uint8
readerPos int
readerEOF bool
writer io.Writer
)
2017-06-16 23:28:25 +02:00
func decodeL3() error {
out := make([]uint32, 576)
2017-06-16 23:43:50 +02:00
// Number of channels(1 for mono and 2 for stereo)
2017-06-16 23:28:25 +02:00
nch := 2
if C.g_frame_header.mode == C.mpeg1_mode_single_channel {
nch = 1
}
for gr := 0; gr < 2; gr++ {
for ch := 0; ch < nch; ch++ {
2017-06-16 23:43:50 +02:00
l3Requantize(gr, ch)
// Reorder short blocks
l3Reorder(gr, ch)
2017-06-16 23:28:25 +02:00
}
2017-06-16 23:43:50 +02:00
l3Stereo(gr)
2017-06-16 23:28:25 +02:00
for ch := 0; ch < nch; ch++ {
2017-06-16 23:43:50 +02:00
l3Antialias(gr, ch)
2017-06-16 23:28:25 +02:00
// (IMDCT,windowing,overlapp add)
2017-06-16 23:43:50 +02:00
l3HybridSynthesis(gr, ch)
l3FrequencyInversion(gr, ch)
2017-06-16 23:28:25 +02:00
// Polyphase subband synthesis
l3SubbandSynthesis(gr, ch, out)
}
if err := audioWriteRaw(out); err != nil {
return err
}
}
return nil
}
func audioWriteRaw(samples []uint32) error {
2017-06-16 23:28:25 +02:00
nch := 2
if C.g_frame_header.mode == C.mpeg1_mode_single_channel {
nch = 1
}
s := make([]uint8, len(samples)*2*nch)
for i, v := range samples {
if nch == 1 {
s[2*i] = uint8(v)
s[2*i+1] = uint8(v >> 8)
} else {
s[4*i] = uint8(v)
s[4*i+1] = uint8(v >> 8)
s[4*i+2] = uint8(v >> 16)
s[4*i+3] = uint8(v >> 24)
}
}
if _, err := writer.Write(s); err != nil {
return err
}
return nil
}
2017-06-16 18:36:58 +02:00
func getByte() (uint8, error) {
2017-06-11 11:12:12 +02:00
for len(readerCache) == 0 && !readerEOF {
buf := make([]uint8, 4096)
n, err := reader.Read(buf)
readerCache = append(readerCache, buf[:n]...)
2017-06-11 11:12:12 +02:00
if err != nil {
if err == io.EOF {
readerEOF = true
} else {
2017-06-16 18:36:58 +02:00
return 0, err
2017-06-11 11:12:12 +02:00
}
}
}
2017-06-16 17:34:54 +02:00
if len(readerCache) == 0 {
2017-06-16 18:36:58 +02:00
return 0, io.EOF
2017-06-11 11:12:12 +02:00
}
b := readerCache[0]
readerCache = readerCache[1:]
readerPos++
2017-06-16 18:36:58 +02:00
return b, nil
2017-06-11 11:12:12 +02:00
}
2017-06-16 17:34:54 +02:00
func getBytes(buf []int) (int, error) {
for i := range buf {
2017-06-16 18:36:58 +02:00
v, err := getByte()
buf[i] = int(v)
if err == io.EOF {
2017-06-16 17:34:54 +02:00
return i, io.EOF
2017-06-12 19:49:15 +02:00
}
}
2017-06-16 17:34:54 +02:00
return len(buf), nil
2017-06-12 19:49:15 +02:00
}
2017-06-16 23:59:27 +02:00
func isEOF() bool {
2017-06-11 11:12:12 +02:00
if len(readerCache) == 0 && readerEOF {
2017-06-16 23:59:27 +02:00
return true
2017-06-11 11:12:12 +02:00
}
2017-06-16 23:59:27 +02:00
return false
}
func getFilepos() int {
2017-06-16 23:48:27 +02:00
return readerPos
2017-06-11 11:12:12 +02:00
}
func decode(r io.Reader, w io.Writer) error {
2017-06-16 23:28:25 +02:00
// TODO: Decoder should know number of channels
2017-06-11 11:12:12 +02:00
reader = r
writer = w
2017-06-16 23:59:27 +02:00
for !isEOF() {
2017-06-16 19:43:13 +02:00
err := readFrame()
if err == nil {
2017-06-16 23:28:25 +02:00
if err := decodeL3(); err != nil {
return err
}
2017-06-11 11:12:12 +02:00
continue
}
2017-06-16 23:59:27 +02:00
if isEOF() {
2017-06-11 11:12:12 +02:00
break
}
2017-06-16 20:01:39 +02:00
return err
2017-06-11 11:12:12 +02:00
}
2017-06-16 20:01:39 +02:00
return nil
2017-06-11 11:12:12 +02:00
}