blob: d7eebbd26d0cc990a1b4655c8e5e4d1d7f6eea40 [file] [log] [blame]
Davide Pesavento248d6bd2014-03-09 10:24:08 +01001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2/**
Davide Pesaventob499a602014-11-18 22:36:56 +01003 * Copyright (c) 2014, Regents of the University of California,
4 * Arizona Board of Regents,
5 * Colorado State University,
6 * University Pierre & Marie Curie, Sorbonne University,
7 * Washington University in St. Louis,
8 * Beijing Institute of Technology,
9 * The University of Memphis
Alexander Afanasyev9bcbc7c2014-04-06 19:37:37 -070010 *
11 * This file is part of NFD (Named Data Networking Forwarding Daemon).
12 * See AUTHORS.md for complete list of NFD authors and contributors.
13 *
14 * NFD is free software: you can redistribute it and/or modify it under the terms
15 * of the GNU General Public License as published by the Free Software Foundation,
16 * either version 3 of the License, or (at your option) any later version.
17 *
18 * NFD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
19 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
20 * PURPOSE. See the GNU General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License along with
23 * NFD, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
Davide Pesaventob499a602014-11-18 22:36:56 +010024 */
Davide Pesavento248d6bd2014-03-09 10:24:08 +010025
26#include "network-interface.hpp"
Steve DiBenedettobf6a93d2014-03-21 14:03:02 -060027#include "core/logger.hpp"
Davide Pesavento248d6bd2014-03-09 10:24:08 +010028
Davide Pesaventob499a602014-11-18 22:36:56 +010029#include <cerrno>
30#include <cstring>
31#include <type_traits>
32#include <unordered_map>
Davide Pesavento248d6bd2014-03-09 10:24:08 +010033
34#include <arpa/inet.h> // for inet_ntop()
35#include <netinet/in.h> // for struct sockaddr_in{,6}
36#include <ifaddrs.h> // for getifaddrs()
37
38#if defined(__linux__)
39#include <net/if_arp.h> // for ARPHRD_* constants
40#include <netpacket/packet.h> // for struct sockaddr_ll
Alexander Afanasyev500b2532014-03-31 12:29:25 -070041#elif defined(__APPLE__) || defined(__FreeBSD__)
Davide Pesavento248d6bd2014-03-09 10:24:08 +010042#include <net/if_dl.h> // for struct sockaddr_dl
Davide Pesaventob499a602014-11-18 22:36:56 +010043#include <net/if_types.h> // for IFT_* constants
Davide Pesavento248d6bd2014-03-09 10:24:08 +010044#endif
45
Davide Pesavento1bdef282014-04-08 20:59:50 +020046NFD_LOG_INIT("NetworkInterfaceInfo");
Davide Pesavento248d6bd2014-03-09 10:24:08 +010047
Davide Pesavento1bdef282014-04-08 20:59:50 +020048namespace nfd {
Davide Pesavento248d6bd2014-03-09 10:24:08 +010049
Davide Pesaventob499a602014-11-18 22:36:56 +010050static_assert(std::is_standard_layout<NetworkInterfaceInfo>::value,
51 "NetworkInterfaceInfo must be a standard layout type");
52#ifdef HAVE_IS_DEFAULT_CONSTRUCTIBLE
53static_assert(std::is_default_constructible<NetworkInterfaceInfo>::value,
54 "NetworkInterfaceInfo must provide a default constructor");
55#endif
56
57std::vector<NetworkInterfaceInfo>
Davide Pesavento248d6bd2014-03-09 10:24:08 +010058listNetworkInterfaces()
59{
Davide Pesaventob499a602014-11-18 22:36:56 +010060 using namespace boost::asio::ip;
61 using std::strerror;
Davide Pesavento248d6bd2014-03-09 10:24:08 +010062
Davide Pesaventob499a602014-11-18 22:36:56 +010063 std::unordered_map<std::string, NetworkInterfaceInfo> ifmap;
64 ifaddrs* ifa_list = nullptr;
65
Davide Pesavento248d6bd2014-03-09 10:24:08 +010066 if (::getifaddrs(&ifa_list) < 0)
Davide Pesaventob499a602014-11-18 22:36:56 +010067 throw std::runtime_error(std::string("getifaddrs() failed: ") + strerror(errno));
Davide Pesavento248d6bd2014-03-09 10:24:08 +010068
Davide Pesaventob499a602014-11-18 22:36:56 +010069 for (ifaddrs* ifa = ifa_list; ifa != nullptr; ifa = ifa->ifa_next) {
70 std::string ifname(ifa->ifa_name);
71 NetworkInterfaceInfo& netif = ifmap[ifname];
72 netif.name = ifa->ifa_name;
73 netif.flags = ifa->ifa_flags;
74
75 if (ifa->ifa_addr == nullptr)
76 continue;
77
78 switch (ifa->ifa_addr->sa_family) {
79
80 case AF_INET: {
81 const sockaddr_in* sin = reinterpret_cast<sockaddr_in*>(ifa->ifa_addr);
82 char address[INET_ADDRSTRLEN];
83 if (::inet_ntop(AF_INET, &sin->sin_addr, address, sizeof(address))) {
84 netif.ipv4Addresses.push_back(address_v4::from_string(address));
85 NFD_LOG_TRACE(ifname << ": added IPv4 address " << address);
86 }
Davide Pesavento248d6bd2014-03-09 10:24:08 +010087 else
Davide Pesaventob499a602014-11-18 22:36:56 +010088 NFD_LOG_WARN(ifname << ": inet_ntop(AF_INET) failed: " << strerror(errno));
89 break;
Davide Pesavento248d6bd2014-03-09 10:24:08 +010090 }
91
Davide Pesaventob499a602014-11-18 22:36:56 +010092 case AF_INET6: {
93 const sockaddr_in6* sin6 = reinterpret_cast<sockaddr_in6*>(ifa->ifa_addr);
94 char address[INET6_ADDRSTRLEN];
95 if (::inet_ntop(AF_INET6, &sin6->sin6_addr, address, sizeof(address))) {
96 netif.ipv6Addresses.push_back(address_v6::from_string(address));
97 NFD_LOG_TRACE(ifname << ": added IPv6 address " << address);
98 }
99 else
100 NFD_LOG_WARN(ifname << ": inet_ntop(AF_INET6) failed: " << strerror(errno));
101 break;
102 }
103
104#if defined(__linux__)
105 case AF_PACKET: {
106 const sockaddr_ll* sll = reinterpret_cast<sockaddr_ll*>(ifa->ifa_addr);
107 netif.index = sll->sll_ifindex;
108 if (sll->sll_hatype == ARPHRD_ETHER && sll->sll_halen == ethernet::ADDR_LEN) {
109 netif.etherAddress = ethernet::Address(sll->sll_addr);
110 NFD_LOG_TRACE(ifname << ": added Ethernet address " << netif.etherAddress);
111 }
112 else if (sll->sll_hatype != ARPHRD_LOOPBACK) {
113 NFD_LOG_DEBUG(ifname << ": ignoring link-layer address for unhandled hardware type "
114 << sll->sll_hatype);
115 }
116 break;
117 }
118
119#elif defined(__APPLE__) || defined(__FreeBSD__)
120 case AF_LINK: {
121 const sockaddr_dl* sdl = reinterpret_cast<sockaddr_dl*>(ifa->ifa_addr);
122 netif.index = sdl->sdl_index;
123 if (sdl->sdl_type == IFT_ETHER && sdl->sdl_alen == ethernet::ADDR_LEN) {
124 netif.etherAddress = ethernet::Address(reinterpret_cast<uint8_t*>(LLADDR(sdl)));
125 NFD_LOG_TRACE(ifname << ": added Ethernet address " << netif.etherAddress);
126 }
127 else if (sdl->sdl_type != IFT_LOOP) {
128 NFD_LOG_DEBUG(ifname << ": ignoring link-layer address for unhandled interface type "
129 << sdl->sdl_type);
130 }
131 break;
132 }
133#endif
134 }
135
136 if (netif.isBroadcastCapable() && ifa->ifa_broadaddr != nullptr) {
137 const sockaddr_in* sin = reinterpret_cast<sockaddr_in*>(ifa->ifa_broadaddr);
138 char address[INET_ADDRSTRLEN];
139 if (::inet_ntop(AF_INET, &sin->sin_addr, address, sizeof(address))) {
140 netif.broadcastAddress = address_v4::from_string(address);
141 NFD_LOG_TRACE(ifname << ": added IPv4 broadcast address " << address);
142 }
143 else
144 NFD_LOG_WARN(ifname << ": inet_ntop(AF_INET) for broadaddr failed: " << strerror(errno));
145 }
146 }
147
Davide Pesavento248d6bd2014-03-09 10:24:08 +0100148 ::freeifaddrs(ifa_list);
149
Davide Pesaventob499a602014-11-18 22:36:56 +0100150 std::vector<NetworkInterfaceInfo> v;
151 v.reserve(ifmap.size());
152 for (auto&& element : ifmap) {
153 v.push_back(element.second);
Davide Pesavento248d6bd2014-03-09 10:24:08 +0100154 }
155
Davide Pesaventob499a602014-11-18 22:36:56 +0100156 return v;
Davide Pesavento248d6bd2014-03-09 10:24:08 +0100157}
158
159} // namespace nfd