hayai/vendor/github.com/hajimehoshi/oto/player.go
2024-12-21 17:26:50 +01:00

119 lines
2.8 KiB
Go

// Copyright 2017 Hajime Hoshi
//
// 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 oto offers io.Writer to play sound on multiple platforms.
package oto
import (
"io"
"runtime"
)
// Player is a PCM (pulse-code modulation) audio player.
// Player implements io.WriteCloser.
// Use Write method to play samples.
type Player struct {
context *Context
r io.ReadCloser
w io.WriteCloser
}
func newPlayer(context *Context) *Player {
r, w := pipe()
p := &Player{
context: context,
r: r,
w: w,
}
context.mux.AddSource(r)
runtime.SetFinalizer(p, (*Player).Close)
return p
}
// Write writes PCM samples to the Player.
//
// The format is as follows:
// [data] = [sample 1] [sample 2] [sample 3] ...
// [sample *] = [channel 1] ...
// [channel *] = [byte 1] [byte 2] ...
// Byte ordering is little endian.
//
// The data is first put into the Player's buffer. Once the buffer is full, Player starts playing
// the data and empties the buffer.
//
// If the supplied data doesn't fit into the Player's buffer, Write block until a sufficient amount
// of data has been played (or at least started playing) and the remaining unplayed data fits into
// the buffer.
//
// Note, that the Player won't start playing anything until the buffer is full.
func (p *Player) Write(buf []byte) (int, error) {
select {
case err := <-p.context.errCh:
return 0, err
default:
}
n, err := p.w.Write(buf)
// When the error is io.ErrClosedPipe, the context is already closed.
if err == io.ErrClosedPipe {
select {
case err := <-p.context.errCh:
return n, err
default:
}
}
return n, err
}
// Close closes the Player and frees any resources associated with it. The Player is no longer
// usable after calling Close.
func (p *Player) Close() error {
runtime.SetFinalizer(p, nil)
// Already closed
if p.context == nil {
return nil
}
select {
case err := <-p.context.errCh:
return err
default:
}
// Close the pipe writer before RemoveSource, or Read-ing in the mux takes forever.
if err := p.w.Close(); err != nil {
return err
}
p.context.mux.RemoveSource(p.r)
p.context = nil
// Close the pipe reader after RemoveSource, or ErrClosedPipe happens at Read-ing.
return p.r.Close()
}
func max(a, b int) int {
if a < b {
return b
}
return a
}
func min(a, b int) int {
if a < b {
return a
}
return b
}