diff options
author | Jukka Rissanen <jukka.rissanen@linux.intel.com> | 2012-04-24 16:40:41 +0300 |
---|---|---|
committer | Marcel Holtmann <marcel@holtmann.org> | 2012-04-24 16:21:41 +0200 |
commit | 1c249af7946667f24cd2b995cc13149b77dd62df (patch) | |
tree | acfb81f855260ff6278c6d0be830630f7f7fed53 /src/resolver.c | |
parent | 51f6e0a79fffaea92d044a2b541da316bd57ffcd (diff) | |
download | connman-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.c | 33 |
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; |