xref: /freebsd/contrib/unbound/iterator/iter_priv.c (revision b7579f77d18196a58ff700756c84dc9a302a7f67)
1*b7579f77SDag-Erling Smørgrav /*
2*b7579f77SDag-Erling Smørgrav  * iterator/iter_priv.c - iterative resolver private address and domain store
3*b7579f77SDag-Erling Smørgrav  *
4*b7579f77SDag-Erling Smørgrav  * Copyright (c) 2008, NLnet Labs. All rights reserved.
5*b7579f77SDag-Erling Smørgrav  *
6*b7579f77SDag-Erling Smørgrav  * This software is open source.
7*b7579f77SDag-Erling Smørgrav  *
8*b7579f77SDag-Erling Smørgrav  * Redistribution and use in source and binary forms, with or without
9*b7579f77SDag-Erling Smørgrav  * modification, are permitted provided that the following conditions
10*b7579f77SDag-Erling Smørgrav  * are met:
11*b7579f77SDag-Erling Smørgrav  *
12*b7579f77SDag-Erling Smørgrav  * Redistributions of source code must retain the above copyright notice,
13*b7579f77SDag-Erling Smørgrav  * this list of conditions and the following disclaimer.
14*b7579f77SDag-Erling Smørgrav  *
15*b7579f77SDag-Erling Smørgrav  * Redistributions in binary form must reproduce the above copyright notice,
16*b7579f77SDag-Erling Smørgrav  * this list of conditions and the following disclaimer in the documentation
17*b7579f77SDag-Erling Smørgrav  * and/or other materials provided with the distribution.
18*b7579f77SDag-Erling Smørgrav  *
19*b7579f77SDag-Erling Smørgrav  * Neither the name of the NLNET LABS nor the names of its contributors may
20*b7579f77SDag-Erling Smørgrav  * be used to endorse or promote products derived from this software without
21*b7579f77SDag-Erling Smørgrav  * specific prior written permission.
22*b7579f77SDag-Erling Smørgrav  *
23*b7579f77SDag-Erling Smørgrav  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
24*b7579f77SDag-Erling Smørgrav  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
25*b7579f77SDag-Erling Smørgrav  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
26*b7579f77SDag-Erling Smørgrav  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE
27*b7579f77SDag-Erling Smørgrav  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
28*b7579f77SDag-Erling Smørgrav  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
29*b7579f77SDag-Erling Smørgrav  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
30*b7579f77SDag-Erling Smørgrav  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
31*b7579f77SDag-Erling Smørgrav  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
32*b7579f77SDag-Erling Smørgrav  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
33*b7579f77SDag-Erling Smørgrav  * POSSIBILITY OF SUCH DAMAGE.
34*b7579f77SDag-Erling Smørgrav  */
35*b7579f77SDag-Erling Smørgrav 
36*b7579f77SDag-Erling Smørgrav /**
37*b7579f77SDag-Erling Smørgrav  * \file
38*b7579f77SDag-Erling Smørgrav  *
39*b7579f77SDag-Erling Smørgrav  * This file contains functions to assist the iterator module.
40*b7579f77SDag-Erling Smørgrav  * Keep track of the private addresses and lookup fast.
41*b7579f77SDag-Erling Smørgrav  */
42*b7579f77SDag-Erling Smørgrav 
43*b7579f77SDag-Erling Smørgrav #include "config.h"
44*b7579f77SDag-Erling Smørgrav #include <ldns/dname.h>
45*b7579f77SDag-Erling Smørgrav #include "iterator/iter_priv.h"
46*b7579f77SDag-Erling Smørgrav #include "util/regional.h"
47*b7579f77SDag-Erling Smørgrav #include "util/log.h"
48*b7579f77SDag-Erling Smørgrav #include "util/config_file.h"
49*b7579f77SDag-Erling Smørgrav #include "util/data/dname.h"
50*b7579f77SDag-Erling Smørgrav #include "util/data/msgparse.h"
51*b7579f77SDag-Erling Smørgrav #include "util/net_help.h"
52*b7579f77SDag-Erling Smørgrav #include "util/storage/dnstree.h"
53*b7579f77SDag-Erling Smørgrav 
54*b7579f77SDag-Erling Smørgrav struct iter_priv* priv_create(void)
55*b7579f77SDag-Erling Smørgrav {
56*b7579f77SDag-Erling Smørgrav 	struct iter_priv* priv = (struct iter_priv*)calloc(1, sizeof(*priv));
57*b7579f77SDag-Erling Smørgrav 	if(!priv)
58*b7579f77SDag-Erling Smørgrav 		return NULL;
59*b7579f77SDag-Erling Smørgrav 	priv->region = regional_create();
60*b7579f77SDag-Erling Smørgrav 	if(!priv->region) {
61*b7579f77SDag-Erling Smørgrav 		priv_delete(priv);
62*b7579f77SDag-Erling Smørgrav 		return NULL;
63*b7579f77SDag-Erling Smørgrav 	}
64*b7579f77SDag-Erling Smørgrav 	addr_tree_init(&priv->a);
65*b7579f77SDag-Erling Smørgrav 	name_tree_init(&priv->n);
66*b7579f77SDag-Erling Smørgrav 	return priv;
67*b7579f77SDag-Erling Smørgrav }
68*b7579f77SDag-Erling Smørgrav 
69*b7579f77SDag-Erling Smørgrav void priv_delete(struct iter_priv* priv)
70*b7579f77SDag-Erling Smørgrav {
71*b7579f77SDag-Erling Smørgrav 	if(!priv) return;
72*b7579f77SDag-Erling Smørgrav 	regional_destroy(priv->region);
73*b7579f77SDag-Erling Smørgrav 	free(priv);
74*b7579f77SDag-Erling Smørgrav }
75*b7579f77SDag-Erling Smørgrav 
76*b7579f77SDag-Erling Smørgrav /** Read private-addr declarations from config */
77*b7579f77SDag-Erling Smørgrav static int read_addrs(struct iter_priv* priv, struct config_file* cfg)
78*b7579f77SDag-Erling Smørgrav {
79*b7579f77SDag-Erling Smørgrav 	/* parse addresses, report errors, insert into tree */
80*b7579f77SDag-Erling Smørgrav 	struct config_strlist* p;
81*b7579f77SDag-Erling Smørgrav 	struct addr_tree_node* n;
82*b7579f77SDag-Erling Smørgrav 	struct sockaddr_storage addr;
83*b7579f77SDag-Erling Smørgrav 	int net;
84*b7579f77SDag-Erling Smørgrav 	socklen_t addrlen;
85*b7579f77SDag-Erling Smørgrav 
86*b7579f77SDag-Erling Smørgrav 	for(p = cfg->private_address; p; p = p->next) {
87*b7579f77SDag-Erling Smørgrav 		log_assert(p->str);
88*b7579f77SDag-Erling Smørgrav 		if(!netblockstrtoaddr(p->str, UNBOUND_DNS_PORT, &addr,
89*b7579f77SDag-Erling Smørgrav 			&addrlen, &net)) {
90*b7579f77SDag-Erling Smørgrav 			log_err("cannot parse private-address: %s", p->str);
91*b7579f77SDag-Erling Smørgrav 			return 0;
92*b7579f77SDag-Erling Smørgrav 		}
93*b7579f77SDag-Erling Smørgrav 		n = (struct addr_tree_node*)regional_alloc(priv->region,
94*b7579f77SDag-Erling Smørgrav 			sizeof(*n));
95*b7579f77SDag-Erling Smørgrav 		if(!n) {
96*b7579f77SDag-Erling Smørgrav 			log_err("out of memory");
97*b7579f77SDag-Erling Smørgrav 			return 0;
98*b7579f77SDag-Erling Smørgrav 		}
99*b7579f77SDag-Erling Smørgrav 		if(!addr_tree_insert(&priv->a, n, &addr, addrlen, net)) {
100*b7579f77SDag-Erling Smørgrav 			verbose(VERB_QUERY, "ignoring duplicate "
101*b7579f77SDag-Erling Smørgrav 				"private-address: %s", p->str);
102*b7579f77SDag-Erling Smørgrav 		}
103*b7579f77SDag-Erling Smørgrav 	}
104*b7579f77SDag-Erling Smørgrav 	return 1;
105*b7579f77SDag-Erling Smørgrav }
106*b7579f77SDag-Erling Smørgrav 
107*b7579f77SDag-Erling Smørgrav /** Read private-domain declarations from config */
108*b7579f77SDag-Erling Smørgrav static int read_names(struct iter_priv* priv, struct config_file* cfg)
109*b7579f77SDag-Erling Smørgrav {
110*b7579f77SDag-Erling Smørgrav 	/* parse names, report errors, insert into tree */
111*b7579f77SDag-Erling Smørgrav 	struct config_strlist* p;
112*b7579f77SDag-Erling Smørgrav 	struct name_tree_node* n;
113*b7579f77SDag-Erling Smørgrav 	uint8_t* nm;
114*b7579f77SDag-Erling Smørgrav 	size_t nm_len;
115*b7579f77SDag-Erling Smørgrav 	int nm_labs;
116*b7579f77SDag-Erling Smørgrav 	ldns_rdf* rdf;
117*b7579f77SDag-Erling Smørgrav 
118*b7579f77SDag-Erling Smørgrav 	for(p = cfg->private_domain; p; p = p->next) {
119*b7579f77SDag-Erling Smørgrav 		log_assert(p->str);
120*b7579f77SDag-Erling Smørgrav 		rdf = ldns_dname_new_frm_str(p->str);
121*b7579f77SDag-Erling Smørgrav 		if(!rdf) {
122*b7579f77SDag-Erling Smørgrav 			log_err("cannot parse private-domain: %s", p->str);
123*b7579f77SDag-Erling Smørgrav 			return 0;
124*b7579f77SDag-Erling Smørgrav 		}
125*b7579f77SDag-Erling Smørgrav 		nm = ldns_rdf_data(rdf);
126*b7579f77SDag-Erling Smørgrav 		nm_labs = dname_count_size_labels(nm, &nm_len);
127*b7579f77SDag-Erling Smørgrav 		nm = (uint8_t*)regional_alloc_init(priv->region, nm, nm_len);
128*b7579f77SDag-Erling Smørgrav 		ldns_rdf_deep_free(rdf);
129*b7579f77SDag-Erling Smørgrav 		if(!nm) {
130*b7579f77SDag-Erling Smørgrav 			log_err("out of memory");
131*b7579f77SDag-Erling Smørgrav 			return 0;
132*b7579f77SDag-Erling Smørgrav 		}
133*b7579f77SDag-Erling Smørgrav 		n = (struct name_tree_node*)regional_alloc(priv->region,
134*b7579f77SDag-Erling Smørgrav 			sizeof(*n));
135*b7579f77SDag-Erling Smørgrav 		if(!n) {
136*b7579f77SDag-Erling Smørgrav 			log_err("out of memory");
137*b7579f77SDag-Erling Smørgrav 			return 0;
138*b7579f77SDag-Erling Smørgrav 		}
139*b7579f77SDag-Erling Smørgrav 		if(!name_tree_insert(&priv->n, n, nm, nm_len, nm_labs,
140*b7579f77SDag-Erling Smørgrav 			LDNS_RR_CLASS_IN)) {
141*b7579f77SDag-Erling Smørgrav 			verbose(VERB_QUERY, "ignoring duplicate "
142*b7579f77SDag-Erling Smørgrav 				"private-domain: %s", p->str);
143*b7579f77SDag-Erling Smørgrav 		}
144*b7579f77SDag-Erling Smørgrav 	}
145*b7579f77SDag-Erling Smørgrav 	return 1;
146*b7579f77SDag-Erling Smørgrav }
147*b7579f77SDag-Erling Smørgrav 
148*b7579f77SDag-Erling Smørgrav int priv_apply_cfg(struct iter_priv* priv, struct config_file* cfg)
149*b7579f77SDag-Erling Smørgrav {
150*b7579f77SDag-Erling Smørgrav 	/* empty the current contents */
151*b7579f77SDag-Erling Smørgrav 	regional_free_all(priv->region);
152*b7579f77SDag-Erling Smørgrav 	addr_tree_init(&priv->a);
153*b7579f77SDag-Erling Smørgrav 	name_tree_init(&priv->n);
154*b7579f77SDag-Erling Smørgrav 
155*b7579f77SDag-Erling Smørgrav 	/* read new contents */
156*b7579f77SDag-Erling Smørgrav 	if(!read_addrs(priv, cfg))
157*b7579f77SDag-Erling Smørgrav 		return 0;
158*b7579f77SDag-Erling Smørgrav 	if(!read_names(priv, cfg))
159*b7579f77SDag-Erling Smørgrav 		return 0;
160*b7579f77SDag-Erling Smørgrav 
161*b7579f77SDag-Erling Smørgrav 	/* prepare for lookups */
162*b7579f77SDag-Erling Smørgrav 	addr_tree_init_parents(&priv->a);
163*b7579f77SDag-Erling Smørgrav 	name_tree_init_parents(&priv->n);
164*b7579f77SDag-Erling Smørgrav 	return 1;
165*b7579f77SDag-Erling Smørgrav }
166*b7579f77SDag-Erling Smørgrav 
167*b7579f77SDag-Erling Smørgrav /**
168*b7579f77SDag-Erling Smørgrav  * See if an address is blocked.
169*b7579f77SDag-Erling Smørgrav  * @param priv: structure for address storage.
170*b7579f77SDag-Erling Smørgrav  * @param addr: address to check
171*b7579f77SDag-Erling Smørgrav  * @param addrlen: length of addr.
172*b7579f77SDag-Erling Smørgrav  * @return: true if the address must not be queried. false if unlisted.
173*b7579f77SDag-Erling Smørgrav  */
174*b7579f77SDag-Erling Smørgrav static int
175*b7579f77SDag-Erling Smørgrav priv_lookup_addr(struct iter_priv* priv, struct sockaddr_storage* addr,
176*b7579f77SDag-Erling Smørgrav 	socklen_t addrlen)
177*b7579f77SDag-Erling Smørgrav {
178*b7579f77SDag-Erling Smørgrav 	return addr_tree_lookup(&priv->a, addr, addrlen) != NULL;
179*b7579f77SDag-Erling Smørgrav }
180*b7579f77SDag-Erling Smørgrav 
181*b7579f77SDag-Erling Smørgrav /**
182*b7579f77SDag-Erling Smørgrav  * See if a name is whitelisted.
183*b7579f77SDag-Erling Smørgrav  * @param priv: structure for address storage.
184*b7579f77SDag-Erling Smørgrav  * @param pkt: the packet (for compression ptrs).
185*b7579f77SDag-Erling Smørgrav  * @param name: name to check.
186*b7579f77SDag-Erling Smørgrav  * @param name_len: uncompressed length of the name to check.
187*b7579f77SDag-Erling Smørgrav  * @param dclass: class to check.
188*b7579f77SDag-Erling Smørgrav  * @return: true if the name is OK. false if unlisted.
189*b7579f77SDag-Erling Smørgrav  */
190*b7579f77SDag-Erling Smørgrav static int
191*b7579f77SDag-Erling Smørgrav priv_lookup_name(struct iter_priv* priv, ldns_buffer* pkt,
192*b7579f77SDag-Erling Smørgrav 	uint8_t* name, size_t name_len, uint16_t dclass)
193*b7579f77SDag-Erling Smørgrav {
194*b7579f77SDag-Erling Smørgrav 	size_t len;
195*b7579f77SDag-Erling Smørgrav 	uint8_t decomp[256];
196*b7579f77SDag-Erling Smørgrav 	int labs;
197*b7579f77SDag-Erling Smørgrav 	if(name_len >= sizeof(decomp))
198*b7579f77SDag-Erling Smørgrav 		return 0;
199*b7579f77SDag-Erling Smørgrav 	dname_pkt_copy(pkt, decomp, name);
200*b7579f77SDag-Erling Smørgrav 	labs = dname_count_size_labels(decomp, &len);
201*b7579f77SDag-Erling Smørgrav 	log_assert(name_len == len);
202*b7579f77SDag-Erling Smørgrav 	return name_tree_lookup(&priv->n, decomp, len, labs, dclass) != NULL;
203*b7579f77SDag-Erling Smørgrav }
204*b7579f77SDag-Erling Smørgrav 
205*b7579f77SDag-Erling Smørgrav size_t priv_get_mem(struct iter_priv* priv)
206*b7579f77SDag-Erling Smørgrav {
207*b7579f77SDag-Erling Smørgrav 	if(!priv) return 0;
208*b7579f77SDag-Erling Smørgrav 	return sizeof(*priv) + regional_get_mem(priv->region);
209*b7579f77SDag-Erling Smørgrav }
210*b7579f77SDag-Erling Smørgrav 
211*b7579f77SDag-Erling Smørgrav int priv_rrset_bad(struct iter_priv* priv, ldns_buffer* pkt,
212*b7579f77SDag-Erling Smørgrav 	struct rrset_parse* rrset)
213*b7579f77SDag-Erling Smørgrav {
214*b7579f77SDag-Erling Smørgrav 	if(priv->a.count == 0)
215*b7579f77SDag-Erling Smørgrav 		return 0; /* there are no blocked addresses */
216*b7579f77SDag-Erling Smørgrav 
217*b7579f77SDag-Erling Smørgrav 	/* see if it is a private name, that is allowed to have any */
218*b7579f77SDag-Erling Smørgrav 	if(priv_lookup_name(priv, pkt, rrset->dname, rrset->dname_len,
219*b7579f77SDag-Erling Smørgrav 		ntohs(rrset->rrset_class))) {
220*b7579f77SDag-Erling Smørgrav 		return 0;
221*b7579f77SDag-Erling Smørgrav 	} else {
222*b7579f77SDag-Erling Smørgrav 		/* so its a public name, check the address */
223*b7579f77SDag-Erling Smørgrav 		socklen_t len;
224*b7579f77SDag-Erling Smørgrav 		struct rr_parse* rr;
225*b7579f77SDag-Erling Smørgrav 		if(rrset->type == LDNS_RR_TYPE_A) {
226*b7579f77SDag-Erling Smørgrav 			struct sockaddr_storage addr;
227*b7579f77SDag-Erling Smørgrav 			struct sockaddr_in sa;
228*b7579f77SDag-Erling Smørgrav 
229*b7579f77SDag-Erling Smørgrav 			len = (socklen_t)sizeof(sa);
230*b7579f77SDag-Erling Smørgrav 			memset(&sa, 0, len);
231*b7579f77SDag-Erling Smørgrav 			sa.sin_family = AF_INET;
232*b7579f77SDag-Erling Smørgrav 			sa.sin_port = (in_port_t)htons(UNBOUND_DNS_PORT);
233*b7579f77SDag-Erling Smørgrav 			for(rr = rrset->rr_first; rr; rr = rr->next) {
234*b7579f77SDag-Erling Smørgrav 				if(ldns_read_uint16(rr->ttl_data+4)
235*b7579f77SDag-Erling Smørgrav 					!= INET_SIZE)
236*b7579f77SDag-Erling Smørgrav 					continue;
237*b7579f77SDag-Erling Smørgrav 				memmove(&sa.sin_addr, rr->ttl_data+4+2,
238*b7579f77SDag-Erling Smørgrav 					INET_SIZE);
239*b7579f77SDag-Erling Smørgrav 				memmove(&addr, &sa, len);
240*b7579f77SDag-Erling Smørgrav 				if(priv_lookup_addr(priv, &addr, len))
241*b7579f77SDag-Erling Smørgrav 					return 1;
242*b7579f77SDag-Erling Smørgrav 			}
243*b7579f77SDag-Erling Smørgrav 		} else if(rrset->type == LDNS_RR_TYPE_AAAA) {
244*b7579f77SDag-Erling Smørgrav 			struct sockaddr_storage addr;
245*b7579f77SDag-Erling Smørgrav 			struct sockaddr_in6 sa;
246*b7579f77SDag-Erling Smørgrav 			len = (socklen_t)sizeof(sa);
247*b7579f77SDag-Erling Smørgrav 			memset(&sa, 0, len);
248*b7579f77SDag-Erling Smørgrav 			sa.sin6_family = AF_INET6;
249*b7579f77SDag-Erling Smørgrav 			sa.sin6_port = (in_port_t)htons(UNBOUND_DNS_PORT);
250*b7579f77SDag-Erling Smørgrav 			for(rr = rrset->rr_first; rr; rr = rr->next) {
251*b7579f77SDag-Erling Smørgrav 				if(ldns_read_uint16(rr->ttl_data+4)
252*b7579f77SDag-Erling Smørgrav 					!= INET6_SIZE)
253*b7579f77SDag-Erling Smørgrav 					continue;
254*b7579f77SDag-Erling Smørgrav 				memmove(&sa.sin6_addr, rr->ttl_data+4+2,
255*b7579f77SDag-Erling Smørgrav 					INET6_SIZE);
256*b7579f77SDag-Erling Smørgrav 				memmove(&addr, &sa, len);
257*b7579f77SDag-Erling Smørgrav 				if(priv_lookup_addr(priv, &addr, len))
258*b7579f77SDag-Erling Smørgrav 					return 1;
259*b7579f77SDag-Erling Smørgrav 			}
260*b7579f77SDag-Erling Smørgrav 		}
261*b7579f77SDag-Erling Smørgrav 	}
262*b7579f77SDag-Erling Smørgrav 	return 0;
263*b7579f77SDag-Erling Smørgrav }
264