Update to latest upstream version
PR: 209177 Reported by: Vitaly Magerya MFC after: 1 week
This commit is contained in:
parent
91dc125e2d
commit
a8ae169218
@ -6,11 +6,11 @@ LOCALBASE?=/usr/local
|
||||
PREFIX?=${LOCALBASE}
|
||||
MANDIR?=${PREFIX}/man
|
||||
|
||||
XCFLAGS=${CFLAGS} -I${LOCALBASE}/include
|
||||
XCFLAGS=${CFLAGS} -I${LOCALBASE}/include -std=c99 -Wall -Wextra -pedantic
|
||||
XLDFLAGS=${LDFLAGS} -L${LOCALBASE}/lib -lldns
|
||||
|
||||
${PROG}: ${SRC}
|
||||
${CC} -o $@ ${XCFLAGS} ${XLDFLAGS} ${SRC}
|
||||
${CC} -o $@ ${XCFLAGS} ${SRC} ${XLDFLAGS}
|
||||
|
||||
clean:
|
||||
rm -f ${PROG}
|
||||
|
@ -236,9 +236,6 @@ commands, while
|
||||
.Sq host
|
||||
from BIND9 uses whatever command was specified last.
|
||||
.It
|
||||
Multi-packet zone transfers are not supported; only the first
|
||||
response packet is printed.
|
||||
.It
|
||||
.Sq Pseudosection TSIG
|
||||
is missing from verbose packet output.
|
||||
.El
|
||||
|
@ -7,16 +7,14 @@
|
||||
* without any warranty.
|
||||
*/
|
||||
|
||||
#include <netinet/in.h>
|
||||
|
||||
#include <ldns/ldns.h>
|
||||
#include <limits.h>
|
||||
#include <netdb.h>
|
||||
#include <netinet/in.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include <ldns/ldns.h>
|
||||
|
||||
/* General utilities.
|
||||
*/
|
||||
|
||||
@ -158,12 +156,109 @@ memerr:
|
||||
return LDNS_STATUS_MEM_ERR;
|
||||
}
|
||||
|
||||
static ldns_status
|
||||
ldns_tcp_start(ldns_resolver *res, ldns_pkt *qpkt, int nameserver) {
|
||||
/* This routine is based on ldns_axfr_start, with the major
|
||||
* difference in that it takes a query packet explicitly.
|
||||
*/
|
||||
struct sockaddr_storage *ns = NULL;
|
||||
size_t ns_len = 0;
|
||||
ldns_buffer *qbuf = NULL;
|
||||
ldns_status status;
|
||||
|
||||
ns = ldns_rdf2native_sockaddr_storage(
|
||||
res->_nameservers[nameserver], ldns_resolver_port(res), &ns_len);
|
||||
if (ns == NULL) {
|
||||
status = LDNS_STATUS_MEM_ERR;
|
||||
goto error;
|
||||
}
|
||||
|
||||
res->_socket = ldns_tcp_connect(
|
||||
ns, (socklen_t)ns_len, ldns_resolver_timeout(res));
|
||||
if (res->_socket <= 0) {
|
||||
status = LDNS_STATUS_ADDRESS_ERR;
|
||||
goto error;
|
||||
}
|
||||
|
||||
qbuf = ldns_buffer_new(LDNS_MAX_PACKETLEN);
|
||||
if (qbuf == NULL) {
|
||||
status = LDNS_STATUS_MEM_ERR;
|
||||
goto error;
|
||||
}
|
||||
|
||||
status = ldns_pkt2buffer_wire(qbuf, qpkt);
|
||||
if (status != LDNS_STATUS_OK)
|
||||
goto error;
|
||||
|
||||
if (ldns_tcp_send_query(qbuf, res->_socket, ns, (socklen_t)ns_len) == 0) {
|
||||
status = LDNS_STATUS_NETWORK_ERR;
|
||||
goto error;
|
||||
}
|
||||
|
||||
ldns_buffer_free(qbuf);
|
||||
free(ns);
|
||||
return LDNS_STATUS_OK;
|
||||
|
||||
error:
|
||||
ldns_buffer_free(qbuf);
|
||||
free(ns);
|
||||
if (res->_socket > 0) {
|
||||
close(res->_socket);
|
||||
res->_socket = 0;
|
||||
}
|
||||
return status;
|
||||
}
|
||||
|
||||
static ldns_status
|
||||
ldns_tcp_read(ldns_pkt **answer, ldns_resolver *res) {
|
||||
ldns_status status;
|
||||
struct timeval t1, t2;
|
||||
uint8_t *data;
|
||||
size_t size;
|
||||
|
||||
if (res->_socket <= 0)
|
||||
return LDNS_STATUS_ERR;
|
||||
|
||||
gettimeofday(&t1, NULL);
|
||||
data = ldns_tcp_read_wire_timeout(
|
||||
res->_socket, &size, ldns_resolver_timeout(res));
|
||||
if (data == NULL)
|
||||
goto error;
|
||||
|
||||
status = ldns_wire2pkt(answer, data, size);
|
||||
free(data);
|
||||
if (status != LDNS_STATUS_OK)
|
||||
goto error;
|
||||
|
||||
gettimeofday(&t2, NULL);
|
||||
ldns_pkt_set_querytime(*answer,
|
||||
(uint32_t)((t2.tv_sec - t1.tv_sec)*1000) +
|
||||
(t2.tv_usec - t1.tv_usec)/1000);
|
||||
ldns_pkt_set_timestamp(*answer, t2);
|
||||
return status;
|
||||
|
||||
error:
|
||||
close(res->_socket);
|
||||
res->_socket = 0;
|
||||
return LDNS_STATUS_ERR;
|
||||
}
|
||||
|
||||
static void
|
||||
ldns_tcp_close(ldns_resolver *res) {
|
||||
if (res->_socket > 0) {
|
||||
close(res->_socket);
|
||||
res->_socket = 0;
|
||||
}
|
||||
}
|
||||
|
||||
static ldns_status
|
||||
ldns_resolver_send_to(ldns_pkt **answer, ldns_resolver *res,
|
||||
const ldns_rdf *name, ldns_rr_type t, ldns_rr_class c,
|
||||
uint16_t flags, uint32_t ixfr_serial, int nameserver) {
|
||||
ldns_status status;
|
||||
uint16_t flags, uint32_t ixfr_serial, int nameserver,
|
||||
bool close_tcp) {
|
||||
ldns_status status = LDNS_STATUS_OK;
|
||||
ldns_pkt *qpkt;
|
||||
struct timeval now;
|
||||
|
||||
int nscnt = ldns_resolver_nameserver_count(res);
|
||||
ldns_rdf **ns = ldns_resolver_nameservers(res);
|
||||
@ -173,12 +268,38 @@ ldns_resolver_send_to(ldns_pkt **answer, ldns_resolver *res,
|
||||
ldns_resolver_set_rtt(res, &rtt[nameserver]);
|
||||
ldns_resolver_set_nameserver_count(res, 1);
|
||||
|
||||
status = ldns_resolver_prepare_query_pkt(&qpkt, res, name, t, c, flags);
|
||||
if (status == LDNS_STATUS_OK && t == LDNS_RR_TYPE_IXFR)
|
||||
/* The next fragment should have been a call to
|
||||
* ldns_resolver_prepare_query_pkt(), but starting with ldns
|
||||
* version 1.6.17 that function tries to add it's own SOA
|
||||
* records when rr_type is LDNS_RR_TYPE_IXFR, and we don't
|
||||
* want that.
|
||||
*/
|
||||
qpkt = ldns_pkt_query_new(ldns_rdf_clone(name), t, c, flags);
|
||||
if (qpkt == NULL) {
|
||||
status = LDNS_STATUS_ERR;
|
||||
goto done;
|
||||
}
|
||||
now.tv_sec = time(NULL);
|
||||
now.tv_usec = 0;
|
||||
ldns_pkt_set_timestamp(qpkt, now);
|
||||
ldns_pkt_set_random_id(qpkt);
|
||||
|
||||
if (t == LDNS_RR_TYPE_IXFR) {
|
||||
status = ldns_pkt_push_rr_soa(qpkt,
|
||||
LDNS_SECTION_AUTHORITY, name, c, ixfr_serial);
|
||||
if (status == LDNS_STATUS_OK)
|
||||
if (status != LDNS_STATUS_OK) goto done;
|
||||
}
|
||||
if (close_tcp) {
|
||||
status = ldns_resolver_send_pkt(answer, res, qpkt);
|
||||
} else {
|
||||
status = ldns_tcp_start(res, qpkt, 0);
|
||||
if (status != LDNS_STATUS_OK) goto done;
|
||||
status = ldns_tcp_read(answer, res);
|
||||
if (status != LDNS_STATUS_OK) goto done;
|
||||
ldns_pkt_set_answerfrom(*answer, ldns_rdf_clone(ns[0]));
|
||||
}
|
||||
|
||||
done:
|
||||
ldns_pkt_free(qpkt);
|
||||
|
||||
ldns_resolver_set_nameservers(res, ns);
|
||||
@ -203,9 +324,9 @@ ldns_pkt_filter_answer(ldns_pkt *pkt, ldns_rr_type type) {
|
||||
type == rrtype ||
|
||||
(type == LDNS_RR_TYPE_AXFR &&
|
||||
(rrtype == LDNS_RR_TYPE_A ||
|
||||
rrtype == LDNS_RR_TYPE_AAAA ||
|
||||
rrtype == LDNS_RR_TYPE_NS ||
|
||||
rrtype == LDNS_RR_TYPE_PTR)))
|
||||
rrtype == LDNS_RR_TYPE_AAAA ||
|
||||
rrtype == LDNS_RR_TYPE_NS ||
|
||||
rrtype == LDNS_RR_TYPE_PTR)))
|
||||
ldns_rr_list_set_rr(rrlist, rr, j++);
|
||||
}
|
||||
ldns_rr_list_set_rr_count(rrlist, j);
|
||||
@ -438,7 +559,7 @@ print_received_line(ldns_resolver *res, ldns_pkt *pkt) {
|
||||
#define DEFAULT_TCP_TIMEOUT 10
|
||||
#define DEFAULT_UDP_TIMEOUT 5
|
||||
|
||||
enum operation_mode { M_AXFR, M_DEFAULT_Q, M_SINGLE_Q, M_SOA };
|
||||
enum operation_mode { M_AXFR, M_IXFR, M_DEFAULT_Q, M_SINGLE_Q, M_SOA };
|
||||
|
||||
static enum operation_mode o_mode = M_DEFAULT_Q;
|
||||
static bool o_ignore_servfail = true;
|
||||
@ -454,15 +575,15 @@ static int o_ipversion = LDNS_RESOLV_INETANY;
|
||||
static int o_ndots = 1;
|
||||
static int o_retries = 1;
|
||||
static ldns_rr_class o_rrclass = LDNS_RR_CLASS_IN;
|
||||
static ldns_rr_type o_rrtype = LDNS_RR_TYPE_A;
|
||||
static ldns_rr_type o_rrtype = (ldns_rr_type)-1;
|
||||
static time_t o_timeout = 0;
|
||||
static uint32_t o_ixfr_serial = 0;
|
||||
|
||||
static void
|
||||
usage(void) {
|
||||
fputs(
|
||||
"Usage: host [-aCdilrsTvw46] [-c class] [-N ndots] [-R number]\n"
|
||||
" [-t type] [-W wait] name [server]\n"
|
||||
fprintf(stderr,
|
||||
"Usage: %s [-aCdilrsTvw46] [-c class] [-N ndots] [-R number]\n"
|
||||
" %*c [-t type] [-W wait] name [server]\n"
|
||||
"\t-a same as -v -t ANY\n"
|
||||
"\t-C query SOA records from all authoritative name servers\n"
|
||||
"\t-c use this query class (IN, CH, HS, etc)\n"
|
||||
@ -480,7 +601,7 @@ usage(void) {
|
||||
"\t-W wait this many seconds for a reply\n"
|
||||
"\t-4 use IPv4 only\n"
|
||||
"\t-6 use IPv6 only\n",
|
||||
stderr);
|
||||
progname, (int)strlen(progname), ' ');
|
||||
exit(1);
|
||||
}
|
||||
|
||||
@ -513,7 +634,8 @@ parse_args(int argc, char *argv[]) {
|
||||
case 'i': o_ip6_int = true; break;
|
||||
case 'l':
|
||||
o_mode = M_AXFR;
|
||||
o_rrtype = LDNS_RR_TYPE_AXFR;
|
||||
if (o_rrtype == (ldns_rr_type)-1)
|
||||
o_rrtype = LDNS_RR_TYPE_AXFR;
|
||||
o_tcp = true;
|
||||
break;
|
||||
case 'N':
|
||||
@ -542,13 +664,15 @@ parse_args(int argc, char *argv[]) {
|
||||
if (o_rrtype <= 0)
|
||||
die(2, "invalid type: %s\n", optarg);
|
||||
}
|
||||
if (o_rrtype == LDNS_RR_TYPE_AXFR || o_rrtype == LDNS_RR_TYPE_IXFR)
|
||||
o_tcp = true;
|
||||
if (o_rrtype == LDNS_RR_TYPE_AXFR) {
|
||||
o_mode = M_AXFR;
|
||||
o_rrtype = LDNS_RR_TYPE_ANY;
|
||||
o_verbose = true;
|
||||
}
|
||||
if (o_rrtype == LDNS_RR_TYPE_IXFR) {
|
||||
o_mode = M_IXFR;
|
||||
o_rrtype = LDNS_RR_TYPE_ANY;
|
||||
}
|
||||
break;
|
||||
case 'v': o_verbose = true; break;
|
||||
case 'w':
|
||||
@ -574,6 +698,8 @@ parse_args(int argc, char *argv[]) {
|
||||
o_server = argv[1];
|
||||
o_print_pkt_server = true;
|
||||
}
|
||||
if (o_rrtype == (ldns_rr_type)-1)
|
||||
o_rrtype = LDNS_RR_TYPE_A;
|
||||
}
|
||||
|
||||
static ldns_rdf*
|
||||
@ -602,7 +728,7 @@ safe_dname_cat_clone(const ldns_rdf *rd1, const ldns_rdf *rd2) {
|
||||
}
|
||||
|
||||
static bool
|
||||
query(ldns_resolver *res, ldns_rdf *domain, ldns_pkt **pkt) {
|
||||
query(ldns_resolver *res, ldns_rdf *domain, ldns_pkt **pkt, bool close_tcp) {
|
||||
ldns_status status;
|
||||
ldns_pkt_rcode rcode;
|
||||
int i, cnt;
|
||||
@ -614,7 +740,8 @@ query(ldns_resolver *res, ldns_rdf *domain, ldns_pkt **pkt) {
|
||||
}
|
||||
for (cnt = ldns_resolver_nameserver_count(res), i = 0; i < cnt; i++) {
|
||||
status = ldns_resolver_send_to(pkt, res, domain, o_rrtype,
|
||||
o_rrclass, o_recursive ? LDNS_RD : 0, o_ixfr_serial, i);
|
||||
o_rrclass, o_recursive ? LDNS_RD : 0, o_ixfr_serial, i,
|
||||
close_tcp);
|
||||
if (status != LDNS_STATUS_OK) {
|
||||
*pkt = NULL;
|
||||
continue;
|
||||
@ -624,7 +751,8 @@ query(ldns_resolver *res, ldns_rdf *domain, ldns_pkt **pkt) {
|
||||
printf(";; Truncated, retrying in TCP mode.\n");
|
||||
ldns_resolver_set_usevc(res, true);
|
||||
status = ldns_resolver_send_to(pkt, res, domain, o_rrtype,
|
||||
o_rrclass, o_recursive ? LDNS_RD : 0, o_ixfr_serial, i);
|
||||
o_rrclass, o_recursive ? LDNS_RD : 0, o_ixfr_serial, i,
|
||||
close_tcp);
|
||||
ldns_resolver_set_usevc(res, false);
|
||||
if (status != LDNS_STATUS_OK)
|
||||
continue;
|
||||
@ -642,16 +770,17 @@ query(ldns_resolver *res, ldns_rdf *domain, ldns_pkt **pkt) {
|
||||
}
|
||||
|
||||
static ldns_rdf *
|
||||
search(ldns_resolver *res, ldns_rdf *domain, ldns_pkt **pkt, bool absolute) {
|
||||
search(ldns_resolver *res, ldns_rdf *domain, ldns_pkt **pkt,
|
||||
bool absolute, bool close_tcp) {
|
||||
ldns_rdf *dname, **searchlist;
|
||||
int i, n;
|
||||
|
||||
if (absolute && query(res, domain, pkt))
|
||||
if (absolute && query(res, domain, pkt, close_tcp))
|
||||
return domain;
|
||||
|
||||
if ((dname = ldns_resolver_domain(res)) != NULL) {
|
||||
dname = safe_dname_cat_clone(domain, dname);
|
||||
if (query(res, dname, pkt))
|
||||
if (query(res, dname, pkt, close_tcp))
|
||||
return dname;
|
||||
}
|
||||
|
||||
@ -659,11 +788,11 @@ search(ldns_resolver *res, ldns_rdf *domain, ldns_pkt **pkt, bool absolute) {
|
||||
n = ldns_resolver_searchlist_count(res);
|
||||
for (i = 0; i < n; i++) {
|
||||
dname = safe_dname_cat_clone(domain, searchlist[i]);
|
||||
if (query(res, dname, pkt))
|
||||
if (query(res, dname, pkt, close_tcp))
|
||||
return dname;
|
||||
}
|
||||
|
||||
if (!absolute && query(res, domain, pkt))
|
||||
if (!absolute && query(res, domain, pkt, close_tcp))
|
||||
return domain;
|
||||
|
||||
return NULL;
|
||||
@ -691,7 +820,7 @@ report(ldns_resolver *res, ldns_rdf *domain, ldns_pkt *pkt) {
|
||||
print_pkt_verbose(pkt);
|
||||
} else {
|
||||
print_pkt_short(pkt, o_print_rr_server);
|
||||
if (o_mode != M_DEFAULT_Q &&
|
||||
if (o_mode == M_SINGLE_Q &&
|
||||
ldns_rr_list_rr_count(ldns_pkt_answer(pkt)) == 0) {
|
||||
print_rdf_nodot(domain);
|
||||
printf(" has no ");
|
||||
@ -709,7 +838,7 @@ doquery(ldns_resolver *res, ldns_rdf *domain) {
|
||||
ldns_pkt *pkt;
|
||||
bool q;
|
||||
|
||||
q = query(res, domain, &pkt);
|
||||
q = query(res, domain, &pkt, true);
|
||||
report(res, domain, pkt);
|
||||
return q;
|
||||
}
|
||||
@ -719,7 +848,7 @@ doquery_filtered(ldns_resolver *res, ldns_rdf *domain) {
|
||||
ldns_pkt *pkt;
|
||||
bool q;
|
||||
|
||||
q = query(res, domain, &pkt);
|
||||
q = query(res, domain, &pkt, true);
|
||||
ldns_pkt_filter_answer(pkt, o_rrtype);
|
||||
report(res, domain, pkt);
|
||||
return q;
|
||||
@ -730,7 +859,7 @@ dosearch(ldns_resolver *res, ldns_rdf *domain, bool absolute) {
|
||||
ldns_pkt *pkt;
|
||||
ldns_rdf *dname;
|
||||
|
||||
dname = search(res, domain, &pkt, absolute);
|
||||
dname = search(res, domain, &pkt, absolute, true);
|
||||
report(res, dname != NULL ? dname : domain, pkt);
|
||||
return o_mode != M_DEFAULT_Q ? (dname != NULL) :
|
||||
(dname != NULL) &&
|
||||
@ -739,17 +868,44 @@ dosearch(ldns_resolver *res, ldns_rdf *domain, bool absolute) {
|
||||
}
|
||||
|
||||
static bool
|
||||
doaxfr(ldns_resolver *res, ldns_rdf *domain, bool absolute) {
|
||||
ldns_pkt *pkt;
|
||||
dozonetransfer(ldns_resolver *res, ldns_rdf *domain, bool absolute) {
|
||||
ldns_pkt *pkt, *nextpkt;
|
||||
ldns_rdf *dname;
|
||||
ldns_rr_type rrtype;
|
||||
ldns_rr_list *rrl;
|
||||
int i, nsoa = 0;
|
||||
|
||||
rrtype = o_rrtype;
|
||||
o_rrtype = LDNS_RR_TYPE_AXFR;
|
||||
dname = search(res, domain, &pkt, absolute);
|
||||
ldns_pkt_filter_answer(pkt, rrtype);
|
||||
report(res, dname != NULL ? dname : domain, pkt);
|
||||
return dname != NULL;
|
||||
o_rrtype = (o_mode == M_AXFR) ? LDNS_RR_TYPE_AXFR : LDNS_RR_TYPE_IXFR;
|
||||
dname = search(res, domain, &pkt, absolute, false);
|
||||
|
||||
for (;;) {
|
||||
rrl = ldns_pkt_answer(pkt);
|
||||
for (i = ldns_rr_list_rr_count(rrl) - 1; i >= 0; i--) {
|
||||
if (ldns_rr_get_type(ldns_rr_list_rr(rrl, i)) == LDNS_RR_TYPE_SOA)
|
||||
nsoa++;
|
||||
}
|
||||
ldns_pkt_filter_answer(pkt, rrtype);
|
||||
report(res, dname != NULL ? dname : domain, pkt);
|
||||
if ((dname == NULL) ||
|
||||
(ldns_pkt_get_rcode(pkt) != LDNS_RCODE_NOERROR)) {
|
||||
printf("; Transfer failed.\n");
|
||||
ldns_tcp_close(res);
|
||||
return false;
|
||||
}
|
||||
if (nsoa >= 2) {
|
||||
ldns_tcp_close(res);
|
||||
return true;
|
||||
}
|
||||
if (ldns_tcp_read(&nextpkt, res) != LDNS_STATUS_OK) {
|
||||
printf("; Transfer failed.\n");
|
||||
return false;
|
||||
}
|
||||
ldns_pkt_set_answerfrom(nextpkt,
|
||||
ldns_rdf_clone(ldns_pkt_answerfrom(pkt)));
|
||||
ldns_pkt_free(pkt);
|
||||
pkt = nextpkt;
|
||||
}
|
||||
}
|
||||
|
||||
static bool
|
||||
@ -760,7 +916,7 @@ dosoa(ldns_resolver *res, ldns_rdf *domain, bool absolute) {
|
||||
ldns_rr *rr;
|
||||
size_t i, j, n, cnt;
|
||||
|
||||
if ((dname = search(res, domain, &pkt, absolute)) == NULL)
|
||||
if ((dname = search(res, domain, &pkt, absolute, true)) == NULL)
|
||||
return false;
|
||||
|
||||
answer = ldns_pkt_answer(pkt);
|
||||
@ -780,9 +936,9 @@ dosoa(ldns_resolver *res, ldns_rdf *domain, bool absolute) {
|
||||
ldns_resolver_remove_nameservers(res);
|
||||
rr = ldns_rr_list_rr(nsaddrs[i], j);
|
||||
if ((ldns_resolver_ip6(res) == LDNS_RESOLV_INET &&
|
||||
ldns_rr_get_type(rr) == LDNS_RR_TYPE_AAAA) ||
|
||||
ldns_rr_get_type(rr) == LDNS_RR_TYPE_AAAA) ||
|
||||
(ldns_resolver_ip6(res) == LDNS_RESOLV_INET6 &&
|
||||
ldns_rr_get_type(rr) == LDNS_RR_TYPE_A))
|
||||
ldns_rr_get_type(rr) == LDNS_RR_TYPE_A))
|
||||
continue;
|
||||
if (ldns_resolver_push_nameserver_rr(res, rr) == LDNS_STATUS_OK)
|
||||
/* bind9-host queries for domain, not dname here */
|
||||
@ -879,6 +1035,9 @@ main(int argc, char *argv[]) {
|
||||
o_rrtype = LDNS_RR_TYPE_PTR;
|
||||
return !doquery(res, dname);
|
||||
}
|
||||
return !(o_mode == M_SOA ? dosoa : o_mode == M_AXFR ? doaxfr : dosearch)
|
||||
return !(o_mode == M_SOA ? dosoa :
|
||||
o_mode == M_AXFR ? dozonetransfer :
|
||||
o_mode == M_IXFR ? dozonetransfer :
|
||||
dosearch)
|
||||
(res, safe_str2rdf_dname(o_name), ndots(o_name) >= o_ndots);
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user