|  | .\" | 
|  | .\" Copyright (c) 2006 Niels Provos <provos@citi.umich.edu> | 
|  | .\" All rights reserved. | 
|  | .\" | 
|  | .\" Redistribution and use in source and binary forms, with or without | 
|  | .\" modification, are permitted provided that the following conditions | 
|  | .\" are met: | 
|  | .\" | 
|  | .\" 1. Redistributions of source code must retain the above copyright | 
|  | .\"    notice, this list of conditions and the following disclaimer. | 
|  | .\" 2. Redistributions in binary form must reproduce the above copyright | 
|  | .\"    notice, this list of conditions and the following disclaimer in the | 
|  | .\"    documentation and/or other materials provided with the distribution. | 
|  | .\" 3. The name of the author may not be used to endorse or promote products | 
|  | .\"    derived from this software without specific prior written permission. | 
|  | .\" | 
|  | .\" THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, | 
|  | .\" INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY | 
|  | .\" AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL | 
|  | .\" THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | 
|  | .\" EXEMPLARY, OR CONSEQUENTIAL  DAMAGES (INCLUDING, BUT NOT LIMITED TO, | 
|  | .\" PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; | 
|  | .\" OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, | 
|  | .\" WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR | 
|  | .\" OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF | 
|  | .\" ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|  | .\" | 
|  | .Dd October 7, 2006 | 
|  | .Dt EVDNS 3 | 
|  | .Os | 
|  | .Sh NAME | 
|  | .Nm evdns_init | 
|  | .Nm evdns_shutdown | 
|  | .Nm evdns_err_to_string | 
|  | .Nm evdns_nameserver_add | 
|  | .Nm evdns_count_nameservers | 
|  | .Nm evdns_clear_nameservers_and_suspend | 
|  | .Nm evdns_resume | 
|  | .Nm evdns_nameserver_ip_add | 
|  | .Nm evdns_resolve_ipv4 | 
|  | .Nm evdns_resolve_reverse | 
|  | .Nm evdns_resolv_conf_parse | 
|  | .Nm evdns_config_windows_nameservers | 
|  | .Nm evdns_search_clear | 
|  | .Nm evdns_search_add | 
|  | .Nm evdns_search_ndots_set | 
|  | .Nm evdns_set_log_fn | 
|  | .Nd asynchronous functions for DNS resolution. | 
|  | .Sh SYNOPSIS | 
|  | .Fd #include <sys/time.h> | 
|  | .Fd #include <event.h> | 
|  | .Fd #include <evdns.h> | 
|  | .Ft int | 
|  | .Fn evdns_init | 
|  | .Ft void | 
|  | .Fn evdns_shutdown "int fail_requests" | 
|  | .Ft "const char *" | 
|  | .Fn evdns_err_to_string "int err" | 
|  | .Ft int | 
|  | .Fn evdns_nameserver_add "unsigned long int address" | 
|  | .Ft int | 
|  | .Fn evdns_count_nameservers | 
|  | .Ft int | 
|  | .Fn evdns_clear_nameservers_and_suspend | 
|  | .Ft int | 
|  | .Fn evdns_resume | 
|  | .Ft int | 
|  | .Fn evdns_nameserver_ip_add(const char *ip_as_string); | 
|  | .Ft int | 
|  | .Fn evdns_resolve_ipv4 "const char *name" "int flags" "evdns_callback_type callback" "void *ptr" | 
|  | .Ft int | 
|  | .Fn evdns_resolve_reverse "struct in_addr *in" "int flags" "evdns_callback_type callback" "void *ptr" | 
|  | .Ft int | 
|  | .Fn evdns_resolv_conf_parse "int flags" "const char *" | 
|  | .Ft void | 
|  | .Fn evdns_search_clear | 
|  | .Ft void | 
|  | .Fn evdns_search_add "const char *domain" | 
|  | .Ft void | 
|  | .Fn evdns_search_ndots_set "const int ndots" | 
|  | .Ft void | 
|  | .Fn evdns_set_log_fn "evdns_debug_log_fn_type fn" | 
|  | .Ft int | 
|  | .Fn evdns_config_windows_nameservers | 
|  | .Sh DESCRIPTION | 
|  | Welcome, gentle reader | 
|  | .Pp | 
|  | Async DNS lookups are really a whole lot harder than they should be, | 
|  | mostly stemming from the fact that the libc resolver has never been | 
|  | very good at them. Before you use this library you should see if libc | 
|  | can do the job for you with the modern async call getaddrinfo_a | 
|  | (see http://www.imperialviolet.org/page25.html#e498). Otherwise, | 
|  | please continue. | 
|  | .Pp | 
|  | This code is based on libevent and you must call event_init before | 
|  | any of the APIs in this file. You must also seed the OpenSSL random | 
|  | source if you are using OpenSSL for ids (see below). | 
|  | .Pp | 
|  | This library is designed to be included and shipped with your source | 
|  | code. You statically link with it. You should also test for the | 
|  | existence of strtok_r and define HAVE_STRTOK_R if you have it. | 
|  | .Pp | 
|  | The DNS protocol requires a good source of id numbers and these | 
|  | numbers should be unpredictable for spoofing reasons. There are | 
|  | three methods for generating them here and you must define exactly | 
|  | one of them. In increasing order of preference: | 
|  | .Pp | 
|  | .Bl -tag -width "DNS_USE_GETTIMEOFDAY_FOR_ID" -compact -offset indent | 
|  | .It DNS_USE_GETTIMEOFDAY_FOR_ID | 
|  | Using the bottom 16 bits of the usec result from gettimeofday. This | 
|  | is a pretty poor solution but should work anywhere. | 
|  | .It DNS_USE_CPU_CLOCK_FOR_ID | 
|  | Using the bottom 16 bits of the nsec result from the CPU's time | 
|  | counter. This is better, but may not work everywhere. Requires | 
|  | POSIX realtime support and you'll need to link against -lrt on | 
|  | glibc systems at least. | 
|  | .It DNS_USE_OPENSSL_FOR_ID | 
|  | Uses the OpenSSL RAND_bytes call to generate the data. You must | 
|  | have seeded the pool before making any calls to this library. | 
|  | .El | 
|  | .Pp | 
|  | The library keeps track of the state of nameservers and will avoid | 
|  | them when they go down. Otherwise it will round robin between them. | 
|  | .Pp | 
|  | Quick start guide: | 
|  | #include "evdns.h" | 
|  | void callback(int result, char type, int count, int ttl, | 
|  | void *addresses, void *arg); | 
|  | evdns_resolv_conf_parse(DNS_OPTIONS_ALL, "/etc/resolv.conf"); | 
|  | evdns_resolve("www.hostname.com", 0, callback, NULL); | 
|  | .Pp | 
|  | When the lookup is complete the callback function is called. The | 
|  | first argument will be one of the DNS_ERR_* defines in evdns.h. | 
|  | Hopefully it will be DNS_ERR_NONE, in which case type will be | 
|  | DNS_IPv4_A, count will be the number of IP addresses, ttl is the time | 
|  | which the data can be cached for (in seconds), addresses will point | 
|  | to an array of uint32_t's and arg will be whatever you passed to | 
|  | evdns_resolve. | 
|  | .Pp | 
|  | Searching: | 
|  | .Pp | 
|  | In order for this library to be a good replacement for glibc's resolver it | 
|  | supports searching. This involves setting a list of default domains, in | 
|  | which names will be queried for. The number of dots in the query name | 
|  | determines the order in which this list is used. | 
|  | .Pp | 
|  | Searching appears to be a single lookup from the point of view of the API, | 
|  | although many DNS queries may be generated from a single call to | 
|  | evdns_resolve. Searching can also drastically slow down the resolution | 
|  | of names. | 
|  | .Pp | 
|  | To disable searching: | 
|  | .Bl -enum -compact -offset indent | 
|  | .It | 
|  | Never set it up. If you never call | 
|  | .Fn evdns_resolv_conf_parse, | 
|  | .Fn evdns_init, | 
|  | or | 
|  | .Fn evdns_search_add | 
|  | then no searching will occur. | 
|  | .It | 
|  | If you do call | 
|  | .Fn evdns_resolv_conf_parse | 
|  | then don't pass | 
|  | .Va DNS_OPTION_SEARCH | 
|  | (or | 
|  | .Va DNS_OPTIONS_ALL, | 
|  | which implies it). | 
|  | .It | 
|  | When calling | 
|  | .Fn evdns_resolve, | 
|  | pass the | 
|  | .Va DNS_QUERY_NO_SEARCH | 
|  | flag. | 
|  | .El | 
|  | .Pp | 
|  | The order of searches depends on the number of dots in the name. If the | 
|  | number is greater than the ndots setting then the names is first tried | 
|  | globally. Otherwise each search domain is appended in turn. | 
|  | .Pp | 
|  | The ndots setting can either be set from a resolv.conf, or by calling | 
|  | evdns_search_ndots_set. | 
|  | .Pp | 
|  | For example, with ndots set to 1 (the default) and a search domain list of | 
|  | ["myhome.net"]: | 
|  | Query: www | 
|  | Order: www.myhome.net, www. | 
|  | .Pp | 
|  | Query: www.abc | 
|  | Order: www.abc., www.abc.myhome.net | 
|  | .Pp | 
|  | .Sh API reference | 
|  | .Pp | 
|  | .Bl -tag -width 0123456 | 
|  | .It Ft int Fn evdns_init | 
|  | Initializes support for non-blocking name resolution by calling | 
|  | .Fn evdns_resolv_conf_parse | 
|  | on UNIX and | 
|  | .Fn evdns_config_windows_nameservers | 
|  | on Windows. | 
|  | .It Ft int Fn evdns_nameserver_add "unsigned long int address" | 
|  | Add a nameserver. The address should be an IP address in | 
|  | network byte order. The type of address is chosen so that | 
|  | it matches in_addr.s_addr. | 
|  | Returns non-zero on error. | 
|  | .It Ft int Fn evdns_nameserver_ip_add "const char *ip_as_string" | 
|  | This wraps the above function by parsing a string as an IP | 
|  | address and adds it as a nameserver. | 
|  | Returns non-zero on error | 
|  | .It Ft int Fn evdns_resolve "const char *name" "int flags" "evdns_callback_type callback" "void *ptr" | 
|  | Resolve a name. The name parameter should be a DNS name. | 
|  | The flags parameter should be 0, or DNS_QUERY_NO_SEARCH | 
|  | which disables searching for this query. (see defn of | 
|  | searching above). | 
|  | .Pp | 
|  | The callback argument is a function which is called when | 
|  | this query completes and ptr is an argument which is passed | 
|  | to that callback function. | 
|  | .Pp | 
|  | Returns non-zero on error | 
|  | .It Ft void Fn evdns_search_clear | 
|  | Clears the list of search domains | 
|  | .It Ft void Fn evdns_search_add "const char *domain" | 
|  | Add a domain to the list of search domains | 
|  | .It Ft void Fn evdns_search_ndots_set "int ndots" | 
|  | Set the number of dots which, when found in a name, causes | 
|  | the first query to be without any search domain. | 
|  | .It Ft int Fn evdns_count_nameservers "void" | 
|  | Return the number of configured nameservers (not necessarily the | 
|  | number of running nameservers).  This is useful for double-checking | 
|  | whether our calls to the various nameserver configuration functions | 
|  | have been successful. | 
|  | .It Ft int Fn evdns_clear_nameservers_and_suspend "void" | 
|  | Remove all currently configured nameservers, and suspend all pending | 
|  | resolves.  Resolves will not necessarily be re-attempted until | 
|  | evdns_resume() is called. | 
|  | .It Ft int Fn evdns_resume "void" | 
|  | Re-attempt resolves left in limbo after an earlier call to | 
|  | evdns_clear_nameservers_and_suspend(). | 
|  | .It Ft int Fn evdns_config_windows_nameservers "void" | 
|  | Attempt to configure a set of nameservers based on platform settings on | 
|  | a win32 host.  Preferentially tries to use GetNetworkParams; if that fails, | 
|  | looks in the registry.  Returns 0 on success, nonzero on failure. | 
|  | .It Ft int Fn evdns_resolv_conf_parse "int flags" "const char *filename" | 
|  | Parse a resolv.conf like file from the given filename. | 
|  | .Pp | 
|  | See the man page for resolv.conf for the format of this file. | 
|  | The flags argument determines what information is parsed from | 
|  | this file: | 
|  | .Bl -tag -width "DNS_OPTION_NAMESERVERS" -offset indent -compact -nested | 
|  | .It DNS_OPTION_SEARCH | 
|  | domain, search and ndots options | 
|  | .It DNS_OPTION_NAMESERVERS | 
|  | nameserver lines | 
|  | .It DNS_OPTION_MISC | 
|  | timeout and attempts options | 
|  | .It DNS_OPTIONS_ALL | 
|  | all of the above | 
|  | .El | 
|  | .Pp | 
|  | The following directives are not parsed from the file: | 
|  | sortlist, rotate, no-check-names, inet6, debug | 
|  | .Pp | 
|  | Returns non-zero on error: | 
|  | .Bl -tag -width "0" -offset indent -compact -nested | 
|  | .It 0 | 
|  | no errors | 
|  | .It 1 | 
|  | failed to open file | 
|  | .It 2 | 
|  | failed to stat file | 
|  | .It 3 | 
|  | file too large | 
|  | .It 4 | 
|  | out of memory | 
|  | .It 5 | 
|  | short read from file | 
|  | .El | 
|  | .El | 
|  | .Sh Internals: | 
|  | Requests are kept in two queues. The first is the inflight queue. In | 
|  | this queue requests have an allocated transaction id and nameserver. | 
|  | They will soon be transmitted if they haven't already been. | 
|  | .Pp | 
|  | The second is the waiting queue. The size of the inflight ring is | 
|  | limited and all other requests wait in waiting queue for space. This | 
|  | bounds the number of concurrent requests so that we don't flood the | 
|  | nameserver. Several algorithms require a full walk of the inflight | 
|  | queue and so bounding its size keeps thing going nicely under huge | 
|  | (many thousands of requests) loads. | 
|  | .Pp | 
|  | If a nameserver loses too many requests it is considered down and we | 
|  | try not to use it. After a while we send a probe to that nameserver | 
|  | (a lookup for google.com) and, if it replies, we consider it working | 
|  | again. If the nameserver fails a probe we wait longer to try again | 
|  | with the next probe. | 
|  | .Sh SEE ALSO | 
|  | .Xr event 3 , | 
|  | .Xr gethostbyname 3 , | 
|  | .Xr resolv.conf 5 | 
|  | .Sh HISTORY | 
|  | The | 
|  | .Nm evdns | 
|  | API was developed by Adam Langley on top of the | 
|  | .Nm libevent | 
|  | API. | 
|  | The code was integrate into | 
|  | .Nm Tor | 
|  | by Nick Mathewson and finally put into | 
|  | .Nm libevent | 
|  | itself by Niels Provos. | 
|  | .Sh AUTHORS | 
|  | The | 
|  | .Nm evdns | 
|  | API and code was written by Adam Langley with significant | 
|  | contributions by Nick Mathewson. | 
|  | .Sh BUGS | 
|  | This documentation is neither complete nor authoritative. | 
|  | If you are in doubt about the usage of this API then | 
|  | check the source code to find out how it works, write | 
|  | up the missing piece of documentation and send it to | 
|  | me for inclusion in this man page. |