blob: 8c118241b5cb4f797e725393d844ce2571861f41 [file] [log] [blame]
Davide Pesavento43ff2a92017-05-18 19:50:57 -04001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Junxiao Shi0d82d042017-07-07 06:15:27 +00002/*
Davide Pesaventoa599d2a2022-02-16 18:52:43 -05003 * Copyright (c) 2014-2022, Regents of the University of California,
Davide Pesavento43ff2a92017-05-18 19:50:57 -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#ifndef NFD_DAEMON_FACE_ETHERNET_CHANNEL_HPP
27#define NFD_DAEMON_FACE_ETHERNET_CHANNEL_HPP
28
29#include "channel.hpp"
Junxiao Shi0d82d042017-07-07 06:15:27 +000030#include "ethernet-protocol.hpp"
Davide Pesavento43ff2a92017-05-18 19:50:57 -040031#include "pcap-helper.hpp"
Davide Pesavento3db98072021-03-09 23:03:27 -050032
Davide Pesaventoa9b09b62022-06-04 14:07:25 -040033#include <boost/asio/posix/stream_descriptor.hpp>
Junxiao Shi0d82d042017-07-07 06:15:27 +000034#include <ndn-cxx/net/network-interface.hpp>
Davide Pesavento43ff2a92017-05-18 19:50:57 -040035
Davide Pesaventoe422f9e2022-06-03 01:30:23 -040036namespace nfd::face {
Davide Pesavento43ff2a92017-05-18 19:50:57 -040037
38/**
39 * \brief Class implementing Ethernet-based channel to create faces
40 */
Davide Pesavento3db98072021-03-09 23:03:27 -050041class EthernetChannel final : public Channel
Davide Pesavento43ff2a92017-05-18 19:50:57 -040042{
43public:
44 /**
45 * \brief EthernetChannel-related error
46 */
47 class Error : public std::runtime_error
48 {
49 public:
Davide Pesaventoaa9e3b22022-10-21 17:00:07 -040050 using std::runtime_error::runtime_error;
Davide Pesavento43ff2a92017-05-18 19:50:57 -040051 };
52
53 /**
54 * \brief Create an Ethernet channel on the given \p localEndpoint (network interface)
55 *
56 * To enable creation of faces upon incoming connections,
57 * one needs to explicitly call EthernetChannel::listen method.
58 */
Junxiao Shi0d82d042017-07-07 06:15:27 +000059 EthernetChannel(shared_ptr<const ndn::net::NetworkInterface> localEndpoint,
Davide Pesavento43ff2a92017-05-18 19:50:57 -040060 time::nanoseconds idleTimeout);
61
62 bool
Davide Pesavento3db98072021-03-09 23:03:27 -050063 isListening() const final
Davide Pesavento43ff2a92017-05-18 19:50:57 -040064 {
65 return m_isListening;
66 }
67
68 size_t
Davide Pesavento3db98072021-03-09 23:03:27 -050069 size() const final
Davide Pesavento43ff2a92017-05-18 19:50:57 -040070 {
71 return m_channelFaces.size();
72 }
73
74 /**
75 * \brief Create a unicast Ethernet face toward \p remoteEndpoint
Davide Pesavento43ff2a92017-05-18 19:50:57 -040076 */
77 void
78 connect(const ethernet::Address& remoteEndpoint,
Davide Pesavento15b55052018-01-27 19:09:28 -050079 const FaceParams& params,
Davide Pesavento43ff2a92017-05-18 19:50:57 -040080 const FaceCreatedCallback& onFaceCreated,
81 const FaceCreationFailedCallback& onConnectFailed);
82
83 /**
84 * \brief Start listening
85 *
86 * Enable listening on the local endpoint, waiting for incoming frames,
87 * and creating a face when a frame is received from a new remote host.
88 *
89 * Faces created in this way will have on-demand persistency.
90 *
91 * \param onFaceCreated Callback to notify successful creation of a face
92 * \param onFaceCreationFailed Callback to notify errors
93 * \throw Error
94 */
95 void
96 listen(const FaceCreatedCallback& onFaceCreated,
97 const FaceCreationFailedCallback& onFaceCreationFailed);
98
99private:
100 void
101 asyncRead(const FaceCreatedCallback& onFaceCreated,
102 const FaceCreationFailedCallback& onReceiveFailed);
103
104 void
105 handleRead(const boost::system::error_code& error,
106 const FaceCreatedCallback& onFaceCreated,
107 const FaceCreationFailedCallback& onReceiveFailed);
108
109 void
Davide Pesaventoa599d2a2022-02-16 18:52:43 -0500110 processIncomingPacket(span<const uint8_t> packet,
Davide Pesavento43ff2a92017-05-18 19:50:57 -0400111 const ethernet::Address& sender,
112 const FaceCreatedCallback& onFaceCreated,
113 const FaceCreationFailedCallback& onReceiveFailed);
114
115 std::pair<bool, shared_ptr<Face>>
116 createFace(const ethernet::Address& remoteEndpoint,
Davide Pesavento15b55052018-01-27 19:09:28 -0500117 const FaceParams& params);
Davide Pesavento43ff2a92017-05-18 19:50:57 -0400118
119 void
120 updateFilter();
121
122private:
Junxiao Shi0d82d042017-07-07 06:15:27 +0000123 shared_ptr<const ndn::net::NetworkInterface> m_localEndpoint;
Davide Pesavento43ff2a92017-05-18 19:50:57 -0400124 bool m_isListening;
125 boost::asio::posix::stream_descriptor m_socket;
126 PcapHelper m_pcap;
127 std::map<ethernet::Address, shared_ptr<Face>> m_channelFaces;
128 const time::nanoseconds m_idleFaceTimeout; ///< Timeout for automatic closure of idle on-demand faces
129
130#ifdef _DEBUG
131 /// number of frames dropped by the kernel, as reported by libpcap
132 size_t m_nDropped;
133#endif
134};
135
Davide Pesaventoe422f9e2022-06-03 01:30:23 -0400136} // namespace nfd::face
Davide Pesavento43ff2a92017-05-18 19:50:57 -0400137
138#endif // NFD_DAEMON_FACE_ETHERNET_CHANNEL_HPP