blob: 465482ff662b542307a551a4dec7bcfad8a74d15 [file] [log] [blame]
Junxiao Shi70911652014-08-12 10:14:24 -07001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Junxiao Shi5d75fd92017-08-08 18:09:20 +00002/*
Davide Pesaventoe78eeca2017-02-23 23:22:32 -05003 * Copyright (c) 2013-2017 Regents of the University of California.
Junxiao Shi70911652014-08-12 10:14:24 -07004 *
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
Junxiao Shi7357ef22016-09-07 02:39:37 +000022#include "control-parameters.hpp"
Junxiao Shi65f1a712014-11-20 14:59:36 -070023#include "encoding/block-helpers.hpp"
Junxiao Shi5d75fd92017-08-08 18:09:20 +000024#include "encoding/tlv-nfd.hpp"
Junxiao Shi65f1a712014-11-20 14:59:36 -070025#include "util/concepts.hpp"
Davide Pesaventoe78eeca2017-02-23 23:22:32 -050026#include "util/string-helper.hpp"
Junxiao Shi70911652014-08-12 10:14:24 -070027
28namespace ndn {
29namespace nfd {
30
Junxiao Shi65f1a712014-11-20 14:59:36 -070031//BOOST_CONCEPT_ASSERT((boost::EqualityComparable<ControlParameters>));
32BOOST_CONCEPT_ASSERT((WireEncodable<ControlParameters>));
33BOOST_CONCEPT_ASSERT((WireDecodable<ControlParameters>));
34static_assert(std::is_base_of<tlv::Error, ControlParameters::Error>::value,
35 "ControlParameters::Error must inherit from tlv::Error");
36
Junxiao Shi70911652014-08-12 10:14:24 -070037ControlParameters::ControlParameters()
38 : m_hasFields(CONTROL_PARAMETER_UBOUND)
39{
40}
41
42ControlParameters::ControlParameters(const Block& block)
43 : m_hasFields(CONTROL_PARAMETER_UBOUND)
44{
45 wireDecode(block);
46}
47
Alexander Afanasyev74633892015-02-08 18:08:46 -080048template<encoding::Tag TAG>
Junxiao Shi70911652014-08-12 10:14:24 -070049size_t
Alexander Afanasyev74633892015-02-08 18:08:46 -080050ControlParameters::wireEncode(EncodingImpl<TAG>& encoder) const
Junxiao Shi70911652014-08-12 10:14:24 -070051{
52 size_t totalLength = 0;
53
Yukai Tud93c5fc2015-08-25 11:37:16 +080054 if (this->hasFacePersistency()) {
55 totalLength += prependNonNegativeIntegerBlock(encoder,
56 tlv::nfd::FacePersistency, m_facePersistency);
57 }
Junxiao Shi70911652014-08-12 10:14:24 -070058 if (this->hasExpirationPeriod()) {
59 totalLength += prependNonNegativeIntegerBlock(encoder,
60 tlv::nfd::ExpirationPeriod, m_expirationPeriod.count());
61 }
62 if (this->hasStrategy()) {
63 totalLength += prependNestedBlock(encoder, tlv::nfd::Strategy, m_strategy);
64 }
Eric Newberryda916d62016-08-11 23:04:34 -070065 if (this->hasMask()) {
66 totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::Mask, m_mask);
67 }
Junxiao Shi70911652014-08-12 10:14:24 -070068 if (this->hasFlags()) {
69 totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::Flags, m_flags);
70 }
71 if (this->hasCost()) {
72 totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::Cost, m_cost);
73 }
74 if (this->hasOrigin()) {
75 totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::Origin, m_origin);
76 }
Eric Newberryd7f5b282017-03-28 19:55:20 -070077 if (this->hasLocalUri()) {
Junxiao Shi5d75fd92017-08-08 18:09:20 +000078 totalLength += prependStringBlock(encoder, tlv::nfd::LocalUri, m_localUri);
Eric Newberryd7f5b282017-03-28 19:55:20 -070079 }
Junxiao Shi70911652014-08-12 10:14:24 -070080 if (this->hasUri()) {
Junxiao Shi5d75fd92017-08-08 18:09:20 +000081 totalLength += prependStringBlock(encoder, tlv::nfd::Uri, m_uri);
Junxiao Shi70911652014-08-12 10:14:24 -070082 }
83 if (this->hasFaceId()) {
84 totalLength += prependNonNegativeIntegerBlock(encoder, tlv::nfd::FaceId, m_faceId);
85 }
86 if (this->hasName()) {
87 totalLength += m_name.wireEncode(encoder);
88 }
89
90 totalLength += encoder.prependVarNumber(totalLength);
91 totalLength += encoder.prependVarNumber(tlv::nfd::ControlParameters);
92 return totalLength;
93}
94
95template size_t
Alexander Afanasyev74633892015-02-08 18:08:46 -080096ControlParameters::wireEncode<encoding::EncoderTag>(EncodingImpl<encoding::EncoderTag>&) const;
Junxiao Shi70911652014-08-12 10:14:24 -070097
98template size_t
Alexander Afanasyev74633892015-02-08 18:08:46 -080099ControlParameters::wireEncode<encoding::EstimatorTag>(EncodingImpl<encoding::EstimatorTag>&) const;
Junxiao Shi70911652014-08-12 10:14:24 -0700100
Yanbiao Li8ee37ed2015-05-19 12:44:04 -0700101Block
Junxiao Shi70911652014-08-12 10:14:24 -0700102ControlParameters::wireEncode() const
103{
104 if (m_wire.hasWire())
105 return m_wire;
106
107 EncodingEstimator estimator;
108 size_t estimatedSize = wireEncode(estimator);
109
110 EncodingBuffer buffer(estimatedSize, 0);
111 wireEncode(buffer);
112
113 m_wire = buffer.block();
114 return m_wire;
115}
116
117void
118ControlParameters::wireDecode(const Block& block)
119{
120 if (block.type() != tlv::nfd::ControlParameters) {
Spyridon Mastorakis0d2ed2e2015-07-27 19:09:12 -0700121 BOOST_THROW_EXCEPTION(Error("Expecting TLV-TYPE ControlParameters"));
Junxiao Shi70911652014-08-12 10:14:24 -0700122 }
123 m_wire = block;
124 m_wire.parse();
125 Block::element_const_iterator val;
126
127 val = m_wire.find(tlv::Name);
128 m_hasFields[CONTROL_PARAMETER_NAME] = val != m_wire.elements_end();
129 if (this->hasName()) {
130 m_name.wireDecode(*val);
131 }
132
133 val = m_wire.find(tlv::nfd::FaceId);
134 m_hasFields[CONTROL_PARAMETER_FACE_ID] = val != m_wire.elements_end();
135 if (this->hasFaceId()) {
Junxiao Shi5d75fd92017-08-08 18:09:20 +0000136 m_faceId = readNonNegativeInteger(*val);
Junxiao Shi70911652014-08-12 10:14:24 -0700137 }
138
139 val = m_wire.find(tlv::nfd::Uri);
140 m_hasFields[CONTROL_PARAMETER_URI] = val != m_wire.elements_end();
141 if (this->hasUri()) {
Junxiao Shi5d75fd92017-08-08 18:09:20 +0000142 m_uri = readString(*val);
Junxiao Shi70911652014-08-12 10:14:24 -0700143 }
144
Eric Newberryd7f5b282017-03-28 19:55:20 -0700145 val = m_wire.find(tlv::nfd::LocalUri);
146 m_hasFields[CONTROL_PARAMETER_LOCAL_URI] = val != m_wire.elements_end();
147 if (this->hasLocalUri()) {
Junxiao Shi5d75fd92017-08-08 18:09:20 +0000148 m_localUri = readString(*val);
Eric Newberryd7f5b282017-03-28 19:55:20 -0700149 }
150
Junxiao Shi70911652014-08-12 10:14:24 -0700151 val = m_wire.find(tlv::nfd::Origin);
152 m_hasFields[CONTROL_PARAMETER_ORIGIN] = val != m_wire.elements_end();
153 if (this->hasOrigin()) {
Junxiao Shi5d75fd92017-08-08 18:09:20 +0000154 m_origin = readNonNegativeIntegerAs<RouteOrigin>(*val);
Junxiao Shi70911652014-08-12 10:14:24 -0700155 }
156
157 val = m_wire.find(tlv::nfd::Cost);
158 m_hasFields[CONTROL_PARAMETER_COST] = val != m_wire.elements_end();
159 if (this->hasCost()) {
Junxiao Shi5d75fd92017-08-08 18:09:20 +0000160 m_cost = readNonNegativeInteger(*val);
Junxiao Shi70911652014-08-12 10:14:24 -0700161 }
162
163 val = m_wire.find(tlv::nfd::Flags);
164 m_hasFields[CONTROL_PARAMETER_FLAGS] = val != m_wire.elements_end();
165 if (this->hasFlags()) {
Junxiao Shi5d75fd92017-08-08 18:09:20 +0000166 m_flags = readNonNegativeInteger(*val);
Junxiao Shi70911652014-08-12 10:14:24 -0700167 }
168
Eric Newberryda916d62016-08-11 23:04:34 -0700169 val = m_wire.find(tlv::nfd::Mask);
170 m_hasFields[CONTROL_PARAMETER_MASK] = val != m_wire.elements_end();
171 if (this->hasMask()) {
Junxiao Shi5d75fd92017-08-08 18:09:20 +0000172 m_mask = readNonNegativeInteger(*val);
Eric Newberryda916d62016-08-11 23:04:34 -0700173 }
174
Junxiao Shi70911652014-08-12 10:14:24 -0700175 val = m_wire.find(tlv::nfd::Strategy);
176 m_hasFields[CONTROL_PARAMETER_STRATEGY] = val != m_wire.elements_end();
177 if (this->hasStrategy()) {
178 val->parse();
179 if (val->elements().empty()) {
Spyridon Mastorakis0d2ed2e2015-07-27 19:09:12 -0700180 BOOST_THROW_EXCEPTION(Error("Expecting Strategy/Name"));
Junxiao Shi70911652014-08-12 10:14:24 -0700181 }
182 else {
183 m_strategy.wireDecode(*val->elements_begin());
184 }
185 }
186
187 val = m_wire.find(tlv::nfd::ExpirationPeriod);
188 m_hasFields[CONTROL_PARAMETER_EXPIRATION_PERIOD] = val != m_wire.elements_end();
189 if (this->hasExpirationPeriod()) {
190 m_expirationPeriod = time::milliseconds(readNonNegativeInteger(*val));
191 }
Yukai Tud93c5fc2015-08-25 11:37:16 +0800192
193 val = m_wire.find(tlv::nfd::FacePersistency);
194 m_hasFields[CONTROL_PARAMETER_FACE_PERSISTENCY] = val != m_wire.elements_end();
195 if (this->hasFacePersistency()) {
Junxiao Shi5d75fd92017-08-08 18:09:20 +0000196 m_facePersistency = readNonNegativeIntegerAs<FacePersistency>(*val);
Yukai Tud93c5fc2015-08-25 11:37:16 +0800197 }
Junxiao Shi70911652014-08-12 10:14:24 -0700198}
199
Eric Newberryda916d62016-08-11 23:04:34 -0700200bool
201ControlParameters::hasFlagBit(size_t bit) const
202{
203 if (bit >= 64) {
204 BOOST_THROW_EXCEPTION(std::out_of_range("bit must be within range [0, 64)"));
205 }
206
207 if (!hasMask()) {
208 return false;
209 }
210
211 return getMask() & (1 << bit);
212}
213
214bool
215ControlParameters::getFlagBit(size_t bit) const
216{
217 if (bit >= 64) {
218 BOOST_THROW_EXCEPTION(std::out_of_range("bit must be within range [0, 64)"));
219 }
220
221 if (!hasFlags()) {
222 return false;
223 }
224
225 return getFlags() & (1 << bit);
226}
227
228ControlParameters&
229ControlParameters::setFlagBit(size_t bit, bool value, bool wantMask/* = true*/)
230{
231 if (bit >= 64) {
232 BOOST_THROW_EXCEPTION(std::out_of_range("bit must be within range [0, 64)"));
233 }
234
235 uint64_t flags = hasFlags() ? getFlags() : 0;
236 if (value) {
237 flags |= (1 << bit);
238 }
239 else {
240 flags &= ~(1 << bit);
241 }
242 setFlags(flags);
243
244 if (wantMask) {
245 uint64_t mask = hasMask() ? getMask() : 0;
246 mask |= (1 << bit);
247 setMask(mask);
248 }
249
250 return *this;
251}
252
253ControlParameters&
254ControlParameters::unsetFlagBit(size_t bit)
255{
256 if (bit >= 64) {
257 BOOST_THROW_EXCEPTION(std::out_of_range("bit must be within range [0, 64)"));
258 }
259
260 uint64_t mask = hasMask() ? getMask() : 0;
261 mask &= ~(1 << bit);
262 if (mask == 0) {
263 unsetMask();
264 unsetFlags();
265 }
266 else {
267 setMask(mask);
268 }
269
270 return *this;
271}
272
Junxiao Shi70911652014-08-12 10:14:24 -0700273std::ostream&
274operator<<(std::ostream& os, const ControlParameters& parameters)
275{
276 os << "ControlParameters(";
277
278 if (parameters.hasName()) {
279 os << "Name: " << parameters.getName() << ", ";
280 }
281
282 if (parameters.hasFaceId()) {
283 os << "FaceId: " << parameters.getFaceId() << ", ";
284 }
285
286 if (parameters.hasUri()) {
287 os << "Uri: " << parameters.getUri() << ", ";
288 }
289
Eric Newberryd7f5b282017-03-28 19:55:20 -0700290 if (parameters.hasLocalUri()) {
291 os << "LocalUri: " << parameters.getLocalUri() << ", ";
292 }
293
Junxiao Shi70911652014-08-12 10:14:24 -0700294 if (parameters.hasOrigin()) {
295 os << "Origin: " << parameters.getOrigin() << ", ";
296 }
297
298 if (parameters.hasCost()) {
299 os << "Cost: " << parameters.getCost() << ", ";
300 }
301
302 if (parameters.hasFlags()) {
Davide Pesaventoe78eeca2017-02-23 23:22:32 -0500303 os << "Flags: " << AsHex{parameters.getFlags()} << ", ";
Eric Newberryda916d62016-08-11 23:04:34 -0700304 }
305
306 if (parameters.hasMask()) {
Davide Pesaventoe78eeca2017-02-23 23:22:32 -0500307 os << "Mask: " << AsHex{parameters.getMask()} << ", ";
Junxiao Shi70911652014-08-12 10:14:24 -0700308 }
309
310 if (parameters.hasStrategy()) {
311 os << "Strategy: " << parameters.getStrategy() << ", ";
312 }
313
314 if (parameters.hasExpirationPeriod()) {
315 os << "ExpirationPeriod: " << parameters.getExpirationPeriod() << ", ";
316 }
317
Yanbiao Licbdacb22016-08-02 16:02:35 +0800318 if (parameters.hasFacePersistency()) {
319 os << "FacePersistency: " << parameters.getFacePersistency() << ", ";
320 }
321
Junxiao Shi70911652014-08-12 10:14:24 -0700322 os << ")";
323 return os;
324}
325
326} // namespace nfd
327} // namespace ndn