blob: f289c866d5d6dec79b013d9b6c9ffc15c597fc6a [file] [log] [blame]
Eric Newberry38982622015-08-06 21:39:55 -07001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Davide Pesavento88a0d812017-08-19 21:31:42 -04002/*
3 * Copyright (c) 2013-2017 Regents of the University of California.
Eric Newberry38982622015-08-06 21:39:55 -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 * @author Eric Newberry <enewberry@email.arizona.edu>
22 */
23
24#include "nack-header.hpp"
25
26namespace ndn {
27namespace lp {
28
29std::ostream&
30operator<<(std::ostream& os, NackReason reason)
31{
32 switch (reason) {
33 case NackReason::CONGESTION:
34 os << "Congestion";
35 break;
36 case NackReason::DUPLICATE:
37 os << "Duplicate";
38 break;
39 case NackReason::NO_ROUTE:
40 os << "NoRoute";
41 break;
42 default:
43 os << "None";
44 break;
45 }
46 return os;
47}
48
Junxiao Shi79a7a162017-09-09 08:33:57 +000049bool
50isLessSevere(lp::NackReason x, lp::NackReason y)
51{
52 if (x == lp::NackReason::NONE) {
53 return false;
54 }
55 if (y == lp::NackReason::NONE) {
56 return true;
57 }
58
59 return static_cast<int>(x) < static_cast<int>(y);
60}
61
Eric Newberry38982622015-08-06 21:39:55 -070062NackHeader::NackHeader()
63 : m_reason(NackReason::NONE)
64{
65}
66
67NackHeader::NackHeader(const Block& block)
68{
69 wireDecode(block);
70}
71
72template<encoding::Tag TAG>
73size_t
74NackHeader::wireEncode(EncodingImpl<TAG>& encoder) const
75{
76 size_t length = 0;
Davide Pesavento88a0d812017-08-19 21:31:42 -040077 length += prependNonNegativeIntegerBlock(encoder, tlv::NackReason, static_cast<uint32_t>(m_reason));
Eric Newberry38982622015-08-06 21:39:55 -070078 length += encoder.prependVarNumber(length);
79 length += encoder.prependVarNumber(tlv::Nack);
80 return length;
81}
82
Davide Pesavento88a0d812017-08-19 21:31:42 -040083NDN_CXX_DEFINE_WIRE_ENCODE_INSTANTIATIONS(NackHeader);
Eric Newberry38982622015-08-06 21:39:55 -070084
85const Block&
86NackHeader::wireEncode() const
87{
88 if (m_wire.hasWire()) {
89 return m_wire;
90 }
91
92 EncodingEstimator estimator;
93 size_t estimatedSize = wireEncode(estimator);
94
95 EncodingBuffer buffer(estimatedSize, 0);
96 wireEncode(buffer);
97
98 m_wire = buffer.block();
99
100 return m_wire;
101}
102
103void
104NackHeader::wireDecode(const Block& wire)
105{
106 if (wire.type() != tlv::Nack) {
Spyridon Mastorakis0d2ed2e2015-07-27 19:09:12 -0700107 BOOST_THROW_EXCEPTION(ndn::tlv::Error("expecting Nack block"));
Eric Newberry38982622015-08-06 21:39:55 -0700108 }
109
110 m_wire = wire;
111 m_wire.parse();
112 m_reason = NackReason::NONE;
113
114 if (m_wire.elements_size() > 0) {
115 Block::element_const_iterator it = m_wire.elements_begin();
116
117 if (it->type() == tlv::NackReason) {
118 m_reason = static_cast<NackReason>(readNonNegativeInteger(*it));
119 }
120 else {
Spyridon Mastorakis0d2ed2e2015-07-27 19:09:12 -0700121 BOOST_THROW_EXCEPTION(ndn::tlv::Error("expecting NackReason block"));
Eric Newberry38982622015-08-06 21:39:55 -0700122 }
123 }
124}
125
126NackReason
127NackHeader::getReason() const
128{
129 switch (m_reason) {
130 case NackReason::CONGESTION:
131 case NackReason::DUPLICATE:
132 case NackReason::NO_ROUTE:
133 return m_reason;
134 default:
135 return NackReason::NONE;
136 }
137}
138
139NackHeader&
140NackHeader::setReason(NackReason reason)
141{
142 m_reason = reason;
143 m_wire.reset();
144 return *this;
145}
146
147} // namespace lp
148} // namespace ndn