audio: Bug fix: GC audio players correctly

Fixes #746
This commit is contained in:
Hajime Hoshi 2018-12-15 22:14:42 +09:00
parent 22b11aafac
commit 08a369b8fd
3 changed files with 152 additions and 32 deletions

View File

@ -48,7 +48,7 @@ import (
) )
type players struct { type players struct {
players map[*Player]struct{} players map[*playerImpl]struct{}
sync.RWMutex sync.RWMutex
} }
@ -117,7 +117,7 @@ func (p *players) Read(b []byte) (int, error) {
b[2*i+1] = byte(x >> 8) b[2*i+1] = byte(x >> 8)
} }
closed := []*Player{} closed := []*playerImpl{}
for player := range p.players { for player := range p.players {
if player.eof() { if player.eof() {
closed = append(closed, player) closed = append(closed, player)
@ -130,19 +130,19 @@ func (p *players) Read(b []byte) (int, error) {
return l, nil return l, nil
} }
func (p *players) addPlayer(player *Player) { func (p *players) addPlayer(player *playerImpl) {
p.Lock() p.Lock()
p.players[player] = struct{}{} p.players[player] = struct{}{}
p.Unlock() p.Unlock()
} }
func (p *players) removePlayer(player *Player) { func (p *players) removePlayer(player *playerImpl) {
p.Lock() p.Lock()
delete(p.players, player) delete(p.players, player)
p.Unlock() p.Unlock()
} }
func (p *players) hasPlayer(player *Player) bool { func (p *players) hasPlayer(player *playerImpl) bool {
p.RLock() p.RLock()
_, ok := p.players[player] _, ok := p.players[player]
p.RUnlock() p.RUnlock()
@ -217,7 +217,7 @@ func NewContext(sampleRate int) (*Context, error) {
} }
theContext = c theContext = c
c.players = &players{ c.players = &players{
players: map[*Player]struct{}{}, players: map[*playerImpl]struct{}{},
} }
go c.loop() go c.loop()
@ -332,6 +332,10 @@ func BytesReadSeekCloser(b []byte) ReadSeekCloser {
// Player is an audio player which has one stream. // Player is an audio player which has one stream.
type Player struct { type Player struct {
p *playerImpl
}
type playerImpl struct {
players *players players *players
src io.ReadCloser src io.ReadCloser
srcEOF bool srcEOF bool
@ -379,32 +383,34 @@ func NewPlayer(context *Context, src io.ReadCloser) (*Player, error) {
return nil, errors.New("audio: src cannot be shared with another Player") return nil, errors.New("audio: src cannot be shared with another Player")
} }
p := &Player{ p := &Player{
players: context.players, &playerImpl{
src: src, players: context.players,
sampleRate: context.sampleRate, src: src,
buf: nil, sampleRate: context.sampleRate,
volume: 1, buf: nil,
closeCh: make(chan struct{}), volume: 1,
closedCh: make(chan struct{}), closeCh: make(chan struct{}),
readLoopEndedCh: make(chan struct{}), closedCh: make(chan struct{}),
seekCh: make(chan seekArgs), readLoopEndedCh: make(chan struct{}),
seekedCh: make(chan error), seekCh: make(chan seekArgs),
proceedCh: make(chan []int16), seekedCh: make(chan error),
proceededCh: make(chan proceededValues), proceedCh: make(chan []int16),
syncCh: make(chan func()), proceededCh: make(chan proceededValues),
syncCh: make(chan func()),
},
} }
if seeker, ok := p.src.(io.Seeker); ok { if seeker, ok := p.p.src.(io.Seeker); ok {
// Get the current position of the source. // Get the current position of the source.
pos, err := seeker.Seek(0, io.SeekCurrent) pos, err := seeker.Seek(0, io.SeekCurrent)
if err != nil { if err != nil {
return nil, err return nil, err
} }
p.pos = pos p.p.pos = pos
} }
runtime.SetFinalizer(p, (*Player).Close) runtime.SetFinalizer(p, (*Player).Close)
go func() { go func() {
p.readLoop() p.p.readLoop()
}() }()
return p, nil return p, nil
} }
@ -435,6 +441,10 @@ func NewPlayerFromBytes(context *Context, src []byte) (*Player, error) {
// Close returns error when closing the source returns error. // Close returns error when closing the source returns error.
func (p *Player) Close() error { func (p *Player) Close() error {
runtime.SetFinalizer(p, nil) runtime.SetFinalizer(p, nil)
return p.p.Close()
}
func (p *playerImpl) Close() error {
p.players.removePlayer(p) p.players.removePlayer(p)
select { select {
@ -446,7 +456,7 @@ func (p *Player) Close() error {
} }
} }
func (p *Player) bufferToInt16(lengthInBytes int) ([]int16, error) { func (p *playerImpl) bufferToInt16(lengthInBytes int) ([]int16, error) {
select { select {
case p.proceedCh <- make([]int16, lengthInBytes/2): case p.proceedCh <- make([]int16, lengthInBytes/2):
r := <-p.proceededCh r := <-p.proceededCh
@ -460,11 +470,15 @@ func (p *Player) bufferToInt16(lengthInBytes int) ([]int16, error) {
// //
// Play always returns nil. // Play always returns nil.
func (p *Player) Play() error { func (p *Player) Play() error {
p.players.addPlayer(p) p.p.Play()
return nil return nil
} }
func (p *Player) readLoop() { func (p *playerImpl) Play() {
p.players.addPlayer(p)
}
func (p *playerImpl) readLoop() {
defer func() { defer func() {
// Note: the error is ignored // Note: the error is ignored
p.src.Close() p.src.Close()
@ -586,7 +600,7 @@ func (p *Player) readLoop() {
} }
} }
func (p *Player) sync(f func()) bool { func (p *playerImpl) sync(f func()) bool {
ch := make(chan struct{}) ch := make(chan struct{})
ff := func() { ff := func() {
f() f()
@ -601,7 +615,7 @@ func (p *Player) sync(f func()) bool {
} }
} }
func (p *Player) shouldSkip() bool { func (p *playerImpl) shouldSkip() bool {
r := false r := false
p.sync(func() { p.sync(func() {
r = p.shouldSkipImpl() r = p.shouldSkipImpl()
@ -609,7 +623,7 @@ func (p *Player) shouldSkip() bool {
return r return r
} }
func (p *Player) shouldSkipImpl() bool { func (p *playerImpl) shouldSkipImpl() bool {
// When p.buf is nil, the player just starts playing or seeking. // When p.buf is nil, the player just starts playing or seeking.
// Note that this is different from len(p.buf) == 0 && p.buf != nil. // Note that this is different from len(p.buf) == 0 && p.buf != nil.
if p.buf == nil { if p.buf == nil {
@ -621,7 +635,7 @@ func (p *Player) shouldSkipImpl() bool {
return false return false
} }
func (p *Player) bufferSizeInBytes() int { func (p *playerImpl) bufferSizeInBytes() int {
s := 0 s := 0
p.sync(func() { p.sync(func() {
s = len(p.buf) s = len(p.buf)
@ -629,7 +643,7 @@ func (p *Player) bufferSizeInBytes() int {
return s return s
} }
func (p *Player) eof() bool { func (p *playerImpl) eof() bool {
r := false r := false
p.sync(func() { p.sync(func() {
r = p.eofImpl() r = p.eofImpl()
@ -637,12 +651,16 @@ func (p *Player) eof() bool {
return r return r
} }
func (p *Player) eofImpl() bool { func (p *playerImpl) eofImpl() bool {
return p.srcEOF && len(p.buf) == 0 return p.srcEOF && len(p.buf) == 0
} }
// IsPlaying returns boolean indicating whether the player is playing. // IsPlaying returns boolean indicating whether the player is playing.
func (p *Player) IsPlaying() bool { func (p *Player) IsPlaying() bool {
return p.p.IsPlaying()
}
func (p *playerImpl) IsPlaying() bool {
return p.players.hasPlayer(p) return p.players.hasPlayer(p)
} }
@ -652,6 +670,10 @@ func (p *Player) IsPlaying() bool {
// //
// Rewind returns error when seeking the source stream returns error. // Rewind returns error when seeking the source stream returns error.
func (p *Player) Rewind() error { func (p *Player) Rewind() error {
return p.p.Rewind()
}
func (p *playerImpl) Rewind() error {
if _, ok := p.src.(io.Seeker); !ok { if _, ok := p.src.(io.Seeker); !ok {
panic("audio: player to be rewound must be io.Seeker") panic("audio: player to be rewound must be io.Seeker")
} }
@ -664,6 +686,10 @@ func (p *Player) Rewind() error {
// //
// Seek returns error when seeking the source stream returns error. // Seek returns error when seeking the source stream returns error.
func (p *Player) Seek(offset time.Duration) error { func (p *Player) Seek(offset time.Duration) error {
return p.p.Seek(offset)
}
func (p *playerImpl) Seek(offset time.Duration) error {
if _, ok := p.src.(io.Seeker); !ok { if _, ok := p.src.(io.Seeker); !ok {
panic("audio: player to be sought must be io.Seeker") panic("audio: player to be sought must be io.Seeker")
} }
@ -681,12 +707,20 @@ func (p *Player) Seek(offset time.Duration) error {
// //
// Pause always returns nil. // Pause always returns nil.
func (p *Player) Pause() error { func (p *Player) Pause() error {
p.players.removePlayer(p) p.p.Pause()
return nil return nil
} }
func (p *playerImpl) Pause() {
p.players.removePlayer(p)
}
// Current returns the current position. // Current returns the current position.
func (p *Player) Current() time.Duration { func (p *Player) Current() time.Duration {
return p.p.Current()
}
func (p *playerImpl) Current() time.Duration {
sample := int64(0) sample := int64(0)
p.sync(func() { p.sync(func() {
sample = p.pos / bytesPerSample sample = p.pos / bytesPerSample
@ -696,6 +730,10 @@ func (p *Player) Current() time.Duration {
// Volume returns the current volume of this player [0-1]. // Volume returns the current volume of this player [0-1].
func (p *Player) Volume() float64 { func (p *Player) Volume() float64 {
return p.p.Volume()
}
func (p *playerImpl) Volume() float64 {
v := 0.0 v := 0.0
p.sync(func() { p.sync(func() {
v = p.volume v = p.volume
@ -706,6 +744,10 @@ func (p *Player) Volume() float64 {
// SetVolume sets the volume of this player. // SetVolume sets the volume of this player.
// volume must be in between 0 and 1. SetVolume panics otherwise. // volume must be in between 0 and 1. SetVolume panics otherwise.
func (p *Player) SetVolume(volume float64) { func (p *Player) SetVolume(volume float64) {
p.p.SetVolume(volume)
}
func (p *playerImpl) SetVolume(volume float64) {
// The condition must be true when volume is NaN. // The condition must be true when volume is NaN.
if !(0 <= volume && volume <= 1) { if !(0 <= volume && volume <= 1) {
panic("audio: volume must be in between 0 and 1") panic("audio: volume must be in between 0 and 1")

55
audio/audio_test.go Normal file
View File

@ -0,0 +1,55 @@
// Copyright 2018 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.
package audio_test
import (
"runtime"
"testing"
. "github.com/hajimehoshi/ebiten/audio"
)
var context *Context
func init() {
var err error
context, err = NewContext(44100)
if err != nil {
panic(err)
}
}
// Issue #746
func TestGC(t *testing.T) {
p, _ := NewPlayer(context, BytesReadSeekCloser(make([]byte, 4)))
got := PlayersNumForTesting()
if want := 0; got != want {
t.Errorf("PlayersNum(): got: %d, want: %d", got, want)
}
p.Play()
got = PlayersNumForTesting()
if want := 1; got != want {
t.Errorf("PlayersNum() after Play: got: %d, want: %d", got, want)
}
runtime.KeepAlive(p)
p = nil
runtime.GC()
got = PlayersNumForTesting()
if want := 0; got != want {
t.Errorf("PlayersNum() after GC: got: %d, want: %d", got, want)
}
}

View File

@ -0,0 +1,23 @@
// Copyright 2018 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.
package audio
func PlayersNumForTesting() int {
c := CurrentContext()
c.players.Lock()
n := len(c.players.players)
c.players.Unlock()
return n
}