blob: 8efa544fbbe02d3f5fe6f1cad468c5d8407574ce [file] [log] [blame]
QUICHE teama6ef0a62019-03-07 20:34:33 -05001// 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
17namespace quic {
18
19QuartcFactory::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
27std::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
51void 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
94QuicConfig 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
194std::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 team4d9d6292019-03-11 14:25:33 -0700200 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 teama6ef0a62019-03-07 20:34:33 -0500206}
207
208std::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
249std::unique_ptr<QuartcFactory> CreateQuartcFactory(
250 const QuartcFactoryConfig& factory_config) {
251 return std::unique_ptr<QuartcFactory>(new QuartcFactory(factory_config));
252}
253
254} // namespace quic