ebiten/audio/mp3/decode_notjs.go

114 lines
2.3 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
import (
2017-06-17 12:33:49 +02:00
"errors"
2017-06-11 11:12:12 +02:00
"io"
)
func (f *frame) decodeL3() []uint8 {
out := make([]uint8, 576*4*2)
nch := f.header.numberOfChannels()
2017-06-16 23:28:25 +02:00
for gr := 0; gr < 2; gr++ {
for ch := 0; ch < nch; ch++ {
2017-06-17 11:42:43 +02:00
f.l3Requantize(gr, ch)
2017-06-16 23:43:50 +02:00
// Reorder short blocks
2017-06-17 11:42:43 +02:00
f.l3Reorder(gr, ch)
2017-06-16 23:28:25 +02:00
}
2017-06-17 11:42:43 +02:00
f.l3Stereo(gr)
2017-06-16 23:28:25 +02:00
for ch := 0; ch < nch; ch++ {
2017-06-17 11:42:43 +02:00
f.l3Antialias(gr, ch)
2017-06-16 23:28:25 +02:00
// (IMDCT,windowing,overlapp add)
2017-06-17 11:42:43 +02:00
f.l3HybridSynthesis(gr, ch)
f.l3FrequencyInversion(gr, ch)
2017-06-16 23:28:25 +02:00
// Polyphase subband synthesis
f.l3SubbandSynthesis(gr, ch, out[576*4*gr:])
2017-06-16 23:28:25 +02:00
}
}
return out
2017-06-16 23:28:25 +02:00
}
type source struct {
reader io.Reader
readerCache []uint8
readerPos int
readerEOF bool
}
func (s *source) getByte() (uint8, error) {
for len(s.readerCache) == 0 && !s.readerEOF {
2017-06-11 11:12:12 +02:00
buf := make([]uint8, 4096)
n, err := s.reader.Read(buf)
s.readerCache = append(s.readerCache, buf[:n]...)
2017-06-11 11:12:12 +02:00
if err != nil {
if err == io.EOF {
s.readerEOF = true
2017-06-11 11:12:12 +02:00
} else {
2017-06-16 18:36:58 +02:00
return 0, err
2017-06-11 11:12:12 +02:00
}
}
}
if len(s.readerCache) == 0 {
2017-06-16 18:36:58 +02:00
return 0, io.EOF
2017-06-11 11:12:12 +02:00
}
b := s.readerCache[0]
s.readerCache = s.readerCache[1:]
s.readerPos++
2017-06-16 18:36:58 +02:00
return b, nil
2017-06-11 11:12:12 +02:00
}
func (s *source) getBytes(buf []int) (int, error) {
2017-06-16 17:34:54 +02:00
for i := range buf {
v, err := s.getByte()
2017-06-16 18:36:58 +02:00
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
}
func (s *source) getFilepos() int {
return s.readerPos
2017-06-11 11:12:12 +02:00
}
2017-06-17 12:33:49 +02:00
var eof = errors.New("mp3: expected EOF")
2017-06-11 11:12:12 +02:00
func decode(r io.Reader, w io.Writer) error {
s := &source{
reader: r,
}
var f *frame
2017-06-17 12:33:49 +02:00
for {
var err error
f, err = s.readNextFrame(f)
2017-06-16 19:43:13 +02:00
if err == nil {
out := f.decodeL3()
if _, err := w.Write(out); err != nil {
2017-06-16 23:28:25 +02:00
return err
}
2017-06-11 11:12:12 +02:00
continue
}
2017-06-17 12:33:49 +02:00
if err == eof {
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
}