blob: 4a5d5b83909c035056a1214a5ef9490bde005e5a [file] [log] [blame]
Jeff Thompson25b4e612013-10-10 16:03:24 -07001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil -*- */
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -07002/**
Alexander Afanasyevdfa52c42014-04-24 21:10:11 -07003 * Copyright (c) 2013-2014, Regents of the University of California.
4 * All rights reserved.
5 *
6 * This file is part of ndn-cxx library (NDN C++ library with eXperimental eXtensions).
7 * See AUTHORS.md for complete list of ndn-cxx authors and contributors.
8 *
9 * This file licensed under New BSD License. See COPYING for detailed information about
10 * ndn-cxx library copyright, permissions, and redistribution restrictions.
11 *
12 * Based on code originally written by Jeff Thompson <jefft0@remap.ucla.edu>
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -070013 */
14
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080015#include "common.hpp"
16
Alexander Afanasyev09c613f2014-01-29 00:23:58 -080017#include "face.hpp"
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -070018
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080019#include "security/signature-sha256-with-rsa.hpp"
20
21#include "util/time.hpp"
22#include "util/random.hpp"
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -060023#include "util/config-file.hpp"
Alexander Afanasyevf7ca3202014-02-14 22:28:31 -080024#include <cstdlib>
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080025
26#include "management/ndnd-controller.hpp"
27#include "management/nfd-controller.hpp"
Alexander Afanasyevefe3ab22014-02-19 14:57:50 -080028#include "management/nrd-controller.hpp"
Jeff Thompsonb982b6d2013-07-15 18:15:45 -070029
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -070030namespace ndn {
Alexander Afanasyevb790d952014-01-24 12:07:53 -080031
Alexander Afanasyevf7ca3202014-02-14 22:28:31 -080032Face::Face()
Jeff Thompsonfb29cda2013-08-24 10:26:54 -070033{
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -060034 const std::string socketName = UnixTransport::getDefaultSocketName(m_config);
35 construct(shared_ptr<Transport>(new UnixTransport(socketName)),
Alexander Afanasyev505646e2014-02-24 20:13:37 -080036 make_shared<boost::asio::io_service>());
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080037}
38
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -080039Face::Face(const shared_ptr<boost::asio::io_service>& ioService)
40{
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -060041 const std::string socketName = UnixTransport::getDefaultSocketName(m_config);
42 construct(shared_ptr<Transport>(new UnixTransport(socketName)),
Alexander Afanasyev505646e2014-02-24 20:13:37 -080043 ioService);
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -080044}
45
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -070046class NullIoDeleter
47{
48public:
49 void
50 operator()(boost::asio::io_service*)
51 {
52 }
53};
54
55Face::Face(boost::asio::io_service& ioService)
56{
57 const std::string socketName = UnixTransport::getDefaultSocketName(m_config);
58 construct(shared_ptr<Transport>(new UnixTransport(socketName)),
59 shared_ptr<boost::asio::io_service>(&ioService, NullIoDeleter()));
60}
61
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -080062Face::Face(const std::string& host, const std::string& port/* = "6363"*/)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080063{
64 construct(shared_ptr<Transport>(new TcpTransport(host, port)),
Alexander Afanasyev505646e2014-02-24 20:13:37 -080065 make_shared<boost::asio::io_service>());
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080066}
67
Alexander Afanasyevf7ca3202014-02-14 22:28:31 -080068Face::Face(const shared_ptr<Transport>& transport)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080069{
70 construct(transport,
Alexander Afanasyev505646e2014-02-24 20:13:37 -080071 make_shared<boost::asio::io_service>());
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080072}
73
74Face::Face(const shared_ptr<Transport>& transport,
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -070075 boost::asio::io_service& ioService)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080076{
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -070077 construct(transport,
78 shared_ptr<boost::asio::io_service>(&ioService, NullIoDeleter()));
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -080079}
80
Alexander Afanasyev505646e2014-02-24 20:13:37 -080081void
82Face::setController(const shared_ptr<Controller>& controller)
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -080083{
Alexander Afanasyev505646e2014-02-24 20:13:37 -080084 m_fwController = controller;
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080085}
86
87void
88Face::construct(const shared_ptr<Transport>& transport,
Alexander Afanasyev505646e2014-02-24 20:13:37 -080089 const shared_ptr<boost::asio::io_service>& ioService)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080090{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -080091 m_pitTimeoutCheckTimerActive = false;
92 m_transport = transport;
93 m_ioService = ioService;
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080094
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -070095 m_pitTimeoutCheckTimer = make_shared<monotonic_deadline_timer>(boost::ref(*m_ioService));
96 m_processEventsTimeoutTimer = make_shared<monotonic_deadline_timer>(boost::ref(*m_ioService));
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -080097
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -060098 std::string protocol = "nrd-0.1";
99
100 try
Alexander Afanasyevefe3ab22014-02-19 14:57:50 -0800101 {
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600102 protocol = m_config.getParsedConfiguration().get<std::string>("protocol");
103 }
Alexander Afanasyev2a7f7202014-04-23 14:25:29 -0700104 catch (boost::property_tree::ptree_bad_path& error)
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600105 {
106 // protocol not specified
107 }
Alexander Afanasyev2a7f7202014-04-23 14:25:29 -0700108 catch (boost::property_tree::ptree_bad_data& error)
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600109 {
110 throw ConfigFile::Error(error.what());
111 }
112
113 if (isSupportedNrdProtocol(protocol))
114 {
115 m_fwController = make_shared<nrd::Controller>(boost::ref(*this));
116 }
Steve DiBenedettoacab8802014-03-24 11:15:57 -0600117 else if (isSupportedNfdProtocol(protocol))
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600118 {
119 m_fwController = make_shared<nfd::Controller>(boost::ref(*this));
120 }
121 else if (isSupportedNdndProtocol(protocol))
122 {
123 m_fwController = make_shared<ndnd::Controller>(boost::ref(*this));
Alexander Afanasyevefe3ab22014-02-19 14:57:50 -0800124 }
Jeff Thompsonfb29cda2013-08-24 10:26:54 -0700125 else
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600126 {
127 throw Face::Error("Cannot create controller for unsupported protocol \"" + protocol + "\"");
128 }
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800129}
130
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800131const PendingInterestId*
132Face::expressInterest(const Interest& interest, const OnData& onData, const OnTimeout& onTimeout)
133{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800134 if (!m_transport->isConnected())
135 m_transport->connect(*m_ioService,
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800136 bind(&Face::onReceiveElement, this, _1));
137
138 shared_ptr<const Interest> interestToExpress(new Interest(interest));
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800139
Alexander Afanasyev4e50b972014-03-25 10:57:50 -0700140 // If the same ioService thread, dispatch directly calls the method
141 m_ioService->dispatch(bind(&Face::asyncExpressInterest, this,
142 interestToExpress, onData, onTimeout));
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800143
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800144 return reinterpret_cast<const PendingInterestId*>(interestToExpress.get());
145}
146
147const PendingInterestId*
148Face::expressInterest(const Name& name,
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800149 const Interest& tmpl,
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800150 const OnData& onData, const OnTimeout& onTimeout/* = OnTimeout()*/)
151{
152 return expressInterest(Interest(name,
153 tmpl.getMinSuffixComponents(),
154 tmpl.getMaxSuffixComponents(),
155 tmpl.getExclude(),
156 tmpl.getChildSelector(),
157 tmpl.getMustBeFresh(),
158 tmpl.getScope(),
159 tmpl.getInterestLifetime()),
160 onData, onTimeout);
161}
162
163void
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800164Face::asyncExpressInterest(const shared_ptr<const Interest>& interest,
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800165 const OnData& onData, const OnTimeout& onTimeout)
166{
Alexander Afanasyev4e50b972014-03-25 10:57:50 -0700167 if (!m_transport->isExpectingData())
168 m_transport->resume();
169
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800170 m_pendingInterestTable.push_back(shared_ptr<PendingInterest>(new PendingInterest
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800171 (interest, onData, onTimeout)));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800172
Alexander Afanasyev5964fb72014-02-18 12:42:45 -0800173 if (!interest->getLocalControlHeader().empty(false, true))
174 {
175 // encode only NextHopFaceId towards the forwarder
176 m_transport->send(interest->getLocalControlHeader().wireEncode(*interest, false, true),
177 interest->wireEncode());
178 }
179 else
180 {
181 m_transport->send(interest->wireEncode());
182 }
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800183
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800184 if (!m_pitTimeoutCheckTimerActive) {
185 m_pitTimeoutCheckTimerActive = true;
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700186 m_pitTimeoutCheckTimer->expires_from_now(time::milliseconds(100));
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800187 m_pitTimeoutCheckTimer->async_wait(bind(&Face::checkPitExpire, this));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800188 }
189}
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800190
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800191void
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800192Face::put(const Data& data)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800193{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800194 if (!m_transport->isConnected())
195 m_transport->connect(*m_ioService,
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800196 bind(&Face::onReceiveElement, this, _1));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800197
Alexander Afanasyev5964fb72014-02-18 12:42:45 -0800198 if (!data.getLocalControlHeader().empty(false, true))
199 {
Alexander Afanasyev83018352014-02-18 19:52:15 -0800200 m_transport->send(data.getLocalControlHeader().wireEncode(data, false, true),
201 data.wireEncode());
Alexander Afanasyev5964fb72014-02-18 12:42:45 -0800202 }
203 else
204 {
Alexander Afanasyev83018352014-02-18 19:52:15 -0800205 m_transport->send(data.wireEncode());
Alexander Afanasyev5964fb72014-02-18 12:42:45 -0800206 }
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800207}
208
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800209void
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800210Face::removePendingInterest(const PendingInterestId* pendingInterestId)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800211{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800212 m_ioService->post(bind(&Face::asyncRemovePendingInterest, this, pendingInterestId));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800213}
214
215
216void
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800217Face::asyncRemovePendingInterest(const PendingInterestId* pendingInterestId)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800218{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800219 m_pendingInterestTable.remove_if(MatchPendingInterestId(pendingInterestId));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800220}
221
222const RegisteredPrefixId*
223Face::setInterestFilter(const Name& prefix,
224 const OnInterest& onInterest,
225 const OnSetInterestFilterFailed& onSetInterestFilterFailed)
226{
227 shared_ptr<RegisteredPrefix> prefixToRegister(new RegisteredPrefix(prefix, onInterest));
228
229 m_fwController->selfRegisterPrefix(prefixToRegister->getPrefix(),
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -0700230 bind(&RegisteredPrefixTable::push_back,
231 &m_registeredPrefixTable, prefixToRegister),
232 bind(onSetInterestFilterFailed,
233 prefixToRegister->getPrefix(), _1));
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800234
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800235 return reinterpret_cast<const RegisteredPrefixId*>(prefixToRegister.get());
236}
237
238void
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800239Face::unsetInterestFilter(const RegisteredPrefixId* registeredPrefixId)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800240{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800241 m_ioService->post(bind(&Face::asyncUnsetInterestFilter, this, registeredPrefixId));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800242}
243
244void
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800245Face::asyncUnsetInterestFilter(const RegisteredPrefixId* registeredPrefixId)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800246{
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -0700247 RegisteredPrefixTable::iterator i = std::find_if(m_registeredPrefixTable.begin(),
248 m_registeredPrefixTable.end(),
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800249 MatchRegisteredPrefixId(registeredPrefixId));
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800250 if (i != m_registeredPrefixTable.end())
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800251 {
252 m_fwController->selfDeregisterPrefix((*i)->getPrefix(),
Alexander Afanasyev52afb3f2014-03-07 09:05:35 +0000253 bind(&Face::finalizeUnsetInterestFilter, this, i),
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800254 Controller::FailCallback());
255 }
256
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -0700257 // there cannot be two registered prefixes with the same id
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800258}
259
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800260void
Alexander Afanasyev52afb3f2014-03-07 09:05:35 +0000261Face::finalizeUnsetInterestFilter(RegisteredPrefixTable::iterator item)
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800262{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800263 m_registeredPrefixTable.erase(item);
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800264
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800265 if (!m_pitTimeoutCheckTimerActive && m_registeredPrefixTable.empty())
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800266 {
Alexander Afanasyev52afb3f2014-03-07 09:05:35 +0000267 m_transport->pause();
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800268 if (!m_ioServiceWork) {
269 m_processEventsTimeoutTimer->cancel();
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800270 }
271 }
272}
273
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800274void
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700275Face::processEvents(const time::milliseconds& timeout/* = time::milliseconds::zero()*/,
276 bool keepThread/* = false*/)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800277{
278 try
279 {
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700280 if (timeout < time::milliseconds::zero())
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800281 {
282 // do not block if timeout is negative, but process pending events
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800283 m_ioService->poll();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800284 return;
285 }
286
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700287 if (timeout > time::milliseconds::zero())
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800288 {
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700289 m_processEventsTimeoutTimer->expires_from_now(time::milliseconds(timeout));
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800290 m_processEventsTimeoutTimer->async_wait(&fireProcessEventsTimeout);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800291 }
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800292
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800293 if (keepThread) {
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800294 // work will ensure that m_ioService is running until work object exists
295 m_ioServiceWork = make_shared<boost::asio::io_service::work>(boost::ref(*m_ioService));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800296 }
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800297
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800298 m_ioService->run();
299 m_ioService->reset(); // so it is possible to run processEvents again (if necessary)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800300 }
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -0700301 catch (Face::ProcessEventsTimeout&)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800302 {
303 // break
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800304 m_ioService->reset();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800305 }
Alexander Afanasyev2a7f7202014-04-23 14:25:29 -0700306 catch (std::exception&)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800307 {
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800308 m_ioService->reset();
309 m_pendingInterestTable.clear();
310 m_registeredPrefixTable.clear();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800311 throw;
312 }
Jeff Thompsonfb29cda2013-08-24 10:26:54 -0700313}
314
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800315void
Jeff Thompson0050abe2013-09-17 12:50:25 -0700316Face::shutdown()
Jeff Thompson517ffa82013-08-05 16:04:34 -0700317{
Alexander Afanasyev7dced462014-03-19 15:12:32 -0700318 m_ioService->post(bind(&Face::asyncShutdown, this));
319}
320
321void
322Face::asyncShutdown()
323{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800324 m_pendingInterestTable.clear();
325 m_registeredPrefixTable.clear();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800326
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800327 m_transport->close();
328 m_pitTimeoutCheckTimer->cancel();
329 m_processEventsTimeoutTimer->cancel();
330 m_pitTimeoutCheckTimerActive = false;
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -0700331}
332
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800333void
334Face::fireProcessEventsTimeout(const boost::system::error_code& error)
335{
336 if (!error) // can fire for some other reason, e.g., cancelled
337 throw Face::ProcessEventsTimeout();
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -0700338}
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800339
340void
341Face::checkPitExpire()
342{
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700343 // Check for PIT entry timeouts.
344 time::steady_clock::TimePoint now = time::steady_clock::now();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800345
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800346 PendingInterestTable::iterator i = m_pendingInterestTable.begin();
347 while (i != m_pendingInterestTable.end())
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800348 {
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700349 if ((*i)->isTimedOut(now))
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800350 {
351 // Save the PendingInterest and remove it from the PIT. Then call the callback.
352 shared_ptr<PendingInterest> pendingInterest = *i;
353
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800354 i = m_pendingInterestTable.erase(i);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800355
356 pendingInterest->callTimeout();
357 }
358 else
359 ++i;
360 }
361
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800362 if (!m_pendingInterestTable.empty()) {
363 m_pitTimeoutCheckTimerActive = true;
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800364
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700365 m_pitTimeoutCheckTimer->expires_from_now(time::milliseconds(100));
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800366 m_pitTimeoutCheckTimer->async_wait(bind(&Face::checkPitExpire, this));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800367 }
368 else {
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800369 m_pitTimeoutCheckTimerActive = false;
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800370
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800371 if (m_registeredPrefixTable.empty()) {
Alexander Afanasyev52afb3f2014-03-07 09:05:35 +0000372 m_transport->pause();
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800373 if (!m_ioServiceWork) {
374 m_processEventsTimeoutTimer->cancel();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800375 }
376 }
377 }
378}
379
380
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800381void
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800382Face::onReceiveElement(const Block& blockFromDaemon)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800383{
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800384 const Block& block = nfd::LocalControlHeader::getPayload(blockFromDaemon);
385
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800386 if (block.type() == Tlv::Interest)
387 {
388 shared_ptr<Interest> interest(new Interest());
389 interest->wireDecode(block);
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800390 if (&block != &blockFromDaemon)
391 interest->getLocalControlHeader().wireDecode(blockFromDaemon);
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800392
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800393 processInterestFilters(*interest);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800394 }
395 else if (block.type() == Tlv::Data)
396 {
397 shared_ptr<Data> data(new Data());
398 data->wireDecode(block);
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800399 if (&block != &blockFromDaemon)
400 data->getLocalControlHeader().wireDecode(blockFromDaemon);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800401
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800402 satisfyPendingInterests(*data);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800403
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800404 if (m_pendingInterestTable.empty()) {
405 m_pitTimeoutCheckTimer->cancel(); // this will cause checkPitExpire invocation
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800406 }
407 }
Yingdi Yuf9fa52f2014-02-06 12:27:32 -0800408 // ignore any other type
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800409}
410
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800411void
412Face::satisfyPendingInterests(Data& data)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800413{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800414 for (PendingInterestTable::iterator i = m_pendingInterestTable.begin ();
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800415 i != m_pendingInterestTable.end();
416 )
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800417 {
Junxiao Shiaf8eeea2014-03-31 20:10:56 -0700418 if ((*i)->getInterest()->matchesData(data))
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800419 {
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -0700420 // Copy pointers to the objects and remove the PIT entry before calling the callback.
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800421 OnData onData = (*i)->getOnData();
422 shared_ptr<const Interest> interest = (*i)->getInterest();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800423
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800424 PendingInterestTable::iterator next = i;
425 ++next;
426 m_pendingInterestTable.erase(i);
427 i = next;
428
429 if (static_cast<bool>(onData)) {
430 onData(*interest, data);
431 }
432 }
433 else
434 ++i;
435 }
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800436}
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800437
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800438void
439Face::processInterestFilters(Interest& interest)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800440{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800441 for (RegisteredPrefixTable::iterator i = m_registeredPrefixTable.begin();
442 i != m_registeredPrefixTable.end();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800443 ++i)
444 {
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800445 if ((*i)->getPrefix().isPrefixOf(interest.getName()))
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800446 {
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800447 (*i)->getOnInterest()((*i)->getPrefix(), interest);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800448 }
449 }
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800450}
451
452} // namespace ndn