gvisor.dev/gvisor@v0.0.0-20240520182842-f9d4d51c7e0f/test/packetimpact/tests/tcp_rack_test.go (about) 1 // Copyright 2020 The gVisor Authors. 2 // 3 // Licensed under the Apache License, Version 2.0 (the "License"); 4 // you may not use this file except in compliance with the License. 5 // You may obtain a copy of the License at 6 // 7 // http://www.apache.org/licenses/LICENSE-2.0 8 // 9 // Unless required by applicable law or agreed to in writing, software 10 // distributed under the License is distributed on an "AS IS" BASIS, 11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 // See the License for the specific language governing permissions and 13 // limitations under the License. 14 15 package tcp_rack_test 16 17 import ( 18 "flag" 19 "testing" 20 "time" 21 22 "golang.org/x/sys/unix" 23 "gvisor.dev/gvisor/pkg/abi/linux" 24 "gvisor.dev/gvisor/pkg/tcpip/header" 25 "gvisor.dev/gvisor/pkg/tcpip/seqnum" 26 "gvisor.dev/gvisor/test/packetimpact/testbench" 27 ) 28 29 func init() { 30 testbench.Initialize(flag.CommandLine) 31 } 32 33 const ( 34 // payloadSize is the size used to send packets. 35 payloadSize = header.TCPDefaultMSS 36 37 // simulatedRTT is the time delay between packets sent and acked to 38 // increase the RTT. 39 simulatedRTT = 30 * time.Millisecond 40 41 // numPktsForRTT is the number of packets sent and acked to establish 42 // RTT. 43 numPktsForRTT = 10 44 ) 45 46 func createSACKConnection(t *testing.T) (testbench.DUT, testbench.TCPIPv4, int32, int32) { 47 dut := testbench.NewDUT(t) 48 listenFd, remotePort := dut.CreateListener(t, unix.SOCK_STREAM, unix.IPPROTO_TCP, 1) 49 conn := dut.Net.NewTCPIPv4(t, testbench.TCP{DstPort: &remotePort}, testbench.TCP{SrcPort: &remotePort}) 50 51 // Enable SACK. 52 opts := make([]byte, 40) 53 optsOff := 0 54 optsOff += header.EncodeNOP(opts[optsOff:]) 55 optsOff += header.EncodeNOP(opts[optsOff:]) 56 optsOff += header.EncodeSACKPermittedOption(opts[optsOff:]) 57 58 conn.ConnectWithOptions(t, opts[:optsOff]) 59 acceptFd, _ := dut.Accept(t, listenFd) 60 return dut, conn, acceptFd, listenFd 61 } 62 63 func closeSACKConnection(t *testing.T, dut testbench.DUT, conn testbench.TCPIPv4, acceptFd, listenFd int32) { 64 dut.Close(t, acceptFd) 65 dut.Close(t, listenFd) 66 conn.Close(t) 67 } 68 69 func getRTTAndRTO(t *testing.T, dut testbench.DUT, acceptFd int32) (rtt, rto time.Duration) { 70 info := dut.GetSockOptTCPInfo(t, acceptFd) 71 return time.Duration(info.RTT) * time.Microsecond, time.Duration(info.RTO) * time.Microsecond 72 } 73 74 func sendAndReceive(t *testing.T, dut testbench.DUT, conn testbench.TCPIPv4, numPkts int, acceptFd int32, sendACK bool) time.Time { 75 seqNum1 := *conn.RemoteSeqNum(t) 76 payload := make([]byte, payloadSize) 77 var lastSent time.Time 78 for i, sn := 0, seqNum1; i < numPkts; i++ { 79 lastSent = time.Now() 80 dut.Send(t, acceptFd, payload, 0) 81 gotOne, err := conn.Expect(t, testbench.TCP{SeqNum: testbench.Uint32(uint32(sn))}, time.Second) 82 if err != nil { 83 t.Fatalf("Expect #%d: %s", i+1, err) 84 continue 85 } 86 if gotOne == nil { 87 t.Fatalf("#%d: expected a packet within a second but got none", i+1) 88 } 89 sn.UpdateForward(seqnum.Size(payloadSize)) 90 91 if sendACK { 92 time.Sleep(simulatedRTT) 93 conn.Send(t, testbench.TCP{Flags: testbench.TCPFlags(header.TCPFlagAck), AckNum: testbench.Uint32(uint32(sn))}) 94 } 95 } 96 return lastSent 97 } 98 99 // TestRACKTLPAllPacketsLost tests TLP when an entire flight of data is lost. 100 func TestRACKTLPAllPacketsLost(t *testing.T) { 101 dut, conn, acceptFd, listenFd := createSACKConnection(t) 102 seqNum1 := *conn.RemoteSeqNum(t) 103 104 // Send ACK for data packets to establish RTT. 105 sendAndReceive(t, dut, conn, numPktsForRTT, acceptFd, true /* sendACK */) 106 seqNum1.UpdateForward(seqnum.Size(numPktsForRTT * payloadSize)) 107 108 // We are not sending ACK for these packets. 109 const numPkts = 5 110 lastSent := sendAndReceive(t, dut, conn, numPkts, acceptFd, false /* sendACK */) 111 112 // Probe Timeout (PTO) should be two times RTT. Check that the last 113 // packet is retransmitted after probe timeout. 114 rtt, _ := getRTTAndRTO(t, dut, acceptFd) 115 pto := rtt * 2 116 // We expect the 5th packet (the last unacknowledged packet) to be 117 // retransmitted. 118 tlpProbe := testbench.Uint32(uint32(seqNum1) + uint32((numPkts-1)*payloadSize)) 119 if _, err := conn.Expect(t, testbench.TCP{SeqNum: tlpProbe}, time.Second); err != nil { 120 t.Fatalf("expected payload was not received: %s %v %v", err, rtt, pto) 121 } 122 diff := time.Now().Sub(lastSent) 123 if diff < pto { 124 t.Fatalf("expected payload was received before the probe timeout, got: %v, want: %v", diff, pto) 125 } 126 closeSACKConnection(t, dut, conn, acceptFd, listenFd) 127 } 128 129 // TestRACKTLPLost tests TLP when there are tail losses. 130 // See: https://tools.ietf.org/html/draft-ietf-tcpm-rack-08#section-7.4 131 func TestRACKTLPLost(t *testing.T) { 132 dut, conn, acceptFd, listenFd := createSACKConnection(t) 133 seqNum1 := *conn.RemoteSeqNum(t) 134 135 // Send ACK for data packets to establish RTT. 136 sendAndReceive(t, dut, conn, numPktsForRTT, acceptFd, true /* sendACK */) 137 seqNum1.UpdateForward(seqnum.Size(numPktsForRTT * payloadSize)) 138 139 // We are not sending ACK for these packets. 140 const numPkts = 10 141 lastSent := sendAndReceive(t, dut, conn, numPkts, acceptFd, false /* sendACK */) 142 143 // Cumulative ACK for #[1-5] packets. 144 ackNum := seqNum1.Add(seqnum.Size(6 * payloadSize)) 145 conn.Send(t, testbench.TCP{Flags: testbench.TCPFlags(header.TCPFlagAck), AckNum: testbench.Uint32(uint32(ackNum))}) 146 147 // Probe Timeout (PTO) should be two times RTT. Check that the last 148 // packet is retransmitted after probe timeout. 149 rtt, _ := getRTTAndRTO(t, dut, acceptFd) 150 pto := rtt * 2 151 // We expect the 10th packet (the last unacknowledged packet) to be 152 // retransmitted. 153 tlpProbe := testbench.Uint32(uint32(seqNum1) + uint32((numPkts-1)*payloadSize)) 154 if _, err := conn.Expect(t, testbench.TCP{SeqNum: tlpProbe}, time.Second); err != nil { 155 t.Fatalf("expected payload was not received: %s", err) 156 } 157 diff := time.Now().Sub(lastSent) 158 if diff < pto { 159 t.Fatalf("expected payload was received before the probe timeout, got: %v, want: %v", diff, pto) 160 } 161 closeSACKConnection(t, dut, conn, acceptFd, listenFd) 162 } 163 164 // TestRACKWithSACK tests that RACK marks the packets as lost after receiving 165 // the ACK for retransmitted packets. 166 // See: https://tools.ietf.org/html/draft-ietf-tcpm-rack-08#section-8.1 167 func TestRACKWithSACK(t *testing.T) { 168 dut, conn, acceptFd, listenFd := createSACKConnection(t) 169 seqNum1 := *conn.RemoteSeqNum(t) 170 171 // Send ACK for data packets to establish RTT. 172 sendAndReceive(t, dut, conn, numPktsForRTT, acceptFd, true /* sendACK */) 173 seqNum1.UpdateForward(seqnum.Size(numPktsForRTT * payloadSize)) 174 175 // We are not sending ACK for these packets. 176 const numPkts = 3 177 sendAndReceive(t, dut, conn, numPkts, acceptFd, false /* sendACK */) 178 179 time.Sleep(simulatedRTT) 180 // SACK for #2 packet. 181 sackBlock := make([]byte, 40) 182 start := seqNum1.Add(seqnum.Size(payloadSize)) 183 end := start.Add(seqnum.Size(payloadSize)) 184 sbOff := 0 185 sbOff += header.EncodeNOP(sackBlock[sbOff:]) 186 sbOff += header.EncodeNOP(sackBlock[sbOff:]) 187 sbOff += header.EncodeSACKBlocks([]header.SACKBlock{{ 188 start, end, 189 }}, sackBlock[sbOff:]) 190 conn.Send(t, testbench.TCP{Flags: testbench.TCPFlags(header.TCPFlagAck), AckNum: testbench.Uint32(uint32(seqNum1)), Options: sackBlock[:sbOff]}) 191 192 rtt, _ := getRTTAndRTO(t, dut, acceptFd) 193 timeout := 2 * rtt 194 // RACK marks #1 packet as lost after RTT+reorderWindow(RTT/4) and 195 // retransmits it. 196 if _, err := conn.Expect(t, testbench.TCP{SeqNum: testbench.Uint32(uint32(seqNum1))}, timeout); err != nil { 197 t.Fatalf("expected payload was not received: %s", err) 198 } 199 200 time.Sleep(simulatedRTT) 201 // ACK for #1 packet. 202 conn.Send(t, testbench.TCP{Flags: testbench.TCPFlags(header.TCPFlagAck), AckNum: testbench.Uint32(uint32(end))}) 203 204 // RACK considers transmission times of the packets to mark them lost. 205 // As the 3rd packet was sent before the retransmitted 1st packet, RACK 206 // marks it as lost and retransmits it.. 207 expectedSeqNum := testbench.Uint32(uint32(seqNum1) + uint32((numPkts-1)*payloadSize)) 208 if _, err := conn.Expect(t, testbench.TCP{SeqNum: expectedSeqNum}, timeout); err != nil { 209 t.Fatalf("expected payload was not received: %s", err) 210 } 211 closeSACKConnection(t, dut, conn, acceptFd, listenFd) 212 } 213 214 // TestRACKWithoutReorder tests that without reordering RACK will retransmit the 215 // lost packets after reorder timer expires. 216 func TestRACKWithoutReorder(t *testing.T) { 217 dut, conn, acceptFd, listenFd := createSACKConnection(t) 218 seqNum1 := *conn.RemoteSeqNum(t) 219 220 // Send ACK for data packets to establish RTT. 221 sendAndReceive(t, dut, conn, numPktsForRTT, acceptFd, true /* sendACK */) 222 seqNum1.UpdateForward(seqnum.Size(numPktsForRTT * payloadSize)) 223 224 // We are not sending ACK for these packets. 225 const numPkts = 4 226 sendAndReceive(t, dut, conn, numPkts, acceptFd, false /* sendACK */) 227 228 // SACK for [3,4] packets. 229 sackBlock := make([]byte, 40) 230 start := seqNum1.Add(seqnum.Size(2 * payloadSize)) 231 end := start.Add(seqnum.Size(2 * payloadSize)) 232 sbOff := 0 233 sbOff += header.EncodeNOP(sackBlock[sbOff:]) 234 sbOff += header.EncodeNOP(sackBlock[sbOff:]) 235 sbOff += header.EncodeSACKBlocks([]header.SACKBlock{{ 236 start, end, 237 }}, sackBlock[sbOff:]) 238 time.Sleep(simulatedRTT) 239 conn.Send(t, testbench.TCP{Flags: testbench.TCPFlags(header.TCPFlagAck), AckNum: testbench.Uint32(uint32(seqNum1)), Options: sackBlock[:sbOff]}) 240 241 // RACK marks #1 and #2 packets as lost and retransmits both after 242 // RTT + reorderWindow. The reorderWindow initially will be a small 243 // fraction of RTT. 244 rtt, _ := getRTTAndRTO(t, dut, acceptFd) 245 timeout := 2 * rtt 246 for i, sn := 0, seqNum1; i < 2; i++ { 247 if _, err := conn.Expect(t, testbench.TCP{SeqNum: testbench.Uint32(uint32(sn))}, timeout); err != nil { 248 t.Fatalf("expected payload was not received: %s", err) 249 } 250 sn.UpdateForward(seqnum.Size(payloadSize)) 251 } 252 closeSACKConnection(t, dut, conn, acceptFd, listenFd) 253 } 254 255 // TestRACKWithReorder tests that RACK will retransmit segments when there is 256 // reordering in the connection and reorder timer expires. 257 func TestRACKWithReorder(t *testing.T) { 258 dut, conn, acceptFd, listenFd := createSACKConnection(t) 259 seqNum1 := *conn.RemoteSeqNum(t) 260 261 // Send ACK for data packets to establish RTT. 262 sendAndReceive(t, dut, conn, numPktsForRTT, acceptFd, true /* sendACK */) 263 seqNum1.UpdateForward(seqnum.Size(numPktsForRTT * payloadSize)) 264 265 // We are not sending ACK for these packets. 266 const numPkts = 4 267 sendAndReceive(t, dut, conn, numPkts, acceptFd, false /* sendACK */) 268 269 time.Sleep(simulatedRTT) 270 // SACK in reverse order for the connection to detect reorder. 271 var start seqnum.Value 272 var end seqnum.Value 273 for i := 0; i < numPkts-1; i++ { 274 sackBlock := make([]byte, 40) 275 sbOff := 0 276 start = seqNum1.Add(seqnum.Size((numPkts - i - 1) * payloadSize)) 277 end = start.Add(seqnum.Size((i + 1) * payloadSize)) 278 sackBlock = make([]byte, 40) 279 sbOff = 0 280 sbOff += header.EncodeNOP(sackBlock[sbOff:]) 281 sbOff += header.EncodeNOP(sackBlock[sbOff:]) 282 sbOff += header.EncodeSACKBlocks([]header.SACKBlock{{ 283 start, end, 284 }}, sackBlock[sbOff:]) 285 conn.Send(t, testbench.TCP{Flags: testbench.TCPFlags(header.TCPFlagAck), AckNum: testbench.Uint32(uint32(seqNum1)), Options: sackBlock[:sbOff]}) 286 } 287 288 // Send a DSACK block indicating both original and retransmitted 289 // packets are received, RACK will increase the reordering window on 290 // every DSACK. 291 dsackBlock := make([]byte, 40) 292 dbOff := 0 293 start = seqNum1 294 end = start.Add(seqnum.Size(2 * payloadSize)) 295 dbOff += header.EncodeNOP(dsackBlock[dbOff:]) 296 dbOff += header.EncodeNOP(dsackBlock[dbOff:]) 297 dbOff += header.EncodeSACKBlocks([]header.SACKBlock{{ 298 start, end, 299 }}, dsackBlock[dbOff:]) 300 conn.Send(t, testbench.TCP{Flags: testbench.TCPFlags(header.TCPFlagAck), AckNum: testbench.Uint32(uint32(seqNum1 + numPkts*payloadSize)), Options: dsackBlock[:dbOff]}) 301 302 seqNum1.UpdateForward(seqnum.Size(numPkts * payloadSize)) 303 sendTime := time.Now() 304 sendAndReceive(t, dut, conn, numPkts, acceptFd, false /* sendACK */) 305 306 time.Sleep(simulatedRTT) 307 // Send SACK for [2-5] packets. 308 sackBlock := make([]byte, 40) 309 sbOff := 0 310 start = seqNum1.Add(seqnum.Size(payloadSize)) 311 end = start.Add(seqnum.Size(3 * payloadSize)) 312 sbOff += header.EncodeNOP(sackBlock[sbOff:]) 313 sbOff += header.EncodeNOP(sackBlock[sbOff:]) 314 sbOff += header.EncodeSACKBlocks([]header.SACKBlock{{ 315 start, end, 316 }}, sackBlock[sbOff:]) 317 conn.Send(t, testbench.TCP{Flags: testbench.TCPFlags(header.TCPFlagAck), AckNum: testbench.Uint32(uint32(seqNum1)), Options: sackBlock[:sbOff]}) 318 319 // Expect the retransmission of #1 packet after RTT+ReorderWindow. 320 if _, err := conn.Expect(t, testbench.TCP{SeqNum: testbench.Uint32(uint32(seqNum1))}, time.Second); err != nil { 321 t.Fatalf("expected payload was not received: %s", err) 322 } 323 rtt, _ := getRTTAndRTO(t, dut, acceptFd) 324 diff := time.Now().Sub(sendTime) 325 if diff < rtt { 326 t.Fatalf("expected payload was received too sonn, within RTT") 327 } 328 329 closeSACKConnection(t, dut, conn, acceptFd, listenFd) 330 } 331 332 // TestRACKWithLostRetransmission tests that RACK will not enter RTO when a 333 // retransmitted segment is lost and enters fast recovery. 334 func TestRACKWithLostRetransmission(t *testing.T) { 335 dut, conn, acceptFd, listenFd := createSACKConnection(t) 336 seqNum1 := *conn.RemoteSeqNum(t) 337 338 // Send ACK for data packets to establish RTT. 339 sendAndReceive(t, dut, conn, numPktsForRTT, acceptFd, true /* sendACK */) 340 seqNum1.UpdateForward(seqnum.Size(numPktsForRTT * payloadSize)) 341 342 // We are not sending ACK for these packets. 343 const numPkts = 5 344 sendAndReceive(t, dut, conn, numPkts, acceptFd, false /* sendACK */) 345 346 // SACK for [2-5] packets. 347 sackBlock := make([]byte, 40) 348 start := seqNum1.Add(seqnum.Size(payloadSize)) 349 end := start.Add(seqnum.Size(4 * payloadSize)) 350 sbOff := 0 351 sbOff += header.EncodeNOP(sackBlock[sbOff:]) 352 sbOff += header.EncodeNOP(sackBlock[sbOff:]) 353 sbOff += header.EncodeSACKBlocks([]header.SACKBlock{{ 354 start, end, 355 }}, sackBlock[sbOff:]) 356 time.Sleep(simulatedRTT) 357 conn.Send(t, testbench.TCP{Flags: testbench.TCPFlags(header.TCPFlagAck), AckNum: testbench.Uint32(uint32(seqNum1)), Options: sackBlock[:sbOff]}) 358 359 // RACK marks #1 packet as lost and retransmits it after 360 // RTT + reorderWindow. The reorderWindow is bounded between a small 361 // fraction of RTT and 1 RTT. 362 rtt, _ := getRTTAndRTO(t, dut, acceptFd) 363 timeout := 2 * rtt 364 if _, err := conn.Expect(t, testbench.TCP{SeqNum: testbench.Uint32(uint32(seqNum1))}, timeout); err != nil { 365 t.Fatalf("expected payload was not received: %s", err) 366 } 367 368 // Send #6 packet. 369 payload := make([]byte, payloadSize) 370 dut.Send(t, acceptFd, payload, 0) 371 gotOne, err := conn.Expect(t, testbench.TCP{SeqNum: testbench.Uint32(uint32(seqNum1 + 5*payloadSize))}, time.Second) 372 if err != nil { 373 t.Fatalf("Expect #6: %s", err) 374 } 375 if gotOne == nil { 376 t.Fatalf("#6: expected a packet within a second but got none") 377 } 378 379 // SACK for [2-6] packets. 380 sackBlock1 := make([]byte, 40) 381 start = seqNum1.Add(seqnum.Size(payloadSize)) 382 end = start.Add(seqnum.Size(5 * payloadSize)) 383 sbOff1 := 0 384 sbOff1 += header.EncodeNOP(sackBlock1[sbOff1:]) 385 sbOff1 += header.EncodeNOP(sackBlock1[sbOff1:]) 386 sbOff1 += header.EncodeSACKBlocks([]header.SACKBlock{{ 387 start, end, 388 }}, sackBlock1[sbOff1:]) 389 time.Sleep(simulatedRTT) 390 conn.Send(t, testbench.TCP{Flags: testbench.TCPFlags(header.TCPFlagAck), AckNum: testbench.Uint32(uint32(seqNum1)), Options: sackBlock1[:sbOff1]}) 391 392 // Expect re-retransmission of #1 packet without entering an RTO. 393 if _, err := conn.Expect(t, testbench.TCP{SeqNum: testbench.Uint32(uint32(seqNum1))}, timeout); err != nil { 394 t.Fatalf("expected payload was not received: %s", err) 395 } 396 397 // Check the congestion control state. 398 info := dut.GetSockOptTCPInfo(t, acceptFd) 399 if info.CaState != linux.TCP_CA_Recovery { 400 t.Fatalf("expected connection to be in fast recovery, want: %v got: %v", linux.TCP_CA_Recovery, info.CaState) 401 } 402 403 closeSACKConnection(t, dut, conn, acceptFd, listenFd) 404 }