| /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */ |
| /* |
| * Copyright (c) 2013-2018, Regents of the University of California, |
| * Colorado State University, |
| * University Pierre & Marie Curie, Sorbonne University. |
| * |
| * This file is part of ndn-cxx library (NDN C++ library with eXperimental eXtensions). |
| * |
| * ndn-cxx library is free software: you can redistribute it and/or modify it under the |
| * terms of the GNU Lesser General Public License as published by the Free Software |
| * Foundation, either version 3 of the License, or (at your option) any later version. |
| * |
| * ndn-cxx library is distributed in the hope that it will be useful, but WITHOUT ANY |
| * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A |
| * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. |
| * |
| * You should have received copies of the GNU General Public License and GNU Lesser |
| * General Public License along with ndn-cxx, e.g., in COPYING.md file. If not, see |
| * <http://www.gnu.org/licenses/>. |
| * |
| * See AUTHORS.md for complete list of ndn-cxx authors and contributors. |
| * |
| * @author Shuo Yang |
| * @author Weiwei Liu |
| * @author Chavoosh Ghasemi |
| */ |
| |
| #include "ndn-cxx/util/segment-fetcher.hpp" |
| #include "ndn-cxx/name-component.hpp" |
| #include "ndn-cxx/encoding/buffer-stream.hpp" |
| #include "ndn-cxx/lp/nack.hpp" |
| #include "ndn-cxx/lp/nack-header.hpp" |
| |
| #include <boost/asio/io_service.hpp> |
| #include <boost/lexical_cast.hpp> |
| #include <cmath> |
| |
| namespace ndn { |
| namespace util { |
| |
| constexpr double SegmentFetcher::MIN_SSTHRESH; |
| |
| void |
| SegmentFetcher::Options::validate() |
| { |
| if (maxTimeout < 1_ms) { |
| BOOST_THROW_EXCEPTION(std::invalid_argument("maxTimeout must be greater than or equal to 1 millisecond")); |
| } |
| |
| if (initCwnd < 1.0) { |
| BOOST_THROW_EXCEPTION(std::invalid_argument("initCwnd must be greater than or equal to 1")); |
| } |
| |
| if (aiStep < 0.0) { |
| BOOST_THROW_EXCEPTION(std::invalid_argument("aiStep must be greater than or equal to 0")); |
| } |
| |
| if (mdCoef < 0.0 || mdCoef > 1.0) { |
| BOOST_THROW_EXCEPTION(std::invalid_argument("mdCoef must be in range [0, 1]")); |
| } |
| } |
| |
| SegmentFetcher::SegmentFetcher(Face& face, |
| security::v2::Validator& validator, |
| const SegmentFetcher::Options& options) |
| : m_options(options) |
| , m_face(face) |
| , m_scheduler(m_face.getIoService()) |
| , m_validator(validator) |
| , m_rttEstimator(options.rttOptions) |
| , m_timeLastSegmentReceived(time::steady_clock::now()) |
| , m_nextSegmentNum(0) |
| , m_cwnd(options.initCwnd) |
| , m_ssthresh(options.initSsthresh) |
| , m_nSegmentsInFlight(0) |
| , m_nSegments(0) |
| , m_highInterest(0) |
| , m_highData(0) |
| , m_recPoint(0) |
| , m_nReceived(0) |
| , m_nBytesReceived(0) |
| { |
| m_options.validate(); |
| } |
| |
| shared_ptr<SegmentFetcher> |
| SegmentFetcher::start(Face& face, |
| const Interest& baseInterest, |
| security::v2::Validator& validator, |
| const SegmentFetcher::Options& options) |
| { |
| shared_ptr<SegmentFetcher> fetcher(new SegmentFetcher(face, validator, options)); |
| fetcher->m_this = fetcher; |
| fetcher->fetchFirstSegment(baseInterest, false); |
| return fetcher; |
| } |
| |
| void |
| SegmentFetcher::stop() |
| { |
| if (!m_this) { |
| return; |
| } |
| |
| for (const auto& pendingSegment : m_pendingSegments) { |
| m_face.removePendingInterest(pendingSegment.second.id); |
| if (pendingSegment.second.timeoutEvent) { |
| m_scheduler.cancelEvent(pendingSegment.second.timeoutEvent); |
| } |
| } |
| m_face.getIoService().post([self = std::move(m_this)] {}); |
| } |
| |
| bool |
| SegmentFetcher::shouldStop(const weak_ptr<SegmentFetcher>& weakSelf) |
| { |
| auto self = weakSelf.lock(); |
| return self == nullptr || self->m_this == nullptr; |
| } |
| |
| void |
| SegmentFetcher::fetchFirstSegment(const Interest& baseInterest, bool isRetransmission) |
| { |
| Interest interest(baseInterest); |
| interest.setCanBePrefix(true); |
| interest.setMustBeFresh(true); |
| interest.setInterestLifetime(m_options.interestLifetime); |
| if (isRetransmission) { |
| interest.refreshNonce(); |
| } |
| |
| weak_ptr<SegmentFetcher> weakSelf = m_this; |
| |
| m_nSegmentsInFlight++; |
| auto pendingInterest = m_face.expressInterest(interest, |
| bind(&SegmentFetcher::afterSegmentReceivedCb, |
| this, _1, _2, weakSelf), |
| bind(&SegmentFetcher::afterNackReceivedCb, |
| this, _1, _2, weakSelf), |
| nullptr); |
| auto timeoutEvent = |
| m_scheduler.scheduleEvent(m_options.useConstantInterestTimeout ? m_options.maxTimeout : getEstimatedRto(), |
| bind(&SegmentFetcher::afterTimeoutCb, this, interest, weakSelf)); |
| |
| if (isRetransmission) { |
| updateRetransmittedSegment(0, pendingInterest, timeoutEvent); |
| } |
| else { |
| BOOST_ASSERT(m_pendingSegments.count(0) == 0); |
| m_pendingSegments.emplace(0, PendingSegment{SegmentState::FirstInterest, time::steady_clock::now(), |
| pendingInterest, timeoutEvent}); |
| } |
| } |
| |
| void |
| SegmentFetcher::fetchSegmentsInWindow(const Interest& origInterest) |
| { |
| weak_ptr<SegmentFetcher> weakSelf = m_this; |
| |
| if (checkAllSegmentsReceived()) { |
| // All segments have been retrieved |
| return finalizeFetch(); |
| } |
| |
| int64_t availableWindowSize = static_cast<int64_t>(m_cwnd) - m_nSegmentsInFlight; |
| std::vector<std::pair<uint64_t, bool>> segmentsToRequest; // The boolean indicates whether a retx or not |
| |
| while (availableWindowSize > 0) { |
| if (!m_retxQueue.empty()) { |
| auto pendingSegmentIt = m_pendingSegments.find(m_retxQueue.front()); |
| m_retxQueue.pop(); |
| if (pendingSegmentIt == m_pendingSegments.end()) { |
| // Skip re-requesting this segment, since it was received after RTO timeout |
| continue; |
| } |
| BOOST_ASSERT(pendingSegmentIt->second.state == SegmentState::InRetxQueue); |
| segmentsToRequest.emplace_back(pendingSegmentIt->first, true); |
| } |
| else if (m_nSegments == 0 || m_nextSegmentNum < static_cast<uint64_t>(m_nSegments)) { |
| if (m_receivedSegments.count(m_nextSegmentNum) > 0) { |
| // Don't request a segment a second time if received in response to first "discovery" Interest |
| m_nextSegmentNum++; |
| continue; |
| } |
| segmentsToRequest.emplace_back(m_nextSegmentNum++, false); |
| } |
| else { |
| break; |
| } |
| availableWindowSize--; |
| } |
| |
| for (const auto& segment : segmentsToRequest) { |
| Interest interest(origInterest); // to preserve Interest elements |
| interest.setName(Name(m_versionedDataName).appendSegment(segment.first)); |
| interest.setCanBePrefix(false); |
| interest.setMustBeFresh(false); |
| interest.setInterestLifetime(m_options.interestLifetime); |
| interest.refreshNonce(); |
| |
| m_nSegmentsInFlight++; |
| auto pendingInterest = m_face.expressInterest(interest, |
| bind(&SegmentFetcher::afterSegmentReceivedCb, |
| this, _1, _2, weakSelf), |
| bind(&SegmentFetcher::afterNackReceivedCb, |
| this, _1, _2, weakSelf), |
| nullptr); |
| auto timeoutEvent = |
| m_scheduler.scheduleEvent(m_options.useConstantInterestTimeout ? m_options.maxTimeout : getEstimatedRto(), |
| bind(&SegmentFetcher::afterTimeoutCb, this, interest, weakSelf)); |
| |
| if (segment.second) { // Retransmission |
| updateRetransmittedSegment(segment.first, pendingInterest, timeoutEvent); |
| } |
| else { // First request for segment |
| BOOST_ASSERT(m_pendingSegments.count(segment.first) == 0); |
| m_pendingSegments.emplace(segment.first, PendingSegment{SegmentState::FirstInterest, |
| time::steady_clock::now(), |
| pendingInterest, timeoutEvent}); |
| m_highInterest = segment.first; |
| } |
| } |
| } |
| |
| void |
| SegmentFetcher::afterSegmentReceivedCb(const Interest& origInterest, const Data& data, |
| const weak_ptr<SegmentFetcher>& weakSelf) |
| { |
| if (shouldStop(weakSelf)) |
| return; |
| |
| BOOST_ASSERT(m_nSegmentsInFlight > 0); |
| m_nSegmentsInFlight--; |
| |
| name::Component currentSegmentComponent = data.getName().get(-1); |
| if (!currentSegmentComponent.isSegment()) { |
| return signalError(DATA_HAS_NO_SEGMENT, "Data Name has no segment number"); |
| } |
| |
| uint64_t currentSegment = currentSegmentComponent.toSegment(); |
| |
| // The first received Interest could have any segment ID |
| std::map<uint64_t, PendingSegment>::iterator pendingSegmentIt; |
| if (m_receivedSegments.size() > 0) { |
| pendingSegmentIt = m_pendingSegments.find(currentSegment); |
| } |
| else { |
| pendingSegmentIt = m_pendingSegments.begin(); |
| } |
| |
| if (pendingSegmentIt == m_pendingSegments.end()) { |
| return; |
| } |
| |
| afterSegmentReceived(data); |
| |
| // Cancel timeout event |
| m_scheduler.cancelEvent(pendingSegmentIt->second.timeoutEvent); |
| pendingSegmentIt->second.timeoutEvent = nullptr; |
| |
| m_validator.validate(data, |
| bind(&SegmentFetcher::afterValidationSuccess, this, _1, origInterest, |
| pendingSegmentIt, weakSelf), |
| bind(&SegmentFetcher::afterValidationFailure, this, _1, _2, weakSelf)); |
| } |
| |
| void |
| SegmentFetcher::afterValidationSuccess(const Data& data, const Interest& origInterest, |
| std::map<uint64_t, PendingSegment>::iterator pendingSegmentIt, |
| const weak_ptr<SegmentFetcher>& weakSelf) |
| { |
| if (shouldStop(weakSelf)) |
| return; |
| |
| // We update the last receive time here instead of in the segment received callback so that the |
| // transfer will not fail to terminate if we only received invalid Data packets. |
| m_timeLastSegmentReceived = time::steady_clock::now(); |
| |
| m_nReceived++; |
| |
| // It was verified in afterSegmentReceivedCb that the last Data name component is a segment number |
| uint64_t currentSegment = data.getName().get(-1).toSegment(); |
| // Add measurement to RTO estimator (if not retransmission) |
| if (pendingSegmentIt->second.state == SegmentState::FirstInterest) { |
| m_rttEstimator.addMeasurement(m_timeLastSegmentReceived - pendingSegmentIt->second.sendTime, |
| std::max<int64_t>(m_nSegmentsInFlight + 1, 1)); |
| } |
| |
| // Remove from pending segments map |
| m_pendingSegments.erase(pendingSegmentIt); |
| |
| // Copy data in segment to temporary buffer |
| auto receivedSegmentIt = m_receivedSegments.emplace(std::piecewise_construct, |
| std::forward_as_tuple(currentSegment), |
| std::forward_as_tuple(data.getContent().value_size())); |
| std::copy(data.getContent().value_begin(), data.getContent().value_end(), |
| receivedSegmentIt.first->second.begin()); |
| m_nBytesReceived += data.getContent().value_size(); |
| afterSegmentValidated(data); |
| |
| if (data.getFinalBlock()) { |
| if (!data.getFinalBlock()->isSegment()) { |
| return signalError(FINALBLOCKID_NOT_SEGMENT, |
| "Received FinalBlockId did not contain a segment component"); |
| } |
| |
| if (data.getFinalBlock()->toSegment() + 1 != static_cast<uint64_t>(m_nSegments)) { |
| m_nSegments = data.getFinalBlock()->toSegment() + 1; |
| cancelExcessInFlightSegments(); |
| } |
| } |
| |
| if (m_receivedSegments.size() == 1) { |
| m_versionedDataName = data.getName().getPrefix(-1); |
| if (currentSegment == 0) { |
| // We received the first segment in response, so we can increment the next segment number |
| m_nextSegmentNum++; |
| } |
| } |
| |
| if (m_highData < currentSegment) { |
| m_highData = currentSegment; |
| } |
| |
| if (data.getCongestionMark() > 0 && !m_options.ignoreCongMarks) { |
| windowDecrease(); |
| } |
| else { |
| windowIncrease(); |
| } |
| |
| fetchSegmentsInWindow(origInterest); |
| } |
| |
| void |
| SegmentFetcher::afterValidationFailure(const Data& data, |
| const security::v2::ValidationError& error, |
| const weak_ptr<SegmentFetcher>& weakSelf) |
| { |
| if (shouldStop(weakSelf)) |
| return; |
| |
| signalError(SEGMENT_VALIDATION_FAIL, "Segment validation failed: " + boost::lexical_cast<std::string>(error)); |
| } |
| |
| void |
| SegmentFetcher::afterNackReceivedCb(const Interest& origInterest, const lp::Nack& nack, |
| const weak_ptr<SegmentFetcher>& weakSelf) |
| { |
| if (shouldStop(weakSelf)) |
| return; |
| |
| afterSegmentNacked(); |
| |
| BOOST_ASSERT(m_nSegmentsInFlight > 0); |
| m_nSegmentsInFlight--; |
| |
| switch (nack.getReason()) { |
| case lp::NackReason::DUPLICATE: |
| case lp::NackReason::CONGESTION: |
| afterNackOrTimeout(origInterest); |
| break; |
| default: |
| signalError(NACK_ERROR, "Nack Error"); |
| break; |
| } |
| } |
| |
| void |
| SegmentFetcher::afterTimeoutCb(const Interest& origInterest, |
| const weak_ptr<SegmentFetcher>& weakSelf) |
| { |
| if (shouldStop(weakSelf)) |
| return; |
| |
| afterSegmentTimedOut(); |
| |
| BOOST_ASSERT(m_nSegmentsInFlight > 0); |
| m_nSegmentsInFlight--; |
| afterNackOrTimeout(origInterest); |
| } |
| |
| void |
| SegmentFetcher::afterNackOrTimeout(const Interest& origInterest) |
| { |
| if (time::steady_clock::now() >= m_timeLastSegmentReceived + m_options.maxTimeout) { |
| // Fail transfer due to exceeding the maximum timeout between the successful receipt of segments |
| return signalError(INTEREST_TIMEOUT, "Timeout exceeded"); |
| } |
| |
| name::Component lastNameComponent = origInterest.getName().get(-1); |
| std::map<uint64_t, PendingSegment>::iterator pendingSegmentIt; |
| BOOST_ASSERT(m_pendingSegments.size() > 0); |
| if (lastNameComponent.isSegment()) { |
| BOOST_ASSERT(m_pendingSegments.count(lastNameComponent.toSegment()) > 0); |
| pendingSegmentIt = m_pendingSegments.find(lastNameComponent.toSegment()); |
| } |
| else { // First Interest |
| BOOST_ASSERT(m_pendingSegments.size() > 0); |
| pendingSegmentIt = m_pendingSegments.begin(); |
| } |
| |
| // Cancel timeout event and set status to InRetxQueue |
| m_scheduler.cancelEvent(pendingSegmentIt->second.timeoutEvent); |
| pendingSegmentIt->second.timeoutEvent = nullptr; |
| pendingSegmentIt->second.state = SegmentState::InRetxQueue; |
| |
| m_rttEstimator.backoffRto(); |
| |
| if (m_receivedSegments.size() == 0) { |
| // Resend first Interest (until maximum receive timeout exceeded) |
| fetchFirstSegment(origInterest, true); |
| } |
| else { |
| windowDecrease(); |
| m_retxQueue.push(pendingSegmentIt->first); |
| fetchSegmentsInWindow(origInterest); |
| } |
| } |
| |
| void |
| SegmentFetcher::finalizeFetch() |
| { |
| // Combine segments into final buffer |
| OBufferStream buf; |
| // We may have received more segments than exist in the object. |
| BOOST_ASSERT(m_receivedSegments.size() >= static_cast<uint64_t>(m_nSegments)); |
| |
| for (int64_t i = 0; i < m_nSegments; i++) { |
| buf.write(m_receivedSegments[i].get<const char>(), m_receivedSegments[i].size()); |
| } |
| |
| onComplete(buf.buf()); |
| stop(); |
| } |
| |
| void |
| SegmentFetcher::windowIncrease() |
| { |
| if (m_options.useConstantCwnd) { |
| BOOST_ASSERT(m_cwnd == m_options.initCwnd); |
| return; |
| } |
| |
| if (m_cwnd < m_ssthresh) { |
| m_cwnd += m_options.aiStep; // additive increase |
| } |
| else { |
| m_cwnd += m_options.aiStep / std::floor(m_cwnd); // congestion avoidance |
| } |
| } |
| |
| void |
| SegmentFetcher::windowDecrease() |
| { |
| if (m_options.disableCwa || m_highData > m_recPoint) { |
| m_recPoint = m_highInterest; |
| |
| if (m_options.useConstantCwnd) { |
| BOOST_ASSERT(m_cwnd == m_options.initCwnd); |
| return; |
| } |
| |
| // Refer to RFC 5681, Section 3.1 for the rationale behind the code below |
| m_ssthresh = std::max(MIN_SSTHRESH, m_cwnd * m_options.mdCoef); // multiplicative decrease |
| m_cwnd = m_options.resetCwndToInit ? m_options.initCwnd : m_ssthresh; |
| } |
| } |
| |
| void |
| SegmentFetcher::signalError(uint32_t code, const std::string& msg) |
| { |
| onError(code, msg); |
| stop(); |
| } |
| |
| void |
| SegmentFetcher::updateRetransmittedSegment(uint64_t segmentNum, |
| const PendingInterestId* pendingInterest, |
| scheduler::EventId timeoutEvent) |
| { |
| auto pendingSegmentIt = m_pendingSegments.find(segmentNum); |
| BOOST_ASSERT(pendingSegmentIt != m_pendingSegments.end()); |
| BOOST_ASSERT(pendingSegmentIt->second.state == SegmentState::InRetxQueue); |
| pendingSegmentIt->second.state = SegmentState::Retransmitted; |
| m_face.removePendingInterest(pendingSegmentIt->second.id); |
| pendingSegmentIt->second.id = pendingInterest; |
| pendingSegmentIt->second.timeoutEvent = timeoutEvent; |
| } |
| |
| void |
| SegmentFetcher::cancelExcessInFlightSegments() |
| { |
| for (auto it = m_pendingSegments.begin(); it != m_pendingSegments.end();) { |
| if (it->first >= static_cast<uint64_t>(m_nSegments)) { |
| m_face.removePendingInterest(it->second.id); |
| if (it->second.timeoutEvent) { |
| m_scheduler.cancelEvent(it->second.timeoutEvent); |
| } |
| it = m_pendingSegments.erase(it); |
| BOOST_ASSERT(m_nSegmentsInFlight > 0); |
| m_nSegmentsInFlight--; |
| } |
| else { |
| ++it; |
| } |
| } |
| } |
| |
| bool |
| SegmentFetcher::checkAllSegmentsReceived() |
| { |
| bool haveReceivedAllSegments = false; |
| |
| if (m_nSegments != 0 && m_nReceived >= m_nSegments) { |
| haveReceivedAllSegments = true; |
| // Verify that all segments in window have been received. If not, send Interests for missing segments. |
| for (uint64_t i = 0; i < static_cast<uint64_t>(m_nSegments); i++) { |
| if (m_receivedSegments.count(i) == 0) { |
| m_retxQueue.push(i); |
| haveReceivedAllSegments = false; |
| } |
| } |
| } |
| |
| return haveReceivedAllSegments; |
| } |
| |
| time::milliseconds |
| SegmentFetcher::getEstimatedRto() |
| { |
| // We don't want an Interest timeout greater than the maximum allowed timeout between the |
| // succesful receipt of segments |
| return std::min(m_options.maxTimeout, |
| time::duration_cast<time::milliseconds>(m_rttEstimator.getEstimatedRto())); |
| } |
| |
| } // namespace util |
| } // namespace ndn |