NS-3 based Named Data Networking (NDN) simulator
ndnSIM 2.5: NDN, CCN, CCNx, content centric networks
API Documentation
control-parameters.cpp
Go to the documentation of this file.
1 /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2 /*
3  * Copyright (c) 2013-2018 Regents of the University of California.
4  *
5  * This file is part of ndn-cxx library (NDN C++ library with eXperimental eXtensions).
6  *
7  * ndn-cxx library is free software: you can redistribute it and/or modify it under the
8  * terms of the GNU Lesser General Public License as published by the Free Software
9  * Foundation, either version 3 of the License, or (at your option) any later version.
10  *
11  * ndn-cxx library is distributed in the hope that it will be useful, but WITHOUT ANY
12  * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
13  * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.
14  *
15  * You should have received copies of the GNU General Public License and GNU Lesser
16  * General Public License along with ndn-cxx, e.g., in COPYING.md file. If not, see
17  * <http://www.gnu.org/licenses/>.
18  *
19  * See AUTHORS.md for complete list of ndn-cxx authors and contributors.
20  */
21 
27 
28 namespace ndn {
29 namespace nfd {
30 
31 //BOOST_CONCEPT_ASSERT((boost::EqualityComparable<ControlParameters>));
32 BOOST_CONCEPT_ASSERT((WireEncodable<ControlParameters>));
33 BOOST_CONCEPT_ASSERT((WireDecodable<ControlParameters>));
34 static_assert(std::is_base_of<tlv::Error, ControlParameters::Error>::value,
35  "ControlParameters::Error must inherit from tlv::Error");
36 
38  : m_hasFields(CONTROL_PARAMETER_UBOUND)
39 {
40 }
41 
43  : m_hasFields(CONTROL_PARAMETER_UBOUND)
44 {
45  wireDecode(block);
46 }
47 
48 template<encoding::Tag TAG>
49 size_t
51 {
52  size_t totalLength = 0;
53 
54  if (this->hasMtu()) {
55  totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::Mtu, m_mtu);
56  }
57  if (this->hasDefaultCongestionThreshold()) {
59  m_defaultCongestionThreshold);
60  }
63  m_baseCongestionMarkingInterval.count());
64  }
65  if (this->hasFacePersistency()) {
66  totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::FacePersistency, m_facePersistency);
67  }
68  if (this->hasExpirationPeriod()) {
70  m_expirationPeriod.count());
71  }
72  if (this->hasStrategy()) {
73  totalLength += prependNestedBlock(encoder, tlv::nfd::Strategy, m_strategy);
74  }
75  if (this->hasMask()) {
76  totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::Mask, m_mask);
77  }
78  if (this->hasFlags()) {
79  totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::Flags, m_flags);
80  }
81  if (this->hasCount()) {
82  totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::Count, m_count);
83  }
84  if (this->hasCapacity()) {
85  totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::Capacity, m_capacity);
86  }
87  if (this->hasCost()) {
88  totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::Cost, m_cost);
89  }
90  if (this->hasOrigin()) {
91  totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::Origin, m_origin);
92  }
93  if (this->hasLocalUri()) {
94  totalLength += prependStringBlock(encoder, tlv::nfd::LocalUri, m_localUri);
95  }
96  if (this->hasUri()) {
97  totalLength += prependStringBlock(encoder, tlv::nfd::Uri, m_uri);
98  }
99  if (this->hasFaceId()) {
100  totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::FaceId, m_faceId);
101  }
102  if (this->hasName()) {
103  totalLength += m_name.wireEncode(encoder);
104  }
105 
106  totalLength += encoder.prependVarNumber(totalLength);
107  totalLength += encoder.prependVarNumber(tlv::nfd::ControlParameters);
108  return totalLength;
109 }
110 
112 
113 Block
115 {
116  if (m_wire.hasWire())
117  return m_wire;
118 
119  EncodingEstimator estimator;
120  size_t estimatedSize = wireEncode(estimator);
121 
122  EncodingBuffer buffer(estimatedSize, 0);
123  wireEncode(buffer);
124 
125  m_wire = buffer.block();
126  return m_wire;
127 }
128 
129 void
131 {
132  if (block.type() != tlv::nfd::ControlParameters) {
133  BOOST_THROW_EXCEPTION(Error("Expecting TLV-TYPE ControlParameters"));
134  }
135  m_wire = block;
136  m_wire.parse();
138 
139  val = m_wire.find(tlv::Name);
140  m_hasFields[CONTROL_PARAMETER_NAME] = val != m_wire.elements_end();
141  if (this->hasName()) {
142  m_name.wireDecode(*val);
143  }
144 
145  val = m_wire.find(tlv::nfd::FaceId);
146  m_hasFields[CONTROL_PARAMETER_FACE_ID] = val != m_wire.elements_end();
147  if (this->hasFaceId()) {
148  m_faceId = readNonNegativeInteger(*val);
149  }
150 
151  val = m_wire.find(tlv::nfd::Uri);
152  m_hasFields[CONTROL_PARAMETER_URI] = val != m_wire.elements_end();
153  if (this->hasUri()) {
154  m_uri = readString(*val);
155  }
156 
157  val = m_wire.find(tlv::nfd::LocalUri);
158  m_hasFields[CONTROL_PARAMETER_LOCAL_URI] = val != m_wire.elements_end();
159  if (this->hasLocalUri()) {
160  m_localUri = readString(*val);
161  }
162 
163  val = m_wire.find(tlv::nfd::Origin);
164  m_hasFields[CONTROL_PARAMETER_ORIGIN] = val != m_wire.elements_end();
165  if (this->hasOrigin()) {
166  m_origin = readNonNegativeIntegerAs<RouteOrigin>(*val);
167  }
168 
169  val = m_wire.find(tlv::nfd::Cost);
170  m_hasFields[CONTROL_PARAMETER_COST] = val != m_wire.elements_end();
171  if (this->hasCost()) {
172  m_cost = readNonNegativeInteger(*val);
173  }
174 
175  val = m_wire.find(tlv::nfd::Capacity);
176  m_hasFields[CONTROL_PARAMETER_CAPACITY] = val != m_wire.elements_end();
177  if (this->hasCapacity()) {
178  m_capacity = readNonNegativeInteger(*val);
179  }
180 
181  val = m_wire.find(tlv::nfd::Count);
182  m_hasFields[CONTROL_PARAMETER_COUNT] = val != m_wire.elements_end();
183  if (this->hasCount()) {
184  m_count = readNonNegativeInteger(*val);
185  }
186 
187  val = m_wire.find(tlv::nfd::Flags);
188  m_hasFields[CONTROL_PARAMETER_FLAGS] = val != m_wire.elements_end();
189  if (this->hasFlags()) {
190  m_flags = readNonNegativeInteger(*val);
191  }
192 
193  val = m_wire.find(tlv::nfd::Mask);
194  m_hasFields[CONTROL_PARAMETER_MASK] = val != m_wire.elements_end();
195  if (this->hasMask()) {
196  m_mask = readNonNegativeInteger(*val);
197  }
198 
199  val = m_wire.find(tlv::nfd::Strategy);
200  m_hasFields[CONTROL_PARAMETER_STRATEGY] = val != m_wire.elements_end();
201  if (this->hasStrategy()) {
202  val->parse();
203  if (val->elements().empty()) {
204  BOOST_THROW_EXCEPTION(Error("Expecting Strategy/Name"));
205  }
206  else {
207  m_strategy.wireDecode(*val->elements_begin());
208  }
209  }
210 
211  val = m_wire.find(tlv::nfd::ExpirationPeriod);
212  m_hasFields[CONTROL_PARAMETER_EXPIRATION_PERIOD] = val != m_wire.elements_end();
213  if (this->hasExpirationPeriod()) {
214  m_expirationPeriod = time::milliseconds(readNonNegativeInteger(*val));
215  }
216 
217  val = m_wire.find(tlv::nfd::FacePersistency);
218  m_hasFields[CONTROL_PARAMETER_FACE_PERSISTENCY] = val != m_wire.elements_end();
219  if (this->hasFacePersistency()) {
220  m_facePersistency = readNonNegativeIntegerAs<FacePersistency>(*val);
221  }
222 
225  if (this->hasBaseCongestionMarkingInterval()) {
226  m_baseCongestionMarkingInterval = time::nanoseconds(readNonNegativeInteger(*val));
227  }
228 
230  m_hasFields[CONTROL_PARAMETER_DEFAULT_CONGESTION_THRESHOLD] = val != m_wire.elements_end();
231  if (this->hasDefaultCongestionThreshold()) {
232  m_defaultCongestionThreshold = readNonNegativeInteger(*val);
233  }
234 
235  val = m_wire.find(tlv::nfd::Mtu);
236  m_hasFields[CONTROL_PARAMETER_MTU] = val != m_wire.elements_end();
237  if (this->hasMtu()) {
238  m_mtu = readNonNegativeInteger(*val);
239  }
240 }
241 
242 bool
244 {
245  if (bit >= 64) {
246  BOOST_THROW_EXCEPTION(std::out_of_range("bit must be within range [0, 64)"));
247  }
248 
249  if (!hasMask()) {
250  return false;
251  }
252 
253  return getMask() & (1 << bit);
254 }
255 
256 bool
258 {
259  if (bit >= 64) {
260  BOOST_THROW_EXCEPTION(std::out_of_range("bit must be within range [0, 64)"));
261  }
262 
263  if (!hasFlags()) {
264  return false;
265  }
266 
267  return getFlags() & (1 << bit);
268 }
269 
271 ControlParameters::setFlagBit(size_t bit, bool value, bool wantMask/* = true*/)
272 {
273  if (bit >= 64) {
274  BOOST_THROW_EXCEPTION(std::out_of_range("bit must be within range [0, 64)"));
275  }
276 
277  uint64_t flags = hasFlags() ? getFlags() : 0;
278  if (value) {
279  flags |= (1 << bit);
280  }
281  else {
282  flags &= ~(1 << bit);
283  }
284  setFlags(flags);
285 
286  if (wantMask) {
287  uint64_t mask = hasMask() ? getMask() : 0;
288  mask |= (1 << bit);
289  setMask(mask);
290  }
291 
292  return *this;
293 }
294 
297 {
298  if (bit >= 64) {
299  BOOST_THROW_EXCEPTION(std::out_of_range("bit must be within range [0, 64)"));
300  }
301 
302  uint64_t mask = hasMask() ? getMask() : 0;
303  mask &= ~(1 << bit);
304  if (mask == 0) {
305  unsetMask();
306  unsetFlags();
307  }
308  else {
309  setMask(mask);
310  }
311 
312  return *this;
313 }
314 
315 std::ostream&
316 operator<<(std::ostream& os, const ControlParameters& parameters)
317 {
318  os << "ControlParameters(";
319 
320  if (parameters.hasName()) {
321  os << "Name: " << parameters.getName() << ", ";
322  }
323 
324  if (parameters.hasFaceId()) {
325  os << "FaceId: " << parameters.getFaceId() << ", ";
326  }
327 
328  if (parameters.hasUri()) {
329  os << "Uri: " << parameters.getUri() << ", ";
330  }
331 
332  if (parameters.hasLocalUri()) {
333  os << "LocalUri: " << parameters.getLocalUri() << ", ";
334  }
335 
336  if (parameters.hasOrigin()) {
337  os << "Origin: " << parameters.getOrigin() << ", ";
338  }
339 
340  if (parameters.hasCost()) {
341  os << "Cost: " << parameters.getCost() << ", ";
342  }
343 
344  if (parameters.hasCapacity()) {
345  os << "Capacity: " << parameters.getCapacity() << ", ";
346  }
347 
348  if (parameters.hasCount()) {
349  os << "Count: " << parameters.getCount() << ", ";
350  }
351 
352  if (parameters.hasFlags()) {
353  os << "Flags: " << AsHex{parameters.getFlags()} << ", ";
354  }
355 
356  if (parameters.hasMask()) {
357  os << "Mask: " << AsHex{parameters.getMask()} << ", ";
358  }
359 
360  if (parameters.hasStrategy()) {
361  os << "Strategy: " << parameters.getStrategy() << ", ";
362  }
363 
364  if (parameters.hasExpirationPeriod()) {
365  os << "ExpirationPeriod: " << parameters.getExpirationPeriod() << ", ";
366  }
367 
368  if (parameters.hasFacePersistency()) {
369  os << "FacePersistency: " << parameters.getFacePersistency() << ", ";
370  }
371 
372  if (parameters.hasBaseCongestionMarkingInterval()) {
373  os << "BaseCongestionMarkingInterval: " << parameters.getBaseCongestionMarkingInterval() << ", ";
374  }
375 
376  if (parameters.hasDefaultCongestionThreshold()) {
377  os << "DefaultCongestionThreshold: " << parameters.getDefaultCongestionThreshold() << ", ";
378  }
379 
380  if (parameters.hasMtu()) {
381  os << "Mtu: " << parameters.getMtu() << ", ";
382  }
383 
384  os << ")";
385  return os;
386 }
387 
388 } // namespace nfd
389 } // namespace ndn
bool hasWire() const
Check if the Block has fully encoded wire.
Definition: block.cpp:249
void wireDecode(const Block &wire) final
Copyright (c) 2011-2015 Regents of the University of California.
size_t prependNonNegativeIntegerBlock(EncodingImpl< TAG > &encoder, uint32_t type, uint64_t value)
Prepend a TLV element containing a non-negative integer.
represents parameters in a ControlCommand request or response
time::nanoseconds getBaseCongestionMarkingInterval() const
element_container::const_iterator element_const_iterator
Definition: block.hpp:47
Helper class to convert a number to hexadecimal format, for use with stream insertion operators.
size_t prependNestedBlock(EncodingImpl< TAG > &encoder, uint32_t type, const U &value)
Prepend a TLV element containing a nested TLV element.
void parse() const
Parse TLV-VALUE into sub elements.
Definition: block.cpp:333
ControlParameters & setFlags(uint64_t flags)
ControlParameters & unsetMask()
NDN_CXX_DEFINE_WIRE_ENCODE_INSTANTIATIONS(ChannelStatus)
Represents a TLV element of NDN packet format.
Definition: block.hpp:42
size_t prependStringBlock(EncodingImpl< TAG > &encoder, uint32_t type, const std::string &value)
Prepend a TLV element containing a string.
std::string readString(const Block &block)
Read TLV-VALUE of a TLV element as a string.
ControlParameters & unsetFlags()
uint64_t readNonNegativeInteger(const Block &block)
Read a non-negative integer from a TLV element.
ControlParameters & unsetFlagBit(size_t bit)
disable a bit in Mask
FacePersistency getFacePersistency() const
uint64_t getMtu() const
get MTU (measured in bytes)
element_const_iterator elements_end() const
Equivalent to elements().end()
Definition: block.hpp:377
Copyright (c) 2011-2015 Regents of the University of California.
Definition: ndn-common.hpp:40
element_const_iterator find(uint32_t type) const
Find the first sub element of specified TLV-TYPE.
Definition: block.cpp:434
const std::string & getLocalUri() const
uint64_t getDefaultCongestionThreshold() const
get default congestion threshold (measured in bytes)
std::ostream & operator<<(std::ostream &os, FaceScope faceScope)
bool hasFlagBit(size_t bit) const
const std::string & getUri() const
const time::milliseconds & getExpirationPeriod() const
size_t wireEncode(EncodingImpl< TAG > &encoder) const
Fast encoding or block size estimation.
Definition: name.cpp:125
bool getFlagBit(size_t bit) const
void wireDecode(const Block &wire)
Decode name from wire encoding.
Definition: name.cpp:158
a concept check for TLV abstraction with .wireEncode method
Definition: concepts.hpp:44
const Name & getStrategy() const
ControlParameters & setMask(uint64_t mask)
a concept check for TLV abstraction with .wireDecode method and constructible from Block
Definition: concepts.hpp:80
ControlParameters & setFlagBit(size_t bit, bool value, bool wantMask=true)
set a bit in Flags
EncodingImpl< EncoderTag > EncodingBuffer
uint32_t type() const
Get TLV-TYPE.
Definition: block.hpp:249
EncodingImpl< EstimatorTag > EncodingEstimator