QUICHE team | a6ef0a6 | 2019-03-07 20:34:33 -0500 | [diff] [blame] | 1 | // Copyright (c) 2017 The Chromium Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
| 5 | #include "net/third_party/quiche/src/quic/quartc/quartc_factory.h" |
| 6 | |
| 7 | #include "net/third_party/quiche/src/quic/core/crypto/quic_random.h" |
| 8 | #include "net/third_party/quiche/src/quic/core/quic_utils.h" |
| 9 | #include "net/third_party/quiche/src/quic/core/tls_client_handshaker.h" |
| 10 | #include "net/third_party/quiche/src/quic/core/tls_server_handshaker.h" |
| 11 | #include "net/third_party/quiche/src/quic/platform/api/quic_ptr_util.h" |
| 12 | #include "net/third_party/quiche/src/quic/platform/api/quic_socket_address.h" |
| 13 | #include "net/third_party/quiche/src/quic/quartc/quartc_connection_helper.h" |
| 14 | #include "net/third_party/quiche/src/quic/quartc/quartc_crypto_helpers.h" |
| 15 | #include "net/third_party/quiche/src/quic/quartc/quartc_session.h" |
| 16 | |
| 17 | namespace quic { |
| 18 | |
| 19 | QuartcFactory::QuartcFactory(const QuartcFactoryConfig& factory_config) |
| 20 | : alarm_factory_(factory_config.alarm_factory), |
| 21 | clock_(factory_config.clock), |
| 22 | connection_helper_(QuicMakeUnique<QuartcConnectionHelper>(clock_)), |
| 23 | compressed_certs_cache_(QuicMakeUnique<QuicCompressedCertsCache>( |
| 24 | QuicCompressedCertsCache::kQuicCompressedCertsCacheSize)), |
| 25 | stream_helper_(QuicMakeUnique<QuartcCryptoServerStreamHelper>()) {} |
| 26 | |
| 27 | std::unique_ptr<QuartcSession> QuartcFactory::CreateQuartcClientSession( |
| 28 | const QuartcSessionConfig& quartc_session_config, |
| 29 | const ParsedQuicVersionVector& supported_versions, |
| 30 | QuicStringPiece server_crypto_config, |
| 31 | QuartcPacketTransport* packet_transport) { |
| 32 | DCHECK(packet_transport); |
| 33 | |
| 34 | // QuartcSession will eventually own both |writer| and |quic_connection|. |
| 35 | auto writer = QuicMakeUnique<QuartcPacketWriter>( |
| 36 | packet_transport, quartc_session_config.max_packet_size); |
| 37 | |
| 38 | // While the QuicConfig is not directly used by the connection, creating it |
| 39 | // also sets flag values which must be set before creating the connection. |
| 40 | QuicConfig quic_config = CreateQuicConfig(quartc_session_config); |
| 41 | std::unique_ptr<QuicConnection> quic_connection = CreateQuicConnection( |
| 42 | Perspective::IS_CLIENT, supported_versions, writer.get()); |
| 43 | |
| 44 | return QuicMakeUnique<QuartcClientSession>( |
| 45 | std::move(quic_connection), quic_config, supported_versions, clock_, |
| 46 | std::move(writer), |
| 47 | CreateCryptoClientConfig(quartc_session_config.pre_shared_key), |
| 48 | server_crypto_config); |
| 49 | } |
| 50 | |
| 51 | void ConfigureGlobalQuicSettings() { |
| 52 | // Fixes behavior of StopReading() with level-triggered stream sequencers. |
| 53 | SetQuicReloadableFlag(quic_stop_reading_when_level_triggered, true); |
| 54 | |
| 55 | // Fix b/110259444. |
| 56 | SetQuicReloadableFlag(quic_fix_spurious_ack_alarm, true); |
| 57 | |
| 58 | // Enable version 46+ to enable SendMessage API and 'quic bit' per draft 17. |
| 59 | SetQuicReloadableFlag(quic_enable_version_46, true); |
| 60 | |
| 61 | // Fix for inconsistent reporting of crypto handshake. |
| 62 | SetQuicReloadableFlag(quic_fix_has_pending_crypto_data, true); |
| 63 | |
| 64 | // Ensure that we don't drop data because QUIC streams refuse to buffer it. |
| 65 | // TODO(b/120099046): Replace this with correct handling of WriteMemSlices(). |
| 66 | SetQuicFlag(&FLAGS_quic_buffered_data_threshold, |
| 67 | std::numeric_limits<int>::max()); |
| 68 | |
| 69 | // TODO(b/117157454): Perform version negotiation for Quartc outside of |
| 70 | // QuicSession/QuicConnection. Currently default of |
| 71 | // quic_restart_flag_quic_no_server_conn_ver_negotiation2 is false, |
| 72 | // but we fail blueprint test that sets all QUIC flags to true. |
| 73 | // |
| 74 | // Forcing flag to false to pass blueprint tests, but eventually we'll have |
| 75 | // to implement negotiation outside of QuicConnection. |
| 76 | SetQuicRestartFlag(quic_no_server_conn_ver_negotiation2, false); |
| 77 | SetQuicReloadableFlag(quic_no_client_conn_ver_negotiation, false); |
| 78 | |
| 79 | // Enable and request QUIC to include receive timestamps in ACK frames. |
| 80 | SetQuicReloadableFlag(quic_send_timestamps, true); |
| 81 | |
| 82 | // Enable ACK_DECIMATION_WITH_REORDERING. It requires ack_decimation to be |
| 83 | // false. |
| 84 | SetQuicReloadableFlag(quic_enable_ack_decimation, false); |
| 85 | |
| 86 | // Note: flag settings have no effect for Exoblaze builds since |
| 87 | // SetQuicReloadableFlag() gets stubbed out. |
| 88 | SetQuicReloadableFlag(quic_bbr_less_probe_rtt, true); // Enable BBR6,7,8. |
| 89 | SetQuicReloadableFlag(quic_unified_iw_options, true); // Enable IWXX opts. |
| 90 | SetQuicReloadableFlag(quic_bbr_slower_startup3, true); // Enable BBQX opts. |
| 91 | SetQuicReloadableFlag(quic_bbr_flexible_app_limited, true); // Enable BBR9. |
| 92 | } |
| 93 | |
| 94 | QuicConfig CreateQuicConfig(const QuartcSessionConfig& quartc_session_config) { |
| 95 | // TODO(b/124398962): Figure out a better way to initialize QUIC flags. |
| 96 | // Creating a config shouldn't have global side-effects on flags. However, |
| 97 | // this has the advantage of ensuring that flag values stay in sync with the |
| 98 | // options requested by configs, so simply splitting the config and flag |
| 99 | // settings doesn't seem preferable. |
| 100 | ConfigureGlobalQuicSettings(); |
| 101 | |
| 102 | // In exoblaze this may return false. DCHECK to avoid problems caused by |
| 103 | // incorrect flags configuration. |
| 104 | DCHECK(GetQuicReloadableFlag(quic_enable_version_46)) |
| 105 | << "Your build does not support quic reloadable flags and shouldn't " |
| 106 | "place Quartc calls"; |
| 107 | |
| 108 | QuicTagVector copt; |
| 109 | copt.push_back(kNSTP); |
| 110 | |
| 111 | // Enable and request QUIC to include receive timestamps in ACK frames. |
| 112 | copt.push_back(kSTMP); |
| 113 | |
| 114 | // Enable ACK_DECIMATION_WITH_REORDERING. It requires ack_decimation to be |
| 115 | // false. |
| 116 | copt.push_back(kAKD2); |
| 117 | |
| 118 | // Use unlimited decimation in order to reduce number of unbundled ACKs. |
| 119 | copt.push_back(kAKDU); |
| 120 | |
| 121 | // Enable time-based loss detection. |
| 122 | copt.push_back(kTIME); |
| 123 | |
| 124 | copt.push_back(kBBR3); // Stay in low-gain until in-flight < BDP. |
| 125 | copt.push_back(kBBR5); // 40 RTT ack aggregation. |
| 126 | copt.push_back(kBBR6); // Use a 0.75 * BDP cwnd during PROBE_RTT. |
| 127 | copt.push_back(kBBR8); // Skip PROBE_RTT if app-limited. |
| 128 | copt.push_back(kBBR9); // Ignore app-limited if enough data is in flight. |
| 129 | copt.push_back(kBBQ1); // 2.773 pacing gain in STARTUP. |
| 130 | copt.push_back(kBBQ2); // 2.0 CWND gain in STARTUP. |
| 131 | copt.push_back(kBBQ4); // 0.75 pacing gain in DRAIN. |
| 132 | copt.push_back(k1RTT); // Exit STARTUP after 1 RTT with no gains. |
| 133 | copt.push_back(kIW10); // 10-packet (14600 byte) initial cwnd. |
| 134 | |
| 135 | if (!quartc_session_config.enable_tail_loss_probe) { |
| 136 | copt.push_back(kNTLP); |
| 137 | } |
| 138 | |
| 139 | // TODO(b/112192153): Test and possible enable slower startup when pipe |
| 140 | // filling is ready to use. Slower startup is kBBRS. |
| 141 | |
| 142 | QuicConfig quic_config; |
| 143 | |
| 144 | // Use the limits for the session & stream flow control. The default 16KB |
| 145 | // limit leads to significantly undersending (not reaching BWE on the outgoing |
| 146 | // bitrate) due to blocked frames, and it leads to high latency (and one-way |
| 147 | // delay). Setting it to its limits is not going to cause issues (our streams |
| 148 | // are small generally, and if we were to buffer 24MB it wouldn't be the end |
| 149 | // of the world). We can consider setting different limits in future (e.g. 1MB |
| 150 | // stream, 1.5MB session). It's worth noting that on 1mbps bitrate, limit of |
| 151 | // 24MB can capture approx 4 minutes of the call, and the default increase in |
| 152 | // size of the window (half of the window size) is approximately 2 minutes of |
| 153 | // the call. |
| 154 | quic_config.SetInitialSessionFlowControlWindowToSend( |
| 155 | kSessionReceiveWindowLimit); |
| 156 | quic_config.SetInitialStreamFlowControlWindowToSend( |
| 157 | kStreamReceiveWindowLimit); |
| 158 | quic_config.SetConnectionOptionsToSend(copt); |
| 159 | quic_config.SetClientConnectionOptions(copt); |
| 160 | if (quartc_session_config.max_time_before_crypto_handshake > |
| 161 | QuicTime::Delta::Zero()) { |
| 162 | quic_config.set_max_time_before_crypto_handshake( |
| 163 | quartc_session_config.max_time_before_crypto_handshake); |
| 164 | } |
| 165 | if (quartc_session_config.max_idle_time_before_crypto_handshake > |
| 166 | QuicTime::Delta::Zero()) { |
| 167 | quic_config.set_max_idle_time_before_crypto_handshake( |
| 168 | quartc_session_config.max_idle_time_before_crypto_handshake); |
| 169 | } |
| 170 | if (quartc_session_config.idle_network_timeout > QuicTime::Delta::Zero()) { |
| 171 | quic_config.SetIdleNetworkTimeout( |
| 172 | quartc_session_config.idle_network_timeout, |
| 173 | quartc_session_config.idle_network_timeout); |
| 174 | } |
| 175 | |
| 176 | // The ICE transport provides a unique 5-tuple for each connection. Save |
| 177 | // overhead by omitting the connection id. |
| 178 | quic_config.SetBytesForConnectionIdToSend(0); |
| 179 | |
| 180 | // Allow up to 1000 incoming streams at once. Quartc streams typically contain |
| 181 | // one audio or video frame and close immediately. However, when a video frame |
| 182 | // becomes larger than one packet, there is some delay between the start and |
| 183 | // end of each stream. The default maximum of 100 only leaves about 1 second |
| 184 | // of headroom (Quartc sends ~30 video frames per second) before QUIC starts |
| 185 | // to refuse incoming streams. Back-pressure should clear backlogs of |
| 186 | // incomplete streams, but targets 1 second for recovery. Increasing the |
| 187 | // number of open streams gives sufficient headroom to recover before QUIC |
| 188 | // refuses new streams. |
| 189 | quic_config.SetMaxIncomingDynamicStreamsToSend(1000); |
| 190 | |
| 191 | return quic_config; |
| 192 | } |
| 193 | |
| 194 | std::unique_ptr<QuicConnection> QuartcFactory::CreateQuicConnection( |
| 195 | Perspective perspective, |
| 196 | const ParsedQuicVersionVector& supported_versions, |
| 197 | QuartcPacketWriter* packet_writer) { |
| 198 | // |dummy_id| and |dummy_address| are used because Quartc network layer will |
| 199 | // not use these two. |
QUICHE team | 4d9d629 | 2019-03-11 14:25:33 -0700 | [diff] [blame] | 200 | QuicConnectionId dummy_id = QuicUtils::CreateZeroConnectionId( |
| 201 | supported_versions[0].transport_version); |
| 202 | QuicSocketAddress dummy_address(QuicIpAddress::Any4(), /*port=*/0); |
| 203 | return quic::CreateQuicConnection( |
| 204 | dummy_id, dummy_address, connection_helper_.get(), alarm_factory_, |
| 205 | packet_writer, perspective, supported_versions); |
QUICHE team | a6ef0a6 | 2019-03-07 20:34:33 -0500 | [diff] [blame] | 206 | } |
| 207 | |
| 208 | std::unique_ptr<QuicConnection> CreateQuicConnection( |
| 209 | QuicConnectionId connection_id, |
| 210 | const QuicSocketAddress& peer_address, |
| 211 | QuicConnectionHelperInterface* connection_helper, |
| 212 | QuicAlarmFactory* alarm_factory, |
| 213 | QuicPacketWriter* packet_writer, |
| 214 | Perspective perspective, |
| 215 | ParsedQuicVersionVector supported_versions) { |
| 216 | auto quic_connection = QuicMakeUnique<QuicConnection>( |
| 217 | connection_id, peer_address, connection_helper, alarm_factory, |
| 218 | packet_writer, |
| 219 | /*owns_writer=*/false, perspective, supported_versions); |
| 220 | |
| 221 | QuicSentPacketManager& sent_packet_manager = |
| 222 | quic_connection->sent_packet_manager(); |
| 223 | |
| 224 | // Default delayed ack time is 25ms. |
| 225 | // If data packets are sent less often (e.g. because p-time was modified), |
| 226 | // we would force acks to be sent every 25ms regardless, increasing |
| 227 | // overhead. Since generally we guarantee a packet every 20ms, changing |
| 228 | // this value should have miniscule effect on quality on good connections, |
| 229 | // but on poor connections, changing this number significantly reduced the |
| 230 | // number of ack-only packets. |
| 231 | // The p-time can go up to as high as 120ms, and when it does, it's |
| 232 | // when the low overhead is the most important thing. Ideally it should be |
| 233 | // above 120ms, but it cannot be higher than 0.5*RTO, which equals to 100ms. |
| 234 | sent_packet_manager.set_delayed_ack_time( |
| 235 | QuicTime::Delta::FromMilliseconds(100)); |
| 236 | |
| 237 | quic_connection->set_fill_up_link_during_probing(true); |
| 238 | |
| 239 | // We start ack decimation after 15 packets. Typically, we would see |
| 240 | // 1-2 crypto handshake packets, one media packet, and 10 probing packets. |
| 241 | // We want to get acks for the probing packets as soon as possible, |
| 242 | // but we can start using ack decimation right after first probing completes. |
| 243 | // The default was to not start ack decimation for the first 100 packets. |
| 244 | quic_connection->set_min_received_before_ack_decimation(15); |
| 245 | |
| 246 | return quic_connection; |
| 247 | } |
| 248 | |
| 249 | std::unique_ptr<QuartcFactory> CreateQuartcFactory( |
| 250 | const QuartcFactoryConfig& factory_config) { |
| 251 | return std::unique_ptr<QuartcFactory>(new QuartcFactory(factory_config)); |
| 252 | } |
| 253 | |
| 254 | } // namespace quic |