@@ -22,47 +22,30 @@ const maxCachedBufSize = 256 * 1024
22
22
// In other words, we can't write and read simultaneously on the same connection.
23
23
// The buffer is similar to bufio.Reader / Writer but zero-copy-ish
24
24
// Also highly optimized for this particular use case.
25
- // This buffer is backed by two byte slices in a double-buffering scheme
26
25
type buffer struct {
27
- buf []byte // buf is a byte buffer who's length and capacity are equal.
28
- nc net.Conn
29
- idx int
30
- length int
31
- timeout time.Duration
32
- dbuf [2 ][]byte // dbuf is an array with the two byte slices that back this buffer
33
- flipcnt uint // flipccnt is the current buffer counter for double-buffering
26
+ buf []byte // read buffer.
27
+ cachedBuf []byte // buffer that will be reused. len(cachedBuf) <= maxCachedBufSize.
28
+ nc net.Conn
29
+ timeout time.Duration
34
30
}
35
31
36
32
// newBuffer allocates and returns a new buffer.
37
33
func newBuffer (nc net.Conn ) buffer {
38
- fg := make ([]byte , defaultBufSize )
39
34
return buffer {
40
- buf : fg ,
41
- nc : nc ,
42
- dbuf : [2 ][]byte {fg , nil },
35
+ cachedBuf : make ([]byte , defaultBufSize ),
36
+ nc : nc ,
43
37
}
44
38
}
45
39
46
- // busy returns true if the buffer contains some read data .
40
+ // busy returns true if the read buffer is not empty .
47
41
func (b * buffer ) busy () bool {
48
- return b . length > 0
42
+ return len ( b . buf ) > 0
49
43
}
50
44
51
- // flip replaces the active buffer with the background buffer
52
- // this is a delayed flip that simply increases the buffer counter;
53
- // the actual flip will be performed the next time we call `buffer.fill`
54
- func (b * buffer ) flip () {
55
- b .flipcnt += 1
56
- }
57
-
58
- // fill reads into the buffer until at least _need_ bytes are in it
45
+ // fill reads into the read buffer until at least _need_ bytes are in it.
59
46
func (b * buffer ) fill (need int ) error {
60
- n := b .length
61
- // fill data into its double-buffering target: if we've called
62
- // flip on this buffer, we'll be copying to the background buffer,
63
- // and then filling it with network data; otherwise we'll just move
64
- // the contents of the current buffer to the front before filling it
65
- dest := b .dbuf [b .flipcnt & 1 ]
47
+ // we'll move the contents of the current buffer to dest before filling it.
48
+ dest := b .cachedBuf
66
49
67
50
// grow buffer if necessary to fit the whole packet.
68
51
if need > len (dest ) {
@@ -72,18 +55,13 @@ func (b *buffer) fill(need int) error {
72
55
// if the allocated buffer is not too large, move it to backing storage
73
56
// to prevent extra allocations on applications that perform large reads
74
57
if len (dest ) <= maxCachedBufSize {
75
- b .dbuf [ b . flipcnt & 1 ] = dest
58
+ b .cachedBuf = dest
76
59
}
77
60
}
78
61
79
- // if we're filling the fg buffer, move the existing data to the start of it.
80
- // if we're filling the bg buffer, copy over the data
81
- if n > 0 {
82
- copy (dest [:n ], b .buf [b .idx :])
83
- }
84
-
85
- b .buf = dest
86
- b .idx = 0
62
+ // move the existing data to the start of the buffer.
63
+ n := len (b .buf )
64
+ copy (dest [:n ], b .buf )
87
65
88
66
for {
89
67
if b .timeout > 0 {
@@ -92,63 +70,58 @@ func (b *buffer) fill(need int) error {
92
70
}
93
71
}
94
72
95
- nn , err := b .nc .Read (b . buf [n :])
73
+ nn , err := b .nc .Read (dest [n :])
96
74
n += nn
97
75
98
- switch err {
99
- case nil :
100
- if n < need {
101
- continue
102
- }
103
- b .length = n
104
- return nil
76
+ if err == nil && n < need {
77
+ continue
78
+ }
105
79
106
- case io .EOF :
107
- if n >= need {
108
- b .length = n
109
- return nil
110
- }
111
- return io .ErrUnexpectedEOF
80
+ b .buf = dest [:n ]
112
81
113
- default :
114
- return err
82
+ if err == io .EOF {
83
+ if n < need {
84
+ err = io .ErrUnexpectedEOF
85
+ } else {
86
+ err = nil
87
+ }
115
88
}
89
+ return err
116
90
}
117
91
}
118
92
119
93
// returns next N bytes from buffer.
120
94
// The returned slice is only guaranteed to be valid until the next read
121
95
func (b * buffer ) readNext (need int ) ([]byte , error ) {
122
- if b . length < need {
96
+ if len ( b . buf ) < need {
123
97
// refill
124
98
if err := b .fill (need ); err != nil {
125
99
return nil , err
126
100
}
127
101
}
128
102
129
- offset := b .idx
130
- b .idx += need
131
- b .length -= need
132
- return b .buf [offset :b .idx ], nil
103
+ data := b .buf [:need ]
104
+ b .buf = b .buf [need :]
105
+ return data , nil
133
106
}
134
107
135
108
// takeBuffer returns a buffer with the requested size.
136
109
// If possible, a slice from the existing buffer is returned.
137
110
// Otherwise a bigger buffer is made.
138
111
// Only one buffer (total) can be used at a time.
139
112
func (b * buffer ) takeBuffer (length int ) ([]byte , error ) {
140
- if b .length > 0 {
113
+ if b .busy () {
141
114
return nil , ErrBusyBuffer
142
115
}
143
116
144
117
// test (cheap) general case first
145
- if length <= cap (b .buf ) {
146
- return b .buf [:length ], nil
118
+ if length <= len (b .cachedBuf ) {
119
+ return b .cachedBuf [:length ], nil
147
120
}
148
121
149
- if length < maxPacketSize {
150
- b .buf = make ([]byte , length )
151
- return b .buf , nil
122
+ if length < maxCachedBufSize {
123
+ b .cachedBuf = make ([]byte , length )
124
+ return b .cachedBuf , nil
152
125
}
153
126
154
127
// buffer is larger than we want to store.
@@ -159,29 +132,26 @@ func (b *buffer) takeBuffer(length int) ([]byte, error) {
159
132
// known to be smaller than defaultBufSize.
160
133
// Only one buffer (total) can be used at a time.
161
134
func (b * buffer ) takeSmallBuffer (length int ) ([]byte , error ) {
162
- if b .length > 0 {
135
+ if b .busy () {
163
136
return nil , ErrBusyBuffer
164
137
}
165
- return b .buf [:length ], nil
138
+ return b .cachedBuf [:length ], nil
166
139
}
167
140
168
141
// takeCompleteBuffer returns the complete existing buffer.
169
142
// This can be used if the necessary buffer size is unknown.
170
143
// cap and len of the returned buffer will be equal.
171
144
// Only one buffer (total) can be used at a time.
172
145
func (b * buffer ) takeCompleteBuffer () ([]byte , error ) {
173
- if b .length > 0 {
146
+ if b .busy () {
174
147
return nil , ErrBusyBuffer
175
148
}
176
- return b .buf , nil
149
+ return b .cachedBuf , nil
177
150
}
178
151
179
152
// store stores buf, an updated buffer, if its suitable to do so.
180
- func (b * buffer ) store (buf []byte ) error {
181
- if b .length > 0 {
182
- return ErrBusyBuffer
183
- } else if cap (buf ) <= maxPacketSize && cap (buf ) > cap (b .buf ) {
184
- b .buf = buf [:cap (buf )]
153
+ func (b * buffer ) store (buf []byte ) {
154
+ if cap (buf ) <= maxCachedBufSize && cap (buf ) > cap (b .cachedBuf ) {
155
+ b .cachedBuf = buf [:cap (buf )]
185
156
}
186
- return nil
187
157
}
0 commit comments