blob: d351c0a74ed7ce5ae9a7d2ead1c9be71c8a764e7 [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
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700238const RegisteredPrefixId*
239Face::setInterestFilter(const Name& prefix,
240 const OnInterest& onInterest,
241 const OnSetInterestFilterFailed& onSetInterestFilterFailed,
242 const IdentityCertificate& certificate)
243{
244 shared_ptr<RegisteredPrefix> prefixToRegister(new RegisteredPrefix(prefix, onInterest));
245
246 m_fwController->selfRegisterPrefix(prefixToRegister->getPrefix(),
247 bind(&RegisteredPrefixTable::push_back,
248 &m_registeredPrefixTable, prefixToRegister),
249 bind(onSetInterestFilterFailed,
250 prefixToRegister->getPrefix(), _1),
251 certificate);
252
253 return reinterpret_cast<const RegisteredPrefixId*>(prefixToRegister.get());
254}
255
256const RegisteredPrefixId*
257Face::setInterestFilter(const Name& prefix,
258 const OnInterest& onInterest,
259 const OnSetInterestFilterFailed& onSetInterestFilterFailed,
260 const Name& identity)
261{
262 shared_ptr<RegisteredPrefix> prefixToRegister(new RegisteredPrefix(prefix, onInterest));
263
264 m_fwController->selfRegisterPrefix(prefixToRegister->getPrefix(),
265 bind(&RegisteredPrefixTable::push_back,
266 &m_registeredPrefixTable, prefixToRegister),
267 bind(onSetInterestFilterFailed,
268 prefixToRegister->getPrefix(), _1),
269 identity);
270
271 return reinterpret_cast<const RegisteredPrefixId*>(prefixToRegister.get());
272}
273
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800274void
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800275Face::unsetInterestFilter(const RegisteredPrefixId* registeredPrefixId)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800276{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800277 m_ioService->post(bind(&Face::asyncUnsetInterestFilter, this, registeredPrefixId));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800278}
279
280void
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700281Face::unsetInterestFilter(const RegisteredPrefixId* registeredPrefixId,
282 const IdentityCertificate& certificate)
283{
284 m_ioService->post(bind(&Face::asyncUnsetInterestFilterWithCertificate, this,
285 registeredPrefixId, certificate));
286}
287
288void
289Face::unsetInterestFilter(const RegisteredPrefixId* registeredPrefixId,
290 const Name& identity)
291{
292 m_ioService->post(bind(&Face::asyncUnsetInterestFilterWithIdentity, this,
293 registeredPrefixId, identity));
294}
295
296void
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800297Face::asyncUnsetInterestFilter(const RegisteredPrefixId* registeredPrefixId)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800298{
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -0700299 RegisteredPrefixTable::iterator i = std::find_if(m_registeredPrefixTable.begin(),
300 m_registeredPrefixTable.end(),
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800301 MatchRegisteredPrefixId(registeredPrefixId));
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800302 if (i != m_registeredPrefixTable.end())
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800303 {
304 m_fwController->selfDeregisterPrefix((*i)->getPrefix(),
Alexander Afanasyev52afb3f2014-03-07 09:05:35 +0000305 bind(&Face::finalizeUnsetInterestFilter, this, i),
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800306 Controller::FailCallback());
307 }
308
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -0700309 // there cannot be two registered prefixes with the same id
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800310}
311
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800312void
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700313Face::asyncUnsetInterestFilterWithCertificate(const RegisteredPrefixId* registeredPrefixId,
314 const IdentityCertificate& certificate)
315{
316 RegisteredPrefixTable::iterator i = std::find_if(m_registeredPrefixTable.begin(),
317 m_registeredPrefixTable.end(),
318 MatchRegisteredPrefixId(registeredPrefixId));
319 if (i != m_registeredPrefixTable.end())
320 {
321 m_fwController->selfDeregisterPrefix((*i)->getPrefix(),
322 bind(&Face::finalizeUnsetInterestFilter, this, i),
323 Controller::FailCallback(),
324 certificate);
325 }
326
327 // there cannot be two registered prefixes with the same id
328}
329
330void
331Face::asyncUnsetInterestFilterWithIdentity(const RegisteredPrefixId* registeredPrefixId,
332 const Name& identity)
333{
334 RegisteredPrefixTable::iterator i = std::find_if(m_registeredPrefixTable.begin(),
335 m_registeredPrefixTable.end(),
336 MatchRegisteredPrefixId(registeredPrefixId));
337 if (i != m_registeredPrefixTable.end())
338 {
339 m_fwController->selfDeregisterPrefix((*i)->getPrefix(),
340 bind(&Face::finalizeUnsetInterestFilter, this, i),
341 Controller::FailCallback(),
342 identity);
343 }
344
345 // there cannot be two registered prefixes with the same id
346}
347
348void
Alexander Afanasyev52afb3f2014-03-07 09:05:35 +0000349Face::finalizeUnsetInterestFilter(RegisteredPrefixTable::iterator item)
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800350{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800351 m_registeredPrefixTable.erase(item);
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800352
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800353 if (!m_pitTimeoutCheckTimerActive && m_registeredPrefixTable.empty())
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800354 {
Alexander Afanasyev52afb3f2014-03-07 09:05:35 +0000355 m_transport->pause();
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800356 if (!m_ioServiceWork) {
357 m_processEventsTimeoutTimer->cancel();
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800358 }
359 }
360}
361
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800362void
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700363Face::processEvents(const time::milliseconds& timeout/* = time::milliseconds::zero()*/,
364 bool keepThread/* = false*/)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800365{
366 try
367 {
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700368 if (timeout < time::milliseconds::zero())
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800369 {
370 // do not block if timeout is negative, but process pending events
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800371 m_ioService->poll();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800372 return;
373 }
374
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700375 if (timeout > time::milliseconds::zero())
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800376 {
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700377 m_processEventsTimeoutTimer->expires_from_now(time::milliseconds(timeout));
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800378 m_processEventsTimeoutTimer->async_wait(&fireProcessEventsTimeout);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800379 }
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800380
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800381 if (keepThread) {
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800382 // work will ensure that m_ioService is running until work object exists
383 m_ioServiceWork = make_shared<boost::asio::io_service::work>(boost::ref(*m_ioService));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800384 }
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800385
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800386 m_ioService->run();
387 m_ioService->reset(); // so it is possible to run processEvents again (if necessary)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800388 }
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -0700389 catch (Face::ProcessEventsTimeout&)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800390 {
391 // break
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800392 m_ioService->reset();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800393 }
Alexander Afanasyev2a7f7202014-04-23 14:25:29 -0700394 catch (std::exception&)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800395 {
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800396 m_ioService->reset();
397 m_pendingInterestTable.clear();
398 m_registeredPrefixTable.clear();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800399 throw;
400 }
Jeff Thompsonfb29cda2013-08-24 10:26:54 -0700401}
402
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800403void
Jeff Thompson0050abe2013-09-17 12:50:25 -0700404Face::shutdown()
Jeff Thompson517ffa82013-08-05 16:04:34 -0700405{
Alexander Afanasyev7dced462014-03-19 15:12:32 -0700406 m_ioService->post(bind(&Face::asyncShutdown, this));
407}
408
409void
410Face::asyncShutdown()
411{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800412 m_pendingInterestTable.clear();
413 m_registeredPrefixTable.clear();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800414
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800415 m_transport->close();
416 m_pitTimeoutCheckTimer->cancel();
417 m_processEventsTimeoutTimer->cancel();
418 m_pitTimeoutCheckTimerActive = false;
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -0700419}
420
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800421void
422Face::fireProcessEventsTimeout(const boost::system::error_code& error)
423{
424 if (!error) // can fire for some other reason, e.g., cancelled
425 throw Face::ProcessEventsTimeout();
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -0700426}
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800427
428void
429Face::checkPitExpire()
430{
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700431 // Check for PIT entry timeouts.
432 time::steady_clock::TimePoint now = time::steady_clock::now();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800433
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800434 PendingInterestTable::iterator i = m_pendingInterestTable.begin();
435 while (i != m_pendingInterestTable.end())
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800436 {
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700437 if ((*i)->isTimedOut(now))
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800438 {
439 // Save the PendingInterest and remove it from the PIT. Then call the callback.
440 shared_ptr<PendingInterest> pendingInterest = *i;
441
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800442 i = m_pendingInterestTable.erase(i);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800443
444 pendingInterest->callTimeout();
445 }
446 else
447 ++i;
448 }
449
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800450 if (!m_pendingInterestTable.empty()) {
451 m_pitTimeoutCheckTimerActive = true;
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800452
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700453 m_pitTimeoutCheckTimer->expires_from_now(time::milliseconds(100));
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800454 m_pitTimeoutCheckTimer->async_wait(bind(&Face::checkPitExpire, this));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800455 }
456 else {
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800457 m_pitTimeoutCheckTimerActive = false;
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800458
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800459 if (m_registeredPrefixTable.empty()) {
Alexander Afanasyev52afb3f2014-03-07 09:05:35 +0000460 m_transport->pause();
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800461 if (!m_ioServiceWork) {
462 m_processEventsTimeoutTimer->cancel();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800463 }
464 }
465 }
466}
467
468
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800469void
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800470Face::onReceiveElement(const Block& blockFromDaemon)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800471{
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800472 const Block& block = nfd::LocalControlHeader::getPayload(blockFromDaemon);
473
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800474 if (block.type() == Tlv::Interest)
475 {
476 shared_ptr<Interest> interest(new Interest());
477 interest->wireDecode(block);
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800478 if (&block != &blockFromDaemon)
479 interest->getLocalControlHeader().wireDecode(blockFromDaemon);
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800480
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800481 processInterestFilters(*interest);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800482 }
483 else if (block.type() == Tlv::Data)
484 {
485 shared_ptr<Data> data(new Data());
486 data->wireDecode(block);
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800487 if (&block != &blockFromDaemon)
488 data->getLocalControlHeader().wireDecode(blockFromDaemon);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800489
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800490 satisfyPendingInterests(*data);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800491
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800492 if (m_pendingInterestTable.empty()) {
493 m_pitTimeoutCheckTimer->cancel(); // this will cause checkPitExpire invocation
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800494 }
495 }
Yingdi Yuf9fa52f2014-02-06 12:27:32 -0800496 // ignore any other type
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800497}
498
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800499void
500Face::satisfyPendingInterests(Data& data)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800501{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800502 for (PendingInterestTable::iterator i = m_pendingInterestTable.begin ();
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800503 i != m_pendingInterestTable.end();
504 )
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800505 {
Junxiao Shiaf8eeea2014-03-31 20:10:56 -0700506 if ((*i)->getInterest()->matchesData(data))
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800507 {
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -0700508 // Copy pointers to the objects and remove the PIT entry before calling the callback.
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800509 OnData onData = (*i)->getOnData();
510 shared_ptr<const Interest> interest = (*i)->getInterest();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800511
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800512 PendingInterestTable::iterator next = i;
513 ++next;
514 m_pendingInterestTable.erase(i);
515 i = next;
516
517 if (static_cast<bool>(onData)) {
518 onData(*interest, data);
519 }
520 }
521 else
522 ++i;
523 }
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800524}
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800525
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800526void
527Face::processInterestFilters(Interest& interest)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800528{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800529 for (RegisteredPrefixTable::iterator i = m_registeredPrefixTable.begin();
530 i != m_registeredPrefixTable.end();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800531 ++i)
532 {
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800533 if ((*i)->getPrefix().isPrefixOf(interest.getName()))
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800534 {
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800535 (*i)->getOnInterest()((*i)->getPrefix(), interest);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800536 }
537 }
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800538}
539
540} // namespace ndn