blob: fa8737e91ce8f34299726de00b5cb9c866afe04c [file] [log] [blame]
Alexander Afanasyevdfe58192013-01-17 17:34:04 -08001/* -*- Mode: C++; c-file-style: "gnu"; indent-tabs-mode:nil -*- */
2/*
3 * Copyright (c) 2012 University of California, Los Angeles
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2 as
7 * published by the Free Software Foundation;
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software
16 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17 *
18 * Author: Alexander Afanasyev <alexander.afanasyev@ucla.edu>
19 * Zhenkai Zhu <zhenkai@cs.ucla.edu>
20 */
21
22#include "fetcher.h"
Alexander Afanasyevd6c2a902013-01-19 21:24:30 -080023#include "fetch-manager.h"
Alexander Afanasyev66f4c492013-01-20 23:32:50 -080024#include "ccnx-pco.h"
Alexander Afanasyev678f3502013-01-23 17:09:37 -080025#include "logging.h"
Alexander Afanasyevd6c2a902013-01-19 21:24:30 -080026
Alexander Afanasyevdfe58192013-01-17 17:34:04 -080027#include <boost/make_shared.hpp>
28#include <boost/ref.hpp>
29#include <boost/throw_exception.hpp>
Alexander Afanasyev21a166e2013-01-20 16:04:41 -080030#include <boost/date_time/posix_time/posix_time.hpp>
Alexander Afanasyevdfe58192013-01-17 17:34:04 -080031
Alexander Afanasyev678f3502013-01-23 17:09:37 -080032INIT_LOGGER ("Fetcher");
33
Alexander Afanasyevdfe58192013-01-17 17:34:04 -080034using namespace boost;
35using namespace std;
Alexander Afanasyev50547892013-01-19 22:03:45 -080036using namespace Ccnx;
Alexander Afanasyevdfe58192013-01-17 17:34:04 -080037
Zhenkai Zhu3d1beca2013-01-23 14:55:32 -080038Fetcher::Fetcher (Ccnx::CcnxWrapperPtr ccnx,
39 const SegmentCallback &segmentCallback,
40 const FinishCallback &finishCallback,
Alexander Afanasyev21a166e2013-01-20 16:04:41 -080041 OnFetchCompleteCallback onFetchComplete, OnFetchFailedCallback onFetchFailed,
Zhenkai Zhu3d1beca2013-01-23 14:55:32 -080042 const Ccnx::Name &deviceName, const Ccnx::Name &name, int64_t minSeqNo, int64_t maxSeqNo,
Alexander Afanasyev21a166e2013-01-20 16:04:41 -080043 boost::posix_time::time_duration timeout/* = boost::posix_time::seconds (30)*/,
Alexander Afanasyev49a18522013-01-18 17:49:04 -080044 const Ccnx::Name &forwardingHint/* = Ccnx::Name ()*/)
Alexander Afanasyev21a166e2013-01-20 16:04:41 -080045 : m_ccnx (ccnx)
46
Zhenkai Zhu3d1beca2013-01-23 14:55:32 -080047 , m_segmentCallback (segmentCallback)
Alexander Afanasyev21a166e2013-01-20 16:04:41 -080048 , m_onFetchComplete (onFetchComplete)
49 , m_onFetchFailed (onFetchFailed)
Zhenkai Zhu3d1beca2013-01-23 14:55:32 -080050 , m_finishCallback (finishCallback)
Alexander Afanasyev21a166e2013-01-20 16:04:41 -080051
Alexander Afanasyev83531a42013-01-19 16:21:54 -080052 , m_active (false)
Alexander Afanasyev49a18522013-01-18 17:49:04 -080053 , m_name (name)
Zhenkai Zhu3d1beca2013-01-23 14:55:32 -080054 , m_deviceName (deviceName)
Alexander Afanasyev49a18522013-01-18 17:49:04 -080055 , m_forwardingHint (forwardingHint)
Alexander Afanasyev21a166e2013-01-20 16:04:41 -080056 , m_maximumNoActivityPeriod (timeout)
57
Alexander Afanasyev28ca3ed2013-01-24 23:17:15 -080058 , m_minSendSeqNo (minSeqNo-1)
59 , m_maxInOrderRecvSeqNo (minSeqNo-1)
Alexander Afanasyev49a18522013-01-18 17:49:04 -080060 , m_minSeqNo (minSeqNo)
61 , m_maxSeqNo (maxSeqNo)
Alexander Afanasyevdfe58192013-01-17 17:34:04 -080062
Alexander Afanasyev49a18522013-01-18 17:49:04 -080063 , m_pipeline (6) // initial "congestion window"
Alexander Afanasyev50547892013-01-19 22:03:45 -080064 , m_activePipeline (0)
Alexander Afanasyev548d38d2013-01-26 16:36:06 -080065 , m_retryPause (0)
Alexander Afanasyevf8b22bf2013-01-26 17:28:11 -080066 , m_nextScheduledRetry (date_time::second_clock<boost::posix_time::ptime>::universal_time ())
Alexander Afanasyevca77d4e2013-01-28 22:50:05 -080067 , m_executor (1) // must be 1
Alexander Afanasyevdfe58192013-01-17 17:34:04 -080068{
Alexander Afanasyevca77d4e2013-01-28 22:50:05 -080069 m_executor.start ();
Alexander Afanasyevdfe58192013-01-17 17:34:04 -080070}
71
72Fetcher::~Fetcher ()
73{
Alexander Afanasyevca77d4e2013-01-28 22:50:05 -080074 m_executor.shutdown ();
Alexander Afanasyevdfe58192013-01-17 17:34:04 -080075}
Alexander Afanasyev83531a42013-01-19 16:21:54 -080076
77void
78Fetcher::RestartPipeline ()
79{
80 m_active = true;
Alexander Afanasyev50547892013-01-19 22:03:45 -080081 m_minSendSeqNo = m_maxInOrderRecvSeqNo;
Alexander Afanasyev650ba282013-01-20 20:14:06 -080082 // cout << "Restart: " << m_minSendSeqNo << endl;
Alexander Afanasyev21a166e2013-01-20 16:04:41 -080083 m_lastPositiveActivity = date_time::second_clock<boost::posix_time::ptime>::universal_time();
Alexander Afanasyev50547892013-01-19 22:03:45 -080084
Alexander Afanasyevca77d4e2013-01-28 22:50:05 -080085 m_executor.execute (bind (&Fetcher::FillPipeline, this));
Alexander Afanasyev50547892013-01-19 22:03:45 -080086}
87
88void
Alexander Afanasyev21a166e2013-01-20 16:04:41 -080089Fetcher::SetForwardingHint (const Ccnx::Name &forwardingHint)
90{
91 m_forwardingHint = forwardingHint;
92}
93
94void
Alexander Afanasyev50547892013-01-19 22:03:45 -080095Fetcher::FillPipeline ()
96{
97 for (; m_minSendSeqNo < m_maxSeqNo && m_activePipeline < m_pipeline; m_minSendSeqNo++)
98 {
Alexander Afanasyev650ba282013-01-20 20:14:06 -080099 if (m_outOfOrderRecvSeqNo.find (m_minSendSeqNo+1) != m_outOfOrderRecvSeqNo.end ())
100 continue;
101
Alexander Afanasyevcfd5dfc2013-01-28 22:21:31 -0800102 if (m_inActivePipeline.find (m_minSendSeqNo+1) != m_inActivePipeline.end ())
103 continue;
104
105 m_inActivePipeline.insert (m_minSendSeqNo+1);
106
Alexander Afanasyevfc720362013-01-24 21:49:48 -0800107 _LOG_DEBUG (" >>> i " << Name (m_forwardingHint)(m_name) << ", seq = " << (m_minSendSeqNo + 1 ));
Alexander Afanasyev678f3502013-01-23 17:09:37 -0800108
Alexander Afanasyev650ba282013-01-20 20:14:06 -0800109 // cout << ">>> " << m_minSendSeqNo+1 << endl;
Alexander Afanasyev66f4c492013-01-20 23:32:50 -0800110 m_ccnx->sendInterest (Name (m_forwardingHint)(m_name)(m_minSendSeqNo+1),
Alexander Afanasyev21a166e2013-01-20 16:04:41 -0800111 Closure (bind(&Fetcher::OnData, this, m_minSendSeqNo+1, _1, _2),
Zhenkai Zhuff4fa8a2013-01-28 22:02:40 -0800112 bind(&Fetcher::OnTimeout, this, m_minSendSeqNo+1, _1, _2, _3)),
Alexander Afanasyev650ba282013-01-20 20:14:06 -0800113 Selectors().interestLifetime (1)); // Alex: this lifetime should be changed to RTO
Alexander Afanasyevff8d9dc2013-01-26 00:45:08 -0800114 _LOG_DEBUG (" >>> i ok");
Alexander Afanasyev50547892013-01-19 22:03:45 -0800115
116 m_activePipeline ++;
117 }
Alexander Afanasyev83531a42013-01-19 16:21:54 -0800118}
Alexander Afanasyevd6c2a902013-01-19 21:24:30 -0800119
120void
Zhenkai Zhu3d1beca2013-01-23 14:55:32 -0800121Fetcher::OnData (uint64_t seqno, const Ccnx::Name &name, PcoPtr data)
Alexander Afanasyevd6c2a902013-01-19 21:24:30 -0800122{
Alexander Afanasyevddc283c2013-01-28 23:11:20 -0800123 m_executor.execute (bind (&Fetcher::OnData_Execute, this, seqno, name, data));
124}
125
126void
127Fetcher::OnData_Execute (uint64_t seqno, Ccnx::Name name, Ccnx::PcoPtr data)
128{
Alexander Afanasyevfc720362013-01-24 21:49:48 -0800129 _LOG_DEBUG (" <<< d " << name.getPartialName (0, name.size () - 1) << ", seq = " << seqno);
130
Alexander Afanasyev66f4c492013-01-20 23:32:50 -0800131 if (m_forwardingHint == Name ())
Zhenkai Zhu3d1beca2013-01-23 14:55:32 -0800132 {
133 // invoke callback
Alexander Afanasyev28ca3ed2013-01-24 23:17:15 -0800134 if (!m_segmentCallback.empty ())
135 {
136 m_segmentCallback (m_deviceName, m_name, seqno, data);
137 }
Zhenkai Zhu3d1beca2013-01-23 14:55:32 -0800138 // we don't have to tell FetchManager about this
139 }
Alexander Afanasyev66f4c492013-01-20 23:32:50 -0800140 else
141 {
142 try {
Alexander Afanasyevf278db32013-01-21 14:41:01 -0800143 PcoPtr pco = make_shared<ParsedContentObject> (*data->contentPtr ());
Alexander Afanasyev28ca3ed2013-01-24 23:17:15 -0800144 if (!m_segmentCallback.empty ())
145 {
146 m_segmentCallback (m_deviceName, m_name, seqno, pco);
147 }
Alexander Afanasyev66f4c492013-01-20 23:32:50 -0800148 }
149 catch (MisformedContentObjectException &e)
150 {
151 cerr << "MisformedContentObjectException..." << endl;
152 // no idea what should do...
153 // let's ignore for now
154 }
155 }
Alexander Afanasyev21a166e2013-01-20 16:04:41 -0800156
Alexander Afanasyev50547892013-01-19 22:03:45 -0800157 m_activePipeline --;
Alexander Afanasyev21a166e2013-01-20 16:04:41 -0800158 m_lastPositiveActivity = date_time::second_clock<boost::posix_time::ptime>::universal_time();
Alexander Afanasyev50547892013-01-19 22:03:45 -0800159
160 ////////////////////////////////////////////////////////////////////////////
161 m_outOfOrderRecvSeqNo.insert (seqno);
Alexander Afanasyevcfd5dfc2013-01-28 22:21:31 -0800162 m_inActivePipeline.erase (seqno);
163 _LOG_DEBUG ("Total segments received: " << m_outOfOrderRecvSeqNo.size ());
Zhenkai Zhu3d1beca2013-01-23 14:55:32 -0800164 set<int64_t>::iterator inOrderSeqNo = m_outOfOrderRecvSeqNo.begin ();
Alexander Afanasyev50547892013-01-19 22:03:45 -0800165 for (; inOrderSeqNo != m_outOfOrderRecvSeqNo.end ();
166 inOrderSeqNo++)
167 {
Alexander Afanasyevcfd5dfc2013-01-28 22:21:31 -0800168 _LOG_TRACE ("Checking " << *inOrderSeqNo << " and " << m_maxInOrderRecvSeqNo+1);
Alexander Afanasyev50547892013-01-19 22:03:45 -0800169 if (*inOrderSeqNo == m_maxInOrderRecvSeqNo+1)
170 {
171 m_maxInOrderRecvSeqNo = *inOrderSeqNo;
172 }
Alexander Afanasyevcfd5dfc2013-01-28 22:21:31 -0800173 else if (*inOrderSeqNo < m_maxInOrderRecvSeqNo+1) // not possible anymore, but just in case
174 {
175 continue;
176 }
Alexander Afanasyev50547892013-01-19 22:03:45 -0800177 else
178 break;
179 }
180 m_outOfOrderRecvSeqNo.erase (m_outOfOrderRecvSeqNo.begin (), inOrderSeqNo);
181 ////////////////////////////////////////////////////////////////////////////
182
Alexander Afanasyevcfd5dfc2013-01-28 22:21:31 -0800183 _LOG_TRACE ("Max in order received: " << m_maxInOrderRecvSeqNo << ", max seqNo to request: " << m_maxSeqNo);
184
Alexander Afanasyev21a166e2013-01-20 16:04:41 -0800185 if (m_maxInOrderRecvSeqNo == m_maxSeqNo)
Alexander Afanasyevd6c2a902013-01-19 21:24:30 -0800186 {
Alexander Afanasyevcfd5dfc2013-01-28 22:21:31 -0800187 _LOG_TRACE ("Fetch finished");
Alexander Afanasyevd6c2a902013-01-19 21:24:30 -0800188 m_active = false;
Zhenkai Zhu3d1beca2013-01-23 14:55:32 -0800189 // invoke callback
Alexander Afanasyev28ca3ed2013-01-24 23:17:15 -0800190 if (!m_finishCallback.empty ())
191 {
Alexander Afanasyevcfd5dfc2013-01-28 22:21:31 -0800192 _LOG_TRACE ("Notifying callback");
Alexander Afanasyev28ca3ed2013-01-24 23:17:15 -0800193 m_finishCallback(m_deviceName, m_name);
194 }
195
Zhenkai Zhu3d1beca2013-01-23 14:55:32 -0800196 // tell FetchManager that we have finish our job
Alexander Afanasyev21a166e2013-01-20 16:04:41 -0800197 m_onFetchComplete (*this);
198 }
199 else
200 {
Alexander Afanasyevca77d4e2013-01-28 22:50:05 -0800201 m_executor.execute (bind (&Fetcher::FillPipeline, this));
Alexander Afanasyevd6c2a902013-01-19 21:24:30 -0800202 }
203}
204
Zhenkai Zhuff4fa8a2013-01-28 22:02:40 -0800205void
206Fetcher::OnTimeout (uint64_t seqno, const Ccnx::Name &name, const Closure &closure, Selectors selectors)
Alexander Afanasyevd6c2a902013-01-19 21:24:30 -0800207{
Alexander Afanasyevddc283c2013-01-28 23:11:20 -0800208 m_executor.execute (bind (&Fetcher::OnTimeout_Execute, this, seqno, name, closure, selectors));
209}
210
211void
Alexander Afanasyev67d757c2013-01-28 23:22:12 -0800212Fetcher::OnTimeout_Execute (uint64_t seqno, Ccnx::Name name, Ccnx::Closure closure, Ccnx::Selectors selectors)
Alexander Afanasyevddc283c2013-01-28 23:11:20 -0800213{
Alexander Afanasyevff8d9dc2013-01-26 00:45:08 -0800214 _LOG_DEBUG (" <<< :( timeout " << name.getPartialName (0, name.size () - 1) << ", seq = " << seqno);
Alexander Afanasyev548d38d2013-01-26 16:36:06 -0800215
Alexander Afanasyev21a166e2013-01-20 16:04:41 -0800216 // cout << "Fetcher::OnTimeout: " << name << endl;
217 // cout << "Last: " << m_lastPositiveActivity << ", config: " << m_maximumNoActivityPeriod
218 // << ", now: " << date_time::second_clock<boost::posix_time::ptime>::universal_time()
219 // << ", oldest: " << (date_time::second_clock<boost::posix_time::ptime>::universal_time() - m_maximumNoActivityPeriod) << endl;
220
221 if (m_lastPositiveActivity <
222 (date_time::second_clock<boost::posix_time::ptime>::universal_time() - m_maximumNoActivityPeriod))
223 {
Alexander Afanasyevcfd5dfc2013-01-28 22:21:31 -0800224 m_inActivePipeline.erase (seqno);
Alexander Afanasyev21a166e2013-01-20 16:04:41 -0800225 m_activePipeline --;
226 if (m_activePipeline == 0)
227 {
Alexander Afanasyevff8d9dc2013-01-26 00:45:08 -0800228 _LOG_DEBUG ("Telling that fetch failed");
Alexander Afanasyevcfd5dfc2013-01-28 22:21:31 -0800229 _LOG_DEBUG ("Active pipeline size should be zero: " << m_inActivePipeline.size ());
230
Alexander Afanasyev650ba282013-01-20 20:14:06 -0800231 m_active = false;
Alexander Afanasyev21a166e2013-01-20 16:04:41 -0800232 m_onFetchFailed (*this);
233 // this is not valid anymore, but we still should be able finish work
234 }
Alexander Afanasyev21a166e2013-01-20 16:04:41 -0800235 }
236 else
Alexander Afanasyevff8d9dc2013-01-26 00:45:08 -0800237 {
Alexander Afanasyevcfd5dfc2013-01-28 22:21:31 -0800238 _LOG_DEBUG ("Asking to reexpress seqno: " << seqno);
Zhenkai Zhuff4fa8a2013-01-28 22:02:40 -0800239 m_ccnx->sendInterest (name, closure, selectors);
Alexander Afanasyevff8d9dc2013-01-26 00:45:08 -0800240 }
Alexander Afanasyevd6c2a902013-01-19 21:24:30 -0800241}