xref: /minix3/external/bsd/bind/dist/lib/dns/forward.c (revision 00b67f09dd46474d133c95011a48590a8e8f94c7)
1*00b67f09SDavid van Moolenbroek /*	$NetBSD: forward.c,v 1.5 2014/12/10 04:37:58 christos Exp $	*/
2*00b67f09SDavid van Moolenbroek 
3*00b67f09SDavid van Moolenbroek /*
4*00b67f09SDavid van Moolenbroek  * Copyright (C) 2004, 2005, 2007, 2009, 2013  Internet Systems Consortium, Inc. ("ISC")
5*00b67f09SDavid van Moolenbroek  * Copyright (C) 2000, 2001  Internet Software Consortium.
6*00b67f09SDavid van Moolenbroek  *
7*00b67f09SDavid van Moolenbroek  * Permission to use, copy, modify, and/or distribute this software for any
8*00b67f09SDavid van Moolenbroek  * purpose with or without fee is hereby granted, provided that the above
9*00b67f09SDavid van Moolenbroek  * copyright notice and this permission notice appear in all copies.
10*00b67f09SDavid van Moolenbroek  *
11*00b67f09SDavid van Moolenbroek  * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
12*00b67f09SDavid van Moolenbroek  * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
13*00b67f09SDavid van Moolenbroek  * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
14*00b67f09SDavid van Moolenbroek  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
15*00b67f09SDavid van Moolenbroek  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
16*00b67f09SDavid van Moolenbroek  * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
17*00b67f09SDavid van Moolenbroek  * PERFORMANCE OF THIS SOFTWARE.
18*00b67f09SDavid van Moolenbroek  */
19*00b67f09SDavid van Moolenbroek 
20*00b67f09SDavid van Moolenbroek /* Id: forward.c,v 1.14 2009/09/02 23:48:02 tbox Exp  */
21*00b67f09SDavid van Moolenbroek 
22*00b67f09SDavid van Moolenbroek /*! \file */
23*00b67f09SDavid van Moolenbroek 
24*00b67f09SDavid van Moolenbroek #include <config.h>
25*00b67f09SDavid van Moolenbroek 
26*00b67f09SDavid van Moolenbroek #include <isc/magic.h>
27*00b67f09SDavid van Moolenbroek #include <isc/mem.h>
28*00b67f09SDavid van Moolenbroek #include <isc/rwlock.h>
29*00b67f09SDavid van Moolenbroek #include <isc/sockaddr.h>
30*00b67f09SDavid van Moolenbroek #include <isc/util.h>
31*00b67f09SDavid van Moolenbroek 
32*00b67f09SDavid van Moolenbroek #include <dns/forward.h>
33*00b67f09SDavid van Moolenbroek #include <dns/rbt.h>
34*00b67f09SDavid van Moolenbroek #include <dns/result.h>
35*00b67f09SDavid van Moolenbroek #include <dns/types.h>
36*00b67f09SDavid van Moolenbroek 
37*00b67f09SDavid van Moolenbroek struct dns_fwdtable {
38*00b67f09SDavid van Moolenbroek 	/* Unlocked. */
39*00b67f09SDavid van Moolenbroek 	unsigned int		magic;
40*00b67f09SDavid van Moolenbroek 	isc_mem_t		*mctx;
41*00b67f09SDavid van Moolenbroek 	isc_rwlock_t		rwlock;
42*00b67f09SDavid van Moolenbroek 	/* Locked by lock. */
43*00b67f09SDavid van Moolenbroek 	dns_rbt_t		*table;
44*00b67f09SDavid van Moolenbroek };
45*00b67f09SDavid van Moolenbroek 
46*00b67f09SDavid van Moolenbroek #define FWDTABLEMAGIC		ISC_MAGIC('F', 'w', 'd', 'T')
47*00b67f09SDavid van Moolenbroek #define VALID_FWDTABLE(ft) 	ISC_MAGIC_VALID(ft, FWDTABLEMAGIC)
48*00b67f09SDavid van Moolenbroek 
49*00b67f09SDavid van Moolenbroek static void
50*00b67f09SDavid van Moolenbroek auto_detach(void *, void *);
51*00b67f09SDavid van Moolenbroek 
52*00b67f09SDavid van Moolenbroek isc_result_t
dns_fwdtable_create(isc_mem_t * mctx,dns_fwdtable_t ** fwdtablep)53*00b67f09SDavid van Moolenbroek dns_fwdtable_create(isc_mem_t *mctx, dns_fwdtable_t **fwdtablep) {
54*00b67f09SDavid van Moolenbroek 	dns_fwdtable_t *fwdtable;
55*00b67f09SDavid van Moolenbroek 	isc_result_t result;
56*00b67f09SDavid van Moolenbroek 
57*00b67f09SDavid van Moolenbroek 	REQUIRE(fwdtablep != NULL && *fwdtablep == NULL);
58*00b67f09SDavid van Moolenbroek 
59*00b67f09SDavid van Moolenbroek 	fwdtable = isc_mem_get(mctx, sizeof(dns_fwdtable_t));
60*00b67f09SDavid van Moolenbroek 	if (fwdtable == NULL)
61*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOMEMORY);
62*00b67f09SDavid van Moolenbroek 
63*00b67f09SDavid van Moolenbroek 	fwdtable->table = NULL;
64*00b67f09SDavid van Moolenbroek 	result = dns_rbt_create(mctx, auto_detach, fwdtable, &fwdtable->table);
65*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
66*00b67f09SDavid van Moolenbroek 		goto cleanup_fwdtable;
67*00b67f09SDavid van Moolenbroek 
68*00b67f09SDavid van Moolenbroek 	result = isc_rwlock_init(&fwdtable->rwlock, 0, 0);
69*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
70*00b67f09SDavid van Moolenbroek 		goto cleanup_rbt;
71*00b67f09SDavid van Moolenbroek 
72*00b67f09SDavid van Moolenbroek 	fwdtable->mctx = NULL;
73*00b67f09SDavid van Moolenbroek 	isc_mem_attach(mctx, &fwdtable->mctx);
74*00b67f09SDavid van Moolenbroek 	fwdtable->magic = FWDTABLEMAGIC;
75*00b67f09SDavid van Moolenbroek 	*fwdtablep = fwdtable;
76*00b67f09SDavid van Moolenbroek 
77*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
78*00b67f09SDavid van Moolenbroek 
79*00b67f09SDavid van Moolenbroek    cleanup_rbt:
80*00b67f09SDavid van Moolenbroek 	dns_rbt_destroy(&fwdtable->table);
81*00b67f09SDavid van Moolenbroek 
82*00b67f09SDavid van Moolenbroek    cleanup_fwdtable:
83*00b67f09SDavid van Moolenbroek 	isc_mem_put(mctx, fwdtable, sizeof(dns_fwdtable_t));
84*00b67f09SDavid van Moolenbroek 
85*00b67f09SDavid van Moolenbroek 	return (result);
86*00b67f09SDavid van Moolenbroek }
87*00b67f09SDavid van Moolenbroek 
88*00b67f09SDavid van Moolenbroek isc_result_t
dns_fwdtable_addfwd(dns_fwdtable_t * fwdtable,dns_name_t * name,dns_forwarderlist_t * fwdrs,dns_fwdpolicy_t fwdpolicy)89*00b67f09SDavid van Moolenbroek dns_fwdtable_addfwd(dns_fwdtable_t *fwdtable, dns_name_t *name,
90*00b67f09SDavid van Moolenbroek 		    dns_forwarderlist_t *fwdrs, dns_fwdpolicy_t fwdpolicy)
91*00b67f09SDavid van Moolenbroek {
92*00b67f09SDavid van Moolenbroek 	isc_result_t result;
93*00b67f09SDavid van Moolenbroek 	dns_forwarders_t *forwarders;
94*00b67f09SDavid van Moolenbroek 	dns_forwarder_t *fwd, *nfwd;
95*00b67f09SDavid van Moolenbroek 
96*00b67f09SDavid van Moolenbroek 	REQUIRE(VALID_FWDTABLE(fwdtable));
97*00b67f09SDavid van Moolenbroek 
98*00b67f09SDavid van Moolenbroek 	forwarders = isc_mem_get(fwdtable->mctx, sizeof(dns_forwarders_t));
99*00b67f09SDavid van Moolenbroek 	if (forwarders == NULL)
100*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOMEMORY);
101*00b67f09SDavid van Moolenbroek 
102*00b67f09SDavid van Moolenbroek 	ISC_LIST_INIT(forwarders->fwdrs);
103*00b67f09SDavid van Moolenbroek 	for (fwd = ISC_LIST_HEAD(*fwdrs);
104*00b67f09SDavid van Moolenbroek 	     fwd != NULL;
105*00b67f09SDavid van Moolenbroek 	     fwd = ISC_LIST_NEXT(fwd, link))
106*00b67f09SDavid van Moolenbroek 	{
107*00b67f09SDavid van Moolenbroek 		nfwd = isc_mem_get(fwdtable->mctx, sizeof(dns_forwarder_t));
108*00b67f09SDavid van Moolenbroek 		if (nfwd == NULL) {
109*00b67f09SDavid van Moolenbroek 			result = ISC_R_NOMEMORY;
110*00b67f09SDavid van Moolenbroek 			goto cleanup;
111*00b67f09SDavid van Moolenbroek 		}
112*00b67f09SDavid van Moolenbroek 		*nfwd = *fwd;
113*00b67f09SDavid van Moolenbroek 		ISC_LINK_INIT(nfwd, link);
114*00b67f09SDavid van Moolenbroek 		ISC_LIST_APPEND(forwarders->fwdrs, nfwd, link);
115*00b67f09SDavid van Moolenbroek 	}
116*00b67f09SDavid van Moolenbroek 	forwarders->fwdpolicy = fwdpolicy;
117*00b67f09SDavid van Moolenbroek 
118*00b67f09SDavid van Moolenbroek 	RWLOCK(&fwdtable->rwlock, isc_rwlocktype_write);
119*00b67f09SDavid van Moolenbroek 	result = dns_rbt_addname(fwdtable->table, name, forwarders);
120*00b67f09SDavid van Moolenbroek 	RWUNLOCK(&fwdtable->rwlock, isc_rwlocktype_write);
121*00b67f09SDavid van Moolenbroek 
122*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
123*00b67f09SDavid van Moolenbroek 		goto cleanup;
124*00b67f09SDavid van Moolenbroek 
125*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
126*00b67f09SDavid van Moolenbroek 
127*00b67f09SDavid van Moolenbroek  cleanup:
128*00b67f09SDavid van Moolenbroek 	while (!ISC_LIST_EMPTY(forwarders->fwdrs)) {
129*00b67f09SDavid van Moolenbroek 		fwd = ISC_LIST_HEAD(forwarders->fwdrs);
130*00b67f09SDavid van Moolenbroek 		ISC_LIST_UNLINK(forwarders->fwdrs, fwd, link);
131*00b67f09SDavid van Moolenbroek 		isc_mem_put(fwdtable->mctx, fwd, sizeof(isc_sockaddr_t));
132*00b67f09SDavid van Moolenbroek 	}
133*00b67f09SDavid van Moolenbroek 	isc_mem_put(fwdtable->mctx, forwarders, sizeof(dns_forwarders_t));
134*00b67f09SDavid van Moolenbroek 	return (result);
135*00b67f09SDavid van Moolenbroek }
136*00b67f09SDavid van Moolenbroek 
137*00b67f09SDavid van Moolenbroek isc_result_t
dns_fwdtable_add(dns_fwdtable_t * fwdtable,dns_name_t * name,isc_sockaddrlist_t * addrs,dns_fwdpolicy_t fwdpolicy)138*00b67f09SDavid van Moolenbroek dns_fwdtable_add(dns_fwdtable_t *fwdtable, dns_name_t *name,
139*00b67f09SDavid van Moolenbroek 		 isc_sockaddrlist_t *addrs, dns_fwdpolicy_t fwdpolicy)
140*00b67f09SDavid van Moolenbroek {
141*00b67f09SDavid van Moolenbroek 	isc_result_t result;
142*00b67f09SDavid van Moolenbroek 	dns_forwarders_t *forwarders;
143*00b67f09SDavid van Moolenbroek 	dns_forwarder_t *fwd;
144*00b67f09SDavid van Moolenbroek 	isc_sockaddr_t *sa;
145*00b67f09SDavid van Moolenbroek 
146*00b67f09SDavid van Moolenbroek 	REQUIRE(VALID_FWDTABLE(fwdtable));
147*00b67f09SDavid van Moolenbroek 
148*00b67f09SDavid van Moolenbroek 	forwarders = isc_mem_get(fwdtable->mctx, sizeof(dns_forwarders_t));
149*00b67f09SDavid van Moolenbroek 	if (forwarders == NULL)
150*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOMEMORY);
151*00b67f09SDavid van Moolenbroek 
152*00b67f09SDavid van Moolenbroek 	ISC_LIST_INIT(forwarders->fwdrs);
153*00b67f09SDavid van Moolenbroek 	for (sa = ISC_LIST_HEAD(*addrs);
154*00b67f09SDavid van Moolenbroek 	     sa != NULL;
155*00b67f09SDavid van Moolenbroek 	     sa = ISC_LIST_NEXT(sa, link))
156*00b67f09SDavid van Moolenbroek 	{
157*00b67f09SDavid van Moolenbroek 		fwd = isc_mem_get(fwdtable->mctx, sizeof(dns_forwarder_t));
158*00b67f09SDavid van Moolenbroek 		if (fwd == NULL) {
159*00b67f09SDavid van Moolenbroek 			result = ISC_R_NOMEMORY;
160*00b67f09SDavid van Moolenbroek 			goto cleanup;
161*00b67f09SDavid van Moolenbroek 		}
162*00b67f09SDavid van Moolenbroek 		fwd->addr = *sa;
163*00b67f09SDavid van Moolenbroek 		fwd->dscp = -1;
164*00b67f09SDavid van Moolenbroek 		ISC_LINK_INIT(fwd, link);
165*00b67f09SDavid van Moolenbroek 		ISC_LIST_APPEND(forwarders->fwdrs, fwd, link);
166*00b67f09SDavid van Moolenbroek 	}
167*00b67f09SDavid van Moolenbroek 	forwarders->fwdpolicy = fwdpolicy;
168*00b67f09SDavid van Moolenbroek 
169*00b67f09SDavid van Moolenbroek 	RWLOCK(&fwdtable->rwlock, isc_rwlocktype_write);
170*00b67f09SDavid van Moolenbroek 	result = dns_rbt_addname(fwdtable->table, name, forwarders);
171*00b67f09SDavid van Moolenbroek 	RWUNLOCK(&fwdtable->rwlock, isc_rwlocktype_write);
172*00b67f09SDavid van Moolenbroek 
173*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS)
174*00b67f09SDavid van Moolenbroek 		goto cleanup;
175*00b67f09SDavid van Moolenbroek 
176*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
177*00b67f09SDavid van Moolenbroek 
178*00b67f09SDavid van Moolenbroek  cleanup:
179*00b67f09SDavid van Moolenbroek 	while (!ISC_LIST_EMPTY(forwarders->fwdrs)) {
180*00b67f09SDavid van Moolenbroek 		fwd = ISC_LIST_HEAD(forwarders->fwdrs);
181*00b67f09SDavid van Moolenbroek 		ISC_LIST_UNLINK(forwarders->fwdrs, fwd, link);
182*00b67f09SDavid van Moolenbroek 		isc_mem_put(fwdtable->mctx, fwd, sizeof(dns_forwarder_t));
183*00b67f09SDavid van Moolenbroek 	}
184*00b67f09SDavid van Moolenbroek 	isc_mem_put(fwdtable->mctx, forwarders, sizeof(dns_forwarders_t));
185*00b67f09SDavid van Moolenbroek 	return (result);
186*00b67f09SDavid van Moolenbroek }
187*00b67f09SDavid van Moolenbroek 
188*00b67f09SDavid van Moolenbroek isc_result_t
dns_fwdtable_delete(dns_fwdtable_t * fwdtable,dns_name_t * name)189*00b67f09SDavid van Moolenbroek dns_fwdtable_delete(dns_fwdtable_t *fwdtable, dns_name_t *name) {
190*00b67f09SDavid van Moolenbroek 	isc_result_t result;
191*00b67f09SDavid van Moolenbroek 
192*00b67f09SDavid van Moolenbroek 	REQUIRE(VALID_FWDTABLE(fwdtable));
193*00b67f09SDavid van Moolenbroek 
194*00b67f09SDavid van Moolenbroek 	RWLOCK(&fwdtable->rwlock, isc_rwlocktype_write);
195*00b67f09SDavid van Moolenbroek 	result = dns_rbt_deletename(fwdtable->table, name, ISC_FALSE);
196*00b67f09SDavid van Moolenbroek 	RWUNLOCK(&fwdtable->rwlock, isc_rwlocktype_write);
197*00b67f09SDavid van Moolenbroek 
198*00b67f09SDavid van Moolenbroek 	if (result == DNS_R_PARTIALMATCH)
199*00b67f09SDavid van Moolenbroek 		result = ISC_R_NOTFOUND;
200*00b67f09SDavid van Moolenbroek 
201*00b67f09SDavid van Moolenbroek 	return (result);
202*00b67f09SDavid van Moolenbroek }
203*00b67f09SDavid van Moolenbroek 
204*00b67f09SDavid van Moolenbroek isc_result_t
dns_fwdtable_find(dns_fwdtable_t * fwdtable,dns_name_t * name,dns_forwarders_t ** forwardersp)205*00b67f09SDavid van Moolenbroek dns_fwdtable_find(dns_fwdtable_t *fwdtable, dns_name_t *name,
206*00b67f09SDavid van Moolenbroek 		  dns_forwarders_t **forwardersp)
207*00b67f09SDavid van Moolenbroek {
208*00b67f09SDavid van Moolenbroek 	return (dns_fwdtable_find2(fwdtable, name, NULL, forwardersp));
209*00b67f09SDavid van Moolenbroek }
210*00b67f09SDavid van Moolenbroek 
211*00b67f09SDavid van Moolenbroek isc_result_t
dns_fwdtable_find2(dns_fwdtable_t * fwdtable,dns_name_t * name,dns_name_t * foundname,dns_forwarders_t ** forwardersp)212*00b67f09SDavid van Moolenbroek dns_fwdtable_find2(dns_fwdtable_t *fwdtable, dns_name_t *name,
213*00b67f09SDavid van Moolenbroek 		   dns_name_t *foundname, dns_forwarders_t **forwardersp)
214*00b67f09SDavid van Moolenbroek {
215*00b67f09SDavid van Moolenbroek 	isc_result_t result;
216*00b67f09SDavid van Moolenbroek 
217*00b67f09SDavid van Moolenbroek 	REQUIRE(VALID_FWDTABLE(fwdtable));
218*00b67f09SDavid van Moolenbroek 
219*00b67f09SDavid van Moolenbroek 	RWLOCK(&fwdtable->rwlock, isc_rwlocktype_read);
220*00b67f09SDavid van Moolenbroek 
221*00b67f09SDavid van Moolenbroek 	result = dns_rbt_findname(fwdtable->table, name, 0, foundname,
222*00b67f09SDavid van Moolenbroek 				  (void **)forwardersp);
223*00b67f09SDavid van Moolenbroek 	if (result == DNS_R_PARTIALMATCH)
224*00b67f09SDavid van Moolenbroek 		result = ISC_R_SUCCESS;
225*00b67f09SDavid van Moolenbroek 
226*00b67f09SDavid van Moolenbroek 	RWUNLOCK(&fwdtable->rwlock, isc_rwlocktype_read);
227*00b67f09SDavid van Moolenbroek 
228*00b67f09SDavid van Moolenbroek 	return (result);
229*00b67f09SDavid van Moolenbroek }
230*00b67f09SDavid van Moolenbroek 
231*00b67f09SDavid van Moolenbroek void
dns_fwdtable_destroy(dns_fwdtable_t ** fwdtablep)232*00b67f09SDavid van Moolenbroek dns_fwdtable_destroy(dns_fwdtable_t **fwdtablep) {
233*00b67f09SDavid van Moolenbroek 	dns_fwdtable_t *fwdtable;
234*00b67f09SDavid van Moolenbroek 	isc_mem_t *mctx;
235*00b67f09SDavid van Moolenbroek 
236*00b67f09SDavid van Moolenbroek 	REQUIRE(fwdtablep != NULL && VALID_FWDTABLE(*fwdtablep));
237*00b67f09SDavid van Moolenbroek 
238*00b67f09SDavid van Moolenbroek 	fwdtable = *fwdtablep;
239*00b67f09SDavid van Moolenbroek 
240*00b67f09SDavid van Moolenbroek 	dns_rbt_destroy(&fwdtable->table);
241*00b67f09SDavid van Moolenbroek 	isc_rwlock_destroy(&fwdtable->rwlock);
242*00b67f09SDavid van Moolenbroek 	fwdtable->magic = 0;
243*00b67f09SDavid van Moolenbroek 	mctx = fwdtable->mctx;
244*00b67f09SDavid van Moolenbroek 	isc_mem_put(mctx, fwdtable, sizeof(dns_fwdtable_t));
245*00b67f09SDavid van Moolenbroek 	isc_mem_detach(&mctx);
246*00b67f09SDavid van Moolenbroek 
247*00b67f09SDavid van Moolenbroek 	*fwdtablep = NULL;
248*00b67f09SDavid van Moolenbroek }
249*00b67f09SDavid van Moolenbroek 
250*00b67f09SDavid van Moolenbroek /***
251*00b67f09SDavid van Moolenbroek  *** Private
252*00b67f09SDavid van Moolenbroek  ***/
253*00b67f09SDavid van Moolenbroek 
254*00b67f09SDavid van Moolenbroek static void
auto_detach(void * data,void * arg)255*00b67f09SDavid van Moolenbroek auto_detach(void *data, void *arg) {
256*00b67f09SDavid van Moolenbroek 	dns_forwarders_t *forwarders = data;
257*00b67f09SDavid van Moolenbroek 	dns_fwdtable_t *fwdtable = arg;
258*00b67f09SDavid van Moolenbroek 	dns_forwarder_t *fwd;
259*00b67f09SDavid van Moolenbroek 
260*00b67f09SDavid van Moolenbroek 	UNUSED(arg);
261*00b67f09SDavid van Moolenbroek 
262*00b67f09SDavid van Moolenbroek 	while (!ISC_LIST_EMPTY(forwarders->fwdrs)) {
263*00b67f09SDavid van Moolenbroek 		fwd = ISC_LIST_HEAD(forwarders->fwdrs);
264*00b67f09SDavid van Moolenbroek 		ISC_LIST_UNLINK(forwarders->fwdrs, fwd, link);
265*00b67f09SDavid van Moolenbroek 		isc_mem_put(fwdtable->mctx, fwd, sizeof(dns_forwarder_t));
266*00b67f09SDavid van Moolenbroek 	}
267*00b67f09SDavid van Moolenbroek 	isc_mem_put(fwdtable->mctx, forwarders, sizeof(dns_forwarders_t));
268*00b67f09SDavid van Moolenbroek }
269