blob: 7e7f3435f0cab88a5d88ae4533ad74573d098ac5 [file] [log] [blame]
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -04001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2/**
3 * Copyright (c) 2014-2017, 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.
10 *
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/>.
24 */
25
26#include "multicast-ethernet-transport.hpp"
27#include "core/global-io.hpp"
28
29#include <pcap/pcap.h>
30
31#include <cerrno> // for errno
32#include <cstring> // for memcpy(), strerror(), strncpy()
33#include <net/if.h> // for struct ifreq
34#include <stdio.h> // for snprintf()
35#include <sys/ioctl.h> // for ioctl()
36
37#if defined(__linux__)
38#include <netpacket/packet.h> // for struct packet_mreq
39#include <sys/socket.h> // for setsockopt()
40#endif
41
42#ifdef SIOCADDMULTI
43#if defined(__APPLE__) || defined(__FreeBSD__)
44#include <net/if_dl.h> // for struct sockaddr_dl
45#endif
46#endif
47
48namespace nfd {
49namespace face {
50
51NFD_LOG_INIT("MulticastEthernetTransport");
52
53MulticastEthernetTransport::MulticastEthernetTransport(const NetworkInterfaceInfo& localEndpoint,
54 const ethernet::Address& mcastAddress,
55 ndn::nfd::LinkType linkType)
56 : EthernetTransport(localEndpoint, mcastAddress)
57{
58 this->setLocalUri(FaceUri::fromDev(m_interfaceName));
59 this->setRemoteUri(FaceUri(m_destAddress));
60 this->setScope(ndn::nfd::FACE_SCOPE_NON_LOCAL);
61 this->setPersistency(ndn::nfd::FACE_PERSISTENCY_PERMANENT);
62 this->setLinkType(linkType);
63
64 NFD_LOG_FACE_INFO("Creating transport");
65
66 char filter[110];
67 // note #1: we cannot use std::snprintf because it's not available
68 // on some platforms (see #2299)
69 // note #2: "not vlan" must appear last in the filter expression, or the
70 // rest of the filter won't work as intended (see pcap-filter(7))
71 snprintf(filter, sizeof(filter),
72 "(ether proto 0x%x) && (ether dst %s) && (not ether src %s) && (not vlan)",
73 ethernet::ETHERTYPE_NDN,
74 m_destAddress.toString().c_str(),
75 m_srcAddress.toString().c_str());
76 setPacketFilter(filter);
77
78 if (!m_destAddress.isBroadcast() && !joinMulticastGroup()) {
79 NFD_LOG_FACE_WARN("Falling back to promiscuous mode");
80 pcap_set_promisc(m_pcap.get(), 1);
81 }
82}
83
84bool
85MulticastEthernetTransport::joinMulticastGroup()
86{
87#if defined(__linux__)
88 packet_mreq mr{};
89 mr.mr_ifindex = m_interfaceIndex;
90 mr.mr_type = PACKET_MR_MULTICAST;
91 mr.mr_alen = m_destAddress.size();
92 std::memcpy(mr.mr_address, m_destAddress.data(), m_destAddress.size());
93
94 if (::setsockopt(m_socket.native_handle(), SOL_PACKET,
95 PACKET_ADD_MEMBERSHIP, &mr, sizeof(mr)) == 0)
96 return true; // success
97
98 NFD_LOG_FACE_WARN("setsockopt(PACKET_ADD_MEMBERSHIP) failed: " << std::strerror(errno));
99#endif
100
101#if defined(SIOCADDMULTI)
102 ifreq ifr{};
103 std::strncpy(ifr.ifr_name, m_interfaceName.c_str(), sizeof(ifr.ifr_name) - 1);
104
105#if defined(__APPLE__) || defined(__FreeBSD__)
106 // see bug #2327
107 using boost::asio::ip::udp;
108 udp::socket sock(getGlobalIoService(), udp::v4());
109 int fd = sock.native_handle();
110
111 // Differences between Linux and the BSDs (including macOS):
112 // o BSD does not have ifr_hwaddr; use ifr_addr instead.
113 // o While macOS seems to accept both AF_LINK and AF_UNSPEC as the address
114 // family, FreeBSD explicitly requires AF_LINK, so we have to use AF_LINK
115 // and sockaddr_dl instead of the generic sockaddr structure.
116 // o BSD's sockaddr (and sockaddr_dl in particular) contains an additional
117 // field, sa_len (sdl_len), which must be set to the total length of the
118 // structure, including the length field itself.
119 // o We do not specify the interface name, thus sdl_nlen is left at 0 and
120 // LLADDR is effectively the same as sdl_data.
121
122 sockaddr_dl* sdl = reinterpret_cast<sockaddr_dl*>(&ifr.ifr_addr);
123 sdl->sdl_len = sizeof(ifr.ifr_addr);
124 sdl->sdl_family = AF_LINK;
125 sdl->sdl_alen = m_destAddress.size();
126 std::memcpy(LLADDR(sdl), m_destAddress.data(), m_destAddress.size());
127
128 static_assert(sizeof(ifr.ifr_addr) >= offsetof(sockaddr_dl, sdl_data) + ethernet::ADDR_LEN,
129 "ifr_addr in struct ifreq is too small on this platform");
130#else
131 int fd = m_socket.native_handle();
132
133 ifr.ifr_hwaddr.sa_family = AF_UNSPEC;
134 std::memcpy(ifr.ifr_hwaddr.sa_data, m_destAddress.data(), m_destAddress.size());
135
136 static_assert(sizeof(ifr.ifr_hwaddr.sa_data) >= ethernet::ADDR_LEN,
137 "ifr_hwaddr in struct ifreq is too small on this platform");
138#endif
139
140 if (::ioctl(fd, SIOCADDMULTI, &ifr) == 0)
141 return true; // success
142
143 NFD_LOG_FACE_WARN("ioctl(SIOCADDMULTI) failed: " << std::strerror(errno));
144#endif
145
146 return false;
147}
148
149} // namespace face
150} // namespace nfd