| // Copyright 2009 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| // Pipe adapter to connect code expecting an io.Reader |
| // with code expecting an io.Writer. |
| |
| package io |
| |
| import ( |
| "errors" |
| "sync" |
| ) |
| |
| // onceError is an object that will only store an error once. |
| type onceError struct { |
| sync.Mutex // guards following |
| err error |
| } |
| |
| func (a *onceError) Store(err error) { |
| a.Lock() |
| defer a.Unlock() |
| if a.err != nil { |
| return |
| } |
| a.err = err |
| } |
| func (a *onceError) Load() error { |
| a.Lock() |
| defer a.Unlock() |
| return a.err |
| } |
| |
| // ErrClosedPipe is the error used for read or write operations on a closed pipe. |
| var ErrClosedPipe = errors.New("io: read/write on closed pipe") |
| |
| // A pipe is the shared pipe structure underlying PipeReader and PipeWriter. |
| type pipe struct { |
| wrMu sync.Mutex // Serializes Write operations |
| wrCh chan []byte |
| rdCh chan int |
| |
| once sync.Once // Protects closing done |
| done chan struct{} |
| rerr onceError |
| werr onceError |
| } |
| |
| func (p *pipe) read(b []byte) (n int, err error) { |
| select { |
| case <-p.done: |
| return 0, p.readCloseError() |
| default: |
| } |
| |
| select { |
| case bw := <-p.wrCh: |
| nr := copy(b, bw) |
| p.rdCh <- nr |
| return nr, nil |
| case <-p.done: |
| return 0, p.readCloseError() |
| } |
| } |
| |
| func (p *pipe) closeRead(err error) error { |
| if err == nil { |
| err = ErrClosedPipe |
| } |
| p.rerr.Store(err) |
| p.once.Do(func() { close(p.done) }) |
| return nil |
| } |
| |
| func (p *pipe) write(b []byte) (n int, err error) { |
| select { |
| case <-p.done: |
| return 0, p.writeCloseError() |
| default: |
| p.wrMu.Lock() |
| defer p.wrMu.Unlock() |
| } |
| |
| for once := true; once || len(b) > 0; once = false { |
| select { |
| case p.wrCh <- b: |
| nw := <-p.rdCh |
| b = b[nw:] |
| n += nw |
| case <-p.done: |
| return n, p.writeCloseError() |
| } |
| } |
| return n, nil |
| } |
| |
| func (p *pipe) closeWrite(err error) error { |
| if err == nil { |
| err = EOF |
| } |
| p.werr.Store(err) |
| p.once.Do(func() { close(p.done) }) |
| return nil |
| } |
| |
| // readCloseError is considered internal to the pipe type. |
| func (p *pipe) readCloseError() error { |
| rerr := p.rerr.Load() |
| if werr := p.werr.Load(); rerr == nil && werr != nil { |
| return werr |
| } |
| return ErrClosedPipe |
| } |
| |
| // writeCloseError is considered internal to the pipe type. |
| func (p *pipe) writeCloseError() error { |
| werr := p.werr.Load() |
| if rerr := p.rerr.Load(); werr == nil && rerr != nil { |
| return rerr |
| } |
| return ErrClosedPipe |
| } |
| |
| // A PipeReader is the read half of a pipe. |
| type PipeReader struct{ pipe } |
| |
| // Read implements the standard Read interface: |
| // it reads data from the pipe, blocking until a writer |
| // arrives or the write end is closed. |
| // If the write end is closed with an error, that error is |
| // returned as err; otherwise err is EOF. |
| func (r *PipeReader) Read(data []byte) (n int, err error) { |
| return r.pipe.read(data) |
| } |
| |
| // Close closes the reader; subsequent writes to the |
| // write half of the pipe will return the error [ErrClosedPipe]. |
| func (r *PipeReader) Close() error { |
| return r.CloseWithError(nil) |
| } |
| |
| // CloseWithError closes the reader; subsequent writes |
| // to the write half of the pipe will return the error err. |
| // |
| // CloseWithError never overwrites the previous error if it exists |
| // and always returns nil. |
| func (r *PipeReader) CloseWithError(err error) error { |
| return r.pipe.closeRead(err) |
| } |
| |
| // A PipeWriter is the write half of a pipe. |
| type PipeWriter struct{ r PipeReader } |
| |
| // Write implements the standard Write interface: |
| // it writes data to the pipe, blocking until one or more readers |
| // have consumed all the data or the read end is closed. |
| // If the read end is closed with an error, that err is |
| // returned as err; otherwise err is [ErrClosedPipe]. |
| func (w *PipeWriter) Write(data []byte) (n int, err error) { |
| return w.r.pipe.write(data) |
| } |
| |
| // Close closes the writer; subsequent reads from the |
| // read half of the pipe will return no bytes and EOF. |
| func (w *PipeWriter) Close() error { |
| return w.CloseWithError(nil) |
| } |
| |
| // CloseWithError closes the writer; subsequent reads from the |
| // read half of the pipe will return no bytes and the error err, |
| // or EOF if err is nil. |
| // |
| // CloseWithError never overwrites the previous error if it exists |
| // and always returns nil. |
| func (w *PipeWriter) CloseWithError(err error) error { |
| return w.r.pipe.closeWrite(err) |
| } |
| |
| // Pipe creates a synchronous in-memory pipe. |
| // It can be used to connect code expecting an [io.Reader] |
| // with code expecting an [io.Writer]. |
| // |
| // Reads and Writes on the pipe are matched one to one |
| // except when multiple Reads are needed to consume a single Write. |
| // That is, each Write to the [PipeWriter] blocks until it has satisfied |
| // one or more Reads from the [PipeReader] that fully consume |
| // the written data. |
| // The data is copied directly from the Write to the corresponding |
| // Read (or Reads); there is no internal buffering. |
| // |
| // It is safe to call Read and Write in parallel with each other or with Close. |
| // Parallel calls to Read and parallel calls to Write are also safe: |
| // the individual calls will be gated sequentially. |
| func Pipe() (*PipeReader, *PipeWriter) { |
| pw := &PipeWriter{r: PipeReader{pipe: pipe{ |
| wrCh: make(chan []byte), |
| rdCh: make(chan int), |
| done: make(chan struct{}), |
| }}} |
| return &pw.r, pw |
| } |