net/addr/ip: Add an any addr class
Change-Id: I265d47fa423a4be538ce70174a940c5633ef0720
Signed-off-by: William A. Kennington III <wak@google.com>
diff --git a/include/stdplus/net/addr/ip.hpp b/include/stdplus/net/addr/ip.hpp
index 71bb7f6..cac9e71 100644
--- a/include/stdplus/net/addr/ip.hpp
+++ b/include/stdplus/net/addr/ip.hpp
@@ -2,6 +2,7 @@
#include <netinet/in.h>
#include <stdplus/hash.hpp>
+#include <stdplus/variant.hpp>
#include <algorithm>
#include <variant>
@@ -69,6 +70,24 @@
}
};
+namespace detail
+{
+using InAnyAddrV = std::variant<In4Addr, In6Addr>;
+}
+
+struct InAnyAddr : detail::InAnyAddrV
+{
+ constexpr InAnyAddr(in_addr a) noexcept : detail::InAnyAddrV(In4Addr{a}) {}
+ constexpr InAnyAddr(In4Addr a) noexcept : detail::InAnyAddrV(a) {}
+ constexpr InAnyAddr(in6_addr a) noexcept : detail::InAnyAddrV(In6Addr{a}) {}
+ constexpr InAnyAddr(In6Addr a) noexcept : detail::InAnyAddrV(a) {}
+
+ constexpr bool operator==(auto rhs) const noexcept
+ {
+ return variantEqFuzzy(*this, rhs);
+ }
+};
+
} // namespace stdplus
template <>
@@ -88,3 +107,12 @@
return stdplus::hashMulti(addr.s6_addr32);
}
};
+
+template <>
+struct std::hash<stdplus::InAnyAddr>
+{
+ inline std::size_t operator()(stdplus::InAnyAddr a) const noexcept
+ {
+ return std::hash<stdplus::detail::InAnyAddrV>{}(a);
+ }
+};
diff --git a/test/net/addr/ip.cpp b/test/net/addr/ip.cpp
index 2745626..7d0cace 100644
--- a/test/net/addr/ip.cpp
+++ b/test/net/addr/ip.cpp
@@ -23,4 +23,25 @@
std::hash<In6Addr>{}(In6Addr{});
}
+TEST(EqualOperator, InAnyAddr)
+{
+ EXPECT_EQ(InAnyAddr(In6Addr{0xff, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0xff}),
+ (In6Addr{0xff, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff}));
+ EXPECT_NE(InAnyAddr(In6Addr{0xff, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0xff}),
+ (in_addr{}));
+ EXPECT_EQ((In6Addr{}), InAnyAddr(In6Addr{}));
+ EXPECT_NE((In4Addr{}), InAnyAddr(In6Addr{}));
+ EXPECT_EQ((In4Addr{}), InAnyAddr(In4Addr{}));
+ EXPECT_NE((In6Addr{}), InAnyAddr(In4Addr{}));
+ EXPECT_EQ(InAnyAddr(In6Addr{}), (in6_addr{}));
+ EXPECT_EQ(InAnyAddr(In4Addr{}), (in_addr{}));
+ EXPECT_NE(InAnyAddr(In6Addr{1}), InAnyAddr(In6Addr{}));
+ EXPECT_EQ(InAnyAddr(In6Addr{1}), InAnyAddr(In6Addr{1}));
+
+ std::hash<InAnyAddr>{}(In4Addr{});
+ std::hash<InAnyAddr>{}(In6Addr{});
+}
+
} // namespace stdplus