Skip to content

Commit fdfe1f8

Browse files
will-extrahopgopherbot
authored andcommitted
ssh: defer channel window adjustment
Sending a window adjustment after every read is unnecessarily chatty, especially with a series of small reads like with TTY interactions. Copy OpenSSH's logic for deferring these, which seemingly hasn't changed since 2007. Note that since channelWindowSize and c.maxIncomingPayload are currently constants here, the two checks could be combined into a single check for c.myWindow < 2 MiB - 96 KiB (with the current values of the constants). Fixes golang/go#57424. Change-Id: Ifcef5be76fcc3f0b1a6dc396096bed9c50d64f21 Reviewed-on: https://go-review.googlesource.com/c/crypto/+/459915 Reviewed-by: Nicola Murino <nicola.murino@gmail.com> Reviewed-by: Michael Knyszek <mknyszek@google.com> Run-TryBot: Nicola Murino <nicola.murino@gmail.com> Auto-Submit: Nicola Murino <nicola.murino@gmail.com> Reviewed-by: Dmitri Shuralyov <dmitshur@google.com> Commit-Queue: Nicola Murino <nicola.murino@gmail.com> TryBot-Result: Gopher Robot <gobot@golang.org>
1 parent b8ffc16 commit fdfe1f8

File tree

3 files changed

+108
-11
lines changed

3 files changed

+108
-11
lines changed

ssh/channel.go

+20-8
Original file line numberDiff line numberDiff line change
@@ -187,9 +187,11 @@ type channel struct {
187187
pending *buffer
188188
extPending *buffer
189189

190-
// windowMu protects myWindow, the flow-control window.
191-
windowMu sync.Mutex
192-
myWindow uint32
190+
// windowMu protects myWindow, the flow-control window, and myConsumed,
191+
// the number of bytes consumed since we last increased myWindow
192+
windowMu sync.Mutex
193+
myWindow uint32
194+
myConsumed uint32
193195

194196
// writeMu serializes calls to mux.conn.writePacket() and
195197
// protects sentClose and packetPool. This mutex must be
@@ -332,14 +334,24 @@ func (ch *channel) handleData(packet []byte) error {
332334
return nil
333335
}
334336

335-
func (c *channel) adjustWindow(n uint32) error {
337+
func (c *channel) adjustWindow(adj uint32) error {
336338
c.windowMu.Lock()
337-
// Since myWindow is managed on our side, and can never exceed
338-
// the initial window setting, we don't worry about overflow.
339-
c.myWindow += uint32(n)
339+
// Since myConsumed and myWindow are managed on our side, and can never
340+
// exceed the initial window setting, we don't worry about overflow.
341+
c.myConsumed += adj
342+
var sendAdj uint32
343+
if (channelWindowSize-c.myWindow > 3*c.maxIncomingPayload) ||
344+
(c.myWindow < channelWindowSize/2) {
345+
sendAdj = c.myConsumed
346+
c.myConsumed = 0
347+
c.myWindow += sendAdj
348+
}
340349
c.windowMu.Unlock()
350+
if sendAdj == 0 {
351+
return nil
352+
}
341353
return c.sendMessage(windowAdjustMsg{
342-
AdditionalBytes: uint32(n),
354+
AdditionalBytes: sendAdj,
343355
})
344356
}
345357

ssh/mempipe_test.go

+17-3
Original file line numberDiff line numberDiff line change
@@ -13,9 +13,10 @@ import (
1313
// An in-memory packetConn. It is safe to call Close and writePacket
1414
// from different goroutines.
1515
type memTransport struct {
16-
eof bool
17-
pending [][]byte
18-
write *memTransport
16+
eof bool
17+
pending [][]byte
18+
write *memTransport
19+
writeCount uint64
1920
sync.Mutex
2021
*sync.Cond
2122
}
@@ -63,9 +64,16 @@ func (t *memTransport) writePacket(p []byte) error {
6364
copy(c, p)
6465
t.write.pending = append(t.write.pending, c)
6566
t.write.Cond.Signal()
67+
t.writeCount++
6668
return nil
6769
}
6870

71+
func (t *memTransport) getWriteCount() uint64 {
72+
t.write.Lock()
73+
defer t.write.Unlock()
74+
return t.writeCount
75+
}
76+
6977
func memPipe() (a, b packetConn) {
7078
t1 := memTransport{}
7179
t2 := memTransport{}
@@ -81,6 +89,9 @@ func TestMemPipe(t *testing.T) {
8189
if err := a.writePacket([]byte{42}); err != nil {
8290
t.Fatalf("writePacket: %v", err)
8391
}
92+
if wc := a.(*memTransport).getWriteCount(); wc != 1 {
93+
t.Fatalf("got %v, want 1", wc)
94+
}
8495
if err := a.Close(); err != nil {
8596
t.Fatal("Close: ", err)
8697
}
@@ -95,6 +106,9 @@ func TestMemPipe(t *testing.T) {
95106
if err != io.EOF {
96107
t.Fatalf("got %v, %v, want EOF", p, err)
97108
}
109+
if wc := b.(*memTransport).getWriteCount(); wc != 0 {
110+
t.Fatalf("got %v, want 0", wc)
111+
}
98112
}
99113

100114
func TestDoubleClose(t *testing.T) {

ssh/mux_test.go

+71
Original file line numberDiff line numberDiff line change
@@ -182,6 +182,40 @@ func TestMuxChannelOverflow(t *testing.T) {
182182
}
183183
}
184184

185+
func TestMuxChannelReadUnblock(t *testing.T) {
186+
reader, writer, mux := channelPair(t)
187+
defer reader.Close()
188+
defer writer.Close()
189+
defer mux.Close()
190+
191+
var wg sync.WaitGroup
192+
t.Cleanup(wg.Wait)
193+
wg.Add(1)
194+
go func() {
195+
defer wg.Done()
196+
if _, err := writer.Write(make([]byte, channelWindowSize)); err != nil {
197+
t.Errorf("could not fill window: %v", err)
198+
}
199+
if _, err := writer.Write(make([]byte, 1)); err != nil {
200+
t.Errorf("Write: %v", err)
201+
}
202+
writer.Close()
203+
}()
204+
205+
writer.remoteWin.waitWriterBlocked()
206+
207+
buf := make([]byte, 32768)
208+
for {
209+
_, err := reader.Read(buf)
210+
if err == io.EOF {
211+
break
212+
}
213+
if err != nil {
214+
t.Fatalf("Read: %v", err)
215+
}
216+
}
217+
}
218+
185219
func TestMuxChannelCloseWriteUnblock(t *testing.T) {
186220
reader, writer, mux := channelPair(t)
187221
defer reader.Close()
@@ -754,6 +788,43 @@ func TestMuxMaxPacketSize(t *testing.T) {
754788
}
755789
}
756790

791+
func TestMuxChannelWindowDeferredUpdates(t *testing.T) {
792+
s, c, mux := channelPair(t)
793+
cTransport := mux.conn.(*memTransport)
794+
defer s.Close()
795+
defer c.Close()
796+
defer mux.Close()
797+
798+
var wg sync.WaitGroup
799+
t.Cleanup(wg.Wait)
800+
801+
data := make([]byte, 1024)
802+
803+
wg.Add(1)
804+
go func() {
805+
defer wg.Done()
806+
_, err := s.Write(data)
807+
if err != nil {
808+
t.Errorf("Write: %v", err)
809+
return
810+
}
811+
}()
812+
cWritesInit := cTransport.getWriteCount()
813+
buf := make([]byte, 1)
814+
for i := 0; i < len(data); i++ {
815+
n, err := c.Read(buf)
816+
if n != len(buf) || err != nil {
817+
t.Fatalf("Read: %v, %v", n, err)
818+
}
819+
}
820+
cWrites := cTransport.getWriteCount() - cWritesInit
821+
// reading 1 KiB should not cause any window updates to be sent, but allow
822+
// for some unexpected writes
823+
if cWrites > 30 {
824+
t.Fatalf("reading 1 KiB from channel caused %v writes", cWrites)
825+
}
826+
}
827+
757828
// Don't ship code with debug=true.
758829
func TestDebug(t *testing.T) {
759830
if debugMux {

0 commit comments

Comments
 (0)