Skip to content

Commit 4ac1ce1

Browse files
jasnellMylesBorins
authored andcommitted
src: introduce node_sockaddr
Introduce the SocketAddress utility class. The QUIC implementation makes extensive use of this for handling of socket addresses. It was separated out to make it generically reusable throughout core Signed-off-by: James M Snell <[email protected]> PR-URL: #32070 Reviewed-By: David Carlier <[email protected]> Reviewed-By: Richard Lau <[email protected]> Reviewed-By: Anna Henningsen <[email protected]> Reviewed-By: Denys Otrishko <[email protected]>
1 parent a727b13 commit 4ac1ce1

7 files changed

+460
-11
lines changed

node.gyp

+4
Original file line numberDiff line numberDiff line change
@@ -597,6 +597,7 @@
597597
'src/node_process_methods.cc',
598598
'src/node_process_object.cc',
599599
'src/node_serdes.cc',
600+
'src/node_sockaddr.cc',
600601
'src/node_stat_watcher.cc',
601602
'src/node_symbols.cc',
602603
'src/node_task_queue.cc',
@@ -684,6 +685,8 @@
684685
'src/node_process.h',
685686
'src/node_revert.h',
686687
'src/node_root_certs.h',
688+
'src/node_sockaddr.h',
689+
'src/node_sockaddr-inl.h',
687690
'src/node_stat_watcher.h',
688691
'src/node_union_bytes.h',
689692
'src/node_url.h',
@@ -1150,6 +1153,7 @@
11501153
'test/cctest/test_linked_binding.cc',
11511154
'test/cctest/test_per_process.cc',
11521155
'test/cctest/test_platform.cc',
1156+
'test/cctest/test_sockaddr.cc',
11531157
'test/cctest/test_traced_value.cc',
11541158
'test/cctest/test_util.cc',
11551159
'test/cctest/test_url.cc',

src/node_sockaddr-inl.h

+170
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,170 @@
1+
#ifndef SRC_NODE_SOCKADDR_INL_H_
2+
#define SRC_NODE_SOCKADDR_INL_H_
3+
4+
#if defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS
5+
6+
#include "node.h"
7+
#include "node_internals.h"
8+
#include "node_sockaddr.h"
9+
#include "util-inl.h"
10+
11+
#include <string>
12+
13+
namespace node {
14+
15+
static constexpr uint32_t kLabelMask = 0xFFFFF;
16+
17+
inline void hash_combine(size_t* seed) { }
18+
19+
template <typename T, typename... Args>
20+
inline void hash_combine(size_t* seed, const T& value, Args... rest) {
21+
*seed ^= std::hash<T>{}(value) + 0x9e3779b9 + (*seed << 6) + (*seed >> 2);
22+
hash_combine(seed, rest...);
23+
}
24+
25+
bool SocketAddress::is_numeric_host(const char* hostname) {
26+
return is_numeric_host(hostname, AF_INET) ||
27+
is_numeric_host(hostname, AF_INET6);
28+
}
29+
30+
bool SocketAddress::is_numeric_host(const char* hostname, int family) {
31+
in6_addr dst;
32+
return inet_pton(family, hostname, &dst) == 1;
33+
}
34+
35+
int SocketAddress::GetPort(const sockaddr* addr) {
36+
CHECK(addr->sa_family == AF_INET || addr->sa_family == AF_INET6);
37+
return ntohs(addr->sa_family == AF_INET ?
38+
reinterpret_cast<const sockaddr_in*>(addr)->sin_port :
39+
reinterpret_cast<const sockaddr_in6*>(addr)->sin6_port);
40+
}
41+
42+
int SocketAddress::GetPort(const sockaddr_storage* addr) {
43+
return GetPort(reinterpret_cast<const sockaddr*>(addr));
44+
}
45+
46+
std::string SocketAddress::GetAddress(const sockaddr* addr) {
47+
CHECK(addr->sa_family == AF_INET || addr->sa_family == AF_INET6);
48+
char host[INET6_ADDRSTRLEN];
49+
const void* src = addr->sa_family == AF_INET ?
50+
static_cast<const void*>(
51+
&(reinterpret_cast<const sockaddr_in*>(addr)->sin_addr)) :
52+
static_cast<const void*>(
53+
&(reinterpret_cast<const sockaddr_in6*>(addr)->sin6_addr));
54+
uv_inet_ntop(addr->sa_family, src, host, INET6_ADDRSTRLEN);
55+
return std::string(host);
56+
}
57+
58+
std::string SocketAddress::GetAddress(const sockaddr_storage* addr) {
59+
return GetAddress(reinterpret_cast<const sockaddr*>(addr));
60+
}
61+
62+
size_t SocketAddress::GetLength(const sockaddr* addr) {
63+
return addr->sa_family == AF_INET ?
64+
sizeof(sockaddr_in) : sizeof(sockaddr_in6);
65+
}
66+
67+
size_t SocketAddress::GetLength(const sockaddr_storage* addr) {
68+
return GetLength(reinterpret_cast<const sockaddr*>(addr));
69+
}
70+
71+
SocketAddress::SocketAddress(const sockaddr* addr) {
72+
memcpy(&address_, addr, GetLength(addr));
73+
}
74+
75+
SocketAddress::SocketAddress(const SocketAddress& addr) {
76+
memcpy(&address_, &addr.address_, addr.length());
77+
}
78+
79+
SocketAddress& SocketAddress::operator=(const sockaddr* addr) {
80+
memcpy(&address_, addr, GetLength(addr));
81+
return *this;
82+
}
83+
84+
SocketAddress& SocketAddress::operator=(const SocketAddress& addr) {
85+
memcpy(&address_, &addr.address_, addr.length());
86+
return *this;
87+
}
88+
89+
const sockaddr& SocketAddress::operator*() const {
90+
return *this->data();
91+
}
92+
93+
const sockaddr* SocketAddress::operator->() const {
94+
return this->data();
95+
}
96+
97+
size_t SocketAddress::length() const {
98+
return GetLength(&address_);
99+
}
100+
101+
const sockaddr* SocketAddress::data() const {
102+
return reinterpret_cast<const sockaddr*>(&address_);
103+
}
104+
105+
const uint8_t* SocketAddress::raw() const {
106+
return reinterpret_cast<const uint8_t*>(&address_);
107+
}
108+
109+
sockaddr* SocketAddress::storage() {
110+
return reinterpret_cast<sockaddr*>(&address_);
111+
}
112+
113+
int SocketAddress::family() const {
114+
return address_.ss_family;
115+
}
116+
117+
std::string SocketAddress::address() const {
118+
return GetAddress(&address_);
119+
}
120+
121+
int SocketAddress::port() const {
122+
return GetPort(&address_);
123+
}
124+
125+
uint32_t SocketAddress::flow_label() const {
126+
if (family() != AF_INET6)
127+
return 0;
128+
const sockaddr_in6* in = reinterpret_cast<const sockaddr_in6*>(data());
129+
return in->sin6_flowinfo;
130+
}
131+
132+
void SocketAddress::set_flow_label(uint32_t label) {
133+
if (family() != AF_INET6)
134+
return;
135+
CHECK_LE(label, kLabelMask);
136+
sockaddr_in6* in = reinterpret_cast<sockaddr_in6*>(&address_);
137+
in->sin6_flowinfo = label;
138+
}
139+
140+
std::string SocketAddress::ToString() const {
141+
if (family() != AF_INET && family() != AF_INET6) return "";
142+
return (family() == AF_INET6 ?
143+
std::string("[") + address() + "]:" :
144+
address() + ":") +
145+
std::to_string(port());
146+
}
147+
148+
void SocketAddress::Update(uint8_t* data, size_t len) {
149+
CHECK_LE(len, sizeof(address_));
150+
memcpy(&address_, data, len);
151+
}
152+
153+
v8::Local<v8::Object> SocketAddress::ToJS(
154+
Environment* env,
155+
v8::Local<v8::Object> info) const {
156+
return AddressToJS(env, data(), info);
157+
}
158+
159+
bool SocketAddress::operator==(const SocketAddress& other) const {
160+
if (family() != other.family()) return false;
161+
return memcmp(raw(), other.raw(), length()) == 0;
162+
}
163+
164+
bool SocketAddress::operator!=(const SocketAddress& other) const {
165+
return !(*this == other);
166+
}
167+
} // namespace node
168+
169+
#endif // NODE_WANT_INTERNALS
170+
#endif // SRC_NODE_SOCKADDR_INL_H_

src/node_sockaddr.cc

+95
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,95 @@
1+
#include "node_sockaddr-inl.h" // NOLINT(build/include)
2+
#include "uv.h"
3+
4+
namespace node {
5+
6+
namespace {
7+
template <typename T, typename F>
8+
SocketAddress FromUVHandle(F fn, const T& handle) {
9+
SocketAddress addr;
10+
int len = sizeof(sockaddr_storage);
11+
if (fn(&handle, addr.storage(), &len) == 0)
12+
CHECK_EQ(static_cast<size_t>(len), addr.length());
13+
else
14+
addr.storage()->sa_family = 0;
15+
return addr;
16+
}
17+
} // namespace
18+
19+
bool SocketAddress::ToSockAddr(
20+
int32_t family,
21+
const char* host,
22+
uint32_t port,
23+
sockaddr_storage* addr) {
24+
switch (family) {
25+
case AF_INET:
26+
return uv_ip4_addr(
27+
host,
28+
port,
29+
reinterpret_cast<sockaddr_in*>(addr)) == 0;
30+
case AF_INET6:
31+
return uv_ip6_addr(
32+
host,
33+
port,
34+
reinterpret_cast<sockaddr_in6*>(addr)) == 0;
35+
default:
36+
UNREACHABLE();
37+
}
38+
}
39+
40+
bool SocketAddress::New(
41+
const char* host,
42+
uint32_t port,
43+
SocketAddress* addr) {
44+
return New(AF_INET, host, port, addr) || New(AF_INET6, host, port, addr);
45+
}
46+
47+
bool SocketAddress::New(
48+
int32_t family,
49+
const char* host,
50+
uint32_t port,
51+
SocketAddress* addr) {
52+
return ToSockAddr(family, host, port,
53+
reinterpret_cast<sockaddr_storage*>(addr->storage()));
54+
}
55+
56+
size_t SocketAddress::Hash::operator()(const SocketAddress& addr) const {
57+
size_t hash = 0;
58+
switch (addr.family()) {
59+
case AF_INET: {
60+
const sockaddr_in* ipv4 =
61+
reinterpret_cast<const sockaddr_in*>(addr.raw());
62+
hash_combine(&hash, ipv4->sin_port, ipv4->sin_addr.s_addr);
63+
break;
64+
}
65+
case AF_INET6: {
66+
const sockaddr_in6* ipv6 =
67+
reinterpret_cast<const sockaddr_in6*>(addr.raw());
68+
const uint64_t* a =
69+
reinterpret_cast<const uint64_t*>(&ipv6->sin6_addr);
70+
hash_combine(&hash, ipv6->sin6_port, a[0], a[1]);
71+
break;
72+
}
73+
default:
74+
UNREACHABLE();
75+
}
76+
return hash;
77+
}
78+
79+
SocketAddress SocketAddress::FromSockName(const uv_tcp_t& handle) {
80+
return FromUVHandle(uv_tcp_getsockname, handle);
81+
}
82+
83+
SocketAddress SocketAddress::FromSockName(const uv_udp_t& handle) {
84+
return FromUVHandle(uv_udp_getsockname, handle);
85+
}
86+
87+
SocketAddress SocketAddress::FromPeerName(const uv_tcp_t& handle) {
88+
return FromUVHandle(uv_tcp_getpeername, handle);
89+
}
90+
91+
SocketAddress SocketAddress::FromPeerName(const uv_udp_t& handle) {
92+
return FromUVHandle(uv_udp_getpeername, handle);
93+
}
94+
95+
} // namespace node

0 commit comments

Comments
 (0)