Update QUIC transport parameters to draft 20

This CL updates from draft-18 to draft-19/20. The only changes are idle_timeout being in milliseconds instead of seconds, and the removal of the versions before the parameters. We've kept the versions in a custom Google-specific extension for now. This extension will eventually evolve into the IETF compatible version negotiation extension.

gfe-relnote: protected by disabled flag quic_supports_tls_handshake
PiperOrigin-RevId: 246043673
Change-Id: I35929d0f381f506e2275e639af41e840ab16b8ca
diff --git a/quic/core/crypto/transport_parameters.cc b/quic/core/crypto/transport_parameters.cc
index 84022fe..8982ae0 100644
--- a/quic/core/crypto/transport_parameters.cc
+++ b/quic/core/crypto/transport_parameters.cc
@@ -4,6 +4,7 @@
 
 #include "net/third_party/quiche/src/quic/core/crypto/transport_parameters.h"
 
+#include <cstdint>
 #include <cstring>
 
 #include "third_party/boringssl/src/include/openssl/bytestring.h"
@@ -41,6 +42,8 @@
   kPreferredAddress = 0xd,
 
   kGoogleQuicParam = 18257,  // Used for non-standard Google-specific params.
+  kGoogleQuicVersion =
+      18258,  // Used to transmit version and supported_versions.
 };
 
 namespace {
@@ -89,6 +92,8 @@
       return "preferred_address";
     case kGoogleQuicParam:
       return "google";
+    case kGoogleQuicVersion:
+      return "google-version";
   }
   return "Unknown(" + QuicTextUtils::Uint64ToString(param_id) + ")";
 }
@@ -245,7 +250,7 @@
     rv += " " + TransportParameterIdToString(kOriginalConnectionId) + " " +
           original_connection_id.ToString();
   }
-  rv += idle_timeout_seconds.ToString(/*for_use_in_list=*/true);
+  rv += idle_timeout_milliseconds.ToString(/*for_use_in_list=*/true);
   if (!stateless_reset_token.empty()) {
     rv += " " + TransportParameterIdToString(kStatelessResetToken) + " " +
           QuicTextUtils::HexEncode(
@@ -278,7 +283,7 @@
 TransportParameters::TransportParameters()
     : version(0),
       original_connection_id(EmptyQuicConnectionId()),
-      idle_timeout_seconds(kIdleTimeout),
+      idle_timeout_milliseconds(kIdleTimeout),
       max_packet_size(kMaxPacketSize,
                       kDefaultMaxPacketSizeTransportParam,
                       kMinMaxPacketSizeTransportParam,
@@ -338,8 +343,8 @@
     QUIC_BUG << "Preferred address family failure";
     return false;
   }
-  const bool ok = idle_timeout_seconds.IsValid() && max_packet_size.IsValid() &&
-                  initial_max_data.IsValid() &&
+  const bool ok = idle_timeout_milliseconds.IsValid() &&
+                  max_packet_size.IsValid() && initial_max_data.IsValid() &&
                   initial_max_stream_data_bidi_local.IsValid() &&
                   initial_max_stream_data_bidi_remote.IsValid() &&
                   initial_max_stream_data_uni.IsValid() &&
@@ -360,26 +365,19 @@
     QUIC_DLOG(ERROR) << "Not serializing invalid transport parameters " << in;
     return false;
   }
+  if (in.version == 0 || (in.perspective == Perspective::IS_SERVER &&
+                          in.supported_versions.empty())) {
+    QUIC_DLOG(ERROR) << "Refusing to serialize without versions";
+    return false;
+  }
+
   bssl::ScopedCBB cbb;
   // Empirically transport parameters generally fit within 128 bytes.
   // The CBB will grow to fit larger serializations if required.
-  if (!CBB_init(cbb.get(), 128) || !CBB_add_u32(cbb.get(), in.version)) {
-    QUIC_BUG << "Failed to write version for " << in;
+  if (!CBB_init(cbb.get(), 128)) {
+    QUIC_BUG << "Failed to initialize CBB for " << in;
     return false;
   }
-  CBB versions;
-  if (in.perspective == Perspective::IS_SERVER) {
-    if (!CBB_add_u8_length_prefixed(cbb.get(), &versions)) {
-      QUIC_BUG << "Failed to write versions length for " << in;
-      return false;
-    }
-    for (QuicVersionLabel version : in.supported_versions) {
-      if (!CBB_add_u32(&versions, version)) {
-        QUIC_BUG << "Failed to write supported version for " << in;
-        return false;
-      }
-    }
-  }
 
   CBB params;
   // Add length of the transport parameters list.
@@ -404,7 +402,7 @@
     }
   }
 
-  if (!in.idle_timeout_seconds.WriteToCbb(&params)) {
+  if (!in.idle_timeout_milliseconds.WriteToCbb(&params)) {
     QUIC_BUG << "Failed to write idle_timeout for " << in;
     return false;
   }
@@ -504,6 +502,29 @@
       return false;
     }
   }
+
+  // Google-specific version extension.
+  CBB google_version_params;
+  if (!CBB_add_u16(&params, kGoogleQuicVersion) ||
+      !CBB_add_u16_length_prefixed(&params, &google_version_params) ||
+      !CBB_add_u32(&google_version_params, in.version)) {
+    QUIC_BUG << "Failed to write Google version extension for " << in;
+    return false;
+  }
+  CBB versions;
+  if (in.perspective == Perspective::IS_SERVER) {
+    if (!CBB_add_u8_length_prefixed(&google_version_params, &versions)) {
+      QUIC_BUG << "Failed to write versions length for " << in;
+      return false;
+    }
+    for (QuicVersionLabel version : in.supported_versions) {
+      if (!CBB_add_u32(&versions, version)) {
+        QUIC_BUG << "Failed to write supported version for " << in;
+        return false;
+      }
+    }
+  }
+
   if (!CBB_flush(cbb.get())) {
     QUIC_BUG << "Failed to flush CBB for " << in;
     return false;
@@ -519,31 +540,9 @@
                               size_t in_len,
                               Perspective perspective,
                               TransportParameters* out) {
+  out->perspective = perspective;
   CBS cbs;
   CBS_init(&cbs, in, in_len);
-  if (!CBS_get_u32(&cbs, &out->version)) {
-    QUIC_DLOG(ERROR) << "Failed to parse transport parameter version";
-    return false;
-  }
-  if (perspective == Perspective::IS_SERVER) {
-    CBS versions;
-    if (!CBS_get_u8_length_prefixed(&cbs, &versions) ||
-        CBS_len(&versions) % 4 != 0) {
-      QUIC_DLOG(ERROR)
-          << "Failed to parse transport parameter supported versions";
-      return false;
-    }
-    while (CBS_len(&versions) > 0) {
-      QuicVersionLabel version;
-      if (!CBS_get_u32(&versions, &version)) {
-        QUIC_DLOG(ERROR)
-            << "Failed to parse transport parameter supported version";
-        return false;
-      }
-      out->supported_versions.push_back(version);
-    }
-  }
-  out->perspective = perspective;
 
   CBS params;
   if (!CBS_get_u16_length_prefixed(&cbs, &params)) {
@@ -552,19 +551,20 @@
   }
 
   while (CBS_len(&params) > 0) {
-    uint16_t param_id;
+    TransportParameters::TransportParameterId param_id;
     CBS value;
-    if (!CBS_get_u16(&params, &param_id)) {
+    static_assert(sizeof(param_id) == sizeof(uint16_t), "bad size");
+    if (!CBS_get_u16(&params, reinterpret_cast<uint16_t*>(&param_id))) {
       QUIC_DLOG(ERROR) << "Failed to parse transport parameter ID";
       return false;
     }
     if (!CBS_get_u16_length_prefixed(&params, &value)) {
       QUIC_DLOG(ERROR) << "Failed to parse length of transport parameter "
-                       << param_id;
+                       << TransportParameterIdToString(param_id);
       return false;
     }
     bool parse_success = true;
-    switch (static_cast<TransportParameters::TransportParameterId>(param_id)) {
+    switch (param_id) {
       case kOriginalConnectionId:
         if (!out->original_connection_id.IsEmpty()) {
           QUIC_DLOG(ERROR) << "Received a second original connection ID";
@@ -582,7 +582,7 @@
         }
         break;
       case kIdleTimeout:
-        parse_success = out->idle_timeout_seconds.ReadFromCbs(&value);
+        parse_success = out->idle_timeout_milliseconds.ReadFromCbs(&value);
         break;
       case kStatelessResetToken:
         if (!out->stateless_reset_token.empty()) {
@@ -692,7 +692,7 @@
             QuicMakeUnique<TransportParameters::PreferredAddress>(
                 preferred_address);
       } break;
-      case kGoogleQuicParam:
+      case kGoogleQuicParam: {
         if (out->google_quic_params) {
           QUIC_DLOG(ERROR) << "Received a second Google parameter";
           return false;
@@ -700,6 +700,30 @@
         QuicStringPiece serialized_params(
             reinterpret_cast<const char*>(CBS_data(&value)), CBS_len(&value));
         out->google_quic_params = CryptoFramer::ParseMessage(serialized_params);
+      } break;
+      case kGoogleQuicVersion: {
+        if (!CBS_get_u32(&value, &out->version)) {
+          QUIC_DLOG(ERROR) << "Failed to parse Google version extension";
+          return false;
+        }
+        if (perspective == Perspective::IS_SERVER) {
+          CBS versions;
+          if (!CBS_get_u8_length_prefixed(&value, &versions) ||
+              CBS_len(&versions) % 4 != 0) {
+            QUIC_DLOG(ERROR)
+                << "Failed to parse Google supported versions length";
+            return false;
+          }
+          while (CBS_len(&versions) > 0) {
+            QuicVersionLabel version;
+            if (!CBS_get_u32(&versions, &version)) {
+              QUIC_DLOG(ERROR) << "Failed to parse Google supported version";
+              return false;
+            }
+            out->supported_versions.push_back(version);
+          }
+        }
+      } break;
     }
     if (!parse_success) {
       return false;
diff --git a/quic/core/crypto/transport_parameters.h b/quic/core/crypto/transport_parameters.h
index c52b3c8..27b5fe7 100644
--- a/quic/core/crypto/transport_parameters.h
+++ b/quic/core/crypto/transport_parameters.h
@@ -20,7 +20,7 @@
 // TransportParameters contains parameters for QUIC's transport layer that are
 // exchanged during the TLS handshake. This struct is a mirror of the struct in
 // the "Transport Parameter Encoding" section of draft-ietf-quic-transport.
-// This struct currently uses the values from draft 18.
+// This struct currently uses the values from draft 20.
 struct QUIC_EXPORT_PRIVATE TransportParameters {
   // The identifier used to differentiate transport parameters.
   enum TransportParameterId : uint16_t;
@@ -119,8 +119,8 @@
   // Initial packet sent by the client.
   QuicConnectionId original_connection_id;
 
-  // Idle timeout expressed in seconds.
-  IntegerParameter idle_timeout_seconds;
+  // Idle timeout expressed in milliseconds.
+  IntegerParameter idle_timeout_milliseconds;
 
   // Stateless reset token used in verifying stateless resets.
   std::vector<uint8_t> stateless_reset_token;
diff --git a/quic/core/crypto/transport_parameters_test.cc b/quic/core/crypto/transport_parameters_test.cc
index 8deaeb3..bf8ac51 100644
--- a/quic/core/crypto/transport_parameters_test.cc
+++ b/quic/core/crypto/transport_parameters_test.cc
@@ -19,7 +19,7 @@
 const QuicVersionLabel kFakeVersionLabel = 0x01234567;
 const QuicVersionLabel kFakeVersionLabel2 = 0x89ABCDEF;
 const QuicConnectionId kFakeOriginalConnectionId = TestConnectionId(0x1337);
-const uint64_t kFakeIdleTimeout = 12;
+const uint64_t kFakeIdleTimeoutMilliseconds = 12012;
 const uint8_t kFakeStatelessResetTokenData[16] = {
     0x90, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97,
     0x98, 0x99, 0x9A, 0x9B, 0x9C, 0x9D, 0x9E, 0x9F};
@@ -82,7 +82,7 @@
   TransportParameters orig_params;
   orig_params.perspective = Perspective::IS_CLIENT;
   orig_params.version = kFakeVersionLabel;
-  orig_params.idle_timeout_seconds.set_value(kFakeIdleTimeout);
+  orig_params.idle_timeout_milliseconds.set_value(kFakeIdleTimeoutMilliseconds);
   orig_params.max_packet_size.set_value(kFakeMaxPacketSize);
   orig_params.initial_max_data.set_value(kFakeInitialMaxData);
   orig_params.initial_max_stream_data_bidi_local.set_value(
@@ -108,7 +108,8 @@
   EXPECT_EQ(kFakeVersionLabel, new_params.version);
   EXPECT_TRUE(new_params.supported_versions.empty());
   EXPECT_EQ(EmptyQuicConnectionId(), new_params.original_connection_id);
-  EXPECT_EQ(kFakeIdleTimeout, new_params.idle_timeout_seconds.value());
+  EXPECT_EQ(kFakeIdleTimeoutMilliseconds,
+            new_params.idle_timeout_milliseconds.value());
   EXPECT_TRUE(new_params.stateless_reset_token.empty());
   EXPECT_EQ(kFakeMaxPacketSize, new_params.max_packet_size.value());
   EXPECT_EQ(kFakeInitialMaxData, new_params.initial_max_data.value());
@@ -134,7 +135,7 @@
   orig_params.supported_versions.push_back(kFakeVersionLabel);
   orig_params.supported_versions.push_back(kFakeVersionLabel2);
   orig_params.original_connection_id = kFakeOriginalConnectionId;
-  orig_params.idle_timeout_seconds.set_value(kFakeIdleTimeout);
+  orig_params.idle_timeout_milliseconds.set_value(kFakeIdleTimeoutMilliseconds);
   orig_params.stateless_reset_token = kFakeStatelessResetToken;
   orig_params.max_packet_size.set_value(kFakeMaxPacketSize);
   orig_params.initial_max_data.set_value(kFakeInitialMaxData);
@@ -164,7 +165,8 @@
   EXPECT_EQ(kFakeVersionLabel, new_params.supported_versions[0]);
   EXPECT_EQ(kFakeVersionLabel2, new_params.supported_versions[1]);
   EXPECT_EQ(kFakeOriginalConnectionId, new_params.original_connection_id);
-  EXPECT_EQ(kFakeIdleTimeout, new_params.idle_timeout_seconds.value());
+  EXPECT_EQ(kFakeIdleTimeoutMilliseconds,
+            new_params.idle_timeout_milliseconds.value());
   EXPECT_EQ(kFakeStatelessResetToken, new_params.stateless_reset_token);
   EXPECT_EQ(kFakeMaxPacketSize, new_params.max_packet_size.value());
   EXPECT_EQ(kFakeInitialMaxData, new_params.initial_max_data.value());
@@ -202,7 +204,9 @@
     TransportParameters params;
     params.perspective = Perspective::IS_CLIENT;
     EXPECT_TRUE(params.AreValid());
-    params.idle_timeout_seconds.set_value(601);
+    params.idle_timeout_milliseconds.set_value(kFakeIdleTimeoutMilliseconds);
+    EXPECT_TRUE(params.AreValid());
+    params.idle_timeout_milliseconds.set_value(601000);
     EXPECT_TRUE(params.AreValid());
   }
   {
@@ -237,7 +241,7 @@
   TransportParameters orig_params;
   orig_params.perspective = Perspective::IS_CLIENT;
   orig_params.version = kFakeVersionLabel;
-  orig_params.idle_timeout_seconds.set_value(kFakeIdleTimeout);
+  orig_params.idle_timeout_milliseconds.set_value(kFakeIdleTimeoutMilliseconds);
   orig_params.stateless_reset_token = kFakeStatelessResetToken;
   orig_params.max_packet_size.set_value(kFakeMaxPacketSize);
 
@@ -248,12 +252,11 @@
 TEST_F(TransportParametersTest, ParseClientParams) {
   // clang-format off
   const uint8_t kClientParams[] = {
-      0x01, 0x23, 0x45, 0x67,  // initial version
-      0x00, 0x3B,              // length of the parameters array that follows
-      // idle_timeout_seconds
+      0x00, 0x44,              // length of the parameters array that follows
+      // idle_timeout
       0x00, 0x01,  // parameter id
-      0x00, 0x01,  // length
-      0x0c,        // value
+      0x00, 0x02,  // length
+      0x6e, 0xec,  // value
       // max_packet_size
       0x00, 0x03,  // parameter id
       0x00, 0x02,  // length
@@ -293,6 +296,10 @@
       // disable_migration
       0x00, 0x0c,  // parameter id
       0x00, 0x00,  // length
+      // Google version extension
+      0x47, 0x52,  // parameter id
+      0x00, 0x04,  // length
+      0x01, 0x23, 0x45, 0x67,  // initial version
   };
   // clang-format on
 
@@ -305,7 +312,8 @@
   EXPECT_EQ(kFakeVersionLabel, new_params.version);
   EXPECT_TRUE(new_params.supported_versions.empty());
   EXPECT_EQ(EmptyQuicConnectionId(), new_params.original_connection_id);
-  EXPECT_EQ(kFakeIdleTimeout, new_params.idle_timeout_seconds.value());
+  EXPECT_EQ(kFakeIdleTimeoutMilliseconds,
+            new_params.idle_timeout_milliseconds.value());
   EXPECT_TRUE(new_params.stateless_reset_token.empty());
   EXPECT_EQ(kFakeMaxPacketSize, new_params.max_packet_size.value());
   EXPECT_EQ(kFakeInitialMaxData, new_params.initial_max_data.value());
@@ -329,12 +337,11 @@
 
   // clang-format off
   const uint8_t kClientParamsWithFullToken[] = {
-      0x01, 0x23, 0x45, 0x67,  // initial version
-      0x00, 0x25,  // length parameters array that follows
-      // idle_timeout_seconds
+      0x00, 0x26,  // length parameters array that follows
+      // idle_timeout
       0x00, 0x01,  // parameter id
-      0x00, 0x01,  // length
-      0x0c,        // value
+      0x00, 0x02,  // length
+      0x6e, 0xec,  // value
       // stateless_reset_token
       0x00, 0x02,
       0x00, 0x10,
@@ -357,12 +364,11 @@
 
   // clang-format off
   const uint8_t kClientParamsWithEmptyToken[] = {
-      0x01, 0x23, 0x45, 0x67,  // initial version
-      0x00, 0x15,  // length parameters array that follows
-      // idle_timeout_seconds
+      0x00, 0x16,  // length parameters array that follows
+      // idle_timeout
       0x00, 0x01,  // parameter id
-      0x00, 0x01,  // length
-      0x0c,        // value
+      0x00, 0x02,  // length
+      0x6e, 0xec,  // value
       // stateless_reset_token
       0x00, 0x02,
       0x00, 0x00,
@@ -385,12 +391,11 @@
 TEST_F(TransportParametersTest, ParseClientParametersRepeated) {
   // clang-format off
   const uint8_t kClientParamsRepeated[] = {
-      0x01, 0x23, 0x45, 0x67,  // initial version
-      0x00, 0x14,  // length parameters array that follows
-      // idle_timeout_seconds
+      0x00, 0x16,  // length parameters array that follows
+      // idle_timeout
       0x00, 0x01,  // parameter id
-      0x00, 0x01,  // length
-      0x0c,        // value
+      0x00, 0x02,  // length
+      0x6e, 0xec,  // value
       // stateless_reset_token
       0x00, 0x02,
       0x00, 0x00,
@@ -398,10 +403,10 @@
       0x00, 0x03,  // parameter id
       0x00, 0x02,  // length
       0x63, 0x29,  // value
-      // idle_timeout_seconds (repeated)
+      // idle_timeout (repeated)
       0x00, 0x01,  // parameter id
-      0x00, 0x01,  // length
-      0x0c,        // value
+      0x00, 0x02,  // length
+      0x6e, 0xec,  // value
   };
   // clang-format on
   TransportParameters out_params;
@@ -413,19 +418,15 @@
 TEST_F(TransportParametersTest, ParseServerParams) {
   // clang-format off
   const uint8_t kServerParams[] = {
-      0x01, 0x23, 0x45, 0x67,  // negotiated_version
-      0x08,  // length of supported versions array
-      0x01, 0x23, 0x45, 0x67,
-      0x89, 0xab, 0xcd, 0xef,
-      0x00, 0x91,  // length of parameters array that follows
+      0x00, 0xa3,  // length of parameters array that follows
       // original_connection_id
       0x00, 0x00,  // parameter id
       0x00, 0x08,  // length
       0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x13, 0x37,
-      // idle_timeout_seconds
+      // idle_timeout
       0x00, 0x01,  // parameter id
-      0x00, 0x01,  // length
-      0x0c,        // value
+      0x00, 0x02,  // length
+      0x6e, 0xec,  // value
       // stateless_reset_token
       0x00, 0x02,
       0x00, 0x10,
@@ -482,6 +483,13 @@
       0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xBE, 0xEF,  // connection ID
       0x80, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87,  // stateless reset token
       0x88, 0x89, 0x8A, 0x8B, 0x8C, 0x8D, 0x8E, 0x8F,
+      // Google version extension
+      0x47, 0x52,  // parameter id
+      0x00, 0x0d,  // length
+      0x01, 0x23, 0x45, 0x67,  // negotiated_version
+      0x08,  // length of supported versions array
+      0x01, 0x23, 0x45, 0x67,
+      0x89, 0xab, 0xcd, 0xef,
   };
   // clang-format on
 
@@ -496,7 +504,8 @@
   EXPECT_EQ(kFakeVersionLabel, new_params.supported_versions[0]);
   EXPECT_EQ(kFakeVersionLabel2, new_params.supported_versions[1]);
   EXPECT_EQ(kFakeOriginalConnectionId, new_params.original_connection_id);
-  EXPECT_EQ(kFakeIdleTimeout, new_params.idle_timeout_seconds.value());
+  EXPECT_EQ(kFakeIdleTimeoutMilliseconds,
+            new_params.idle_timeout_milliseconds.value());
   EXPECT_EQ(kFakeStatelessResetToken, new_params.stateless_reset_token);
   EXPECT_EQ(kFakeMaxPacketSize, new_params.max_packet_size.value());
   EXPECT_EQ(kFakeInitialMaxData, new_params.initial_max_data.value());
@@ -527,28 +536,24 @@
 TEST_F(TransportParametersTest, ParseServerParametersRepeated) {
   // clang-format off
   const uint8_t kServerParamsRepeated[] = {
-      0x01, 0x23, 0x45, 0x67,  // negotiated_version
-      0x08,  // length of supported versions array
-      0x01, 0x23, 0x45, 0x67,
-      0x89, 0xab, 0xcd, 0xef,
-      0x00, 0x2A,  // length of parameters array that follows
+      0x00, 0x2c,  // length of parameters array that follows
       // original_connection_id
       0x00, 0x00,  // parameter id
       0x00, 0x08,  // length
       0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x13, 0x37,
-      // idle_timeout_seconds
+      // idle_timeout
       0x00, 0x01,  // parameter id
-      0x00, 0x01,  // length
-      0x0c,        // value
+      0x00, 0x02,  // length
+      0x6e, 0xec,  // value
       // stateless_reset_token
       0x00, 0x02,
       0x00, 0x10,
       0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08,
       0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, 0x10,
-      // idle_timeout_seconds (repeated)
+      // idle_timeout (repeated)
       0x00, 0x01,  // parameter id
-      0x00, 0x01,  // length
-      0x0c,        // value
+      0x00, 0x02,  // length
+      0x6e, 0xec,  // value
   };
   // clang-format on
 
@@ -561,6 +566,7 @@
 TEST_F(TransportParametersTest, CryptoHandshakeMessageRoundtrip) {
   TransportParameters orig_params;
   orig_params.perspective = Perspective::IS_CLIENT;
+  orig_params.version = kFakeVersionLabel;
   orig_params.max_packet_size.set_value(kFakeMaxPacketSize);
 
   orig_params.google_quic_params = QuicMakeUnique<CryptoHandshakeMessage>();
@@ -586,6 +592,7 @@
   EXPECT_EQ(new_params.google_quic_params->GetUint32(1337, &test_value),
             QUIC_NO_ERROR);
   EXPECT_EQ(test_value, kTestValue);
+  EXPECT_EQ(kFakeVersionLabel, new_params.version);
   EXPECT_EQ(kFakeMaxPacketSize, new_params.max_packet_size.value());
 }
 
diff --git a/quic/core/quic_config.cc b/quic/core/quic_config.cc
index cad98db..b106f41 100644
--- a/quic/core/quic_config.cc
+++ b/quic/core/quic_config.cc
@@ -750,8 +750,8 @@
 }
 
 bool QuicConfig::FillTransportParameters(TransportParameters* params) const {
-  params->idle_timeout_seconds.set_value(
-      idle_network_timeout_seconds_.GetUint32());
+  params->idle_timeout_milliseconds.set_value(
+      idle_network_timeout_seconds_.GetUint32() * kNumMillisPerSecond);
 
   if (stateless_reset_token_.HasSendValue()) {
     QuicUint128 stateless_reset_token = stateless_reset_token_.GetSendValue();
@@ -807,7 +807,9 @@
     const TransportParameters& params,
     HelloType hello_type,
     std::string* error_details) {
-  uint64_t idle_timeout_seconds = params.idle_timeout_seconds.value();
+  // Intentionally round down to probe too often rather than not often enough.
+  uint64_t idle_timeout_seconds =
+      params.idle_timeout_milliseconds.value() / kNumMillisPerSecond;
   if (idle_timeout_seconds > kMaximumIdleTimeoutSecs) {
     idle_timeout_seconds = kMaximumIdleTimeoutSecs;
   }
diff --git a/quic/core/quic_constants.h b/quic/core/quic_constants.h
index 81bed70..939f746 100644
--- a/quic/core/quic_constants.h
+++ b/quic/core/quic_constants.h
@@ -21,8 +21,9 @@
 const uint64_t kNumSecondsPerMinute = 60;
 const uint64_t kNumSecondsPerHour = kNumSecondsPerMinute * 60;
 const uint64_t kNumSecondsPerWeek = kNumSecondsPerHour * 24 * 7;
+const uint64_t kNumMillisPerSecond = 1000;
 const uint64_t kNumMicrosPerMilli = 1000;
-const uint64_t kNumMicrosPerSecond = 1000 * 1000;
+const uint64_t kNumMicrosPerSecond = kNumMicrosPerMilli * kNumMillisPerSecond;
 
 // Default number of connections for N-connection emulation.
 const uint32_t kDefaultNumConnections = 2;
diff --git a/quic/core/tls_client_handshaker.cc b/quic/core/tls_client_handshaker.cc
index 932d537..3f2a725 100644
--- a/quic/core/tls_client_handshaker.cc
+++ b/quic/core/tls_client_handshaker.cc
@@ -152,6 +152,15 @@
     return false;
   }
 
+  // When interoperating with non-Google implementations that do not send
+  // the version extension, set it to what we expect.
+  if (params.version == 0) {
+    params.version = CreateQuicVersionLabel(session()->connection()->version());
+  }
+  if (params.supported_versions.empty()) {
+    params.supported_versions.push_back(params.version);
+  }
+
   if (params.version !=
       CreateQuicVersionLabel(session()->connection()->version())) {
     *error_details = "Version mismatch detected";
diff --git a/quic/core/tls_server_handshaker.cc b/quic/core/tls_server_handshaker.cc
index ec254b5..ca56e4d 100644
--- a/quic/core/tls_server_handshaker.cc
+++ b/quic/core/tls_server_handshaker.cc
@@ -211,6 +211,14 @@
     *error_details = "Unable to parse Transport Parameters";
     return false;
   }
+
+  // When interoperating with non-Google implementations that do not send
+  // the version extension, set it to what we expect.
+  if (client_params.version == 0) {
+    client_params.version =
+        CreateQuicVersionLabel(session()->connection()->version());
+  }
+
   if (CryptoUtils::ValidateClientHelloVersion(
           client_params.version, session()->connection()->version(),
           session()->supported_versions(), error_details) != QUIC_NO_ERROR ||