blob: 44de6cc05e11c8bb625f22a81c3cd60e074c1989 [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 Afanasyevb67090a2014-04-29 22:31:01 -070095 m_pitTimeoutCheckTimer = make_shared<monotonic_deadline_timer>(ref(*m_ioService));
96 m_processEventsTimeoutTimer = make_shared<monotonic_deadline_timer>(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 {
Alexander Afanasyevb67090a2014-04-29 22:31:01 -0700115 m_fwController = make_shared<nrd::Controller>(ref(*this));
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600116 }
Steve DiBenedettoacab8802014-03-24 11:15:57 -0600117 else if (isSupportedNfdProtocol(protocol))
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600118 {
Alexander Afanasyevb67090a2014-04-29 22:31:01 -0700119 m_fwController = make_shared<nfd::Controller>(ref(*this));
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600120 }
121 else if (isSupportedNdndProtocol(protocol))
122 {
Alexander Afanasyevb67090a2014-04-29 22:31:01 -0700123 m_fwController = make_shared<ndnd::Controller>(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
Alexander Afanasyevb67090a2014-04-29 22:31:01 -0700138 shared_ptr<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
Alexander Afanasyevb67090a2014-04-29 22:31:01 -0700222void
223Face::finalizeSetInterestFilter(const shared_ptr<RegisteredPrefix>& registeredPrefix)
224{
225 m_registeredPrefixTable.push_back(registeredPrefix);
226}
227
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800228const RegisteredPrefixId*
229Face::setInterestFilter(const Name& prefix,
230 const OnInterest& onInterest,
231 const OnSetInterestFilterFailed& onSetInterestFilterFailed)
232{
233 shared_ptr<RegisteredPrefix> prefixToRegister(new RegisteredPrefix(prefix, onInterest));
234
235 m_fwController->selfRegisterPrefix(prefixToRegister->getPrefix(),
Alexander Afanasyevb67090a2014-04-29 22:31:01 -0700236 bind(&Face::finalizeSetInterestFilter, this, prefixToRegister),
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -0700237 bind(onSetInterestFilterFailed,
238 prefixToRegister->getPrefix(), _1));
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800239
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800240 return reinterpret_cast<const RegisteredPrefixId*>(prefixToRegister.get());
241}
242
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700243const RegisteredPrefixId*
244Face::setInterestFilter(const Name& prefix,
245 const OnInterest& onInterest,
246 const OnSetInterestFilterFailed& onSetInterestFilterFailed,
247 const IdentityCertificate& certificate)
248{
249 shared_ptr<RegisteredPrefix> prefixToRegister(new RegisteredPrefix(prefix, onInterest));
250
251 m_fwController->selfRegisterPrefix(prefixToRegister->getPrefix(),
Alexander Afanasyevb67090a2014-04-29 22:31:01 -0700252 bind(&Face::finalizeSetInterestFilter, this, prefixToRegister),
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700253 bind(onSetInterestFilterFailed,
254 prefixToRegister->getPrefix(), _1),
255 certificate);
256
257 return reinterpret_cast<const RegisteredPrefixId*>(prefixToRegister.get());
258}
259
260const RegisteredPrefixId*
261Face::setInterestFilter(const Name& prefix,
262 const OnInterest& onInterest,
263 const OnSetInterestFilterFailed& onSetInterestFilterFailed,
264 const Name& identity)
265{
266 shared_ptr<RegisteredPrefix> prefixToRegister(new RegisteredPrefix(prefix, onInterest));
267
268 m_fwController->selfRegisterPrefix(prefixToRegister->getPrefix(),
Alexander Afanasyevb67090a2014-04-29 22:31:01 -0700269 bind(&Face::finalizeSetInterestFilter, this, prefixToRegister),
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700270 bind(onSetInterestFilterFailed,
271 prefixToRegister->getPrefix(), _1),
272 identity);
273
274 return reinterpret_cast<const RegisteredPrefixId*>(prefixToRegister.get());
275}
276
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800277void
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800278Face::unsetInterestFilter(const RegisteredPrefixId* registeredPrefixId)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800279{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800280 m_ioService->post(bind(&Face::asyncUnsetInterestFilter, this, registeredPrefixId));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800281}
282
283void
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700284Face::unsetInterestFilter(const RegisteredPrefixId* registeredPrefixId,
285 const IdentityCertificate& certificate)
286{
287 m_ioService->post(bind(&Face::asyncUnsetInterestFilterWithCertificate, this,
288 registeredPrefixId, certificate));
289}
290
291void
292Face::unsetInterestFilter(const RegisteredPrefixId* registeredPrefixId,
293 const Name& identity)
294{
295 m_ioService->post(bind(&Face::asyncUnsetInterestFilterWithIdentity, this,
296 registeredPrefixId, identity));
297}
298
299void
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800300Face::asyncUnsetInterestFilter(const RegisteredPrefixId* registeredPrefixId)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800301{
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -0700302 RegisteredPrefixTable::iterator i = std::find_if(m_registeredPrefixTable.begin(),
303 m_registeredPrefixTable.end(),
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800304 MatchRegisteredPrefixId(registeredPrefixId));
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800305 if (i != m_registeredPrefixTable.end())
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800306 {
307 m_fwController->selfDeregisterPrefix((*i)->getPrefix(),
Alexander Afanasyev52afb3f2014-03-07 09:05:35 +0000308 bind(&Face::finalizeUnsetInterestFilter, this, i),
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800309 Controller::FailCallback());
310 }
311
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -0700312 // there cannot be two registered prefixes with the same id
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800313}
314
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800315void
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700316Face::asyncUnsetInterestFilterWithCertificate(const RegisteredPrefixId* registeredPrefixId,
317 const IdentityCertificate& certificate)
318{
319 RegisteredPrefixTable::iterator i = std::find_if(m_registeredPrefixTable.begin(),
320 m_registeredPrefixTable.end(),
321 MatchRegisteredPrefixId(registeredPrefixId));
322 if (i != m_registeredPrefixTable.end())
323 {
324 m_fwController->selfDeregisterPrefix((*i)->getPrefix(),
325 bind(&Face::finalizeUnsetInterestFilter, this, i),
326 Controller::FailCallback(),
327 certificate);
328 }
329
330 // there cannot be two registered prefixes with the same id
331}
332
333void
334Face::asyncUnsetInterestFilterWithIdentity(const RegisteredPrefixId* registeredPrefixId,
335 const Name& identity)
336{
337 RegisteredPrefixTable::iterator i = std::find_if(m_registeredPrefixTable.begin(),
338 m_registeredPrefixTable.end(),
339 MatchRegisteredPrefixId(registeredPrefixId));
340 if (i != m_registeredPrefixTable.end())
341 {
342 m_fwController->selfDeregisterPrefix((*i)->getPrefix(),
343 bind(&Face::finalizeUnsetInterestFilter, this, i),
344 Controller::FailCallback(),
345 identity);
346 }
347
348 // there cannot be two registered prefixes with the same id
349}
350
351void
Alexander Afanasyev52afb3f2014-03-07 09:05:35 +0000352Face::finalizeUnsetInterestFilter(RegisteredPrefixTable::iterator item)
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800353{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800354 m_registeredPrefixTable.erase(item);
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800355
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800356 if (!m_pitTimeoutCheckTimerActive && m_registeredPrefixTable.empty())
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800357 {
Alexander Afanasyev52afb3f2014-03-07 09:05:35 +0000358 m_transport->pause();
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800359 if (!m_ioServiceWork) {
360 m_processEventsTimeoutTimer->cancel();
Alexander Afanasyev12dfbad2014-02-11 14:42:46 -0800361 }
362 }
363}
364
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800365void
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700366Face::processEvents(const time::milliseconds& timeout/* = time::milliseconds::zero()*/,
367 bool keepThread/* = false*/)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800368{
369 try
370 {
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700371 if (timeout < time::milliseconds::zero())
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800372 {
373 // do not block if timeout is negative, but process pending events
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800374 m_ioService->poll();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800375 return;
376 }
377
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700378 if (timeout > time::milliseconds::zero())
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800379 {
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700380 m_processEventsTimeoutTimer->expires_from_now(time::milliseconds(timeout));
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800381 m_processEventsTimeoutTimer->async_wait(&fireProcessEventsTimeout);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800382 }
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800383
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800384 if (keepThread) {
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800385 // work will ensure that m_ioService is running until work object exists
Alexander Afanasyevb67090a2014-04-29 22:31:01 -0700386 m_ioServiceWork = make_shared<boost::asio::io_service::work>(ref(*m_ioService));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800387 }
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800388
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800389 m_ioService->run();
390 m_ioService->reset(); // so it is possible to run processEvents again (if necessary)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800391 }
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -0700392 catch (Face::ProcessEventsTimeout&)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800393 {
394 // break
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800395 m_ioService->reset();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800396 }
Alexander Afanasyev2a7f7202014-04-23 14:25:29 -0700397 catch (std::exception&)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800398 {
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800399 m_ioService->reset();
400 m_pendingInterestTable.clear();
401 m_registeredPrefixTable.clear();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800402 throw;
403 }
Jeff Thompsonfb29cda2013-08-24 10:26:54 -0700404}
405
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800406void
Jeff Thompson0050abe2013-09-17 12:50:25 -0700407Face::shutdown()
Jeff Thompson517ffa82013-08-05 16:04:34 -0700408{
Alexander Afanasyev7dced462014-03-19 15:12:32 -0700409 m_ioService->post(bind(&Face::asyncShutdown, this));
410}
411
412void
413Face::asyncShutdown()
414{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800415 m_pendingInterestTable.clear();
416 m_registeredPrefixTable.clear();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800417
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800418 m_transport->close();
419 m_pitTimeoutCheckTimer->cancel();
420 m_processEventsTimeoutTimer->cancel();
421 m_pitTimeoutCheckTimerActive = false;
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -0700422}
423
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800424void
425Face::fireProcessEventsTimeout(const boost::system::error_code& error)
426{
427 if (!error) // can fire for some other reason, e.g., cancelled
428 throw Face::ProcessEventsTimeout();
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -0700429}
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800430
431void
432Face::checkPitExpire()
433{
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700434 // Check for PIT entry timeouts.
435 time::steady_clock::TimePoint now = time::steady_clock::now();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800436
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800437 PendingInterestTable::iterator i = m_pendingInterestTable.begin();
438 while (i != m_pendingInterestTable.end())
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800439 {
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700440 if ((*i)->isTimedOut(now))
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800441 {
442 // Save the PendingInterest and remove it from the PIT. Then call the callback.
443 shared_ptr<PendingInterest> pendingInterest = *i;
444
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800445 i = m_pendingInterestTable.erase(i);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800446
447 pendingInterest->callTimeout();
448 }
449 else
450 ++i;
451 }
452
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800453 if (!m_pendingInterestTable.empty()) {
454 m_pitTimeoutCheckTimerActive = true;
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800455
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700456 m_pitTimeoutCheckTimer->expires_from_now(time::milliseconds(100));
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800457 m_pitTimeoutCheckTimer->async_wait(bind(&Face::checkPitExpire, this));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800458 }
459 else {
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800460 m_pitTimeoutCheckTimerActive = false;
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800461
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800462 if (m_registeredPrefixTable.empty()) {
Alexander Afanasyev52afb3f2014-03-07 09:05:35 +0000463 m_transport->pause();
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800464 if (!m_ioServiceWork) {
465 m_processEventsTimeoutTimer->cancel();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800466 }
467 }
468 }
469}
470
471
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800472void
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800473Face::onReceiveElement(const Block& blockFromDaemon)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800474{
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800475 const Block& block = nfd::LocalControlHeader::getPayload(blockFromDaemon);
476
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800477 if (block.type() == Tlv::Interest)
478 {
479 shared_ptr<Interest> interest(new Interest());
480 interest->wireDecode(block);
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800481 if (&block != &blockFromDaemon)
482 interest->getLocalControlHeader().wireDecode(blockFromDaemon);
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800483
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800484 processInterestFilters(*interest);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800485 }
486 else if (block.type() == Tlv::Data)
487 {
488 shared_ptr<Data> data(new Data());
489 data->wireDecode(block);
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800490 if (&block != &blockFromDaemon)
491 data->getLocalControlHeader().wireDecode(blockFromDaemon);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800492
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800493 satisfyPendingInterests(*data);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800494
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800495 if (m_pendingInterestTable.empty()) {
496 m_pitTimeoutCheckTimer->cancel(); // this will cause checkPitExpire invocation
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800497 }
498 }
Yingdi Yuf9fa52f2014-02-06 12:27:32 -0800499 // ignore any other type
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800500}
501
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800502void
503Face::satisfyPendingInterests(Data& data)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800504{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800505 for (PendingInterestTable::iterator i = m_pendingInterestTable.begin ();
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800506 i != m_pendingInterestTable.end();
507 )
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800508 {
Junxiao Shiaf8eeea2014-03-31 20:10:56 -0700509 if ((*i)->getInterest()->matchesData(data))
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800510 {
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -0700511 // Copy pointers to the objects and remove the PIT entry before calling the callback.
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800512 OnData onData = (*i)->getOnData();
513 shared_ptr<const Interest> interest = (*i)->getInterest();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800514
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800515 PendingInterestTable::iterator next = i;
516 ++next;
517 m_pendingInterestTable.erase(i);
518 i = next;
519
520 if (static_cast<bool>(onData)) {
521 onData(*interest, data);
522 }
523 }
524 else
525 ++i;
526 }
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800527}
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800528
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800529void
530Face::processInterestFilters(Interest& interest)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800531{
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800532 for (RegisteredPrefixTable::iterator i = m_registeredPrefixTable.begin();
533 i != m_registeredPrefixTable.end();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800534 ++i)
535 {
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800536 if ((*i)->getPrefix().isPrefixOf(interest.getName()))
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800537 {
Alexander Afanasyev42c81852014-02-25 21:37:26 -0800538 (*i)->getOnInterest()((*i)->getPrefix(), interest);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800539 }
540 }
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800541}
542
543} // namespace ndn