blob: 2068012f791bb334a7d7898cba38c5f23f0079f4 [file] [log] [blame]
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -04001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Junxiao Shi0d82d042017-07-07 06:15:27 +00002/*
Davide Pesaventoc52cd5e2022-03-05 20:40:54 -05003 * Copyright (c) 2014-2022, Regents of the University of California,
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -04004 * 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"
Davide Pesavento2cae8ca2019-04-18 20:48:05 -040027#include "common/global.hpp"
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040028
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040029#include <cerrno> // for errno
Davide Pesaventoc52cd5e2022-03-05 20:40:54 -050030#include <cstdio> // for snprintf()
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040031#include <cstring> // for memcpy(), strerror(), strncpy()
32#include <net/if.h> // for struct ifreq
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040033#include <sys/ioctl.h> // for ioctl()
34
35#if defined(__linux__)
36#include <netpacket/packet.h> // for struct packet_mreq
37#include <sys/socket.h> // for setsockopt()
38#endif
39
40#ifdef SIOCADDMULTI
41#if defined(__APPLE__) || defined(__FreeBSD__)
42#include <net/if_dl.h> // for struct sockaddr_dl
43#endif
44#endif
45
Davide Pesaventoe422f9e2022-06-03 01:30:23 -040046namespace nfd::face {
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040047
Davide Pesaventoa3148082018-04-12 18:21:54 -040048NFD_LOG_INIT(MulticastEthernetTransport);
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040049
Junxiao Shi0d82d042017-07-07 06:15:27 +000050MulticastEthernetTransport::MulticastEthernetTransport(const ndn::net::NetworkInterface& localEndpoint,
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040051 const ethernet::Address& mcastAddress,
52 ndn::nfd::LinkType linkType)
53 : EthernetTransport(localEndpoint, mcastAddress)
Davide Pesaventofe0580c2017-05-12 02:02:10 -040054#if defined(__linux__)
Junxiao Shi0d82d042017-07-07 06:15:27 +000055 , m_interfaceIndex(localEndpoint.getIndex())
Davide Pesaventofe0580c2017-05-12 02:02:10 -040056#endif
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040057{
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);
Davide Pesaventob8eebcb2017-08-08 15:03:17 -040063 this->setMtu(localEndpoint.getMtu());
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040064
Davide Pesaventoa681a242019-03-29 23:48:27 -040065 NFD_LOG_FACE_DEBUG("Creating transport");
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040066
67 char filter[110];
Davide Pesaventoc52cd5e2022-03-05 20:40:54 -050068 // Note: "not vlan" must appear last in the filter expression, or the
69 // rest of the filter won't work as intended (see pcap-filter(7))
70 std::snprintf(filter, sizeof(filter),
71 "(ether proto 0x%x) && (ether dst %s) && (not ether src %s) && (not vlan)",
72 ethernet::ETHERTYPE_NDN,
73 m_destAddress.toString().data(),
74 m_srcAddress.toString().data());
Davide Pesaventofe0580c2017-05-12 02:02:10 -040075 m_pcap.setPacketFilter(filter);
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040076
Davide Pesaventoa94f3672017-05-18 01:18:39 -040077 BOOST_ASSERT(m_destAddress.isMulticast());
Davide Pesaventoc52cd5e2022-03-05 20:40:54 -050078 if (!m_destAddress.isBroadcast()) {
Davide Pesaventoa94f3672017-05-18 01:18:39 -040079 joinMulticastGroup();
Davide Pesaventoc52cd5e2022-03-05 20:40:54 -050080 }
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040081}
82
Davide Pesaventoa94f3672017-05-18 01:18:39 -040083void
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040084MulticastEthernetTransport::joinMulticastGroup()
85{
86#if defined(__linux__)
87 packet_mreq mr{};
88 mr.mr_ifindex = m_interfaceIndex;
89 mr.mr_type = PACKET_MR_MULTICAST;
90 mr.mr_alen = m_destAddress.size();
91 std::memcpy(mr.mr_address, m_destAddress.data(), m_destAddress.size());
92
93 if (::setsockopt(m_socket.native_handle(), SOL_PACKET,
94 PACKET_ADD_MEMBERSHIP, &mr, sizeof(mr)) == 0)
Davide Pesaventoa94f3672017-05-18 01:18:39 -040095 return; // success
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -040096
97 NFD_LOG_FACE_WARN("setsockopt(PACKET_ADD_MEMBERSHIP) failed: " << std::strerror(errno));
98#endif
99
100#if defined(SIOCADDMULTI)
101 ifreq ifr{};
Davide Pesaventofe0580c2017-05-12 02:02:10 -0400102 std::strncpy(ifr.ifr_name, m_interfaceName.data(), sizeof(ifr.ifr_name) - 1);
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -0400103
104#if defined(__APPLE__) || defined(__FreeBSD__)
105 // see bug #2327
106 using boost::asio::ip::udp;
107 udp::socket sock(getGlobalIoService(), udp::v4());
108 int fd = sock.native_handle();
109
110 // Differences between Linux and the BSDs (including macOS):
111 // o BSD does not have ifr_hwaddr; use ifr_addr instead.
112 // o While macOS seems to accept both AF_LINK and AF_UNSPEC as the address
113 // family, FreeBSD explicitly requires AF_LINK, so we have to use AF_LINK
114 // and sockaddr_dl instead of the generic sockaddr structure.
115 // o BSD's sockaddr (and sockaddr_dl in particular) contains an additional
116 // field, sa_len (sdl_len), which must be set to the total length of the
117 // structure, including the length field itself.
118 // o We do not specify the interface name, thus sdl_nlen is left at 0 and
119 // LLADDR is effectively the same as sdl_data.
120
121 sockaddr_dl* sdl = reinterpret_cast<sockaddr_dl*>(&ifr.ifr_addr);
122 sdl->sdl_len = sizeof(ifr.ifr_addr);
123 sdl->sdl_family = AF_LINK;
124 sdl->sdl_alen = m_destAddress.size();
125 std::memcpy(LLADDR(sdl), m_destAddress.data(), m_destAddress.size());
126
127 static_assert(sizeof(ifr.ifr_addr) >= offsetof(sockaddr_dl, sdl_data) + ethernet::ADDR_LEN,
128 "ifr_addr in struct ifreq is too small on this platform");
129#else
130 int fd = m_socket.native_handle();
131
132 ifr.ifr_hwaddr.sa_family = AF_UNSPEC;
133 std::memcpy(ifr.ifr_hwaddr.sa_data, m_destAddress.data(), m_destAddress.size());
134
135 static_assert(sizeof(ifr.ifr_hwaddr.sa_data) >= ethernet::ADDR_LEN,
136 "ifr_hwaddr in struct ifreq is too small on this platform");
137#endif
138
139 if (::ioctl(fd, SIOCADDMULTI, &ifr) == 0)
Davide Pesaventoa94f3672017-05-18 01:18:39 -0400140 return; // success
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -0400141
142 NFD_LOG_FACE_WARN("ioctl(SIOCADDMULTI) failed: " << std::strerror(errno));
143#endif
144
Davide Pesavento19779d82019-02-14 13:40:04 -0500145 NDN_THROW(Error("Failed to join multicast group"));
Davide Pesavento6bd6d0b2017-03-25 15:16:40 -0400146}
147
Davide Pesaventoe422f9e2022-06-03 01:30:23 -0400148} // namespace nfd::face