Skip to content

Commit

Permalink
ssh: rationalize rekeying decisions.
Browse files Browse the repository at this point in the history
1) Always force a key exchange if we exchange 2^31 packets. In the past
this might not happen if RekeyThreshold was set to a very large
interval.

2) Follow recommendations from RFC 4344 for block ciphers. For AES, we
can encrypt 2^(blocksize/4) blocks under the same keys.

On modern hardware, the previous default of 1Gb could force a key
exchange within ~10 seconds. Since the key exchange takes 3 roundtrips
(send kex init, send DH init, send NEW_KEYS), this is relatively
expensive on high-latency links.

Change-Id: I1297124a307c541b7bf22d814d136ec0c6d8ed97
Reviewed-on: https://go-review.googlesource.com/35410
Run-TryBot: Han-Wen Nienhuys <[email protected]>
TryBot-Result: Gobot Gobot <[email protected]>
Reviewed-by: Adam Langley <[email protected]>
  • Loading branch information
hanwen committed Jan 30, 2017
1 parent 6fb0668 commit a59c127
Show file tree
Hide file tree
Showing 2 changed files with 70 additions and 17 deletions.
15 changes: 15 additions & 0 deletions ssh/common.go
Original file line number Diff line number Diff line change
Expand Up @@ -104,6 +104,21 @@ type directionAlgorithms struct {
Compression string
}

// rekeyBytes returns a rekeying intervals in bytes.
func (a *directionAlgorithms) rekeyBytes() int64 {
// According to RFC4344 block ciphers should rekey after
// 2^(BLOCKSIZE/4) blocks. For all AES flavors BLOCKSIZE is
// 128.
switch a.Cipher {
case "aes128-ctr", "aes192-ctr", "aes256-ctr", gcmCipherID, aes128cbcID:
return 16 * (1 << 32)

}

// For others, stick with RFC4253 recommendation to rekey after 1 Gb of data.
return 1 << 30
}

type algorithms struct {
kex string
hostKey string
Expand Down
72 changes: 55 additions & 17 deletions ssh/handshake.go
Original file line number Diff line number Diff line change
Expand Up @@ -78,9 +78,14 @@ type handshakeTransport struct {
dialAddress string
remoteAddr net.Addr

readSinceKex uint64
// Algorithms agreed in the last key exchange.
algorithms *algorithms

writtenSinceKex uint64
readPacketsLeft uint32
readBytesLeft int64

writePacketsLeft uint32
writeBytesLeft int64

// The session ID or nil if first kex did not complete yet.
sessionID []byte
Expand Down Expand Up @@ -290,7 +295,12 @@ write:
t.writeError = err
t.sentInitPacket = nil
t.sentInitMsg = nil
t.writtenSinceKex = 0
t.writePacketsLeft = packetRekeyThreshold
if t.config.RekeyThreshold > 0 {
t.writeBytesLeft = int64(t.config.RekeyThreshold)
} else if t.algorithms != nil {
t.writeBytesLeft = t.algorithms.w.rekeyBytes()
}
request.done <- t.writeError

// kex finished. Push packets that we received while
Expand Down Expand Up @@ -320,17 +330,31 @@ write:
t.conn.Close()
}

func (t *handshakeTransport) readOnePacket(first bool) ([]byte, error) {
if t.readSinceKex > t.config.RekeyThreshold {
t.requestKeyExchange()
}
// The protocol uses uint32 for packet counters, so we can't let them
// reach 1<<32. We will actually read and write more packets than
// this, though: the other side may send more packets, and after we
// hit this limit on writing we will send a few more packets for the
// key exchange itself.
const packetRekeyThreshold = (1 << 31)

func (t *handshakeTransport) readOnePacket(first bool) ([]byte, error) {
p, err := t.conn.readPacket()
if err != nil {
return nil, err
}

t.readSinceKex += uint64(len(p))
if t.readPacketsLeft > 0 {
t.readPacketsLeft--
} else {
t.requestKeyExchange()
}

if t.readBytesLeft > 0 {
t.readBytesLeft -= int64(len(p))
} else {
t.requestKeyExchange()
}

if debugHandshake {
t.printPacket(p, false)
}
Expand Down Expand Up @@ -360,7 +384,12 @@ func (t *handshakeTransport) readOnePacket(first bool) ([]byte, error) {
return nil, err
}

t.readSinceKex = 0
t.readPacketsLeft = packetRekeyThreshold
if t.config.RekeyThreshold > 0 {
t.readBytesLeft = int64(t.config.RekeyThreshold)
} else {
t.readBytesLeft = t.algorithms.r.rekeyBytes()
}

// By default, a key exchange is hidden from higher layers by
// translating it into msgIgnore.
Expand Down Expand Up @@ -443,8 +472,16 @@ func (t *handshakeTransport) writePacket(p []byte) error {
t.pendingPackets = append(t.pendingPackets, cp)
return nil
}
t.writtenSinceKex += uint64(len(p))
if t.writtenSinceKex > t.config.RekeyThreshold {

if t.writeBytesLeft > 0 {
t.writeBytesLeft -= int64(len(p))
} else {
t.requestKeyExchange()
}

if t.writePacketsLeft > 0 {
t.writePacketsLeft--
} else {
t.requestKeyExchange()
}

Expand Down Expand Up @@ -485,7 +522,8 @@ func (t *handshakeTransport) enterKeyExchange(otherInitPacket []byte) error {
magics.serverKexInit = otherInitPacket
}

algs, err := findAgreedAlgorithms(clientInit, serverInit)
var err error
t.algorithms, err = findAgreedAlgorithms(clientInit, serverInit)
if err != nil {
return err
}
Expand All @@ -508,16 +546,16 @@ func (t *handshakeTransport) enterKeyExchange(otherInitPacket []byte) error {
}
}

kex, ok := kexAlgoMap[algs.kex]
kex, ok := kexAlgoMap[t.algorithms.kex]
if !ok {
return fmt.Errorf("ssh: unexpected key exchange algorithm %v", algs.kex)
return fmt.Errorf("ssh: unexpected key exchange algorithm %v", t.algorithms.kex)
}

var result *kexResult
if len(t.hostKeys) > 0 {
result, err = t.server(kex, algs, &magics)
result, err = t.server(kex, t.algorithms, &magics)
} else {
result, err = t.client(kex, algs, &magics)
result, err = t.client(kex, t.algorithms, &magics)
}

if err != nil {
Expand All @@ -529,7 +567,7 @@ func (t *handshakeTransport) enterKeyExchange(otherInitPacket []byte) error {
}
result.SessionID = t.sessionID

t.conn.prepareKeyChange(algs, result)
t.conn.prepareKeyChange(t.algorithms, result)
if err = t.conn.writePacket([]byte{msgNewKeys}); err != nil {
return err
}
Expand Down

4 comments on commit a59c127

@ianvernon
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This commit broke SSH for me, and I had to revert to the previous commit: 6fb0668.

We consistently hit issues like the following: connect error: "ssh: handshake failed: ssh: unexpected message type 3 (expected one of [6])" with this commit, but not with the prior commit.

@mboersma
Copy link

@mboersma mboersma commented on a59c127 Feb 2, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We are seeing similar SSH failures in deis/builder as of this commit. Same with current golang/x/crypto tip. Using the previous commit 6fb0668 works.

Please consider reverting this commit. It introduces problematic behavior that apparently isn't caught by unit tests for this package.

@ianvernon
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@hanwen do you have any update on this? We would like to use the latest version if possible, but are stuck using commit 6fb0668 .

@hanwen
Copy link
Contributor Author

@hanwen hanwen commented on a59c127 Feb 13, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can you try with tip? I fixed all bugs that I know of.

Please sign in to comment.