mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-11 19:48:54 +01:00
parent
22b11aafac
commit
08a369b8fd
106
audio/audio.go
106
audio/audio.go
@ -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
55
audio/audio_test.go
Normal 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)
|
||||||
|
}
|
||||||
|
}
|
23
audio/testinternal_test.go
Normal file
23
audio/testinternal_test.go
Normal 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
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user