blob: a6011ff9282f88e3d7202cf4bb8d907a43e91d98 [file] [log] [blame]
Ratan Gupta82549cc2017-04-21 08:45:23 +05301#pragma once
William A. Kennington IIIbb0eacc2022-10-21 15:22:06 -07002#include <fmt/core.h>
3#include <net/ethernet.h>
William A. Kennington III0d7ce482019-01-30 17:14:23 -08004#include <netinet/in.h>
Patrick Venture189d44e2018-07-09 12:30:59 -07005#include <systemd/sd-event.h>
Ratan Gupta3681a502017-06-17 19:20:04 +05306
William A. Kennington IIIbb0eacc2022-10-21 15:22:06 -07007#include <algorithm>
8#include <array>
Patrick Venture189d44e2018-07-09 12:30:59 -07009#include <chrono>
Ratan Gupta3681a502017-06-17 19:20:04 +053010#include <memory>
William A. Kennington III3a70fa22018-09-20 18:48:20 -070011#include <sdeventplus/clock.hpp>
12#include <sdeventplus/utility/timer.hpp>
Patrick Venture189d44e2018-07-09 12:30:59 -070013#include <string>
William A. Kennington IIIbb0eacc2022-10-21 15:22:06 -070014#include <string_view>
15#include <type_traits>
William A. Kennington IIIdd9ef812022-10-05 02:08:02 -070016#include <unordered_map>
Willy Tuf7dce2e2022-10-07 05:48:08 +000017#include <unordered_set>
William A. Kennington III0d7ce482019-01-30 17:14:23 -080018#include <variant>
Ratan Gupta0f9dc1b2017-09-03 17:57:50 +053019
Ratan Gupta82549cc2017-04-21 08:45:23 +053020namespace phosphor
21{
22namespace network
23{
Ratan Guptae05083a2017-09-16 07:12:11 +053024
25using namespace std::chrono_literals;
Ratan Gupta16f12882017-09-22 18:26:11 +053026
William A. Kennington IIIc7cf25f2021-11-09 16:16:59 -080027// wait for three seconds before reloading systemd-networkd
28constexpr auto reloadTimeout = 3s;
Ratan Gupta16f12882017-09-22 18:26:11 +053029
William A. Kennington IIId41db382021-11-09 20:42:29 -080030// refresh the objets after four seconds as network
31// configuration takes 3-4 sec to reconfigure at most.
32constexpr auto refreshTimeout = 4s;
Ratan Guptae05083a2017-09-16 07:12:11 +053033
Ratan Gupta0f9dc1b2017-09-03 17:57:50 +053034/* Need a custom deleter for freeing up sd_event */
35struct EventDeleter
36{
37 void operator()(sd_event* event) const
38 {
William A. Kennington IIIdc2f1b62021-05-13 21:20:19 -070039 sd_event_unref(event);
Ratan Gupta0f9dc1b2017-09-03 17:57:50 +053040 }
41};
42using EventPtr = std::unique_ptr<sd_event, EventDeleter>;
43
William A. Kennington III0d7ce482019-01-30 17:14:23 -080044// Byte representations for common address types in network byte order
William A. Kennington IIIbb0eacc2022-10-21 15:22:06 -070045using InAddrAny = std::variant<in_addr, in6_addr>;
William A. Kennington III0d7ce482019-01-30 17:14:23 -080046
William A. Kennington III3a70fa22018-09-20 18:48:20 -070047using Timer = sdeventplus::utility::Timer<sdeventplus::ClockId::Monotonic>;
48
William A. Kennington IIIdd9ef812022-10-05 02:08:02 -070049struct string_hash : public std::hash<std::string_view>
50{
51 using is_transparent = void;
52};
53template <typename V>
54using string_umap =
55 std::unordered_map<std::string, V, string_hash, std::equal_to<>>;
William A. Kennington III96444792022-10-05 15:16:22 -070056using string_uset =
57 std::unordered_set<std::string, string_hash, std::equal_to<>>;
William A. Kennington IIIdd9ef812022-10-05 02:08:02 -070058
William A. Kennington III3e471c52022-10-27 19:46:07 -070059constexpr std::size_t hash_multi() noexcept
William A. Kennington III991a8e82022-10-11 15:02:47 -070060{
61 return 0;
62}
63
64template <typename T, typename... Args>
William A. Kennington III3e471c52022-10-27 19:46:07 -070065constexpr std::size_t hash_multi(const T& v, Args... args) noexcept
William A. Kennington III991a8e82022-10-11 15:02:47 -070066{
67 const std::size_t seed = hash_multi(args...);
68 return seed ^ (std::hash<T>{}(v) + 0x9e3779b9 + (seed << 6) + (seed >> 2));
69}
70
William A. Kennington IIIbb0eacc2022-10-21 15:22:06 -070071namespace detail
72{
73
74template <typename T>
75constexpr bool vcontains() noexcept
76{
77 return false;
78}
79
80template <typename T, typename V, typename... Vs>
81constexpr bool vcontains() noexcept
82{
83 return vcontains<T, Vs...>() || std::is_same_v<T, V>;
84}
85
86template <typename T, typename... Types>
87constexpr std::enable_if_t<vcontains<T, Types...>(), bool>
88 veq(T t, std::variant<Types...> v) noexcept
89{
90 return std::visit(
91 [t](auto v) {
92 if constexpr (std::is_same_v<T, decltype(v)>)
93 {
94 return v == t;
95 }
96 else
97 {
98 return false;
99 }
100 },
101 v);
102}
103
104template <typename T>
105struct AddrBufMaker
106{
107};
108
109template <>
110struct AddrBufMaker<ether_addr>
111{
112 public:
113 std::string_view operator()(ether_addr val) noexcept;
114
115 private:
116 std::array<char, /*octet*/ 2 * /*octets*/ 6 + /*seps*/ 5> buf;
117};
118
119template <>
120struct AddrBufMaker<in_addr>
121{
122 public:
123 std::string_view operator()(in_addr val) noexcept;
124
125 private:
126 std::array<char, /*octet*/ 3 * /*octets*/ 4 + /*seps*/ 3> buf;
127};
128
129template <>
130struct AddrBufMaker<in6_addr>
131{
132 public:
133 std::string_view operator()(in6_addr val) noexcept;
134
135 private:
136 std::array<char, /*hextet*/ 4 * /*hextets*/ 8 + /*seps*/ 7> buf;
137};
138
139template <typename BufMaker>
140struct FormatFromBuf
141{
142 private:
143 fmt::formatter<std::string_view> formatter;
144
145 public:
146 template <typename ParseContext>
147 constexpr auto parse(ParseContext& ctx)
148 {
149 return ctx.begin();
150 }
151
152 template <typename FormatContext>
153 auto format(auto v, FormatContext& ctx) const
154 {
155 return formatter.format(BufMaker{}(v), ctx);
156 }
157};
158} // namespace detail
Gunnar Mills57d9c502018-09-14 14:42:34 -0500159} // namespace network
160} // namespace phosphor
William A. Kennington III3e471c52022-10-27 19:46:07 -0700161
162template <typename... Ts>
163struct std::hash<std::tuple<Ts...>>
164{
165 constexpr auto operator()(const std::tuple<Ts...>& t) const noexcept
166 {
167 return std::apply(phosphor::network::hash_multi<Ts...>, t);
168 }
169};
William A. Kennington IIIbb0eacc2022-10-21 15:22:06 -0700170
171namespace fmt
172{
173template <>
174struct formatter<ether_addr>
175 : phosphor::network::detail::FormatFromBuf<
176 phosphor::network::detail::AddrBufMaker<ether_addr>>
177{
178};
179template <>
180struct formatter<in_addr>
181 : phosphor::network::detail::FormatFromBuf<
182 phosphor::network::detail::AddrBufMaker<in_addr>>
183{
184};
185template <>
186struct formatter<in6_addr>
187 : phosphor::network::detail::FormatFromBuf<
188 phosphor::network::detail::AddrBufMaker<in6_addr>>
189{
190};
191template <>
192struct formatter<phosphor::network::InAddrAny>
193{
194 private:
195 fmt::formatter<std::string_view> formatter;
196
197 public:
198 template <typename ParseContext>
199 constexpr auto parse(ParseContext& ctx)
200 {
201 return ctx.begin();
202 }
203
204 template <typename FormatContext>
205 auto format(auto v, FormatContext& ctx) const
206 {
207 return std::visit(
208 [&](auto v) {
209 auto abm =
210 phosphor::network::detail::AddrBufMaker<decltype(v)>{};
211 return formatter.format(abm(v), ctx);
212 },
213 v);
214 }
215};
216} // namespace fmt
217
218namespace std
219{
220string to_string(ether_addr value);
221string to_string(in_addr value);
222string to_string(in6_addr value);
223string to_string(phosphor::network::InAddrAny value);
224} // namespace std
225
226constexpr bool operator==(ether_addr lhs, ether_addr rhs) noexcept
227{
228 return std::equal(lhs.ether_addr_octet, lhs.ether_addr_octet + 6,
229 rhs.ether_addr_octet);
230}
231
232constexpr bool operator==(in_addr lhs, in_addr rhs) noexcept
233{
234 return lhs.s_addr == rhs.s_addr;
235}
236
237constexpr bool operator==(in6_addr lhs, in6_addr rhs) noexcept
238{
239 return std::equal(lhs.s6_addr32, lhs.s6_addr32 + 4, rhs.s6_addr32);
240}
241
242template <typename T>
243constexpr std::enable_if_t<!std::is_same_v<phosphor::network::InAddrAny, T>,
244 bool>
245 operator==(phosphor::network::InAddrAny lhs, T rhs) noexcept
246{
247 return phosphor::network::detail::veq(rhs, lhs);
248}
249
250template <typename CharT, typename Traits>
251std::basic_ostream<CharT, Traits>&
252 operator<<(std::basic_ostream<CharT, Traits>& os, ether_addr v)
253{
254 return os << phosphor::network::detail::AddrBufMaker<ether_addr>{}(v);
255}
256
257template <typename CharT, typename Traits>
258std::basic_ostream<CharT, Traits>&
259 operator<<(std::basic_ostream<CharT, Traits>& os, in_addr v)
260{
261 return os << phosphor::network::detail::AddrBufMaker<in_addr>{}(v);
262}
263
264template <typename CharT, typename Traits>
265std::basic_ostream<CharT, Traits>&
266 operator<<(std::basic_ostream<CharT, Traits>& os, in6_addr v)
267{
268 return os << phosphor::network::detail::AddrBufMaker<in6_addr>{}(v);
269}
270
271template <typename CharT, typename Traits>
272std::basic_ostream<CharT, Traits>&
273 operator<<(std::basic_ostream<CharT, Traits>& os,
274 phosphor::network::InAddrAny v)
275{
276 return os << std::visit(
277 [](auto v) {
278 return phosphor::network::detail::AddrBufMaker<
279 decltype(v)>{}(v);
280 },
281 v);
282}