You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
121 lines
3.3 KiB
Go
121 lines
3.3 KiB
Go
// Copyright 2014 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.
|
|
|
|
// Flow control
|
|
|
|
package http2
|
|
|
|
// inflowMinRefresh is the minimum number of bytes we'll send for a
|
|
// flow control window update.
|
|
const inflowMinRefresh = 4 << 10
|
|
|
|
// inflow accounts for an inbound flow control window.
|
|
// It tracks both the latest window sent to the peer (used for enforcement)
|
|
// and the accumulated unsent window.
|
|
type inflow struct {
|
|
avail int32
|
|
unsent int32
|
|
}
|
|
|
|
// set sets the initial window.
|
|
func (f *inflow) init(n int32) {
|
|
f.avail = n
|
|
}
|
|
|
|
// add adds n bytes to the window, with a maximum window size of max,
|
|
// indicating that the peer can now send us more data.
|
|
// For example, the user read from a {Request,Response} body and consumed
|
|
// some of the buffered data, so the peer can now send more.
|
|
// It returns the number of bytes to send in a WINDOW_UPDATE frame to the peer.
|
|
// Window updates are accumulated and sent when the unsent capacity
|
|
// is at least inflowMinRefresh or will at least double the peer's available window.
|
|
func (f *inflow) add(n int) (connAdd int32) {
|
|
if n < 0 {
|
|
panic("negative update")
|
|
}
|
|
unsent := int64(f.unsent) + int64(n)
|
|
// "A sender MUST NOT allow a flow-control window to exceed 2^31-1 octets."
|
|
// RFC 7540 Section 6.9.1.
|
|
const maxWindow = 1<<31 - 1
|
|
if unsent+int64(f.avail) > maxWindow {
|
|
panic("flow control update exceeds maximum window size")
|
|
}
|
|
f.unsent = int32(unsent)
|
|
if f.unsent < inflowMinRefresh && f.unsent < f.avail {
|
|
// If there aren't at least inflowMinRefresh bytes of window to send,
|
|
// and this update won't at least double the window, buffer the update for later.
|
|
return 0
|
|
}
|
|
f.avail += f.unsent
|
|
f.unsent = 0
|
|
return int32(unsent)
|
|
}
|
|
|
|
// take attempts to take n bytes from the peer's flow control window.
|
|
// It reports whether the window has available capacity.
|
|
func (f *inflow) take(n uint32) bool {
|
|
if n > uint32(f.avail) {
|
|
return false
|
|
}
|
|
f.avail -= int32(n)
|
|
return true
|
|
}
|
|
|
|
// takeInflows attempts to take n bytes from two inflows,
|
|
// typically connection-level and stream-level flows.
|
|
// It reports whether both windows have available capacity.
|
|
func takeInflows(f1, f2 *inflow, n uint32) bool {
|
|
if n > uint32(f1.avail) || n > uint32(f2.avail) {
|
|
return false
|
|
}
|
|
f1.avail -= int32(n)
|
|
f2.avail -= int32(n)
|
|
return true
|
|
}
|
|
|
|
// outflow is the outbound flow control window's size.
|
|
type outflow struct {
|
|
_ incomparable
|
|
|
|
// n is the number of DATA bytes we're allowed to send.
|
|
// An outflow is kept both on a conn and a per-stream.
|
|
n int32
|
|
|
|
// conn points to the shared connection-level outflow that is
|
|
// shared by all streams on that conn. It is nil for the outflow
|
|
// that's on the conn directly.
|
|
conn *outflow
|
|
}
|
|
|
|
func (f *outflow) setConnFlow(cf *outflow) { f.conn = cf }
|
|
|
|
func (f *outflow) available() int32 {
|
|
n := f.n
|
|
if f.conn != nil && f.conn.n < n {
|
|
n = f.conn.n
|
|
}
|
|
return n
|
|
}
|
|
|
|
func (f *outflow) take(n int32) {
|
|
if n > f.available() {
|
|
panic("internal error: took too much")
|
|
}
|
|
f.n -= n
|
|
if f.conn != nil {
|
|
f.conn.n -= n
|
|
}
|
|
}
|
|
|
|
// add adds n bytes (positive or negative) to the flow control window.
|
|
// It returns false if the sum would exceed 2^31-1.
|
|
func (f *outflow) add(n int32) bool {
|
|
sum := f.n + n
|
|
if (sum > n) == (f.n > 0) {
|
|
f.n = sum
|
|
return true
|
|
}
|
|
return false
|
|
}
|