1*00b67f09SDavid van Moolenbroek /* $NetBSD: order.c,v 1.4 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 Internet Systems Consortium, Inc. ("ISC")
5*00b67f09SDavid van Moolenbroek * Copyright (C) 2002 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: order.c,v 1.10 2007/06/19 23:47:16 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/types.h>
29*00b67f09SDavid van Moolenbroek #include <isc/util.h>
30*00b67f09SDavid van Moolenbroek #include <isc/refcount.h>
31*00b67f09SDavid van Moolenbroek
32*00b67f09SDavid van Moolenbroek #include <dns/fixedname.h>
33*00b67f09SDavid van Moolenbroek #include <dns/name.h>
34*00b67f09SDavid van Moolenbroek #include <dns/order.h>
35*00b67f09SDavid van Moolenbroek #include <dns/rdataset.h>
36*00b67f09SDavid van Moolenbroek #include <dns/types.h>
37*00b67f09SDavid van Moolenbroek
38*00b67f09SDavid van Moolenbroek typedef struct dns_order_ent dns_order_ent_t;
39*00b67f09SDavid van Moolenbroek struct dns_order_ent {
40*00b67f09SDavid van Moolenbroek dns_fixedname_t name;
41*00b67f09SDavid van Moolenbroek dns_rdataclass_t rdclass;
42*00b67f09SDavid van Moolenbroek dns_rdatatype_t rdtype;
43*00b67f09SDavid van Moolenbroek unsigned int mode;
44*00b67f09SDavid van Moolenbroek ISC_LINK(dns_order_ent_t) link;
45*00b67f09SDavid van Moolenbroek };
46*00b67f09SDavid van Moolenbroek
47*00b67f09SDavid van Moolenbroek struct dns_order {
48*00b67f09SDavid van Moolenbroek unsigned int magic;
49*00b67f09SDavid van Moolenbroek isc_refcount_t references;
50*00b67f09SDavid van Moolenbroek ISC_LIST(dns_order_ent_t) ents;
51*00b67f09SDavid van Moolenbroek isc_mem_t *mctx;
52*00b67f09SDavid van Moolenbroek };
53*00b67f09SDavid van Moolenbroek
54*00b67f09SDavid van Moolenbroek #define DNS_ORDER_MAGIC ISC_MAGIC('O','r','d','r')
55*00b67f09SDavid van Moolenbroek #define DNS_ORDER_VALID(order) ISC_MAGIC_VALID(order, DNS_ORDER_MAGIC)
56*00b67f09SDavid van Moolenbroek
57*00b67f09SDavid van Moolenbroek isc_result_t
dns_order_create(isc_mem_t * mctx,dns_order_t ** orderp)58*00b67f09SDavid van Moolenbroek dns_order_create(isc_mem_t *mctx, dns_order_t **orderp) {
59*00b67f09SDavid van Moolenbroek dns_order_t *order;
60*00b67f09SDavid van Moolenbroek isc_result_t result;
61*00b67f09SDavid van Moolenbroek
62*00b67f09SDavid van Moolenbroek REQUIRE(orderp != NULL && *orderp == NULL);
63*00b67f09SDavid van Moolenbroek
64*00b67f09SDavid van Moolenbroek order = isc_mem_get(mctx, sizeof(*order));
65*00b67f09SDavid van Moolenbroek if (order == NULL)
66*00b67f09SDavid van Moolenbroek return (ISC_R_NOMEMORY);
67*00b67f09SDavid van Moolenbroek
68*00b67f09SDavid van Moolenbroek ISC_LIST_INIT(order->ents);
69*00b67f09SDavid van Moolenbroek
70*00b67f09SDavid van Moolenbroek /* Implicit attach. */
71*00b67f09SDavid van Moolenbroek result = isc_refcount_init(&order->references, 1);
72*00b67f09SDavid van Moolenbroek if (result != ISC_R_SUCCESS) {
73*00b67f09SDavid van Moolenbroek isc_mem_put(mctx, order, sizeof(*order));
74*00b67f09SDavid van Moolenbroek return (result);
75*00b67f09SDavid van Moolenbroek }
76*00b67f09SDavid van Moolenbroek
77*00b67f09SDavid van Moolenbroek order->mctx = NULL;
78*00b67f09SDavid van Moolenbroek isc_mem_attach(mctx, &order->mctx);
79*00b67f09SDavid van Moolenbroek order->magic = DNS_ORDER_MAGIC;
80*00b67f09SDavid van Moolenbroek *orderp = order;
81*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
82*00b67f09SDavid van Moolenbroek }
83*00b67f09SDavid van Moolenbroek
84*00b67f09SDavid van Moolenbroek isc_result_t
dns_order_add(dns_order_t * order,dns_name_t * name,dns_rdatatype_t rdtype,dns_rdataclass_t rdclass,unsigned int mode)85*00b67f09SDavid van Moolenbroek dns_order_add(dns_order_t *order, dns_name_t *name,
86*00b67f09SDavid van Moolenbroek dns_rdatatype_t rdtype, dns_rdataclass_t rdclass,
87*00b67f09SDavid van Moolenbroek unsigned int mode)
88*00b67f09SDavid van Moolenbroek {
89*00b67f09SDavid van Moolenbroek dns_order_ent_t *ent;
90*00b67f09SDavid van Moolenbroek
91*00b67f09SDavid van Moolenbroek REQUIRE(DNS_ORDER_VALID(order));
92*00b67f09SDavid van Moolenbroek REQUIRE(mode == DNS_RDATASETATTR_RANDOMIZE ||
93*00b67f09SDavid van Moolenbroek mode == DNS_RDATASETATTR_FIXEDORDER ||
94*00b67f09SDavid van Moolenbroek mode == 0 /* DNS_RDATASETATTR_CYCLIC */ );
95*00b67f09SDavid van Moolenbroek
96*00b67f09SDavid van Moolenbroek ent = isc_mem_get(order->mctx, sizeof(*ent));
97*00b67f09SDavid van Moolenbroek if (ent == NULL)
98*00b67f09SDavid van Moolenbroek return (ISC_R_NOMEMORY);
99*00b67f09SDavid van Moolenbroek
100*00b67f09SDavid van Moolenbroek dns_fixedname_init(&ent->name);
101*00b67f09SDavid van Moolenbroek RUNTIME_CHECK(dns_name_copy(name, dns_fixedname_name(&ent->name), NULL)
102*00b67f09SDavid van Moolenbroek == ISC_R_SUCCESS);
103*00b67f09SDavid van Moolenbroek ent->rdtype = rdtype;
104*00b67f09SDavid van Moolenbroek ent->rdclass = rdclass;
105*00b67f09SDavid van Moolenbroek ent->mode = mode;
106*00b67f09SDavid van Moolenbroek ISC_LINK_INIT(ent, link);
107*00b67f09SDavid van Moolenbroek ISC_LIST_INITANDAPPEND(order->ents, ent, link);
108*00b67f09SDavid van Moolenbroek return (ISC_R_SUCCESS);
109*00b67f09SDavid van Moolenbroek }
110*00b67f09SDavid van Moolenbroek
111*00b67f09SDavid van Moolenbroek static inline isc_boolean_t
match(dns_name_t * name1,dns_name_t * name2)112*00b67f09SDavid van Moolenbroek match(dns_name_t *name1, dns_name_t *name2) {
113*00b67f09SDavid van Moolenbroek
114*00b67f09SDavid van Moolenbroek if (dns_name_iswildcard(name2))
115*00b67f09SDavid van Moolenbroek return(dns_name_matcheswildcard(name1, name2));
116*00b67f09SDavid van Moolenbroek return (dns_name_equal(name1, name2));
117*00b67f09SDavid van Moolenbroek }
118*00b67f09SDavid van Moolenbroek
119*00b67f09SDavid van Moolenbroek unsigned int
dns_order_find(dns_order_t * order,dns_name_t * name,dns_rdatatype_t rdtype,dns_rdataclass_t rdclass)120*00b67f09SDavid van Moolenbroek dns_order_find(dns_order_t *order, dns_name_t *name,
121*00b67f09SDavid van Moolenbroek dns_rdatatype_t rdtype, dns_rdataclass_t rdclass)
122*00b67f09SDavid van Moolenbroek {
123*00b67f09SDavid van Moolenbroek dns_order_ent_t *ent;
124*00b67f09SDavid van Moolenbroek REQUIRE(DNS_ORDER_VALID(order));
125*00b67f09SDavid van Moolenbroek
126*00b67f09SDavid van Moolenbroek for (ent = ISC_LIST_HEAD(order->ents);
127*00b67f09SDavid van Moolenbroek ent != NULL;
128*00b67f09SDavid van Moolenbroek ent = ISC_LIST_NEXT(ent, link)) {
129*00b67f09SDavid van Moolenbroek if (ent->rdtype != rdtype && ent->rdtype != dns_rdatatype_any)
130*00b67f09SDavid van Moolenbroek continue;
131*00b67f09SDavid van Moolenbroek if (ent->rdclass != rdclass &&
132*00b67f09SDavid van Moolenbroek ent->rdclass != dns_rdataclass_any)
133*00b67f09SDavid van Moolenbroek continue;
134*00b67f09SDavid van Moolenbroek if (match(name, dns_fixedname_name(&ent->name)))
135*00b67f09SDavid van Moolenbroek return (ent->mode);
136*00b67f09SDavid van Moolenbroek }
137*00b67f09SDavid van Moolenbroek return (0);
138*00b67f09SDavid van Moolenbroek }
139*00b67f09SDavid van Moolenbroek
140*00b67f09SDavid van Moolenbroek void
dns_order_attach(dns_order_t * source,dns_order_t ** target)141*00b67f09SDavid van Moolenbroek dns_order_attach(dns_order_t *source, dns_order_t **target) {
142*00b67f09SDavid van Moolenbroek REQUIRE(DNS_ORDER_VALID(source));
143*00b67f09SDavid van Moolenbroek REQUIRE(target != NULL && *target == NULL);
144*00b67f09SDavid van Moolenbroek isc_refcount_increment(&source->references, NULL);
145*00b67f09SDavid van Moolenbroek *target = source;
146*00b67f09SDavid van Moolenbroek }
147*00b67f09SDavid van Moolenbroek
148*00b67f09SDavid van Moolenbroek void
dns_order_detach(dns_order_t ** orderp)149*00b67f09SDavid van Moolenbroek dns_order_detach(dns_order_t **orderp) {
150*00b67f09SDavid van Moolenbroek dns_order_t *order;
151*00b67f09SDavid van Moolenbroek dns_order_ent_t *ent;
152*00b67f09SDavid van Moolenbroek unsigned int references;
153*00b67f09SDavid van Moolenbroek
154*00b67f09SDavid van Moolenbroek REQUIRE(orderp != NULL);
155*00b67f09SDavid van Moolenbroek order = *orderp;
156*00b67f09SDavid van Moolenbroek REQUIRE(DNS_ORDER_VALID(order));
157*00b67f09SDavid van Moolenbroek isc_refcount_decrement(&order->references, &references);
158*00b67f09SDavid van Moolenbroek *orderp = NULL;
159*00b67f09SDavid van Moolenbroek if (references != 0)
160*00b67f09SDavid van Moolenbroek return;
161*00b67f09SDavid van Moolenbroek
162*00b67f09SDavid van Moolenbroek order->magic = 0;
163*00b67f09SDavid van Moolenbroek while ((ent = ISC_LIST_HEAD(order->ents)) != NULL) {
164*00b67f09SDavid van Moolenbroek ISC_LIST_UNLINK(order->ents, ent, link);
165*00b67f09SDavid van Moolenbroek isc_mem_put(order->mctx, ent, sizeof(*ent));
166*00b67f09SDavid van Moolenbroek }
167*00b67f09SDavid van Moolenbroek isc_refcount_destroy(&order->references);
168*00b67f09SDavid van Moolenbroek isc_mem_putanddetach(&order->mctx, order, sizeof(*order));
169*00b67f09SDavid van Moolenbroek }
170