aboutgitcodebugslistschat
path: root/inany.h
diff options
context:
space:
mode:
authorDavid Gibson <david@gibson.dropbear.id.au>2022-11-17 16:59:03 +1100
committerStefano Brivio <sbrivio@redhat.com>2022-11-25 01:35:51 +0100
commitbb6bb5649d382ccb4cce82eb2fbc97c2708f9c0f (patch)
tree949da000bf46277877bb463863b9ae9652ef68eb /inany.h
parent034fa8a58d87ad2ea5f8b56d267d17dbc75798de (diff)
downloadpasst-bb6bb5649d382ccb4cce82eb2fbc97c2708f9c0f.tar
passt-bb6bb5649d382ccb4cce82eb2fbc97c2708f9c0f.tar.gz
passt-bb6bb5649d382ccb4cce82eb2fbc97c2708f9c0f.tar.bz2
passt-bb6bb5649d382ccb4cce82eb2fbc97c2708f9c0f.tar.lz
passt-bb6bb5649d382ccb4cce82eb2fbc97c2708f9c0f.tar.xz
passt-bb6bb5649d382ccb4cce82eb2fbc97c2708f9c0f.tar.zst
passt-bb6bb5649d382ccb4cce82eb2fbc97c2708f9c0f.zip
tcp: NAT IPv4-mapped IPv6 addresses like IPv4 addresses
passt usually doesn't NAT, but it does do so for the remapping of the gateway address to refer to the host. Currently we perform this NAT with slightly different rules on both IPv4 addresses and IPv6 addresses, but not on IPv4-mapped IPv6 addresses. This means we won't correctly handle the case of an IPv4 connection over an IPv6 socket, which is possible on Linux (and probably other platforms). Refactor tcp_conn_from_sock() to perform the NAT after converting either address family into an inany_addr, so IPv4 and and IPv4-mapped addresses have the same representation. With two new helpers this lets us remove the IPv4 and IPv6 specific paths from tcp_conn_from_sock(). Signed-off-by: David Gibson <david@gibson.dropbear.id.au> Signed-off-by: Stefano Brivio <sbrivio@redhat.com>
Diffstat (limited to 'inany.h')
-rw-r--r--inany.h30
1 files changed, 28 insertions, 2 deletions
diff --git a/inany.h b/inany.h
index 03e3e3f..52d43f2 100644
--- a/inany.h
+++ b/inany.h
@@ -30,11 +30,11 @@ union inany_addr {
*
* Return: IPv4 address if @addr is IPv4, NULL otherwise
*/
-static inline const struct in_addr *inany_v4(const union inany_addr *addr)
+static inline struct in_addr *inany_v4(const union inany_addr *addr)
{
if (!IN6_IS_ADDR_V4MAPPED(&addr->a6))
return NULL;
- return &addr->v4mapped.a4;
+ return (struct in_addr *)&addr->v4mapped.a4;
}
/** inany_equals - Compare two IPv[46] addresses
@@ -66,3 +66,29 @@ static inline void inany_from_af(union inany_addr *aa, int af, const void *addr)
assert(0);
}
}
+
+/** inany_from_sockaddr - Extract IPv[46] address and port number from sockaddr
+ * @aa: Pointer to store IPv[46] address
+ * @port: Pointer to store port number, host order
+ * @addr: struct sockaddr_in (IPv4) or struct sockaddr_in6 (IPv6)
+ */
+static inline void inany_from_sockaddr(union inany_addr *aa, in_port_t *port,
+ const void *addr)
+{
+ const struct sockaddr *sa = (const struct sockaddr *)addr;
+
+ if (sa->sa_family == AF_INET6) {
+ struct sockaddr_in6 *sa6 = (struct sockaddr_in6 *)sa;
+
+ inany_from_af(aa, AF_INET6, &sa6->sin6_addr);
+ *port = ntohs(sa6->sin6_port);
+ } else if (sa->sa_family == AF_INET) {
+ struct sockaddr_in *sa4 = (struct sockaddr_in *)sa;
+
+ inany_from_af(aa, AF_INET, &sa4->sin_addr);
+ *port = ntohs(sa4->sin_port);
+ } else {
+ /* Not valid to call with other address families */
+ assert(0);
+ }
+}