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