audio: bug fix: crash with uncomparable source

Closes #3039
This commit is contained in:
Hajime Hoshi 2024-07-15 13:40:37 +09:00
parent cde4c4fd2e
commit e980d59191
2 changed files with 35 additions and 1 deletions

View File

@ -37,6 +37,7 @@ import (
"errors" "errors"
"fmt" "fmt"
"io" "io"
"reflect"
"runtime" "runtime"
"sync" "sync"
"time" "time"
@ -188,11 +189,15 @@ func (c *Context) addPlayingPlayer(p *playerImpl) {
defer c.m.Unlock() defer c.m.Unlock()
c.playingPlayers[p] = struct{}{} c.playingPlayers[p] = struct{}{}
if !reflect.ValueOf(p.sourceIdent()).Comparable() {
return
}
// Check the source duplication // Check the source duplication
srcs := map[any]struct{}{} srcs := map[any]struct{}{}
for p := range c.playingPlayers { for p := range c.playingPlayers {
if _, ok := srcs[p.sourceIdent()]; ok { if _, ok := srcs[p.sourceIdent()]; ok {
c.err = errors.New("audio: a same source is used by multiple Player") c.err = errors.New("audio: the same source must not be used by multiple Player objects")
return return
} }
srcs[p.sourceIdent()] = struct{}{} srcs[p.sourceIdent()] = struct{}{}

View File

@ -16,6 +16,7 @@ package audio_test
import ( import (
"bytes" "bytes"
"io"
"os" "os"
"runtime" "runtime"
"testing" "testing"
@ -147,4 +148,32 @@ func TestNonSeekableSource(t *testing.T) {
p.Play() p.Play()
p.Pause() p.Pause()
if err := audio.UpdateForTesting(); err != nil {
t.Error(err)
}
}
type uncomparableSource []int
func (uncomparableSource) Read(buf []byte) (int, error) {
return 0, io.EOF
}
// Issue #3039
func TestUncomparableSource(t *testing.T) {
setup()
defer teardown()
p, err := context.NewPlayer(uncomparableSource{})
if err != nil {
t.Fatal(err)
}
p.Play()
p.Pause()
if err := audio.UpdateForTesting(); err != nil {
t.Error(err)
}
} }