diff options
Diffstat (limited to 'src/or/dns.c')
-rw-r--r-- | src/or/dns.c | 48 |
1 files changed, 35 insertions, 13 deletions
diff --git a/src/or/dns.c b/src/or/dns.c index 03dc85f421..2b6ea88482 100644 --- a/src/or/dns.c +++ b/src/or/dns.c @@ -33,6 +33,8 @@ const char dns_c_id[] = /** Have we currently configured nameservers with eventdns? */ static int nameservers_configured = 0; +/** Did our most recent attempt to configure nameservers with eventdns fail? */ +static int nameserver_config_failed = 0; /** What was the resolv_conf fname we last used when configuring the * nameservers? Used to check whether we need to reconfigure. */ static char *resolv_conf_fname = NULL; @@ -220,12 +222,20 @@ dns_reset(void) tor_free(resolv_conf_fname); resolv_conf_mtime = 0; } else { - if (configure_nameservers(0) < 0) + if (configure_nameservers(0) < 0) { return -1; + } } return 0; } +/**DOCDOC*/ +int +has_dns_init_failed(void) +{ + return nameserver_config_failed; +} + /** Helper: Given a TTL from a DNS response, determine what TTL to give the * OP that asked us to resolve it. */ uint32_t @@ -1109,10 +1119,11 @@ evdns_err_is_transient(int err) } /** Configure eventdns nameservers if force is true, or if the configuration - * has changed since the last time we called this function. On Unix, this - * reads from options->ServerDNSResolvConfFile or /etc/resolv.conf; on - * Windows, this reads from options->ServerDNSResolvConfFile or the registry. - * Return 0 on success or -1 on failure. */ + * has changed since the last time we called this function, or if we failed on + * our last attempt. On Unix, this reads from /etc/resolv.conf or + * options->ServerDNSResolvConfFile; on Windows, this reads from + * options->ServerDNSResolvConfFile or the registry. Return 0 on success or + * -1 on failure. */ static int configure_nameservers(int force) { @@ -1132,7 +1143,7 @@ configure_nameservers(int force) if (stat(conf_fname, &st)) { log_warn(LD_EXIT, "Unable to stat resolver configuration in '%s': %s", conf_fname, strerror(errno)); - return options->ServerDNSAllowBrokenResolvConf ? 0 : -1; + goto err; } if (!force && resolv_conf_fname && !strcmp(conf_fname,resolv_conf_fname) && st.st_mtime == resolv_conf_mtime) { @@ -1147,11 +1158,11 @@ configure_nameservers(int force) if ((r = evdns_resolv_conf_parse(DNS_OPTIONS_ALL, conf_fname))) { log_warn(LD_EXIT, "Unable to parse '%s', or no nameservers in '%s' (%d)", conf_fname, conf_fname, r); - return options->ServerDNSAllowBrokenResolvConf ? 0 : -1; + goto err; } if (evdns_count_nameservers() == 0) { log_warn(LD_EXIT, "Unable to find any nameservers in '%s'.", conf_fname); - return options->ServerDNSAllowBrokenResolvConf ? 0 : -1; + goto err; } tor_free(resolv_conf_fname); resolv_conf_fname = tor_strdup(conf_fname); @@ -1167,13 +1178,12 @@ configure_nameservers(int force) } if (evdns_config_windows_nameservers()) { log_warn(LD_EXIT,"Could not config nameservers."); - return options->ServerDNSAllowBrokenResolvConf ? 0 : -1; + goto err; } if (evdns_count_nameservers() == 0) { log_warn(LD_EXIT, "Unable to find any platform nameservers in " - "your Windows configuration. Perhaps you should list a " - "ServerDNSResolvConfFile file in your torrc?"); - return options->ServerDNSAllowBrokenResolvConf ? 0 : -1; + "your Windows configuration."); + goto err; } if (nameservers_configured) evdns_resume(); @@ -1193,7 +1203,18 @@ configure_nameservers(int force) dns_servers_relaunch_checks(); nameservers_configured = 1; + if (nameserver_config_failed) { + nameserver_config_failed = 0; + mark_my_descriptor_dirty(); + } return 0; + err: + nameservers_configured = 0; + if (! nameserver_config_failed) { + nameserver_config_failed = 1; + mark_my_descriptor_dirty(); + } + return -1; } /** For eventdns: Called when we get an answer for a request we launched. @@ -1284,8 +1305,9 @@ launch_resolve(edge_connection_t *exitconn) if (!nameservers_configured) { log_warn(LD_EXIT, "(Harmless.) Nameservers not configured, but resolve " "launched. Configuring."); - if (configure_nameservers(1) < 0) + if (configure_nameservers(1) < 0) { return -1; + } } r = parse_inaddr_arpa_address(exitconn->_base.address, &in); |