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
 | |
| }
 | |
| 
 | |
| // init 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
 | |
| }
 |