aboutsummaryrefslogtreecommitdiff
path: root/src/nat/nat.c
diff options
context:
space:
mode:
authorChristian Grothoff <christian@grothoff.org>2013-05-15 09:00:26 +0000
committerChristian Grothoff <christian@grothoff.org>2013-05-15 09:00:26 +0000
commit68239c2fb5ace698568fe16705804e842cba6e70 (patch)
treebd9a4f47a4b86da22e79e695dcff4e3d064e6864 /src/nat/nat.c
parent2399550f72d224717fa1c8b767fc0a9749d348cf (diff)
do not enumerate all IP addresses from our interfaces if the plugin is bound to a specific IP address only
Diffstat (limited to 'src/nat/nat.c')
-rw-r--r--src/nat/nat.c39
1 files changed, 39 insertions, 0 deletions
diff --git a/src/nat/nat.c b/src/nat/nat.c
index 515fa1eb8c..0aefc4312e 100644
--- a/src/nat/nat.c
+++ b/src/nat/nat.c
@@ -638,15 +638,36 @@ process_interfaces (void *cls, const char *name, int isDefault,
const struct sockaddr *broadcast_addr,
const struct sockaddr *netmask, socklen_t addrlen)
{
+ const static struct in6_addr any6 = IN6ADDR_ANY_INIT;
struct GNUNET_NAT_Handle *h = cls;
const struct sockaddr_in *s4;
const struct sockaddr_in6 *s6;
const void *ip;
char buf[INET6_ADDRSTRLEN];
+ unsigned int i;
+ int have_any;
switch (addr->sa_family)
{
case AF_INET:
+ /* check if we're bound to the "ANY" IP address */
+ have_any = GNUNET_NO;
+ for (i=0;i<h->num_local_addrs;i++)
+ {
+ if (h->local_addrs[i]->sa_family != AF_INET)
+ continue;
+#ifndef INADDR_ANY
+#define INADDR_ANY 0
+#endif
+ if (INADDR_ANY == ((struct sockaddr_in*) h->local_addrs[i])->sin_addr.s_addr)
+ {
+ have_any = GNUNET_YES;
+ break;
+ }
+ }
+ if (GNUNET_NO == have_any)
+ return GNUNET_OK; /* not bound to IP 0.0.0.0 but to specific IP addresses,
+ do not use those from interfaces */
s4 = (struct sockaddr_in *) addr;
ip = &s4->sin_addr;
@@ -665,6 +686,24 @@ process_interfaces (void *cls, const char *name, int isDefault,
}
break;
case AF_INET6:
+ /* check if we're bound to the "ANY" IP address */
+ have_any = GNUNET_NO;
+ for (i=0;i<h->num_local_addrs;i++)
+ {
+ if (h->local_addrs[i]->sa_family != AF_INET6)
+ continue;
+ if (0 == memcmp (&any6,
+ &((struct sockaddr_in6*) h->local_addrs[i])->sin6_addr,
+ sizeof (struct in6_addr)))
+ {
+ have_any = GNUNET_YES;
+ break;
+ }
+ }
+ if (GNUNET_NO == have_any)
+ return GNUNET_OK; /* not bound to "ANY" IP (::0) but to specific IP addresses,
+ do not use those from interfaces */
+
s6 = (struct sockaddr_in6 *) addr;
if (IN6_IS_ADDR_LINKLOCAL (&((struct sockaddr_in6 *) addr)->sin6_addr))
{