blob: fd28829aa7b83468f057a535637f0a3d96d601d3 [file] [log] [blame]
Davide Pesaventobf1c0692017-01-15 19:15:09 -05001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +00002/*
Junxiao Shi7664b122019-01-23 16:45:17 +00003 * Copyright (c) 2016-2019, Regents of the University of California,
Davide Pesaventocd65c2c2017-01-15 16:10:38 -05004 * Colorado State University,
5 * University Pierre & Marie Curie, Sorbonne University.
Weiwei Liu245d7912016-07-28 00:04:25 -07006 *
7 * This file is part of ndn-tools (Named Data Networking Essential Tools).
8 * See AUTHORS.md for complete list of ndn-tools authors and contributors.
9 *
10 * ndn-tools is free software: you can redistribute it and/or modify it under the terms
11 * of the GNU General Public License as published by the Free Software Foundation,
12 * either version 3 of the License, or (at your option) any later version.
13 *
14 * ndn-tools is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
15 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
16 * PURPOSE. See the GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License along with
19 * ndn-tools, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
20 *
21 * See AUTHORS.md for complete list of ndn-cxx authors and contributors.
22 *
23 * @author Shuo Yang
24 * @author Weiwei Liu
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +000025 * @author Chavoosh Ghasemi
schneiderklausd8197df2019-03-16 11:31:40 -070026 * @author Klaus Schneider
Weiwei Liu245d7912016-07-28 00:04:25 -070027 */
28
schneiderklausd8197df2019-03-16 11:31:40 -070029#include "pipeline-interests-adaptive.hpp"
Davide Pesavento44b3b232017-12-23 16:58:25 -050030#include "data-fetcher.hpp"
Weiwei Liu245d7912016-07-28 00:04:25 -070031
32#include <cmath>
Chavoosh Ghasemi3dae1092017-12-21 12:39:08 -070033#include <iomanip>
Weiwei Liu245d7912016-07-28 00:04:25 -070034
35namespace ndn {
36namespace chunks {
Weiwei Liu245d7912016-07-28 00:04:25 -070037
schneiderklausd8197df2019-03-16 11:31:40 -070038constexpr double PipelineInterestsAdaptive::MIN_SSTHRESH;
Chavoosh Ghasemi641f5932017-11-06 22:45:11 +000039
schneiderklausd8197df2019-03-16 11:31:40 -070040PipelineInterestsAdaptive::PipelineInterestsAdaptive(Face& face, RttEstimator& rttEstimator,
41 const Options& options)
Weiwei Liu245d7912016-07-28 00:04:25 -070042 : PipelineInterests(face)
43 , m_options(options)
Klaus Schneider9e5122b2019-03-19 17:03:25 -070044 , m_cwnd(m_options.initCwnd)
45 , m_ssthresh(m_options.initSsthresh)
Weiwei Liu245d7912016-07-28 00:04:25 -070046 , m_rttEstimator(rttEstimator)
47 , m_scheduler(m_face.getIoService())
Weiwei Liu245d7912016-07-28 00:04:25 -070048 , m_highData(0)
49 , m_highInterest(0)
50 , m_recPoint(0)
51 , m_nInFlight(0)
schneiderklaus8ff3abd2019-03-12 22:15:12 -070052 , m_nLossDecr(0)
53 , m_nMarkDecr(0)
54 , m_nTimeouts(0)
55 , m_nSkippedRetx(0)
Weiwei Liu245d7912016-07-28 00:04:25 -070056 , m_nRetransmitted(0)
Chavoosh Ghasemi641f5932017-11-06 22:45:11 +000057 , m_nCongMarks(0)
Chavoosh Ghasemi75309ae2018-03-26 14:46:24 -040058 , m_nSent(0)
Weiwei Liu245d7912016-07-28 00:04:25 -070059 , m_hasFailure(false)
60 , m_failedSegNo(0)
61{
Weiwei Liu245d7912016-07-28 00:04:25 -070062}
63
schneiderklausd8197df2019-03-16 11:31:40 -070064PipelineInterestsAdaptive::~PipelineInterestsAdaptive()
Weiwei Liu245d7912016-07-28 00:04:25 -070065{
66 cancel();
67}
68
69void
schneiderklausd8197df2019-03-16 11:31:40 -070070PipelineInterestsAdaptive::doRun()
Weiwei Liu245d7912016-07-28 00:04:25 -070071{
Ryan Wickman034f30f2018-06-06 11:11:11 -050072 if (allSegmentsReceived()) {
73 cancel();
74 if (!m_options.isQuiet) {
75 printSummary();
76 }
77 return;
78 }
79
Weiwei Liu245d7912016-07-28 00:04:25 -070080 // schedule the event to check retransmission timer
Davide Pesaventobf2c5172019-03-20 19:08:09 -040081 m_checkRtoEvent = m_scheduler.schedule(m_options.rtoCheckInterval, [this] { checkRto(); });
Weiwei Liu245d7912016-07-28 00:04:25 -070082
Davide Pesavento958896e2017-01-19 00:52:04 -050083 schedulePackets();
Weiwei Liu245d7912016-07-28 00:04:25 -070084}
85
86void
schneiderklausd8197df2019-03-16 11:31:40 -070087PipelineInterestsAdaptive::doCancel()
Weiwei Liu245d7912016-07-28 00:04:25 -070088{
Davide Pesaventocd65c2c2017-01-15 16:10:38 -050089 m_checkRtoEvent.cancel();
Weiwei Liu245d7912016-07-28 00:04:25 -070090 m_segmentInfo.clear();
Weiwei Liu245d7912016-07-28 00:04:25 -070091}
92
93void
schneiderklausd8197df2019-03-16 11:31:40 -070094PipelineInterestsAdaptive::checkRto()
Weiwei Liu245d7912016-07-28 00:04:25 -070095{
96 if (isStopping())
97 return;
98
Davide Pesavento958896e2017-01-19 00:52:04 -050099 bool hasTimeout = false;
Weiwei Liu245d7912016-07-28 00:04:25 -0700100
101 for (auto& entry : m_segmentInfo) {
102 SegmentInfo& segInfo = entry.second;
Ryan Wickman2c9933c2018-06-12 11:51:51 -0500103 if (segInfo.state != SegmentState::InRetxQueue) { // skip segments already in the retx queue
Davide Pesavento70576402019-06-07 16:42:21 -0400104 auto timeElapsed = time::steady_clock::now() - segInfo.timeSent;
105 if (timeElapsed > segInfo.rto) { // timer expired?
schneiderklaus8ff3abd2019-03-12 22:15:12 -0700106 m_nTimeouts++;
Davide Pesavento958896e2017-01-19 00:52:04 -0500107 hasTimeout = true;
108 enqueueForRetransmission(entry.first);
Weiwei Liu245d7912016-07-28 00:04:25 -0700109 }
110 }
111 }
112
Davide Pesavento958896e2017-01-19 00:52:04 -0500113 if (hasTimeout) {
114 recordTimeout();
115 schedulePackets();
Weiwei Liu245d7912016-07-28 00:04:25 -0700116 }
117
118 // schedule the next check after predefined interval
Davide Pesaventobf2c5172019-03-20 19:08:09 -0400119 m_checkRtoEvent = m_scheduler.schedule(m_options.rtoCheckInterval, [this] { checkRto(); });
Weiwei Liu245d7912016-07-28 00:04:25 -0700120}
121
122void
schneiderklausd8197df2019-03-16 11:31:40 -0700123PipelineInterestsAdaptive::sendInterest(uint64_t segNo, bool isRetransmission)
Weiwei Liu245d7912016-07-28 00:04:25 -0700124{
125 if (isStopping())
126 return;
127
Ryan Wickman034f30f2018-06-06 11:11:11 -0500128 if (m_hasFinalBlockId && segNo > m_lastSegmentNo)
Weiwei Liu245d7912016-07-28 00:04:25 -0700129 return;
130
131 if (!isRetransmission && m_hasFailure)
132 return;
133
134 if (m_options.isVerbose) {
Davide Pesavento958896e2017-01-19 00:52:04 -0500135 std::cerr << (isRetransmission ? "Retransmitting" : "Requesting")
136 << " segment #" << segNo << std::endl;
Weiwei Liu245d7912016-07-28 00:04:25 -0700137 }
138
139 if (isRetransmission) {
Davide Pesavento958896e2017-01-19 00:52:04 -0500140 // keep track of retx count for this segment
141 auto ret = m_retxCount.emplace(segNo, 1);
Weiwei Liu245d7912016-07-28 00:04:25 -0700142 if (ret.second == false) { // not the first retransmission
143 m_retxCount[segNo] += 1;
Davide Pesavento44b3b232017-12-23 16:58:25 -0500144 if (m_options.maxRetriesOnTimeoutOrNack != DataFetcher::MAX_RETRIES_INFINITE &&
145 m_retxCount[segNo] > m_options.maxRetriesOnTimeoutOrNack) {
Weiwei Liu245d7912016-07-28 00:04:25 -0700146 return handleFail(segNo, "Reached the maximum number of retries (" +
147 to_string(m_options.maxRetriesOnTimeoutOrNack) +
148 ") while retrieving segment #" + to_string(segNo));
149 }
150
151 if (m_options.isVerbose) {
152 std::cerr << "# of retries for segment #" << segNo
153 << " is " << m_retxCount[segNo] << std::endl;
154 }
155 }
Weiwei Liu245d7912016-07-28 00:04:25 -0700156 }
157
Davide Pesavento84d84772019-04-07 14:40:07 -0400158 auto interest = Interest()
159 .setName(Name(m_prefix).appendSegment(segNo))
160 .setCanBePrefix(false)
161 .setMustBeFresh(m_options.mustBeFresh)
162 .setInterestLifetime(m_options.interestLifetime);
Weiwei Liu245d7912016-07-28 00:04:25 -0700163
Junxiao Shi06d008c2019-02-04 08:26:59 +0000164 SegmentInfo& segInfo = m_segmentInfo[segNo];
165 segInfo.interestHdl = m_face.expressInterest(interest,
schneiderklausd8197df2019-03-16 11:31:40 -0700166 bind(&PipelineInterestsAdaptive::handleData, this, _1, _2),
167 bind(&PipelineInterestsAdaptive::handleNack, this, _1, _2),
168 bind(&PipelineInterestsAdaptive::handleLifetimeExpiration, this, _1));
Junxiao Shi06d008c2019-02-04 08:26:59 +0000169 segInfo.timeSent = time::steady_clock::now();
170 segInfo.rto = m_rttEstimator.getEstimatedRto();
171
Weiwei Liu245d7912016-07-28 00:04:25 -0700172 m_nInFlight++;
Chavoosh Ghasemi75309ae2018-03-26 14:46:24 -0400173 m_nSent++;
Weiwei Liu245d7912016-07-28 00:04:25 -0700174
175 if (isRetransmission) {
Davide Pesavento958896e2017-01-19 00:52:04 -0500176 segInfo.state = SegmentState::Retransmitted;
Weiwei Liu245d7912016-07-28 00:04:25 -0700177 m_nRetransmitted++;
178 }
179 else {
180 m_highInterest = segNo;
Junxiao Shi06d008c2019-02-04 08:26:59 +0000181 segInfo.state = SegmentState::FirstTimeSent;
Weiwei Liu245d7912016-07-28 00:04:25 -0700182 }
183}
184
185void
schneiderklausd8197df2019-03-16 11:31:40 -0700186PipelineInterestsAdaptive::schedulePackets()
Weiwei Liu245d7912016-07-28 00:04:25 -0700187{
Davide Pesaventobf1c0692017-01-15 19:15:09 -0500188 BOOST_ASSERT(m_nInFlight >= 0);
189 auto availableWindowSize = static_cast<int64_t>(m_cwnd) - m_nInFlight;
190
Weiwei Liu245d7912016-07-28 00:04:25 -0700191 while (availableWindowSize > 0) {
192 if (!m_retxQueue.empty()) { // do retransmission first
193 uint64_t retxSegNo = m_retxQueue.front();
194 m_retxQueue.pop();
Ryan Wickman2c9933c2018-06-12 11:51:51 -0500195 if (m_segmentInfo.count(retxSegNo) == 0) {
schneiderklaus8ff3abd2019-03-12 22:15:12 -0700196 m_nSkippedRetx++;
Weiwei Liu245d7912016-07-28 00:04:25 -0700197 continue;
198 }
Ryan Wickman2c9933c2018-06-12 11:51:51 -0500199 // the segment is still in the map, that means it needs to be retransmitted
Weiwei Liu245d7912016-07-28 00:04:25 -0700200 sendInterest(retxSegNo, true);
201 }
202 else { // send next segment
203 sendInterest(getNextSegmentNo(), false);
204 }
205 availableWindowSize--;
206 }
207}
208
209void
schneiderklausd8197df2019-03-16 11:31:40 -0700210PipelineInterestsAdaptive::handleData(const Interest& interest, const Data& data)
Weiwei Liu245d7912016-07-28 00:04:25 -0700211{
212 if (isStopping())
213 return;
214
Davide Pesavento84d84772019-04-07 14:40:07 -0400215 // Interest was expressed with CanBePrefix=false
Weiwei Liu245d7912016-07-28 00:04:25 -0700216 BOOST_ASSERT(data.getName().equals(interest.getName()));
217
Davide Pesavento969cd5a2018-04-20 16:27:47 -0400218 if (!m_hasFinalBlockId && data.getFinalBlock()) {
219 m_lastSegmentNo = data.getFinalBlock()->toSegment();
Weiwei Liu245d7912016-07-28 00:04:25 -0700220 m_hasFinalBlockId = true;
221 cancelInFlightSegmentsGreaterThan(m_lastSegmentNo);
222 if (m_hasFailure && m_lastSegmentNo >= m_failedSegNo) {
223 // previously failed segment is part of the content
224 return onFailure(m_failureReason);
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +0000225 }
226 else {
Weiwei Liu245d7912016-07-28 00:04:25 -0700227 m_hasFailure = false;
228 }
229 }
230
Davide Pesaventobf1c0692017-01-15 19:15:09 -0500231 uint64_t recvSegNo = getSegmentFromPacket(data);
Ryan Wickman2c9933c2018-06-12 11:51:51 -0500232 auto segIt = m_segmentInfo.find(recvSegNo);
233 if (segIt == m_segmentInfo.end()) {
Weiwei Liu245d7912016-07-28 00:04:25 -0700234 return; // ignore already-received segment
235 }
236
Ryan Wickman2c9933c2018-06-12 11:51:51 -0500237 SegmentInfo& segInfo = segIt->second;
Davide Pesavento70576402019-06-07 16:42:21 -0400238 RttEstimator::MillisecondsDouble rtt = time::steady_clock::now() - segInfo.timeSent;
Weiwei Liu245d7912016-07-28 00:04:25 -0700239 if (m_options.isVerbose) {
240 std::cerr << "Received segment #" << recvSegNo
241 << ", rtt=" << rtt.count() << "ms"
242 << ", rto=" << segInfo.rto.count() << "ms" << std::endl;
243 }
244
Davide Pesavento958896e2017-01-19 00:52:04 -0500245 if (m_highData < recvSegNo) {
246 m_highData = recvSegNo;
247 }
248
249 // for segments in retx queue, we must not decrement m_nInFlight
250 // because it was already decremented when the segment timed out
251 if (segInfo.state != SegmentState::InRetxQueue) {
Weiwei Liu245d7912016-07-28 00:04:25 -0700252 m_nInFlight--;
253 }
254
Chavoosh Ghasemi641f5932017-11-06 22:45:11 +0000255 // upon finding congestion mark, decrease the window size
256 // without retransmitting any packet
257 if (data.getCongestionMark() > 0) {
258 m_nCongMarks++;
259 if (!m_options.ignoreCongMarks) {
260 if (m_options.disableCwa || m_highData > m_recPoint) {
261 m_recPoint = m_highInterest; // react to only one congestion event (timeout or congestion mark)
262 // per RTT (conservative window adaptation)
schneiderklaus8ff3abd2019-03-12 22:15:12 -0700263 m_nMarkDecr++;
Chavoosh Ghasemi641f5932017-11-06 22:45:11 +0000264 decreaseWindow();
265
266 if (m_options.isVerbose) {
267 std::cerr << "Received congestion mark, value = " << data.getCongestionMark()
268 << ", new cwnd = " << m_cwnd << std::endl;
269 }
270 }
271 }
272 else {
273 increaseWindow();
274 }
275 }
276 else {
277 increaseWindow();
278 }
279
Davide Pesaventoe9c69852017-11-04 18:08:37 -0400280 onData(data);
Weiwei Liu245d7912016-07-28 00:04:25 -0700281
Ryan Wickman2c9933c2018-06-12 11:51:51 -0500282 // do not sample RTT for retransmitted segments
283 if ((segInfo.state == SegmentState::FirstTimeSent ||
284 segInfo.state == SegmentState::InRetxQueue) &&
285 m_retxCount.count(recvSegNo) == 0) {
Davide Pesaventobf1c0692017-01-15 19:15:09 -0500286 auto nExpectedSamples = std::max<int64_t>((m_nInFlight + 1) >> 1, 1);
287 BOOST_ASSERT(nExpectedSamples > 0);
Davide Pesavento70576402019-06-07 16:42:21 -0400288 m_rttEstimator.addMeasurement(rtt, static_cast<size_t>(nExpectedSamples), recvSegNo);
Weiwei Liu245d7912016-07-28 00:04:25 -0700289 }
290
Ryan Wickman2c9933c2018-06-12 11:51:51 -0500291 // remove the entry associated with the received segment
292 m_segmentInfo.erase(segIt);
293
Ryan Wickman034f30f2018-06-06 11:11:11 -0500294 if (allSegmentsReceived()) {
Weiwei Liu245d7912016-07-28 00:04:25 -0700295 cancel();
Davide Pesaventof6991e12018-01-08 20:58:50 -0500296 if (!m_options.isQuiet) {
Weiwei Liu245d7912016-07-28 00:04:25 -0700297 printSummary();
298 }
299 }
300 else {
301 schedulePackets();
302 }
303}
304
305void
schneiderklausd8197df2019-03-16 11:31:40 -0700306PipelineInterestsAdaptive::handleNack(const Interest& interest, const lp::Nack& nack)
Weiwei Liu245d7912016-07-28 00:04:25 -0700307{
308 if (isStopping())
309 return;
310
311 if (m_options.isVerbose)
312 std::cerr << "Received Nack with reason " << nack.getReason()
313 << " for Interest " << interest << std::endl;
314
Davide Pesaventobf1c0692017-01-15 19:15:09 -0500315 uint64_t segNo = getSegmentFromPacket(interest);
Weiwei Liu245d7912016-07-28 00:04:25 -0700316
317 switch (nack.getReason()) {
Davide Pesavento958896e2017-01-19 00:52:04 -0500318 case lp::NackReason::DUPLICATE:
319 // ignore duplicates
Weiwei Liu245d7912016-07-28 00:04:25 -0700320 break;
Davide Pesavento958896e2017-01-19 00:52:04 -0500321 case lp::NackReason::CONGESTION:
322 // treated the same as timeout for now
323 enqueueForRetransmission(segNo);
324 recordTimeout();
325 schedulePackets();
326 break;
327 default:
Weiwei Liu245d7912016-07-28 00:04:25 -0700328 handleFail(segNo, "Could not retrieve data for " + interest.getName().toUri() +
329 ", reason: " + boost::lexical_cast<std::string>(nack.getReason()));
330 break;
Weiwei Liu245d7912016-07-28 00:04:25 -0700331 }
332}
333
334void
schneiderklausd8197df2019-03-16 11:31:40 -0700335PipelineInterestsAdaptive::handleLifetimeExpiration(const Interest& interest)
Weiwei Liu245d7912016-07-28 00:04:25 -0700336{
337 if (isStopping())
338 return;
339
schneiderklaus8ff3abd2019-03-12 22:15:12 -0700340 m_nTimeouts++;
Davide Pesavento958896e2017-01-19 00:52:04 -0500341 enqueueForRetransmission(getSegmentFromPacket(interest));
342 recordTimeout();
343 schedulePackets();
Weiwei Liu245d7912016-07-28 00:04:25 -0700344}
345
346void
schneiderklausd8197df2019-03-16 11:31:40 -0700347PipelineInterestsAdaptive::recordTimeout()
Weiwei Liu245d7912016-07-28 00:04:25 -0700348{
Weiwei Liu245d7912016-07-28 00:04:25 -0700349 if (m_options.disableCwa || m_highData > m_recPoint) {
350 // react to only one timeout per RTT (conservative window adaptation)
351 m_recPoint = m_highInterest;
352
353 decreaseWindow();
354 m_rttEstimator.backoffRto();
schneiderklaus8ff3abd2019-03-12 22:15:12 -0700355 m_nLossDecr++;
Weiwei Liu245d7912016-07-28 00:04:25 -0700356
357 if (m_options.isVerbose) {
Chavoosh Ghasemi641f5932017-11-06 22:45:11 +0000358 std::cerr << "Packet loss event, new cwnd = " << m_cwnd
Weiwei Liu245d7912016-07-28 00:04:25 -0700359 << ", ssthresh = " << m_ssthresh << std::endl;
360 }
361 }
Davide Pesavento958896e2017-01-19 00:52:04 -0500362}
Weiwei Liu245d7912016-07-28 00:04:25 -0700363
Davide Pesavento958896e2017-01-19 00:52:04 -0500364void
schneiderklausd8197df2019-03-16 11:31:40 -0700365PipelineInterestsAdaptive::enqueueForRetransmission(uint64_t segNo)
Davide Pesavento958896e2017-01-19 00:52:04 -0500366{
367 BOOST_ASSERT(m_nInFlight > 0);
368 m_nInFlight--;
369 m_retxQueue.push(segNo);
370 m_segmentInfo.at(segNo).state = SegmentState::InRetxQueue;
Weiwei Liu245d7912016-07-28 00:04:25 -0700371}
372
373void
schneiderklausd8197df2019-03-16 11:31:40 -0700374PipelineInterestsAdaptive::handleFail(uint64_t segNo, const std::string& reason)
Weiwei Liu245d7912016-07-28 00:04:25 -0700375{
376 if (isStopping())
377 return;
378
379 // if the failed segment is definitely part of the content, raise a fatal error
380 if (m_hasFinalBlockId && segNo <= m_lastSegmentNo)
381 return onFailure(reason);
382
383 if (!m_hasFinalBlockId) {
384 m_segmentInfo.erase(segNo);
Davide Pesavento958896e2017-01-19 00:52:04 -0500385 m_nInFlight--;
Weiwei Liu245d7912016-07-28 00:04:25 -0700386
387 if (m_segmentInfo.empty()) {
388 onFailure("Fetching terminated but no final segment number has been found");
389 }
390 else {
391 cancelInFlightSegmentsGreaterThan(segNo);
392 m_hasFailure = true;
393 m_failedSegNo = segNo;
394 m_failureReason = reason;
395 }
396 }
397}
398
399void
schneiderklausd8197df2019-03-16 11:31:40 -0700400PipelineInterestsAdaptive::cancelInFlightSegmentsGreaterThan(uint64_t segNo)
Weiwei Liu245d7912016-07-28 00:04:25 -0700401{
402 for (auto it = m_segmentInfo.begin(); it != m_segmentInfo.end();) {
403 // cancel fetching all segments that follow
Davide Pesavento958896e2017-01-19 00:52:04 -0500404 if (it->first > segNo) {
Weiwei Liu245d7912016-07-28 00:04:25 -0700405 it = m_segmentInfo.erase(it);
Davide Pesavento958896e2017-01-19 00:52:04 -0500406 m_nInFlight--;
Weiwei Liu245d7912016-07-28 00:04:25 -0700407 }
408 else {
409 ++it;
410 }
411 }
412}
413
414void
schneiderklausd8197df2019-03-16 11:31:40 -0700415PipelineInterestsAdaptive::printSummary() const
Weiwei Liu245d7912016-07-28 00:04:25 -0700416{
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +0000417 PipelineInterests::printSummary();
schneiderklaus8ff3abd2019-03-12 22:15:12 -0700418 std::cerr << "Congestion marks: " << m_nCongMarks << " (caused " << m_nMarkDecr << " window decreases)\n"
419 << "Timeouts: " << m_nTimeouts << " (caused " << m_nLossDecr << " window decreases)\n"
420 << "Retransmitted segments: " << m_nRetransmitted
421 << " (" << (m_nSent == 0 ? 0 : (static_cast<double>(m_nRetransmitted) / m_nSent * 100.0)) << "%)"
422 << ", skipped: " << m_nSkippedRetx << "\n"
Chavoosh Ghasemi75309ae2018-03-26 14:46:24 -0400423 << "RTT ";
424
Davide Pesavento70576402019-06-07 16:42:21 -0400425 if (m_rttEstimator.getMinRtt().count() == std::numeric_limits<double>::max() ||
426 m_rttEstimator.getMaxRtt().count() == std::numeric_limits<double>::min()) {
427 std::cerr << "stats unavailable\n";
428 }
429 else {
430 std::cerr << "min/avg/max = " << std::fixed << std::setprecision(3)
431 << m_rttEstimator.getMinRtt().count() << "/"
432 << m_rttEstimator.getAvgRtt().count() << "/"
433 << m_rttEstimator.getMaxRtt().count() << " ms\n";
Chavoosh Ghasemi75309ae2018-03-26 14:46:24 -0400434 }
Weiwei Liu245d7912016-07-28 00:04:25 -0700435}
436
437std::ostream&
438operator<<(std::ostream& os, SegmentState state)
439{
440 switch (state) {
441 case SegmentState::FirstTimeSent:
442 os << "FirstTimeSent";
443 break;
444 case SegmentState::InRetxQueue:
445 os << "InRetxQueue";
446 break;
447 case SegmentState::Retransmitted:
448 os << "Retransmitted";
449 break;
Weiwei Liu245d7912016-07-28 00:04:25 -0700450 }
Weiwei Liu245d7912016-07-28 00:04:25 -0700451 return os;
452}
453
454std::ostream&
schneiderklausd8197df2019-03-16 11:31:40 -0700455operator<<(std::ostream& os, const PipelineInterestsAdaptiveOptions& options)
Weiwei Liu245d7912016-07-28 00:04:25 -0700456{
schneiderklausd8197df2019-03-16 11:31:40 -0700457 os << "Adaptive pipeline parameters:\n"
Weiwei Liu245d7912016-07-28 00:04:25 -0700458 << "\tInitial congestion window size = " << options.initCwnd << "\n"
459 << "\tInitial slow start threshold = " << options.initSsthresh << "\n"
Weiwei Liu245d7912016-07-28 00:04:25 -0700460 << "\tAdditive increase step = " << options.aiStep << "\n"
Davide Pesavento958896e2017-01-19 00:52:04 -0500461 << "\tMultiplicative decrease factor = " << options.mdCoef << "\n"
Weiwei Liu245d7912016-07-28 00:04:25 -0700462 << "\tRTO check interval = " << options.rtoCheckInterval << "\n"
Davide Pesavento44b3b232017-12-23 16:58:25 -0500463 << "\tMax retries on timeout or Nack = " << (options.maxRetriesOnTimeoutOrNack == DataFetcher::MAX_RETRIES_INFINITE ?
Ryan Wickman2c9933c2018-06-12 11:51:51 -0500464 "infinite" : to_string(options.maxRetriesOnTimeoutOrNack)) << "\n"
Klaus Schneider9e5122b2019-03-19 17:03:25 -0700465 << "\tReact to congestion marks = " << (options.ignoreCongMarks ? "no" : "yes") << "\n"
466 << "\tConservative window adaptation = " << (options.disableCwa ? "no" : "yes") << "\n"
467 << "\tResetting window to " << (options.resetCwndToInit ? "initCwnd" : "ssthresh") << " upon loss event\n";
Weiwei Liu245d7912016-07-28 00:04:25 -0700468 return os;
469}
470
Weiwei Liu245d7912016-07-28 00:04:25 -0700471} // namespace chunks
472} // namespace ndn