summaryrefslogtreecommitdiff
path: root/src/resolver.c
diff options
context:
space:
mode:
authorJukka Rissanen <jukka.rissanen@linux.intel.com>2012-04-24 16:40:41 +0300
committerMarcel Holtmann <marcel@holtmann.org>2012-04-24 16:21:41 +0200
commit1c249af7946667f24cd2b995cc13149b77dd62df (patch)
treeacfb81f855260ff6278c6d0be830630f7f7fed53 /src/resolver.c
parent51f6e0a79fffaea92d044a2b541da316bd57ffcd (diff)
downloadconnman-1c249af7946667f24cd2b995cc13149b77dd62df.tar.gz
connman-1c249af7946667f24cd2b995cc13149b77dd62df.tar.bz2
connman-1c249af7946667f24cd2b995cc13149b77dd62df.zip
resolver: Use proper IPv6 source address when sending DNS queries
This fix is for following scenario: - New interface is coming up. - There is radvd in the connected network and it is advertising IPv6 addresses and DNS servers. - Kernel receives router advertisement and picks up the DNS server information which is then routed via netlink to rtnl.c:rtnl_newnduseropt() which then creates DNS listener. - Kernel activates DAD (duplicate address detection). - As the DAD takes some time we now have interface up and it only has link local IPv6 address defined. - The DNS listener is now using link local source addresses when sending queries instead of proper autoconfigured addresses. - When DAD is finished, the interface will have autoconfigured addresses assigned and corresponding netlink message will cause function rtnl.c:process_newaddr() to be called. - If all this happens, then we re-create DNS listener in dnsproxy.c so that listener will have a proper (autoconfigured) source address when sending DNS packets.
Diffstat (limited to 'src/resolver.c')
-rw-r--r--src/resolver.c33
1 files changed, 33 insertions, 0 deletions
diff --git a/src/resolver.c b/src/resolver.c
index 9796717f..cead9c32 100644
--- a/src/resolver.c
+++ b/src/resolver.c
@@ -468,6 +468,39 @@ void connman_resolver_flush(void)
return;
}
+int __connman_resolver_redo_servers(const char *interface)
+{
+ GSList *list;
+
+ if (dnsproxy_enabled == FALSE)
+ return 0;
+
+ DBG("interface %s", interface);
+
+ if (interface == NULL)
+ return -EINVAL;
+
+ for (list = entry_list; list; list = list->next) {
+ struct entry_data *entry = list->data;
+
+ if (entry->timeout == 0 ||
+ g_strcmp0(entry->interface, interface) != 0)
+ continue;
+
+ /*
+ * We remove the server, and then re-create so that it will
+ * use proper source addresses when sending DNS queries.
+ */
+ __connman_dnsproxy_remove(entry->interface, entry->domain,
+ entry->server);
+
+ __connman_dnsproxy_append(entry->interface, entry->domain,
+ entry->server);
+ }
+
+ return 0;
+}
+
static void free_entry(gpointer data)
{
struct entry_data *entry = data;