File 0006-CVE-2025-22869-ssh-limit-the-size-of-the-internal-pa.patch of Package podman
xxxxxxxxxx
1
From af4068e561bad90e5b138a2eb79e8f238afb7ce4 Mon Sep 17 00:00:00 2001
2
From: Danish Prakash <contact@danishpraka.sh>
3
Date: Thu, 13 Mar 2025 14:42:02 +0530
4
Subject: [PATCH 6/6] CVE-2025-22869: ssh: limit the size of the internal
5
packet queue while waiting for KEX (#12)
6
7
In the SSH protocol, clients and servers execute the key exchange to
8
generate one-time session keys used for encryption and authentication.
9
The key exchange is performed initially after the connection is
10
established and then periodically after a configurable amount of data.
11
While a key exchange is in progress, we add the received packets to an
12
internal queue until we receive SSH_MSG_KEXINIT from the other side.
13
This can result in high memory usage if the other party is slow to
14
respond to the SSH_MSG_KEXINIT packet, or memory exhaustion if a
15
malicious client never responds to an SSH_MSG_KEXINIT packet during a
16
large file transfer.
17
We now limit the internal queue to 64 packets: this means 2MB with the
18
typical 32KB packet size.
19
When the internal queue is full we block further writes until the
20
pending key exchange is completed or there is a read or write error.
21
22
Thanks to Yuichi Watanabe for reporting this issue.
23
24
Fixes: CVE-2025-22869
25
Bugs: bsc#1239330
26
27
Signed-off-by: Danish Prakash <contact@danishpraka.sh>
28
Co-authored-by: Nicola Murino <nicola.murino@gmail.com>
29
---
30
vendor/golang.org/x/crypto/ssh/handshake.go | 47 ++++++++++++++++-----
31
1 file changed, 37 insertions(+), 10 deletions(-)
32
33
diff --git a/vendor/golang.org/x/crypto/ssh/handshake.go b/vendor/golang.org/x/crypto/ssh/handshake.go
34
index 56cdc7c21c3b..a68d20f7f396 100644
35
--- a/vendor/golang.org/x/crypto/ssh/handshake.go
36
+++ b/vendor/golang.org/x/crypto/ssh/handshake.go
37
38
// quickly.
39
const chanSize = 16
40
41
+// maxPendingPackets sets the maximum number of packets to queue while waiting
42
+// for KEX to complete. This limits the total pending data to maxPendingPackets
43
+// * maxPacket bytes, which is ~16.8MB.
44
+const maxPendingPackets = 64
45
+
46
// keyingTransport is a packet based transport that supports key
47
// changes. It need not be thread-safe. It should pass through
48
// msgNewKeys in both directions.
49
50
incoming chan []byte
51
readError error
52
53
- mu sync.Mutex
54
- writeError error
55
- sentInitPacket []byte
56
- sentInitMsg *kexInitMsg
57
- pendingPackets [][]byte // Used when a key exchange is in progress.
58
+ mu sync.Mutex
59
+ // Condition for the above mutex. It is used to notify a completed key
60
+ // exchange or a write failure. Writes can wait for this condition while a
61
+ // key exchange is in progress.
62
+ writeCond *sync.Cond
63
+ writeError error
64
+ sentInitPacket []byte
65
+ sentInitMsg *kexInitMsg
66
+ // Used to queue writes when a key exchange is in progress. The length is
67
+ // limited by pendingPacketsSize. Once full, writes will block until the key
68
+ // exchange is completed or an error occurs. If not empty, it is emptied
69
+ // all at once when the key exchange is completed in kexLoop.
70
+ pendingPackets [][]byte
71
writePacketsLeft uint32
72
writeBytesLeft int64
73
74
75
76
config: config,
77
}
78
+ t.writeCond = sync.NewCond(&t.mu)
79
t.resetReadThresholds()
80
t.resetWriteThresholds()
81
82
83
defer t.mu.Unlock()
84
if t.writeError == nil && err != nil {
85
t.writeError = err
86
+ t.writeCond.Broadcast()
87
}
88
}
89
90
91
}
92
}
93
t.pendingPackets = t.pendingPackets[:0]
94
+ // Unblock writePacket if waiting for KEX.
95
+ t.writeCond.Broadcast()
96
t.mu.Unlock()
97
}
98
99
100
}
101
102
if t.sentInitMsg != nil {
103
- // Copy the packet so the writer can reuse the buffer.
104
- cp := make([]byte, len(p))
105
- copy(cp, p)
106
- t.pendingPackets = append(t.pendingPackets, cp)
107
- return nil
108
+ if len(t.pendingPackets) < maxPendingPackets {
109
+ // Copy the packet so the writer can reuse the buffer.
110
+ cp := make([]byte, len(p))
111
+ copy(cp, p)
112
+ t.pendingPackets = append(t.pendingPackets, cp)
113
+ return nil
114
+ }
115
+ for t.sentInitMsg != nil {
116
+ // Block and wait for KEX to complete or an error.
117
+ t.writeCond.Wait()
118
+ if t.writeError != nil {
119
+ return t.writeError
120
+ }
121
+ }
122
}
123
124
if t.writeBytesLeft > 0 {
125
126
127
if err := t.pushPacket(p); err != nil {
128
t.writeError = err
129
+ t.writeCond.Broadcast()
130
}
131
132
return nil
133
--
134
2.46.0
135
136