mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-27 11:12:44 +01:00
audio/mp3: Bug fix: Decode error on Safari (#438)
This commit is contained in:
parent
1f051b50db
commit
2c6f6e605e
@ -34,6 +34,8 @@ type Stream struct {
|
|||||||
posInBytes int
|
posInBytes int
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var errTryAgain = errors.New("audio/mp3: try again")
|
||||||
|
|
||||||
func (s *Stream) Read(b []byte) (int, error) {
|
func (s *Stream) Read(b []byte) (int, error) {
|
||||||
l := len(s.leftData)*4 - s.posInBytes
|
l := len(s.leftData)*4 - s.posInBytes
|
||||||
if l > len(b) {
|
if l > len(b) {
|
||||||
@ -90,6 +92,30 @@ func (s *Stream) Size() int64 {
|
|||||||
return int64(len(s.leftData) * 4)
|
return int64(len(s.leftData) * 4)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// seekNextFrame seeks the next frame and returns the new buffer with the new position.
|
||||||
|
// seekNextFrame also returns true when seeking is successful, or false otherwise.
|
||||||
|
//
|
||||||
|
// Seeking is necessary when decoding fails. Safari's MP3 decoder can't treat IDs well (#438).
|
||||||
|
func seekNextFrame(buf []uint8) ([]uint8, bool) {
|
||||||
|
// TODO: Need to skip tags explicitly? (hajimehoshi/go-mp3#9)
|
||||||
|
|
||||||
|
if len(buf) < 1 {
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
buf = buf[1:]
|
||||||
|
|
||||||
|
for {
|
||||||
|
if buf[0] == 0xff && buf[1]&0xfe == 0xfe {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
buf = buf[1:]
|
||||||
|
if len(buf) < 2 {
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return buf, true
|
||||||
|
}
|
||||||
|
|
||||||
func Decode(context *audio.Context, src audio.ReadSeekCloser) (*Stream, error) {
|
func Decode(context *audio.Context, src audio.ReadSeekCloser) (*Stream, error) {
|
||||||
b, err := ioutil.ReadAll(src)
|
b, err := ioutil.ReadAll(src)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -98,20 +124,53 @@ func Decode(context *audio.Context, src audio.ReadSeekCloser) (*Stream, error) {
|
|||||||
if err := src.Close(); err != nil {
|
if err := src.Close(); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
var s *Stream
|
||||||
|
for {
|
||||||
|
s, err = decode(context, b)
|
||||||
|
if err == errTryAgain {
|
||||||
|
buf, ok := seekNextFrame(b)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("audio/mp3: Decode failed: invalid format?")
|
||||||
|
}
|
||||||
|
b = buf
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var offlineAudioContextClass *js.Object
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
if klass := js.Global.Get("OfflineAudioContext"); klass != js.Undefined {
|
||||||
|
offlineAudioContextClass = klass
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if klass := js.Global.Get("webkitOfflineAudioContext"); klass != js.Undefined {
|
||||||
|
offlineAudioContextClass = klass
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func decode(context *audio.Context, buf []uint8) (*Stream, error) {
|
||||||
|
if offlineAudioContextClass == nil {
|
||||||
|
return nil, errors.New("audio/mp3: OfflineAudioContext is not available")
|
||||||
|
}
|
||||||
|
|
||||||
s := &Stream{}
|
s := &Stream{}
|
||||||
ch := make(chan error)
|
ch := make(chan error)
|
||||||
|
|
||||||
klass := js.Global.Get("OfflineAudioContext")
|
// TODO: webkitOfflineAudioContext's constructor might causes 'Syntax' error
|
||||||
if klass == js.Undefined {
|
// when the sample rate is not so usual like 22050 on Safari.
|
||||||
klass = js.Global.Get("webkitOfflineAudioContext")
|
// Should this be handled before calling this?
|
||||||
}
|
|
||||||
if klass == js.Undefined {
|
|
||||||
return nil, errors.New("audio/mp3: OfflineAudioContext is not available")
|
|
||||||
}
|
|
||||||
// This might causes 'Syntax' error when the sample rate is not so usual like 22050 on Safari.
|
|
||||||
// TODO: 1 is a correct second argument?
|
// TODO: 1 is a correct second argument?
|
||||||
oc := klass.New(2, 1, context.SampleRate())
|
oc := offlineAudioContextClass.New(2, 1, context.SampleRate())
|
||||||
oc.Call("decodeAudioData", js.NewArrayBuffer(b), func(buf *js.Object) {
|
oc.Call("decodeAudioData", js.NewArrayBuffer(buf), func(buf *js.Object) {
|
||||||
s.leftData = buf.Call("getChannelData", 0).Interface().([]float32)
|
s.leftData = buf.Call("getChannelData", 0).Interface().([]float32)
|
||||||
switch n := buf.Get("numberOfChannels").Int(); n {
|
switch n := buf.Get("numberOfChannels").Int(); n {
|
||||||
case 1:
|
case 1:
|
||||||
@ -122,8 +181,8 @@ func Decode(context *audio.Context, src audio.ReadSeekCloser) (*Stream, error) {
|
|||||||
ch <- fmt.Errorf("audio/mp3: number of channels must be 1 or 2 but %d", n)
|
ch <- fmt.Errorf("audio/mp3: number of channels must be 1 or 2 but %d", n)
|
||||||
}
|
}
|
||||||
close(ch)
|
close(ch)
|
||||||
}, func(err *js.Object) {
|
}, func() {
|
||||||
ch <- fmt.Errorf("audio/mp3: decoding failed: %s", err.String())
|
ch <- errTryAgain
|
||||||
close(ch)
|
close(ch)
|
||||||
})
|
})
|
||||||
if err := <-ch; err != nil {
|
if err := <-ch; err != nil {
|
||||||
|
Loading…
Reference in New Issue
Block a user