mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-13 20:42:07 +01:00
audio/internal/readerdriver: Share one goroutine to read the source to the buffers (iOS/macOS)
This fix improves the latency of the audio. Closes #1662
This commit is contained in:
parent
67e5fae9c0
commit
95c494f47e
@ -197,35 +197,41 @@ type playerImpl struct {
|
|||||||
state playerState
|
state playerState
|
||||||
err error
|
err error
|
||||||
eof bool
|
eof bool
|
||||||
cond *sync.Cond
|
|
||||||
volume float64
|
volume float64
|
||||||
|
|
||||||
|
m sync.Mutex
|
||||||
}
|
}
|
||||||
|
|
||||||
type players struct {
|
type players struct {
|
||||||
players map[C.AudioQueueRef]*playerImpl
|
players map[C.AudioQueueRef]*playerImpl
|
||||||
toResume map[*playerImpl]struct{}
|
toResume map[*playerImpl]struct{}
|
||||||
m sync.Mutex
|
cond *sync.Cond
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *players) add(player *playerImpl, audioQueue C.AudioQueueRef) {
|
func (p *players) add(player *playerImpl, audioQueue C.AudioQueueRef) {
|
||||||
p.m.Lock()
|
p.cond.L.Lock()
|
||||||
defer p.m.Unlock()
|
defer p.cond.L.Unlock()
|
||||||
|
|
||||||
if p.players == nil {
|
if p.players == nil {
|
||||||
p.players = map[C.AudioQueueRef]*playerImpl{}
|
p.players = map[C.AudioQueueRef]*playerImpl{}
|
||||||
}
|
}
|
||||||
|
runLoop := len(p.players) == 0
|
||||||
p.players[audioQueue] = player
|
p.players[audioQueue] = player
|
||||||
|
if runLoop {
|
||||||
|
// Use the only one loop for multiple players (#1662).
|
||||||
|
go p.loop()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *players) get(audioQueue C.AudioQueueRef) *playerImpl {
|
func (p *players) get(audioQueue C.AudioQueueRef) *playerImpl {
|
||||||
p.m.Lock()
|
p.cond.L.Lock()
|
||||||
defer p.m.Unlock()
|
defer p.cond.L.Unlock()
|
||||||
return p.players[audioQueue]
|
return p.players[audioQueue]
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *players) remove(audioQueue C.AudioQueueRef) {
|
func (p *players) remove(audioQueue C.AudioQueueRef) {
|
||||||
p.m.Lock()
|
p.cond.L.Lock()
|
||||||
defer p.m.Unlock()
|
defer p.cond.L.Unlock()
|
||||||
|
|
||||||
pl, ok := p.players[audioQueue]
|
pl, ok := p.players[audioQueue]
|
||||||
if !ok {
|
if !ok {
|
||||||
@ -233,11 +239,56 @@ func (p *players) remove(audioQueue C.AudioQueueRef) {
|
|||||||
}
|
}
|
||||||
delete(p.players, audioQueue)
|
delete(p.players, audioQueue)
|
||||||
delete(p.toResume, pl)
|
delete(p.toResume, pl)
|
||||||
|
|
||||||
|
p.cond.Signal()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *players) shouldWait() bool {
|
||||||
|
if len(p.players) == 0 {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, pl := range p.players {
|
||||||
|
if pl.canReadSourceToBuffer() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *players) wait() bool {
|
||||||
|
p.cond.L.Lock()
|
||||||
|
defer p.cond.L.Unlock()
|
||||||
|
|
||||||
|
for p.shouldWait() {
|
||||||
|
p.cond.Wait()
|
||||||
|
}
|
||||||
|
return len(p.players) > 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *players) loop() {
|
||||||
|
var players []*playerImpl
|
||||||
|
for {
|
||||||
|
if !p.wait() {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
p.cond.L.Lock()
|
||||||
|
players = players[:0]
|
||||||
|
for _, pl := range p.players {
|
||||||
|
players = append(players, pl)
|
||||||
|
}
|
||||||
|
p.cond.L.Unlock()
|
||||||
|
|
||||||
|
for _, pl := range players {
|
||||||
|
pl.readSourceToBuffer()
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *players) suspend() error {
|
func (p *players) suspend() error {
|
||||||
p.m.Lock()
|
p.cond.L.Lock()
|
||||||
defer p.m.Unlock()
|
defer p.cond.L.Unlock()
|
||||||
|
|
||||||
for _, pl := range p.players {
|
for _, pl := range p.players {
|
||||||
if !pl.IsPlaying() {
|
if !pl.IsPlaying() {
|
||||||
@ -259,13 +310,13 @@ func (p *players) suspend() error {
|
|||||||
|
|
||||||
func (p *players) resume() error {
|
func (p *players) resume() error {
|
||||||
// playerImpl's Play can touch p. Avoid the deadlock.
|
// playerImpl's Play can touch p. Avoid the deadlock.
|
||||||
p.m.Lock()
|
p.cond.L.Lock()
|
||||||
players := map[*playerImpl]struct{}{}
|
players := map[*playerImpl]struct{}{}
|
||||||
for pl := range p.toResume {
|
for pl := range p.toResume {
|
||||||
players[pl] = struct{}{}
|
players[pl] = struct{}{}
|
||||||
delete(p.toResume, pl)
|
delete(p.toResume, pl)
|
||||||
}
|
}
|
||||||
p.m.Unlock()
|
p.cond.L.Unlock()
|
||||||
|
|
||||||
for pl := range players {
|
for pl := range players {
|
||||||
pl.Play()
|
pl.Play()
|
||||||
@ -276,14 +327,15 @@ func (p *players) resume() error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
var thePlayers players
|
var thePlayers = &players{
|
||||||
|
cond: sync.NewCond(&sync.Mutex{}),
|
||||||
|
}
|
||||||
|
|
||||||
func (c *context) NewPlayer(src io.Reader) Player {
|
func (c *context) NewPlayer(src io.Reader) Player {
|
||||||
p := &player{
|
p := &player{
|
||||||
p: &playerImpl{
|
p: &playerImpl{
|
||||||
context: c,
|
context: c,
|
||||||
src: src,
|
src: src,
|
||||||
cond: sync.NewCond(&sync.Mutex{}),
|
|
||||||
volume: 1,
|
volume: 1,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@ -296,8 +348,8 @@ func (p *player) Err() error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *playerImpl) Err() error {
|
func (p *playerImpl) Err() error {
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
defer p.cond.L.Unlock()
|
defer p.m.Unlock()
|
||||||
|
|
||||||
return p.err
|
return p.err
|
||||||
}
|
}
|
||||||
@ -310,8 +362,8 @@ func (p *playerImpl) Play() {
|
|||||||
// Call Play asynchronously since AudioQueuePrime and AudioQueuePlay might take long.
|
// Call Play asynchronously since AudioQueuePrime and AudioQueuePlay might take long.
|
||||||
ch := make(chan struct{})
|
ch := make(chan struct{})
|
||||||
go func() {
|
go func() {
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
defer p.cond.L.Unlock()
|
defer p.m.Unlock()
|
||||||
close(ch)
|
close(ch)
|
||||||
p.playImpl()
|
p.playImpl()
|
||||||
}()
|
}()
|
||||||
@ -328,7 +380,6 @@ func (p *playerImpl) playImpl() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
var runLoop bool
|
|
||||||
if p.audioQueue == nil {
|
if p.audioQueue == nil {
|
||||||
audioQueue, audioQueueBuffers, err := p.context.audioQueuePool.Get()
|
audioQueue, audioQueueBuffers, err := p.context.audioQueuePool.Get()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -338,9 +389,10 @@ func (p *playerImpl) playImpl() {
|
|||||||
p.audioQueue = audioQueue
|
p.audioQueue = audioQueue
|
||||||
p.unqueuedBufs = audioQueueBuffers
|
p.unqueuedBufs = audioQueueBuffers
|
||||||
C.AudioQueueSetParameter(p.audioQueue, C.kAudioQueueParam_Volume, C.AudioQueueParameterValue(p.volume))
|
C.AudioQueueSetParameter(p.audioQueue, C.kAudioQueueParam_Volume, C.AudioQueueParameterValue(p.volume))
|
||||||
thePlayers.add(p, p.audioQueue)
|
|
||||||
|
|
||||||
runLoop = true
|
p.m.Unlock()
|
||||||
|
thePlayers.add(p, p.audioQueue)
|
||||||
|
p.m.Lock()
|
||||||
}
|
}
|
||||||
|
|
||||||
buf := make([]byte, p.context.maxBufferSize())
|
buf := make([]byte, p.context.maxBufferSize())
|
||||||
@ -395,11 +447,6 @@ func (p *playerImpl) playImpl() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
p.state = playerPlay
|
p.state = playerPlay
|
||||||
p.cond.Signal()
|
|
||||||
|
|
||||||
if runLoop {
|
|
||||||
go p.loop()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *player) Pause() {
|
func (p *player) Pause() {
|
||||||
@ -407,8 +454,8 @@ func (p *player) Pause() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *playerImpl) Pause() {
|
func (p *playerImpl) Pause() {
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
defer p.cond.L.Unlock()
|
defer p.m.Unlock()
|
||||||
|
|
||||||
if p.err != nil {
|
if p.err != nil {
|
||||||
return
|
return
|
||||||
@ -425,7 +472,6 @@ func (p *playerImpl) Pause() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
p.state = playerPaused
|
p.state = playerPaused
|
||||||
p.cond.Signal()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *player) Reset() {
|
func (p *player) Reset() {
|
||||||
@ -433,8 +479,8 @@ func (p *player) Reset() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *playerImpl) Reset() {
|
func (p *playerImpl) Reset() {
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
defer p.cond.L.Unlock()
|
defer p.m.Unlock()
|
||||||
p.resetImpl()
|
p.resetImpl()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -456,9 +502,9 @@ func (p *playerImpl) resetImpl() {
|
|||||||
}
|
}
|
||||||
// AudioQueueReset invokes the callback directry.
|
// AudioQueueReset invokes the callback directry.
|
||||||
q := p.audioQueue
|
q := p.audioQueue
|
||||||
p.cond.L.Unlock()
|
p.m.Unlock()
|
||||||
osstatus := C.AudioQueueReset(q)
|
osstatus := C.AudioQueueReset(q)
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
if osstatus != C.noErr && p.err == nil {
|
if osstatus != C.noErr && p.err == nil {
|
||||||
p.setErrorImpl(fmt.Errorf("readerdriver: AudioQueueReset failed: %d", osstatus))
|
p.setErrorImpl(fmt.Errorf("readerdriver: AudioQueueReset failed: %d", osstatus))
|
||||||
return
|
return
|
||||||
@ -468,7 +514,7 @@ func (p *playerImpl) resetImpl() {
|
|||||||
p.state = playerPaused
|
p.state = playerPaused
|
||||||
p.buf = p.buf[:0]
|
p.buf = p.buf[:0]
|
||||||
p.eof = false
|
p.eof = false
|
||||||
p.cond.Signal()
|
thePlayers.cond.Signal()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *player) IsPlaying() bool {
|
func (p *player) IsPlaying() bool {
|
||||||
@ -476,8 +522,8 @@ func (p *player) IsPlaying() bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *playerImpl) IsPlaying() bool {
|
func (p *playerImpl) IsPlaying() bool {
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
defer p.cond.L.Unlock()
|
defer p.m.Unlock()
|
||||||
|
|
||||||
return p.state == playerPlay
|
return p.state == playerPlay
|
||||||
}
|
}
|
||||||
@ -487,8 +533,8 @@ func (p *player) Volume() float64 {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *playerImpl) Volume() float64 {
|
func (p *playerImpl) Volume() float64 {
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
defer p.cond.L.Unlock()
|
defer p.m.Unlock()
|
||||||
return p.volume
|
return p.volume
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -497,8 +543,8 @@ func (p *player) SetVolume(volume float64) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *playerImpl) SetVolume(volume float64) {
|
func (p *playerImpl) SetVolume(volume float64) {
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
defer p.cond.L.Unlock()
|
defer p.m.Unlock()
|
||||||
|
|
||||||
p.volume = volume
|
p.volume = volume
|
||||||
if p.audioQueue == nil {
|
if p.audioQueue == nil {
|
||||||
@ -512,8 +558,8 @@ func (p *player) UnplayedBufferSize() int {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *playerImpl) UnplayedBufferSize() int {
|
func (p *playerImpl) UnplayedBufferSize() int {
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
defer p.cond.L.Unlock()
|
defer p.m.Unlock()
|
||||||
return len(p.buf)
|
return len(p.buf)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -523,8 +569,8 @@ func (p *player) Close() error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *playerImpl) Close() error {
|
func (p *playerImpl) Close() error {
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
defer p.cond.L.Unlock()
|
defer p.m.Unlock()
|
||||||
return p.closeImpl()
|
return p.closeImpl()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -535,9 +581,9 @@ func (p *playerImpl) closeImpl() error {
|
|||||||
|
|
||||||
// AudioQueueStop might invoke AudioQueueReset. Unlock the mutex here to avoid a deadlock.
|
// AudioQueueStop might invoke AudioQueueReset. Unlock the mutex here to avoid a deadlock.
|
||||||
q := p.audioQueue
|
q := p.audioQueue
|
||||||
p.cond.L.Unlock()
|
p.m.Unlock()
|
||||||
osstatus := C.AudioQueueStop(q, C.true)
|
osstatus := C.AudioQueueStop(q, C.true)
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
|
|
||||||
if osstatus != C.noErr && p.err == nil {
|
if osstatus != C.noErr && p.err == nil {
|
||||||
// setErrorImpl calls closeImpl. Do not call this.
|
// setErrorImpl calls closeImpl. Do not call this.
|
||||||
@ -548,18 +594,24 @@ func (p *playerImpl) closeImpl() error {
|
|||||||
if err := p.context.audioQueuePool.Put(p.audioQueue); err != nil && p.err == nil {
|
if err := p.context.audioQueuePool.Put(p.audioQueue); err != nil && p.err == nil {
|
||||||
p.err = err
|
p.err = err
|
||||||
}
|
}
|
||||||
|
p.m.Unlock()
|
||||||
thePlayers.remove(p.audioQueue)
|
thePlayers.remove(p.audioQueue)
|
||||||
|
p.m.Lock()
|
||||||
p.audioQueue = nil
|
p.audioQueue = nil
|
||||||
}
|
}
|
||||||
|
|
||||||
p.state = playerClosed
|
p.state = playerClosed
|
||||||
p.cond.Signal()
|
|
||||||
return p.err
|
return p.err
|
||||||
}
|
}
|
||||||
|
|
||||||
//export ebiten_readerdriver_render
|
//export ebiten_readerdriver_render
|
||||||
func ebiten_readerdriver_render(inUserData unsafe.Pointer, inAQ C.AudioQueueRef, inBuffer C.AudioQueueBufferRef) {
|
func ebiten_readerdriver_render(inUserData unsafe.Pointer, inAQ C.AudioQueueRef, inBuffer C.AudioQueueBufferRef) {
|
||||||
p := thePlayers.get(inAQ)
|
p := thePlayers.get(inAQ)
|
||||||
|
// The player might be already closed.
|
||||||
|
if p == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
queued, err := p.appendBuffer(inBuffer)
|
queued, err := p.appendBuffer(inBuffer)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
p.setError(err)
|
p.setError(err)
|
||||||
@ -569,8 +621,8 @@ func ebiten_readerdriver_render(inUserData unsafe.Pointer, inAQ C.AudioQueueRef,
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
defer p.cond.L.Unlock()
|
defer p.m.Unlock()
|
||||||
p.unqueuedBufs = append(p.unqueuedBufs, inBuffer)
|
p.unqueuedBufs = append(p.unqueuedBufs, inBuffer)
|
||||||
if len(p.unqueuedBufs) == 2 && p.eof {
|
if len(p.unqueuedBufs) == 2 && p.eof {
|
||||||
p.resetImpl()
|
p.resetImpl()
|
||||||
@ -578,8 +630,8 @@ func ebiten_readerdriver_render(inUserData unsafe.Pointer, inAQ C.AudioQueueRef,
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *playerImpl) appendBuffer(inBuffer C.AudioQueueBufferRef) (bool, error) {
|
func (p *playerImpl) appendBuffer(inBuffer C.AudioQueueBufferRef) (bool, error) {
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
defer p.cond.L.Unlock()
|
defer p.m.Unlock()
|
||||||
return p.appendBufferImpl(inBuffer)
|
return p.appendBufferImpl(inBuffer)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -594,7 +646,6 @@ func (p *playerImpl) appendBufferImpl(inBuffer C.AudioQueueBufferRef) (bool, err
|
|||||||
n = len(p.buf)
|
n = len(p.buf)
|
||||||
}
|
}
|
||||||
buf := p.buf[:n]
|
buf := p.buf[:n]
|
||||||
signal := len(p.buf[n:]) < p.context.maxBufferSize()
|
|
||||||
|
|
||||||
for i, b := range buf {
|
for i, b := range buf {
|
||||||
*(*byte)(unsafe.Pointer(uintptr(inBuffer.mAudioData) + uintptr(i))) = b
|
*(*byte)(unsafe.Pointer(uintptr(inBuffer.mAudioData) + uintptr(i))) = b
|
||||||
@ -612,40 +663,13 @@ func (p *playerImpl) appendBufferImpl(inBuffer C.AudioQueueBufferRef) (bool, err
|
|||||||
}
|
}
|
||||||
|
|
||||||
p.buf = p.buf[n:]
|
p.buf = p.buf[n:]
|
||||||
if signal {
|
thePlayers.cond.Signal()
|
||||||
p.cond.Signal()
|
|
||||||
}
|
|
||||||
return true, nil
|
return true, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *playerImpl) shouldWait() bool {
|
|
||||||
switch p.state {
|
|
||||||
case playerPaused:
|
|
||||||
return true
|
|
||||||
case playerPlay:
|
|
||||||
// If the buffer has too much data, wait until the buffer data is consumed.
|
|
||||||
// If the source reaches EOF, wait until the state is reset.
|
|
||||||
return len(p.buf) >= p.context.maxBufferSize() || p.eof
|
|
||||||
case playerClosed:
|
|
||||||
return false
|
|
||||||
default:
|
|
||||||
panic("not reached")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *playerImpl) wait() bool {
|
|
||||||
p.cond.L.Lock()
|
|
||||||
defer p.cond.L.Unlock()
|
|
||||||
|
|
||||||
for p.shouldWait() {
|
|
||||||
p.cond.Wait()
|
|
||||||
}
|
|
||||||
return p.state == playerPlay
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *playerImpl) setError(err error) {
|
func (p *playerImpl) setError(err error) {
|
||||||
p.cond.L.Lock()
|
p.m.Lock()
|
||||||
defer p.cond.L.Unlock()
|
defer p.m.Unlock()
|
||||||
p.setErrorImpl(err)
|
p.setErrorImpl(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -654,25 +678,46 @@ func (p *playerImpl) setErrorImpl(err error) {
|
|||||||
p.closeImpl()
|
p.closeImpl()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *playerImpl) loop() {
|
func (p *playerImpl) canReadSourceToBuffer() bool {
|
||||||
buf := make([]byte, 4096)
|
p.m.Lock()
|
||||||
for {
|
defer p.m.Unlock()
|
||||||
if !p.wait() {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
n, err := p.src.Read(buf)
|
if p.eof {
|
||||||
if err != nil && err != io.EOF {
|
return false
|
||||||
p.setError(err)
|
}
|
||||||
return
|
return len(p.buf) < p.context.maxBufferSize()
|
||||||
}
|
}
|
||||||
|
|
||||||
p.cond.L.Lock()
|
func (p *playerImpl) readSourceToBuffer() {
|
||||||
p.buf = append(p.buf, buf[:n]...)
|
p.m.Lock()
|
||||||
if err == io.EOF && len(p.buf) == 0 {
|
defer p.m.Unlock()
|
||||||
p.eof = true
|
|
||||||
}
|
if p.err != nil {
|
||||||
p.cond.L.Unlock()
|
return
|
||||||
|
}
|
||||||
|
if p.state == playerClosed {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
maxBufferSize := p.context.maxBufferSize()
|
||||||
|
if len(p.buf) >= maxBufferSize {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
src := p.src
|
||||||
|
p.m.Unlock()
|
||||||
|
buf := make([]byte, maxBufferSize)
|
||||||
|
n, err := src.Read(buf)
|
||||||
|
p.m.Lock()
|
||||||
|
|
||||||
|
if err != nil && err != io.EOF {
|
||||||
|
p.setError(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
p.buf = append(p.buf, buf[:n]...)
|
||||||
|
if err == io.EOF && len(p.buf) == 0 {
|
||||||
|
p.eof = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user