File: eager_reader.go

package info (click to toggle)
golang-github-htcat-htcat 1.0.2-1.1
  • links: PTS, VCS
  • area: main
  • in suites: bookworm, bullseye, forky, sid, trixie
  • size: 100 kB
  • sloc: makefile: 5
file content (106 lines) | stat: -rw-r--r-- 1,919 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
package htcat

import (
	"io"
	"sync"
)

type eagerReader struct {
	closeNotify chan struct{}
	rc          io.ReadCloser

	buf   []byte
	more  *sync.Cond
	begin int
	end   int

	lastErr error
}

func newEagerReader(r io.ReadCloser, bufSz int64) *eagerReader {
	er := eagerReader{
		closeNotify: make(chan struct{}),
		rc:          r,
		buf:         make([]byte, bufSz, bufSz),
	}

	er.more = sync.NewCond(new(sync.Mutex))

	go er.buffer()

	return &er
}

func (er *eagerReader) buffer() {
	for er.lastErr == nil && er.end != len(er.buf) {
		var n int

		er.more.L.Lock()
		n, er.lastErr = er.rc.Read(er.buf[er.end:])
		er.end += n

		er.more.Broadcast()
		er.more.L.Unlock()
	}
}

func (er *eagerReader) writeOnce(dst io.Writer) (int64, error) {
	// Make one attempt at writing bytes from the buffer to the
	// destination.
	//
	// It may be necessary to wait for more bytes to arrive.
	er.more.L.Lock()
	defer er.more.L.Unlock()

	for er.begin == er.end {
		if er.lastErr != nil {
			return 0, er.lastErr
		}

		if er.begin == len(er.buf) {
			return 0, io.EOF
		}

		er.more.Wait()
	}

	n, err := dst.Write(er.buf[er.begin:er.end])
	er.begin += n
	return int64(n), err
}

func (er *eagerReader) WriteTo(dst io.Writer) (int64, error) {
	var written int64

	for {
		n, err := er.writeOnce(dst)
		written += n
		switch err {
		case io.EOF:
			// Finished.
			//
			// The EOF originates from the Read half of
			// the eagerReader, and it's not desirable
			// emit that to the caller of WriteTo: it's
			// assumed that a nil error and a return means
			// that all bytes have been written.
			return 0, nil
		case nil:
			// More bytes to be written still.
			continue
		default:
			// Error encountered, stop execution.
			return written, err
		}
	}
}

func (er *eagerReader) Close() error {
	err := er.rc.Close()
	er.closeNotify <- struct{}{}
	return err
}

func (er *eagerReader) WaitClosed() {
	<-er.closeNotify
}