gem5  v21.0.1.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
inet.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2013 ARM Limited
3  * All rights reserved
4  *
5  * The license below extends only to copyright in the software and shall
6  * not be construed as granting a license to any other intellectual
7  * property including but not limited to intellectual property relating
8  * to a hardware implementation of the functionality of the software
9  * licensed hereunder. You may use the software subject to the license
10  * terms below provided that you ensure that this notice is replicated
11  * unmodified and in its entirety in all distributions of the software,
12  * modified or unmodified, in source code or in binary form.
13  *
14  * Copyright (c) 2002-2005 The Regents of The University of Michigan
15  * Copyright (c) 2010 Advanced Micro Devices, Inc.
16  * All rights reserved.
17  *
18  * Redistribution and use in source and binary forms, with or without
19  * modification, are permitted provided that the following conditions are
20  * met: redistributions of source code must retain the above copyright
21  * notice, this list of conditions and the following disclaimer;
22  * redistributions in binary form must reproduce the above copyright
23  * notice, this list of conditions and the following disclaimer in the
24  * documentation and/or other materials provided with the distribution;
25  * neither the name of the copyright holders nor the names of its
26  * contributors may be used to endorse or promote products derived from
27  * this software without specific prior written permission.
28  *
29  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
30  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
31  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
32  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
33  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
34  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
35  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
36  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
37  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
38  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
39  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
40  */
41 
42 #include "base/inet.hh"
43 
44 #include <cstddef>
45 #include <cstdio>
46 #include <sstream>
47 #include <string>
48 
49 #include "base/cprintf.hh"
50 #include "base/logging.hh"
51 #include "base/types.hh"
52 
53 namespace Net {
54 
56 {
57  std::memset(data, 0, ETH_ADDR_LEN);
58 }
59 
60 EthAddr::EthAddr(const uint8_t ea[ETH_ADDR_LEN])
61 {
62  for (int i = 0; i < ETH_ADDR_LEN; ++i)
63  data[i] = ea[i];
64 }
65 
66 EthAddr::EthAddr(const eth_addr &ea)
67 {
68  for (int i = 0; i < ETH_ADDR_LEN; ++i)
69  data[i] = ea.data[i];
70 }
71 
72 EthAddr::EthAddr(const std::string &addr)
73 {
74  parse(addr);
75 }
76 
77 const EthAddr &
78 EthAddr::operator=(const eth_addr &ea)
79 {
80  *data = *ea.data;
81  return *this;
82 }
83 
84 const EthAddr &
85 EthAddr::operator=(const std::string &addr)
86 {
87  parse(addr);
88  return *this;
89 }
90 
91 void
92 EthAddr::parse(const std::string &addr)
93 {
94  // the hack below is to make sure that ETH_ADDR_LEN is 6 otherwise
95  // the sscanf function won't work.
96  int bytes[ETH_ADDR_LEN == 6 ? ETH_ADDR_LEN : -1];
97  if (sscanf(addr.c_str(), "%x:%x:%x:%x:%x:%x", &bytes[0], &bytes[1],
98  &bytes[2], &bytes[3], &bytes[4], &bytes[5]) != ETH_ADDR_LEN) {
99  std::memset(data, 0xff, ETH_ADDR_LEN);
100  return;
101  }
102 
103  for (int i = 0; i < ETH_ADDR_LEN; ++i) {
104  if (bytes[i] & ~0xff) {
105  std::memset(data, 0xff, ETH_ADDR_LEN);
106  return;
107  }
108 
109  data[i] = bytes[i];
110  }
111 }
112 
113 std::string
115 {
116  std::stringstream stream;
117  stream << *this;
118  return stream.str();
119 }
120 
121 bool
122 operator==(const EthAddr &left, const EthAddr &right)
123 {
124  return !std::memcmp(left.bytes(), right.bytes(), ETH_ADDR_LEN);
125 }
126 
127  std::ostream &
128 operator<<(std::ostream &stream, const EthAddr &ea)
129 {
130  const uint8_t *a = ea.addr();
131  ccprintf(stream, "%x:%x:%x:%x:%x:%x", a[0], a[1], a[2], a[3], a[4], a[5]);
132  return stream;
133 }
134 
135 std::string
137 {
138  std::stringstream stream;
139  stream << *this;
140  return stream.str();
141 }
142 
143 bool
144 operator==(const IpAddress &left, const IpAddress &right)
145 {
146  return left.ip() == right.ip();
147 }
148 
149 std::ostream &
150 operator<<(std::ostream &stream, const IpAddress &ia)
151 {
152  uint32_t ip = ia.ip();
153  ccprintf(stream, "%x.%x.%x.%x",
154  (uint8_t)(ip >> 24), (uint8_t)(ip >> 16),
155  (uint8_t)(ip >> 8), (uint8_t)(ip >> 0));
156  return stream;
157 }
158 
159 std::string
161 {
162  std::stringstream stream;
163  stream << *this;
164  return stream.str();
165 }
166 
167 bool
168 operator==(const IpNetmask &left, const IpNetmask &right)
169 {
170  return (left.ip() == right.ip()) &&
171  (left.netmask() == right.netmask());
172 }
173 
174 std::ostream &
175 operator<<(std::ostream &stream, const IpNetmask &in)
176 {
177  ccprintf(stream, "%s/%d", (const IpAddress &)in, in.netmask());
178  return stream;
179 }
180 
181 std::string
183 {
184  std::stringstream stream;
185  stream << *this;
186  return stream.str();
187 }
188 
189 bool
190 operator==(const IpWithPort &left, const IpWithPort &right)
191 {
192  return (left.ip() == right.ip()) && (left.port() == right.port());
193 }
194 
195 std::ostream &
196 operator<<(std::ostream &stream, const IpWithPort &iwp)
197 {
198  ccprintf(stream, "%s:%d", (const IpAddress &)iwp, iwp.port());
199  return stream;
200 }
201 
202 uint16_t
203 cksum(const IpPtr &ptr)
204 {
205  int sum = ip_cksum_add(ptr->bytes(), ptr->hlen(), 0);
206  return ip_cksum_carry(sum);
207 }
208 
209 uint16_t
211 {
212  int tcplen = ip->len() - ip->hlen();
213  int sum = ip_cksum_add(ip->payload(), tcplen, 0);
214  sum = ip_cksum_add(&ip->ip_src, 8, sum); // source and destination
215  sum += htons(ip->ip_p + tcplen);
216  return ip_cksum_carry(sum);
217 }
218 
219 uint16_t
221 {
222  int tcplen = ip6->plen() - ip6->extensionLength();
223  int sum = ip_cksum_add(ip6->payload(), tcplen, 0);
224  sum = ip_cksum_add(ip6->src(), 32, sum);
225  sum += htons(ip6->proto() + tcplen);
226  return ip_cksum_carry(sum);
227 }
228 
229 uint16_t
230 cksum(const TcpPtr &tcp)
231 {
232  if (IpPtr(tcp.packet())) {
233  return __tu_cksum(IpPtr(tcp.packet()));
234  } else if (Ip6Ptr(tcp.packet())) {
235  return __tu_cksum6(Ip6Ptr(tcp.packet()));
236  } else {
237  panic("Unrecognized IP packet format");
238  }
239  // Should never reach here
240  return 0;
241 }
242 
243 uint16_t
244 cksum(const UdpPtr &udp)
245 {
246  if (IpPtr(udp.packet())) {
247  return __tu_cksum(IpPtr(udp.packet()));
248  } else if (Ip6Ptr(udp.packet())) {
249  return __tu_cksum6(Ip6Ptr(udp.packet()));
250  } else {
251  panic("Unrecognized IP packet format");
252  }
253  return 0;
254 }
255 
256 bool
258 {
259  vec.clear();
260 
261  const uint8_t *data = bytes() + sizeof(struct ip_hdr);
262  int all = hlen() - sizeof(struct ip_hdr);
263  while (all > 0) {
264  const IpOpt *opt = (const IpOpt *)data;
265  int len = opt->len();
266  if (all < len)
267  return false;
268 
269  vec.push_back(opt);
270  all -= len;
271  data += len;
272  }
273 
274  return true;
275 }
276 
277 #define IP6_EXTENSION(nxt) (nxt == IP_PROTO_HOPOPTS) ? true : \
278  (nxt == IP_PROTO_ROUTING) ? true : \
279  (nxt == IP_PROTO_FRAGMENT) ? true : \
280  (nxt == IP_PROTO_AH) ? true : \
281  (nxt == IP_PROTO_ESP) ? true: \
282  (nxt == IP_PROTO_DSTOPTS) ? true : false
283 
284 /* Scan the IP6 header for all header extensions
285  * and return the number of headers found
286  */
287 int
289 {
290  const uint8_t *data = bytes() + IP6_HDR_LEN;
291  uint8_t nxt = ip6_nxt;
292  int len = 0;
293  int all = plen();
294 
295  while (IP6_EXTENSION(nxt)) {
296  const Ip6Opt *ext = (const Ip6Opt *)data;
297  nxt = ext->nxt();
298  len += ext->len();
299  data += ext->len();
300  all -= ext->len();
301  assert(all >= 0);
302  }
303  return len;
304 }
305 
306 /* Scan the IP6 header for a particular extension
307  * header type and return a pointer to it if it
308  * exists, otherwise return NULL
309  */
310 const Ip6Opt*
311 Ip6Hdr::getExt(uint8_t ext_type) const
312 {
313  const uint8_t *data = bytes() + IP6_HDR_LEN;
314  uint8_t nxt = ip6_nxt;
315  Ip6Opt* opt = NULL;
316  int all = plen();
317 
318  while (IP6_EXTENSION(nxt)) {
319  opt = (Ip6Opt *)data;
320  if (nxt == ext_type) {
321  break;
322  }
323  nxt = opt->nxt();
324  data += opt->len();
325  all -= opt->len();
326  opt = NULL;
327  assert(all >= 0);
328  }
329  return (const Ip6Opt*)opt;
330 }
331 
332 /* Scan the IP6 header and any extension headers
333  * to find what type of Layer 4 header exists
334  * after this header
335  */
336 uint8_t
338 {
339  const uint8_t *data = bytes() + IP6_HDR_LEN;
340  uint8_t nxt = ip6_nxt;
341  int all = plen();
342 
343  while (IP6_EXTENSION(nxt)) {
344  const Ip6Opt *ext = (const Ip6Opt *)data;
345  nxt = ext->nxt();
346  data += ext->len();
347  all -= ext->len();
348  assert(all >= 0);
349  }
350  return nxt;
351 }
352 
353 bool
355 {
356  vec.clear();
357 
358  const uint8_t *data = bytes() + sizeof(struct tcp_hdr);
359  int all = off() - sizeof(struct tcp_hdr);
360  while (all > 0) {
361  const TcpOpt *opt = (const TcpOpt *)data;
362  int len = opt->len();
363  if (all < len)
364  return false;
365 
366  vec.push_back(opt);
367  all -= len;
368  data += len;
369  }
370 
371  return true;
372 }
373 
374 int
375 hsplit(const EthPacketPtr &ptr)
376 {
377  int split_point = 0;
378 
379  IpPtr ip(ptr);
380  Ip6Ptr ip6(ptr);
381  if (ip) {
382  split_point = ip.pstart();
383 
384  TcpPtr tcp(ip);
385  if (tcp)
386  split_point = tcp.pstart();
387 
388  UdpPtr udp(ip);
389  if (udp)
390  split_point = udp.pstart();
391  } else if (ip6) {
392  split_point = ip6.pstart();
393 
394  TcpPtr tcp(ip6);
395  if (tcp)
396  split_point = tcp.pstart();
397  UdpPtr udp(ip6);
398  if (udp)
399  split_point = udp.pstart();
400  }
401  return split_point;
402 }
403 
404 
405 } // namespace Net
Net::IpNetmask
Definition: inet.hh:266
Net::IpWithPort::string
std::string string() const
Definition: inet.cc:182
Net::Ip6Hdr::plen
uint16_t plen() const
Definition: inet.hh:455
Net::Ip6Hdr::extensionLength
int extensionLength() const
Definition: inet.cc:288
Net::IpHdr::options
bool options(std::vector< const IpOpt * > &vec) const
Definition: inet.cc:257
Net::TcpHdr::options
bool options(std::vector< const TcpOpt * > &vec) const
Definition: inet.cc:354
Net::EthAddr::EthAddr
EthAddr()
Definition: inet.cc:55
data
const char data[]
Definition: circlebuf.test.cc:47
Net::operator==
bool operator==(const EthAddr &left, const EthAddr &right)
Definition: inet.cc:122
Net::EthAddr::bytes
const uint8_t * bytes() const
Definition: inet.hh:100
Net::IpHdr::bytes
const uint8_t * bytes() const
Definition: inet.hh:345
RiscvISA::sum
Bitfield< 18 > sum
Definition: registers.hh:634
Net::Ip6Hdr::getExt
const Ip6Opt * getExt(uint8_t ext) const
Definition: inet.cc:311
ArmISA::i
Bitfield< 7 > i
Definition: miscregs_types.hh:63
Net::EthAddr::string
std::string string() const
Definition: inet.cc:114
Net::Ip6Hdr::proto
uint8_t proto() const
Definition: inet.cc:337
Net::IpAddress
Definition: inet.hh:231
Net::EthAddr
Definition: inet.hh:72
Net::TcpPtr
Definition: inet.hh:637
iGbReg::TxdOp::tcp
bool tcp(TxDesc *d)
Definition: i8254xGBe_defs.hh:262
Net::Ip6Hdr::bytes
const uint8_t * bytes() const
Definition: inet.hh:473
Net::IpNetmask::string
std::string string() const
Definition: inet.cc:160
Net::UdpPtr
Definition: inet.hh:751
Net::IpAddress::ip
uint32_t ip() const
Definition: inet.hh:250
std::vector
STL vector class.
Definition: stl.hh:37
Net::IpHdr::hlen
uint8_t hlen() const
Definition: inet.hh:326
Net::operator<<
std::ostream & operator<<(std::ostream &stream, const EthAddr &ea)
Definition: inet.cc:128
IP6_EXTENSION
#define IP6_EXTENSION(nxt)
Definition: inet.cc:277
Net
Definition: inet.cc:53
Net::IpOpt::len
uint8_t len() const
Definition: inet.hh:428
Net::EthAddr::addr
const uint8_t * addr() const
Definition: inet.hh:108
Net::TcpOpt::len
uint8_t len() const
Definition: inet.hh:709
ArmISA::a
Bitfield< 8 > a
Definition: miscregs_types.hh:62
Net::IpPtr
Definition: inet.hh:351
MipsISA::ia
Bitfield< 18, 16 > ia
Definition: pra_constants.hh:234
Net::IpWithPort
Definition: inet.hh:294
Net::Ip6Opt::nxt
uint8_t nxt() const
Definition: inet.hh:585
Net::__tu_cksum6
uint16_t __tu_cksum6(const Ip6Ptr &ip6)
Definition: inet.cc:220
ArmISA::ext
Bitfield< 12 > ext
Definition: miscregs_types.hh:422
cprintf.hh
Net::IpWithPort::port
uint8_t port() const
Definition: inet.hh:309
Net::__tu_cksum
uint16_t __tu_cksum(const IpPtr &ip)
Definition: inet.cc:210
Net::EthAddr::operator=
const EthAddr & operator=(const eth_addr &ea)
Definition: inet.cc:78
Net::cksum
uint16_t cksum(const IpPtr &ptr)
Definition: inet.cc:203
X86ISA::addr
Bitfield< 3 > addr
Definition: types.hh:80
Net::IpAddress::string
std::string string() const
Definition: inet.cc:136
EthPacketPtr
std::shared_ptr< EthPacketData > EthPacketPtr
Definition: etherpkt.hh:87
Net::Ip6Opt
Definition: inet.hh:583
types.hh
Net::IpOpt
Definition: inet.hh:422
Net::Ip6Ptr
Definition: inet.hh:481
iGbReg::TxdOp::ip
bool ip(TxDesc *d)
Definition: i8254xGBe_defs.hh:261
ArmISA::len
Bitfield< 18, 16 > len
Definition: miscregs_types.hh:439
Net::Ip6Hdr::nxt
uint8_t nxt() const
Definition: inet.hh:457
ccprintf
void ccprintf(cp::Print &print)
Definition: cprintf.hh:127
logging.hh
ArmISA::ea
Bitfield< 3 > ea
Definition: miscregs_types.hh:325
Net::UdpPtr::packet
const EthPacketPtr packet() const
Definition: inet.hh:804
Net::TcpHdr::bytes
const uint8_t * bytes() const
Definition: inet.hh:631
Net::hsplit
int hsplit(const EthPacketPtr &ptr)
Definition: inet.cc:375
inet.hh
Net::EthAddr::parse
void parse(const std::string &addr)
Definition: inet.cc:92
Net::IpHdr::len
uint16_t len() const
Definition: inet.hh:328
Net::TcpHdr::off
uint8_t off() const
Definition: inet.hh:618
MipsISA::ip6
Bitfield< 14 > ip6
Definition: pra_constants.hh:187
Net::TcpOpt
Definition: inet.hh:706
Net::IpNetmask::netmask
uint8_t netmask() const
Definition: inet.hh:281
Net::UdpPtr::pstart
int pstart() const
Definition: inet.hh:809
panic
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:171
Net::Ip6Opt::len
uint8_t len() const
Definition: inet.hh:587

Generated on Tue Jun 22 2021 15:28:25 for gem5 by doxygen 1.8.17