Commit bc995801 authored by Chuck Lever's avatar Chuck Lever Committed by J. Bruce Fields

NLM: Support IPv6 scope IDs in nlm_display_address()

Scope ID support is needed since the kernel's NSM implementation is
about to use these displayed addresses as a mon_name in some cases.

When nsm_use_hostnames is zero, without scope ID support NSM will fail
to handle peers that contact us via a link-local address.  Link-local
addresses do not work without an interface ID, which is stored in the
sockaddr's sin6_scope_id field.
Signed-off-by: default avatarChuck Lever <chuck.lever@oracle.com>
Signed-off-by: default avatarJ. Bruce Fields <bfields@citi.umich.edu>
parent 6999fb40
...@@ -105,22 +105,31 @@ static void nlm_clear_port(struct sockaddr *sap) ...@@ -105,22 +105,31 @@ static void nlm_clear_port(struct sockaddr *sap)
} }
} }
static void nlm_display_ipv6_address(const struct sockaddr *sap, char *buf,
const size_t len)
{
const struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)sap;
if (ipv6_addr_v4mapped(&sin6->sin6_addr))
snprintf(buf, len, "%pI4", &sin6->sin6_addr.s6_addr32[3]);
else if (sin6->sin6_scope_id != 0)
snprintf(buf, len, "%pI6%%%u", &sin6->sin6_addr,
sin6->sin6_scope_id);
else
snprintf(buf, len, "%pI6", &sin6->sin6_addr);
}
static void nlm_display_address(const struct sockaddr *sap, static void nlm_display_address(const struct sockaddr *sap,
char *buf, const size_t len) char *buf, const size_t len)
{ {
const struct sockaddr_in *sin = (struct sockaddr_in *)sap; const struct sockaddr_in *sin = (struct sockaddr_in *)sap;
const struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)sap;
switch (sap->sa_family) { switch (sap->sa_family) {
case AF_INET: case AF_INET:
snprintf(buf, len, "%pI4", &sin->sin_addr.s_addr); snprintf(buf, len, "%pI4", &sin->sin_addr.s_addr);
break; break;
case AF_INET6: case AF_INET6:
if (ipv6_addr_v4mapped(&sin6->sin6_addr)) nlm_display_ipv6_address(sap, buf, len);
snprintf(buf, len, "%pI4",
&sin6->sin6_addr.s6_addr32[3]);
else
snprintf(buf, len, "%pI6", &sin6->sin6_addr);
break; break;
default: default:
snprintf(buf, len, "unsupported address family"); snprintf(buf, len, "unsupported address family");
......
...@@ -68,6 +68,14 @@ struct nlm_host { ...@@ -68,6 +68,14 @@ struct nlm_host {
char *h_addrbuf; /* address eyecatcher */ char *h_addrbuf; /* address eyecatcher */
}; };
/*
* The largest string sm_addrbuf should hold is a full-size IPv6 address
* (no "::" anywhere) with a scope ID. The buffer size is computed to
* hold eight groups of colon-separated four-hex-digit numbers, a
* percent sign, a scope id (at most 32 bits, in decimal), and NUL.
*/
#define NSM_ADDRBUF ((8 * 4 + 7) + (1 + 10) + 1)
struct nsm_handle { struct nsm_handle {
struct list_head sm_link; struct list_head sm_link;
atomic_t sm_count; atomic_t sm_count;
...@@ -76,7 +84,7 @@ struct nsm_handle { ...@@ -76,7 +84,7 @@ struct nsm_handle {
size_t sm_addrlen; size_t sm_addrlen;
unsigned int sm_monitored : 1, unsigned int sm_monitored : 1,
sm_sticky : 1; /* don't unmonitor */ sm_sticky : 1; /* don't unmonitor */
char sm_addrbuf[48]; /* address eyecatcher */ char sm_addrbuf[NSM_ADDRBUF];
}; };
/* /*
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment