xref: /minix3/external/bsd/bind/dist/lib/dns/rdataslab.c (revision 00b67f09dd46474d133c95011a48590a8e8f94c7)
1*00b67f09SDavid van Moolenbroek /*	$NetBSD: rdataslab.c,v 1.11 2015/07/08 17:28:59 christos Exp $	*/
2*00b67f09SDavid van Moolenbroek 
3*00b67f09SDavid van Moolenbroek /*
4*00b67f09SDavid van Moolenbroek  * Copyright (C) 2004-2014  Internet Systems Consortium, Inc. ("ISC")
5*00b67f09SDavid van Moolenbroek  * Copyright (C) 1999-2003  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 */
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 <stdlib.h>
27*00b67f09SDavid van Moolenbroek 
28*00b67f09SDavid van Moolenbroek #include <isc/mem.h>
29*00b67f09SDavid van Moolenbroek #include <isc/region.h>
30*00b67f09SDavid van Moolenbroek #include <isc/string.h>		/* Required for HP/UX (and others?) */
31*00b67f09SDavid van Moolenbroek #include <isc/util.h>
32*00b67f09SDavid van Moolenbroek 
33*00b67f09SDavid van Moolenbroek #include <dns/result.h>
34*00b67f09SDavid van Moolenbroek #include <dns/rdata.h>
35*00b67f09SDavid van Moolenbroek #include <dns/rdataset.h>
36*00b67f09SDavid van Moolenbroek #include <dns/rdataslab.h>
37*00b67f09SDavid van Moolenbroek 
38*00b67f09SDavid van Moolenbroek /*
39*00b67f09SDavid van Moolenbroek  * The rdataslab structure allows iteration to occur in both load order
40*00b67f09SDavid van Moolenbroek  * and DNSSEC order.  The structure is as follows:
41*00b67f09SDavid van Moolenbroek  *
42*00b67f09SDavid van Moolenbroek  *	header		(reservelen bytes)
43*00b67f09SDavid van Moolenbroek  *	record count	(2 bytes)
44*00b67f09SDavid van Moolenbroek  *	offset table	(4 x record count bytes in load order)
45*00b67f09SDavid van Moolenbroek  *	data records
46*00b67f09SDavid van Moolenbroek  *		data length	(2 bytes)
47*00b67f09SDavid van Moolenbroek  *		order		(2 bytes)
48*00b67f09SDavid van Moolenbroek  *		meta data	(1 byte for RRSIG's)
49*00b67f09SDavid van Moolenbroek  *		data		(data length bytes)
50*00b67f09SDavid van Moolenbroek  *
51*00b67f09SDavid van Moolenbroek  * If DNS_RDATASET_FIXED is defined to be zero (0) the format of a
52*00b67f09SDavid van Moolenbroek  * rdataslab is as follows:
53*00b67f09SDavid van Moolenbroek  *
54*00b67f09SDavid van Moolenbroek  *	header		(reservelen bytes)
55*00b67f09SDavid van Moolenbroek  *	record count	(2 bytes)
56*00b67f09SDavid van Moolenbroek  *	data records
57*00b67f09SDavid van Moolenbroek  *		data length	(2 bytes)
58*00b67f09SDavid van Moolenbroek  *		meta data	(1 byte for RRSIG's)
59*00b67f09SDavid van Moolenbroek  *		data		(data length bytes)
60*00b67f09SDavid van Moolenbroek  *
61*00b67f09SDavid van Moolenbroek  * Offsets are from the end of the header.
62*00b67f09SDavid van Moolenbroek  *
63*00b67f09SDavid van Moolenbroek  * Load order traversal is performed by walking the offset table to find
64*00b67f09SDavid van Moolenbroek  * the start of the record (DNS_RDATASET_FIXED = 1).
65*00b67f09SDavid van Moolenbroek  *
66*00b67f09SDavid van Moolenbroek  * DNSSEC order traversal is performed by walking the data records.
67*00b67f09SDavid van Moolenbroek  *
68*00b67f09SDavid van Moolenbroek  * The order is stored with record to allow for efficient reconstruction
69*00b67f09SDavid van Moolenbroek  * of the offset table following a merge or subtraction.
70*00b67f09SDavid van Moolenbroek  *
71*00b67f09SDavid van Moolenbroek  * The iterator methods here currently only support DNSSEC order iteration.
72*00b67f09SDavid van Moolenbroek  *
73*00b67f09SDavid van Moolenbroek  * The iterator methods in rbtdb support both load order and DNSSEC order
74*00b67f09SDavid van Moolenbroek  * iteration.
75*00b67f09SDavid van Moolenbroek  *
76*00b67f09SDavid van Moolenbroek  * WARNING:
77*00b67f09SDavid van Moolenbroek  *	rbtdb.c directly interacts with the slab's raw structures.  If the
78*00b67f09SDavid van Moolenbroek  *	structure changes then rbtdb.c also needs to be updated to reflect
79*00b67f09SDavid van Moolenbroek  *	the changes.  See the areas tagged with "RDATASLAB".
80*00b67f09SDavid van Moolenbroek  */
81*00b67f09SDavid van Moolenbroek 
82*00b67f09SDavid van Moolenbroek struct xrdata {
83*00b67f09SDavid van Moolenbroek 	dns_rdata_t	rdata;
84*00b67f09SDavid van Moolenbroek 	unsigned int	order;
85*00b67f09SDavid van Moolenbroek };
86*00b67f09SDavid van Moolenbroek 
87*00b67f09SDavid van Moolenbroek /*% Note: the "const void *" are just to make qsort happy.  */
88*00b67f09SDavid van Moolenbroek static int
compare_rdata(const void * p1,const void * p2)89*00b67f09SDavid van Moolenbroek compare_rdata(const void *p1, const void *p2) {
90*00b67f09SDavid van Moolenbroek 	const struct xrdata *x1 = p1;
91*00b67f09SDavid van Moolenbroek 	const struct xrdata *x2 = p2;
92*00b67f09SDavid van Moolenbroek 	return (dns_rdata_compare(&x1->rdata, &x2->rdata));
93*00b67f09SDavid van Moolenbroek }
94*00b67f09SDavid van Moolenbroek 
95*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
96*00b67f09SDavid van Moolenbroek static void
fillin_offsets(unsigned char * offsetbase,unsigned int * offsettable,unsigned length)97*00b67f09SDavid van Moolenbroek fillin_offsets(unsigned char *offsetbase, unsigned int *offsettable,
98*00b67f09SDavid van Moolenbroek 	       unsigned length)
99*00b67f09SDavid van Moolenbroek {
100*00b67f09SDavid van Moolenbroek 	unsigned int i, j;
101*00b67f09SDavid van Moolenbroek 	unsigned char *raw;
102*00b67f09SDavid van Moolenbroek 
103*00b67f09SDavid van Moolenbroek 	for (i = 0, j = 0; i < length; i++) {
104*00b67f09SDavid van Moolenbroek 
105*00b67f09SDavid van Moolenbroek 		if (offsettable[i] == 0)
106*00b67f09SDavid van Moolenbroek 			continue;
107*00b67f09SDavid van Moolenbroek 
108*00b67f09SDavid van Moolenbroek 		/*
109*00b67f09SDavid van Moolenbroek 		 * Fill in offset table.
110*00b67f09SDavid van Moolenbroek 		 */
111*00b67f09SDavid van Moolenbroek 		raw = &offsetbase[j*4 + 2];
112*00b67f09SDavid van Moolenbroek 		*raw++ = (offsettable[i] & 0xff000000) >> 24;
113*00b67f09SDavid van Moolenbroek 		*raw++ = (offsettable[i] & 0xff0000) >> 16;
114*00b67f09SDavid van Moolenbroek 		*raw++ = (offsettable[i] & 0xff00) >> 8;
115*00b67f09SDavid van Moolenbroek 		*raw = offsettable[i] & 0xff;
116*00b67f09SDavid van Moolenbroek 
117*00b67f09SDavid van Moolenbroek 		/*
118*00b67f09SDavid van Moolenbroek 		 * Fill in table index.
119*00b67f09SDavid van Moolenbroek 		 */
120*00b67f09SDavid van Moolenbroek 		raw = offsetbase + offsettable[i] + 2;
121*00b67f09SDavid van Moolenbroek 		*raw++ = (j & 0xff00) >> 8;
122*00b67f09SDavid van Moolenbroek 		*raw = j++ & 0xff;
123*00b67f09SDavid van Moolenbroek 	}
124*00b67f09SDavid van Moolenbroek }
125*00b67f09SDavid van Moolenbroek #endif
126*00b67f09SDavid van Moolenbroek 
127*00b67f09SDavid van Moolenbroek isc_result_t
dns_rdataslab_fromrdataset(dns_rdataset_t * rdataset,isc_mem_t * mctx,isc_region_t * region,unsigned int reservelen)128*00b67f09SDavid van Moolenbroek dns_rdataslab_fromrdataset(dns_rdataset_t *rdataset, isc_mem_t *mctx,
129*00b67f09SDavid van Moolenbroek 			   isc_region_t *region, unsigned int reservelen)
130*00b67f09SDavid van Moolenbroek {
131*00b67f09SDavid van Moolenbroek 	/*
132*00b67f09SDavid van Moolenbroek 	 * Use &removed as a sentinal pointer for duplicate
133*00b67f09SDavid van Moolenbroek 	 * rdata as rdata.data == NULL is valid.
134*00b67f09SDavid van Moolenbroek 	 */
135*00b67f09SDavid van Moolenbroek 	static unsigned char removed;
136*00b67f09SDavid van Moolenbroek 	struct xrdata  *x;
137*00b67f09SDavid van Moolenbroek 	unsigned char  *rawbuf;
138*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
139*00b67f09SDavid van Moolenbroek 	unsigned char  *offsetbase;
140*00b67f09SDavid van Moolenbroek #endif
141*00b67f09SDavid van Moolenbroek 	unsigned int	buflen;
142*00b67f09SDavid van Moolenbroek 	isc_result_t	result;
143*00b67f09SDavid van Moolenbroek 	unsigned int	nitems;
144*00b67f09SDavid van Moolenbroek 	unsigned int	nalloc;
145*00b67f09SDavid van Moolenbroek 	unsigned int	i;
146*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
147*00b67f09SDavid van Moolenbroek 	unsigned int   *offsettable;
148*00b67f09SDavid van Moolenbroek #endif
149*00b67f09SDavid van Moolenbroek 	unsigned int	length;
150*00b67f09SDavid van Moolenbroek 
151*00b67f09SDavid van Moolenbroek 	buflen = reservelen + 2;
152*00b67f09SDavid van Moolenbroek 
153*00b67f09SDavid van Moolenbroek 	nitems = dns_rdataset_count(rdataset);
154*00b67f09SDavid van Moolenbroek 
155*00b67f09SDavid van Moolenbroek 	/*
156*00b67f09SDavid van Moolenbroek 	 * If there are no rdata then we can just need to allocate a header
157*00b67f09SDavid van Moolenbroek 	 * with zero a record count.
158*00b67f09SDavid van Moolenbroek 	 */
159*00b67f09SDavid van Moolenbroek 	if (nitems == 0) {
160*00b67f09SDavid van Moolenbroek 		if (rdataset->type != 0)
161*00b67f09SDavid van Moolenbroek 			return (ISC_R_FAILURE);
162*00b67f09SDavid van Moolenbroek 		rawbuf = isc_mem_get(mctx, buflen);
163*00b67f09SDavid van Moolenbroek 		if (rawbuf == NULL)
164*00b67f09SDavid van Moolenbroek 			return (ISC_R_NOMEMORY);
165*00b67f09SDavid van Moolenbroek 		region->base = rawbuf;
166*00b67f09SDavid van Moolenbroek 		region->length = buflen;
167*00b67f09SDavid van Moolenbroek 		rawbuf += reservelen;
168*00b67f09SDavid van Moolenbroek 		*rawbuf++ = 0;
169*00b67f09SDavid van Moolenbroek 		*rawbuf = 0;
170*00b67f09SDavid van Moolenbroek 		return (ISC_R_SUCCESS);
171*00b67f09SDavid van Moolenbroek 	}
172*00b67f09SDavid van Moolenbroek 
173*00b67f09SDavid van Moolenbroek 	if (nitems > 0xffff)
174*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOSPACE);
175*00b67f09SDavid van Moolenbroek 
176*00b67f09SDavid van Moolenbroek 	/*
177*00b67f09SDavid van Moolenbroek 	 * Remember the original number of items.
178*00b67f09SDavid van Moolenbroek 	 */
179*00b67f09SDavid van Moolenbroek 	nalloc = nitems;
180*00b67f09SDavid van Moolenbroek 	x = isc_mem_get(mctx, nalloc * sizeof(struct xrdata));
181*00b67f09SDavid van Moolenbroek 	if (x == NULL)
182*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOMEMORY);
183*00b67f09SDavid van Moolenbroek 
184*00b67f09SDavid van Moolenbroek 	/*
185*00b67f09SDavid van Moolenbroek 	 * Save all of the rdata members into an array.
186*00b67f09SDavid van Moolenbroek 	 */
187*00b67f09SDavid van Moolenbroek 	result = dns_rdataset_first(rdataset);
188*00b67f09SDavid van Moolenbroek 	if (result != ISC_R_SUCCESS && result != ISC_R_NOMORE)
189*00b67f09SDavid van Moolenbroek 		goto free_rdatas;
190*00b67f09SDavid van Moolenbroek 	for (i = 0; i < nalloc && result == ISC_R_SUCCESS; i++) {
191*00b67f09SDavid van Moolenbroek 		INSIST(result == ISC_R_SUCCESS);
192*00b67f09SDavid van Moolenbroek 		dns_rdata_init(&x[i].rdata);
193*00b67f09SDavid van Moolenbroek 		dns_rdataset_current(rdataset, &x[i].rdata);
194*00b67f09SDavid van Moolenbroek 		INSIST(x[i].rdata.data != &removed);
195*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
196*00b67f09SDavid van Moolenbroek 		x[i].order = i;
197*00b67f09SDavid van Moolenbroek #endif
198*00b67f09SDavid van Moolenbroek 		result = dns_rdataset_next(rdataset);
199*00b67f09SDavid van Moolenbroek 	}
200*00b67f09SDavid van Moolenbroek 	if (i != nalloc || result != ISC_R_NOMORE) {
201*00b67f09SDavid van Moolenbroek 		/*
202*00b67f09SDavid van Moolenbroek 		 * Somehow we iterated over fewer rdatas than
203*00b67f09SDavid van Moolenbroek 		 * dns_rdataset_count() said there were or there
204*00b67f09SDavid van Moolenbroek 		 * were more items than dns_rdataset_count said
205*00b67f09SDavid van Moolenbroek 		 * there were.
206*00b67f09SDavid van Moolenbroek 		 */
207*00b67f09SDavid van Moolenbroek 		result = ISC_R_FAILURE;
208*00b67f09SDavid van Moolenbroek 		goto free_rdatas;
209*00b67f09SDavid van Moolenbroek 	}
210*00b67f09SDavid van Moolenbroek 
211*00b67f09SDavid van Moolenbroek 	/*
212*00b67f09SDavid van Moolenbroek 	 * Put into DNSSEC order.
213*00b67f09SDavid van Moolenbroek 	 */
214*00b67f09SDavid van Moolenbroek 	if (nalloc > 1U)
215*00b67f09SDavid van Moolenbroek 		qsort(x, nalloc, sizeof(struct xrdata), compare_rdata);
216*00b67f09SDavid van Moolenbroek 
217*00b67f09SDavid van Moolenbroek 	/*
218*00b67f09SDavid van Moolenbroek 	 * Remove duplicates and compute the total storage required.
219*00b67f09SDavid van Moolenbroek 	 *
220*00b67f09SDavid van Moolenbroek 	 * If an rdata is not a duplicate, accumulate the storage size
221*00b67f09SDavid van Moolenbroek 	 * required for the rdata.  We do not store the class, type, etc,
222*00b67f09SDavid van Moolenbroek 	 * just the rdata, so our overhead is 2 bytes for the number of
223*00b67f09SDavid van Moolenbroek 	 * records, and 8 for each rdata, (length(2), offset(4) and order(2))
224*00b67f09SDavid van Moolenbroek 	 * and then the rdata itself.
225*00b67f09SDavid van Moolenbroek 	 */
226*00b67f09SDavid van Moolenbroek 	for (i = 1; i < nalloc; i++) {
227*00b67f09SDavid van Moolenbroek 		if (compare_rdata(&x[i-1].rdata, &x[i].rdata) == 0) {
228*00b67f09SDavid van Moolenbroek 			x[i-1].rdata.data = &removed;
229*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
230*00b67f09SDavid van Moolenbroek 			/*
231*00b67f09SDavid van Moolenbroek 			 * Preserve the least order so A, B, A -> A, B
232*00b67f09SDavid van Moolenbroek 			 * after duplicate removal.
233*00b67f09SDavid van Moolenbroek 			 */
234*00b67f09SDavid van Moolenbroek 			if (x[i-1].order < x[i].order)
235*00b67f09SDavid van Moolenbroek 				x[i].order = x[i-1].order;
236*00b67f09SDavid van Moolenbroek #endif
237*00b67f09SDavid van Moolenbroek 			nitems--;
238*00b67f09SDavid van Moolenbroek 		} else {
239*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
240*00b67f09SDavid van Moolenbroek 			buflen += (8 + x[i-1].rdata.length);
241*00b67f09SDavid van Moolenbroek #else
242*00b67f09SDavid van Moolenbroek 			buflen += (2 + x[i-1].rdata.length);
243*00b67f09SDavid van Moolenbroek #endif
244*00b67f09SDavid van Moolenbroek 			/*
245*00b67f09SDavid van Moolenbroek 			 * Provide space to store the per RR meta data.
246*00b67f09SDavid van Moolenbroek 			 */
247*00b67f09SDavid van Moolenbroek 			if (rdataset->type == dns_rdatatype_rrsig)
248*00b67f09SDavid van Moolenbroek 				buflen++;
249*00b67f09SDavid van Moolenbroek 		}
250*00b67f09SDavid van Moolenbroek 	}
251*00b67f09SDavid van Moolenbroek 
252*00b67f09SDavid van Moolenbroek 	/*
253*00b67f09SDavid van Moolenbroek 	 * Don't forget the last item!
254*00b67f09SDavid van Moolenbroek 	 */
255*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
256*00b67f09SDavid van Moolenbroek 	buflen += (8 + x[i-1].rdata.length);
257*00b67f09SDavid van Moolenbroek #else
258*00b67f09SDavid van Moolenbroek 	buflen += (2 + x[i-1].rdata.length);
259*00b67f09SDavid van Moolenbroek #endif
260*00b67f09SDavid van Moolenbroek 	/*
261*00b67f09SDavid van Moolenbroek 	 * Provide space to store the per RR meta data.
262*00b67f09SDavid van Moolenbroek 	 */
263*00b67f09SDavid van Moolenbroek 	if (rdataset->type == dns_rdatatype_rrsig)
264*00b67f09SDavid van Moolenbroek 		buflen++;
265*00b67f09SDavid van Moolenbroek 
266*00b67f09SDavid van Moolenbroek 	/*
267*00b67f09SDavid van Moolenbroek 	 * Ensure that singleton types are actually singletons.
268*00b67f09SDavid van Moolenbroek 	 */
269*00b67f09SDavid van Moolenbroek 	if (nitems > 1 && dns_rdatatype_issingleton(rdataset->type)) {
270*00b67f09SDavid van Moolenbroek 		/*
271*00b67f09SDavid van Moolenbroek 		 * We have a singleton type, but there's more than one
272*00b67f09SDavid van Moolenbroek 		 * RR in the rdataset.
273*00b67f09SDavid van Moolenbroek 		 */
274*00b67f09SDavid van Moolenbroek 		result = DNS_R_SINGLETON;
275*00b67f09SDavid van Moolenbroek 		goto free_rdatas;
276*00b67f09SDavid van Moolenbroek 	}
277*00b67f09SDavid van Moolenbroek 
278*00b67f09SDavid van Moolenbroek 	/*
279*00b67f09SDavid van Moolenbroek 	 * Allocate the memory, set up a buffer, start copying in
280*00b67f09SDavid van Moolenbroek 	 * data.
281*00b67f09SDavid van Moolenbroek 	 */
282*00b67f09SDavid van Moolenbroek 	rawbuf = isc_mem_get(mctx, buflen);
283*00b67f09SDavid van Moolenbroek 	if (rawbuf == NULL) {
284*00b67f09SDavid van Moolenbroek 		result = ISC_R_NOMEMORY;
285*00b67f09SDavid van Moolenbroek 		goto free_rdatas;
286*00b67f09SDavid van Moolenbroek 	}
287*00b67f09SDavid van Moolenbroek 
288*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
289*00b67f09SDavid van Moolenbroek 	/* Allocate temporary offset table. */
290*00b67f09SDavid van Moolenbroek 	offsettable = isc_mem_get(mctx, nalloc * sizeof(unsigned int));
291*00b67f09SDavid van Moolenbroek 	if (offsettable == NULL) {
292*00b67f09SDavid van Moolenbroek 		isc_mem_put(mctx, rawbuf, buflen);
293*00b67f09SDavid van Moolenbroek 		result = ISC_R_NOMEMORY;
294*00b67f09SDavid van Moolenbroek 		goto free_rdatas;
295*00b67f09SDavid van Moolenbroek 	}
296*00b67f09SDavid van Moolenbroek 	memset(offsettable, 0, nalloc * sizeof(unsigned int));
297*00b67f09SDavid van Moolenbroek #endif
298*00b67f09SDavid van Moolenbroek 
299*00b67f09SDavid van Moolenbroek 	region->base = rawbuf;
300*00b67f09SDavid van Moolenbroek 	region->length = buflen;
301*00b67f09SDavid van Moolenbroek 
302*00b67f09SDavid van Moolenbroek 	rawbuf += reservelen;
303*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
304*00b67f09SDavid van Moolenbroek 	offsetbase = rawbuf;
305*00b67f09SDavid van Moolenbroek #endif
306*00b67f09SDavid van Moolenbroek 
307*00b67f09SDavid van Moolenbroek 	*rawbuf++ = (nitems & 0xff00) >> 8;
308*00b67f09SDavid van Moolenbroek 	*rawbuf++ = (nitems & 0x00ff);
309*00b67f09SDavid van Moolenbroek 
310*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
311*00b67f09SDavid van Moolenbroek 	/* Skip load order table.  Filled in later. */
312*00b67f09SDavid van Moolenbroek 	rawbuf += nitems * 4;
313*00b67f09SDavid van Moolenbroek #endif
314*00b67f09SDavid van Moolenbroek 
315*00b67f09SDavid van Moolenbroek 	for (i = 0; i < nalloc; i++) {
316*00b67f09SDavid van Moolenbroek 		if (x[i].rdata.data == &removed)
317*00b67f09SDavid van Moolenbroek 			continue;
318*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
319*00b67f09SDavid van Moolenbroek 		offsettable[x[i].order] = rawbuf - offsetbase;
320*00b67f09SDavid van Moolenbroek #endif
321*00b67f09SDavid van Moolenbroek 		length = x[i].rdata.length;
322*00b67f09SDavid van Moolenbroek 		if (rdataset->type == dns_rdatatype_rrsig)
323*00b67f09SDavid van Moolenbroek 			length++;
324*00b67f09SDavid van Moolenbroek 		INSIST(length <= 0xffff);
325*00b67f09SDavid van Moolenbroek 		*rawbuf++ = (length & 0xff00) >> 8;
326*00b67f09SDavid van Moolenbroek 		*rawbuf++ = (length & 0x00ff);
327*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
328*00b67f09SDavid van Moolenbroek 		rawbuf += 2;	/* filled in later */
329*00b67f09SDavid van Moolenbroek #endif
330*00b67f09SDavid van Moolenbroek 		/*
331*00b67f09SDavid van Moolenbroek 		 * Store the per RR meta data.
332*00b67f09SDavid van Moolenbroek 		 */
333*00b67f09SDavid van Moolenbroek 		if (rdataset->type == dns_rdatatype_rrsig) {
334*00b67f09SDavid van Moolenbroek 			*rawbuf++ |= (x[i].rdata.flags & DNS_RDATA_OFFLINE) ?
335*00b67f09SDavid van Moolenbroek 					    DNS_RDATASLAB_OFFLINE : 0;
336*00b67f09SDavid van Moolenbroek 		}
337*00b67f09SDavid van Moolenbroek 		memmove(rawbuf, x[i].rdata.data, x[i].rdata.length);
338*00b67f09SDavid van Moolenbroek 		rawbuf += x[i].rdata.length;
339*00b67f09SDavid van Moolenbroek 	}
340*00b67f09SDavid van Moolenbroek 
341*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
342*00b67f09SDavid van Moolenbroek 	fillin_offsets(offsetbase, offsettable, nalloc);
343*00b67f09SDavid van Moolenbroek 	isc_mem_put(mctx, offsettable, nalloc * sizeof(unsigned int));
344*00b67f09SDavid van Moolenbroek #endif
345*00b67f09SDavid van Moolenbroek 
346*00b67f09SDavid van Moolenbroek 	result = ISC_R_SUCCESS;
347*00b67f09SDavid van Moolenbroek 
348*00b67f09SDavid van Moolenbroek  free_rdatas:
349*00b67f09SDavid van Moolenbroek 	isc_mem_put(mctx, x, nalloc * sizeof(struct xrdata));
350*00b67f09SDavid van Moolenbroek 	return (result);
351*00b67f09SDavid van Moolenbroek }
352*00b67f09SDavid van Moolenbroek 
353*00b67f09SDavid van Moolenbroek static void
rdataset_disassociate(dns_rdataset_t * rdataset)354*00b67f09SDavid van Moolenbroek rdataset_disassociate(dns_rdataset_t *rdataset) {
355*00b67f09SDavid van Moolenbroek 	UNUSED(rdataset);
356*00b67f09SDavid van Moolenbroek }
357*00b67f09SDavid van Moolenbroek 
358*00b67f09SDavid van Moolenbroek static isc_result_t
rdataset_first(dns_rdataset_t * rdataset)359*00b67f09SDavid van Moolenbroek rdataset_first(dns_rdataset_t *rdataset) {
360*00b67f09SDavid van Moolenbroek 	unsigned char *raw = rdataset->private3;
361*00b67f09SDavid van Moolenbroek 	unsigned int count;
362*00b67f09SDavid van Moolenbroek 
363*00b67f09SDavid van Moolenbroek 	count = raw[0] * 256 + raw[1];
364*00b67f09SDavid van Moolenbroek 	if (count == 0) {
365*00b67f09SDavid van Moolenbroek 		rdataset->private5 = NULL;
366*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOMORE);
367*00b67f09SDavid van Moolenbroek 	}
368*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
369*00b67f09SDavid van Moolenbroek 	raw += 2 + (4 * count);
370*00b67f09SDavid van Moolenbroek #else
371*00b67f09SDavid van Moolenbroek 	raw += 2;
372*00b67f09SDavid van Moolenbroek #endif
373*00b67f09SDavid van Moolenbroek 	/*
374*00b67f09SDavid van Moolenbroek 	 * The privateuint4 field is the number of rdata beyond the cursor
375*00b67f09SDavid van Moolenbroek 	 * position, so we decrement the total count by one before storing
376*00b67f09SDavid van Moolenbroek 	 * it.
377*00b67f09SDavid van Moolenbroek 	 */
378*00b67f09SDavid van Moolenbroek 	count--;
379*00b67f09SDavid van Moolenbroek 	rdataset->privateuint4 = count;
380*00b67f09SDavid van Moolenbroek 	rdataset->private5 = raw;
381*00b67f09SDavid van Moolenbroek 
382*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
383*00b67f09SDavid van Moolenbroek }
384*00b67f09SDavid van Moolenbroek 
385*00b67f09SDavid van Moolenbroek static isc_result_t
rdataset_next(dns_rdataset_t * rdataset)386*00b67f09SDavid van Moolenbroek rdataset_next(dns_rdataset_t *rdataset) {
387*00b67f09SDavid van Moolenbroek 	unsigned int count;
388*00b67f09SDavid van Moolenbroek 	unsigned int length;
389*00b67f09SDavid van Moolenbroek 	unsigned char *raw;
390*00b67f09SDavid van Moolenbroek 
391*00b67f09SDavid van Moolenbroek 	count = rdataset->privateuint4;
392*00b67f09SDavid van Moolenbroek 	if (count == 0)
393*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOMORE);
394*00b67f09SDavid van Moolenbroek 	count--;
395*00b67f09SDavid van Moolenbroek 	rdataset->privateuint4 = count;
396*00b67f09SDavid van Moolenbroek 	raw = rdataset->private5;
397*00b67f09SDavid van Moolenbroek 	length = raw[0] * 256 + raw[1];
398*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
399*00b67f09SDavid van Moolenbroek 	raw += length + 4;
400*00b67f09SDavid van Moolenbroek #else
401*00b67f09SDavid van Moolenbroek 	raw += length + 2;
402*00b67f09SDavid van Moolenbroek #endif
403*00b67f09SDavid van Moolenbroek 	rdataset->private5 = raw;
404*00b67f09SDavid van Moolenbroek 
405*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
406*00b67f09SDavid van Moolenbroek }
407*00b67f09SDavid van Moolenbroek 
408*00b67f09SDavid van Moolenbroek static void
rdataset_current(dns_rdataset_t * rdataset,dns_rdata_t * rdata)409*00b67f09SDavid van Moolenbroek rdataset_current(dns_rdataset_t *rdataset, dns_rdata_t *rdata) {
410*00b67f09SDavid van Moolenbroek 	unsigned char *raw = rdataset->private5;
411*00b67f09SDavid van Moolenbroek 	isc_region_t r;
412*00b67f09SDavid van Moolenbroek 	unsigned int length;
413*00b67f09SDavid van Moolenbroek 	unsigned int flags = 0;
414*00b67f09SDavid van Moolenbroek 
415*00b67f09SDavid van Moolenbroek 	REQUIRE(raw != NULL);
416*00b67f09SDavid van Moolenbroek 
417*00b67f09SDavid van Moolenbroek 	length = raw[0] * 256 + raw[1];
418*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
419*00b67f09SDavid van Moolenbroek 	raw += 4;
420*00b67f09SDavid van Moolenbroek #else
421*00b67f09SDavid van Moolenbroek 	raw += 2;
422*00b67f09SDavid van Moolenbroek #endif
423*00b67f09SDavid van Moolenbroek 	if (rdataset->type == dns_rdatatype_rrsig) {
424*00b67f09SDavid van Moolenbroek 		if (*raw & DNS_RDATASLAB_OFFLINE)
425*00b67f09SDavid van Moolenbroek 			flags |= DNS_RDATA_OFFLINE;
426*00b67f09SDavid van Moolenbroek 		length--;
427*00b67f09SDavid van Moolenbroek 		raw++;
428*00b67f09SDavid van Moolenbroek 	}
429*00b67f09SDavid van Moolenbroek 	r.length = length;
430*00b67f09SDavid van Moolenbroek 	r.base = raw;
431*00b67f09SDavid van Moolenbroek 	dns_rdata_fromregion(rdata, rdataset->rdclass, rdataset->type, &r);
432*00b67f09SDavid van Moolenbroek 	rdata->flags |= flags;
433*00b67f09SDavid van Moolenbroek }
434*00b67f09SDavid van Moolenbroek 
435*00b67f09SDavid van Moolenbroek static void
rdataset_clone(dns_rdataset_t * source,dns_rdataset_t * target)436*00b67f09SDavid van Moolenbroek rdataset_clone(dns_rdataset_t *source, dns_rdataset_t *target) {
437*00b67f09SDavid van Moolenbroek 	*target = *source;
438*00b67f09SDavid van Moolenbroek 
439*00b67f09SDavid van Moolenbroek 	/*
440*00b67f09SDavid van Moolenbroek 	 * Reset iterator state.
441*00b67f09SDavid van Moolenbroek 	 */
442*00b67f09SDavid van Moolenbroek 	target->privateuint4 = 0;
443*00b67f09SDavid van Moolenbroek 	target->private5 = NULL;
444*00b67f09SDavid van Moolenbroek }
445*00b67f09SDavid van Moolenbroek 
446*00b67f09SDavid van Moolenbroek static unsigned int
rdataset_count(dns_rdataset_t * rdataset)447*00b67f09SDavid van Moolenbroek rdataset_count(dns_rdataset_t *rdataset) {
448*00b67f09SDavid van Moolenbroek 	unsigned char *raw = rdataset->private3;
449*00b67f09SDavid van Moolenbroek 	unsigned int count;
450*00b67f09SDavid van Moolenbroek 
451*00b67f09SDavid van Moolenbroek 	count = raw[0] * 256 + raw[1];
452*00b67f09SDavid van Moolenbroek 
453*00b67f09SDavid van Moolenbroek 	return (count);
454*00b67f09SDavid van Moolenbroek }
455*00b67f09SDavid van Moolenbroek 
456*00b67f09SDavid van Moolenbroek static dns_rdatasetmethods_t rdataset_methods = {
457*00b67f09SDavid van Moolenbroek 	rdataset_disassociate,
458*00b67f09SDavid van Moolenbroek 	rdataset_first,
459*00b67f09SDavid van Moolenbroek 	rdataset_next,
460*00b67f09SDavid van Moolenbroek 	rdataset_current,
461*00b67f09SDavid van Moolenbroek 	rdataset_clone,
462*00b67f09SDavid van Moolenbroek 	rdataset_count,
463*00b67f09SDavid van Moolenbroek 	NULL,
464*00b67f09SDavid van Moolenbroek 	NULL,
465*00b67f09SDavid van Moolenbroek 	NULL,
466*00b67f09SDavid van Moolenbroek 	NULL,
467*00b67f09SDavid van Moolenbroek 	NULL,
468*00b67f09SDavid van Moolenbroek 	NULL,
469*00b67f09SDavid van Moolenbroek 	NULL,
470*00b67f09SDavid van Moolenbroek 	NULL,
471*00b67f09SDavid van Moolenbroek 	NULL,
472*00b67f09SDavid van Moolenbroek 	NULL
473*00b67f09SDavid van Moolenbroek };
474*00b67f09SDavid van Moolenbroek 
475*00b67f09SDavid van Moolenbroek void
dns_rdataslab_tordataset(unsigned char * slab,unsigned int reservelen,dns_rdataclass_t rdclass,dns_rdatatype_t rdtype,dns_rdatatype_t covers,dns_ttl_t ttl,dns_rdataset_t * rdataset)476*00b67f09SDavid van Moolenbroek dns_rdataslab_tordataset(unsigned char *slab, unsigned int reservelen,
477*00b67f09SDavid van Moolenbroek 			 dns_rdataclass_t rdclass, dns_rdatatype_t rdtype,
478*00b67f09SDavid van Moolenbroek 			 dns_rdatatype_t covers, dns_ttl_t ttl,
479*00b67f09SDavid van Moolenbroek 			 dns_rdataset_t *rdataset)
480*00b67f09SDavid van Moolenbroek {
481*00b67f09SDavid van Moolenbroek 	REQUIRE(slab != NULL);
482*00b67f09SDavid van Moolenbroek 	REQUIRE(!dns_rdataset_isassociated(rdataset));
483*00b67f09SDavid van Moolenbroek 
484*00b67f09SDavid van Moolenbroek 	rdataset->methods = &rdataset_methods;
485*00b67f09SDavid van Moolenbroek 	rdataset->rdclass = rdclass;
486*00b67f09SDavid van Moolenbroek 	rdataset->type = rdtype;
487*00b67f09SDavid van Moolenbroek 	rdataset->covers = covers;
488*00b67f09SDavid van Moolenbroek 	rdataset->ttl = ttl;
489*00b67f09SDavid van Moolenbroek 	rdataset->trust = 0;
490*00b67f09SDavid van Moolenbroek 	rdataset->private1 = NULL;
491*00b67f09SDavid van Moolenbroek 	rdataset->private2 = NULL;
492*00b67f09SDavid van Moolenbroek 	rdataset->private3 = slab + reservelen;
493*00b67f09SDavid van Moolenbroek 
494*00b67f09SDavid van Moolenbroek 	/*
495*00b67f09SDavid van Moolenbroek 	 * Reset iterator state.
496*00b67f09SDavid van Moolenbroek 	 */
497*00b67f09SDavid van Moolenbroek 	rdataset->privateuint4 = 0;
498*00b67f09SDavid van Moolenbroek 	rdataset->private5 = NULL;
499*00b67f09SDavid van Moolenbroek }
500*00b67f09SDavid van Moolenbroek 
501*00b67f09SDavid van Moolenbroek unsigned int
dns_rdataslab_size(unsigned char * slab,unsigned int reservelen)502*00b67f09SDavid van Moolenbroek dns_rdataslab_size(unsigned char *slab, unsigned int reservelen) {
503*00b67f09SDavid van Moolenbroek 	unsigned int count, length;
504*00b67f09SDavid van Moolenbroek 	unsigned char *current;
505*00b67f09SDavid van Moolenbroek 
506*00b67f09SDavid van Moolenbroek 	REQUIRE(slab != NULL);
507*00b67f09SDavid van Moolenbroek 
508*00b67f09SDavid van Moolenbroek 	current = slab + reservelen;
509*00b67f09SDavid van Moolenbroek 	count = *current++ * 256;
510*00b67f09SDavid van Moolenbroek 	count += *current++;
511*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
512*00b67f09SDavid van Moolenbroek 	current += (4 * count);
513*00b67f09SDavid van Moolenbroek #endif
514*00b67f09SDavid van Moolenbroek 	while (count > 0) {
515*00b67f09SDavid van Moolenbroek 		count--;
516*00b67f09SDavid van Moolenbroek 		length = *current++ * 256;
517*00b67f09SDavid van Moolenbroek 		length += *current++;
518*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
519*00b67f09SDavid van Moolenbroek 		current += length + 2;
520*00b67f09SDavid van Moolenbroek #else
521*00b67f09SDavid van Moolenbroek 		current += length;
522*00b67f09SDavid van Moolenbroek #endif
523*00b67f09SDavid van Moolenbroek 	}
524*00b67f09SDavid van Moolenbroek 
525*00b67f09SDavid van Moolenbroek 	return ((unsigned int)(current - slab));
526*00b67f09SDavid van Moolenbroek }
527*00b67f09SDavid van Moolenbroek 
528*00b67f09SDavid van Moolenbroek /*
529*00b67f09SDavid van Moolenbroek  * Make the dns_rdata_t 'rdata' refer to the slab item
530*00b67f09SDavid van Moolenbroek  * beginning at '*current', which is part of a slab of type
531*00b67f09SDavid van Moolenbroek  * 'type' and class 'rdclass', and advance '*current' to
532*00b67f09SDavid van Moolenbroek  * point to the next item in the slab.
533*00b67f09SDavid van Moolenbroek  */
534*00b67f09SDavid van Moolenbroek static inline void
rdata_from_slab(unsigned char ** current,dns_rdataclass_t rdclass,dns_rdatatype_t type,dns_rdata_t * rdata)535*00b67f09SDavid van Moolenbroek rdata_from_slab(unsigned char **current,
536*00b67f09SDavid van Moolenbroek 	      dns_rdataclass_t rdclass, dns_rdatatype_t type,
537*00b67f09SDavid van Moolenbroek 	      dns_rdata_t *rdata)
538*00b67f09SDavid van Moolenbroek {
539*00b67f09SDavid van Moolenbroek 	unsigned char *tcurrent = *current;
540*00b67f09SDavid van Moolenbroek 	isc_region_t region;
541*00b67f09SDavid van Moolenbroek 	unsigned int length;
542*00b67f09SDavid van Moolenbroek 	isc_boolean_t offline = ISC_FALSE;
543*00b67f09SDavid van Moolenbroek 
544*00b67f09SDavid van Moolenbroek 	length = *tcurrent++ * 256;
545*00b67f09SDavid van Moolenbroek 	length += *tcurrent++;
546*00b67f09SDavid van Moolenbroek 
547*00b67f09SDavid van Moolenbroek 	if (type == dns_rdatatype_rrsig) {
548*00b67f09SDavid van Moolenbroek 		if ((*tcurrent & DNS_RDATASLAB_OFFLINE) != 0)
549*00b67f09SDavid van Moolenbroek 			offline = ISC_TRUE;
550*00b67f09SDavid van Moolenbroek 		length--;
551*00b67f09SDavid van Moolenbroek 		tcurrent++;
552*00b67f09SDavid van Moolenbroek 	}
553*00b67f09SDavid van Moolenbroek 	region.length = length;
554*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
555*00b67f09SDavid van Moolenbroek 	tcurrent += 2;
556*00b67f09SDavid van Moolenbroek #endif
557*00b67f09SDavid van Moolenbroek 	region.base = tcurrent;
558*00b67f09SDavid van Moolenbroek 	tcurrent += region.length;
559*00b67f09SDavid van Moolenbroek 	dns_rdata_fromregion(rdata, rdclass, type, &region);
560*00b67f09SDavid van Moolenbroek 	if (offline)
561*00b67f09SDavid van Moolenbroek 		rdata->flags |= DNS_RDATA_OFFLINE;
562*00b67f09SDavid van Moolenbroek 	*current = tcurrent;
563*00b67f09SDavid van Moolenbroek }
564*00b67f09SDavid van Moolenbroek 
565*00b67f09SDavid van Moolenbroek /*
566*00b67f09SDavid van Moolenbroek  * Return true iff 'slab' (slab data of type 'type' and class 'rdclass')
567*00b67f09SDavid van Moolenbroek  * contains an rdata identical to 'rdata'.  This does case insensitive
568*00b67f09SDavid van Moolenbroek  * comparisons per DNSSEC.
569*00b67f09SDavid van Moolenbroek  */
570*00b67f09SDavid van Moolenbroek static inline isc_boolean_t
rdata_in_slab(unsigned char * slab,unsigned int reservelen,dns_rdataclass_t rdclass,dns_rdatatype_t type,dns_rdata_t * rdata)571*00b67f09SDavid van Moolenbroek rdata_in_slab(unsigned char *slab, unsigned int reservelen,
572*00b67f09SDavid van Moolenbroek 	      dns_rdataclass_t rdclass, dns_rdatatype_t type,
573*00b67f09SDavid van Moolenbroek 	      dns_rdata_t *rdata)
574*00b67f09SDavid van Moolenbroek {
575*00b67f09SDavid van Moolenbroek 	unsigned int count, i;
576*00b67f09SDavid van Moolenbroek 	unsigned char *current;
577*00b67f09SDavid van Moolenbroek 	dns_rdata_t trdata = DNS_RDATA_INIT;
578*00b67f09SDavid van Moolenbroek 	int n;
579*00b67f09SDavid van Moolenbroek 
580*00b67f09SDavid van Moolenbroek 	current = slab + reservelen;
581*00b67f09SDavid van Moolenbroek 	count = *current++ * 256;
582*00b67f09SDavid van Moolenbroek 	count += *current++;
583*00b67f09SDavid van Moolenbroek 
584*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
585*00b67f09SDavid van Moolenbroek 	current += (4 * count);
586*00b67f09SDavid van Moolenbroek #endif
587*00b67f09SDavid van Moolenbroek 
588*00b67f09SDavid van Moolenbroek 	for (i = 0; i < count; i++) {
589*00b67f09SDavid van Moolenbroek 		rdata_from_slab(&current, rdclass, type, &trdata);
590*00b67f09SDavid van Moolenbroek 
591*00b67f09SDavid van Moolenbroek 		n = dns_rdata_compare(&trdata, rdata);
592*00b67f09SDavid van Moolenbroek 		if (n == 0)
593*00b67f09SDavid van Moolenbroek 			return (ISC_TRUE);
594*00b67f09SDavid van Moolenbroek 		if (n > 0)	/* In DNSSEC order. */
595*00b67f09SDavid van Moolenbroek 			break;
596*00b67f09SDavid van Moolenbroek 		dns_rdata_reset(&trdata);
597*00b67f09SDavid van Moolenbroek 	}
598*00b67f09SDavid van Moolenbroek 	return (ISC_FALSE);
599*00b67f09SDavid van Moolenbroek }
600*00b67f09SDavid van Moolenbroek 
601*00b67f09SDavid van Moolenbroek isc_result_t
dns_rdataslab_merge(unsigned char * oslab,unsigned char * nslab,unsigned int reservelen,isc_mem_t * mctx,dns_rdataclass_t rdclass,dns_rdatatype_t type,unsigned int flags,unsigned char ** tslabp)602*00b67f09SDavid van Moolenbroek dns_rdataslab_merge(unsigned char *oslab, unsigned char *nslab,
603*00b67f09SDavid van Moolenbroek 		    unsigned int reservelen, isc_mem_t *mctx,
604*00b67f09SDavid van Moolenbroek 		    dns_rdataclass_t rdclass, dns_rdatatype_t type,
605*00b67f09SDavid van Moolenbroek 		    unsigned int flags, unsigned char **tslabp)
606*00b67f09SDavid van Moolenbroek {
607*00b67f09SDavid van Moolenbroek 	unsigned char *ocurrent, *ostart, *ncurrent, *tstart, *tcurrent, *data;
608*00b67f09SDavid van Moolenbroek 	unsigned int ocount, ncount, count, olength, tlength, tcount, length;
609*00b67f09SDavid van Moolenbroek 	dns_rdata_t ordata = DNS_RDATA_INIT;
610*00b67f09SDavid van Moolenbroek 	dns_rdata_t nrdata = DNS_RDATA_INIT;
611*00b67f09SDavid van Moolenbroek 	isc_boolean_t added_something = ISC_FALSE;
612*00b67f09SDavid van Moolenbroek 	unsigned int oadded = 0;
613*00b67f09SDavid van Moolenbroek 	unsigned int nadded = 0;
614*00b67f09SDavid van Moolenbroek 	unsigned int nncount = 0;
615*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
616*00b67f09SDavid van Moolenbroek 	unsigned int oncount;
617*00b67f09SDavid van Moolenbroek 	unsigned int norder = 0;
618*00b67f09SDavid van Moolenbroek 	unsigned int oorder = 0;
619*00b67f09SDavid van Moolenbroek 	unsigned char *offsetbase;
620*00b67f09SDavid van Moolenbroek 	unsigned int *offsettable;
621*00b67f09SDavid van Moolenbroek #endif
622*00b67f09SDavid van Moolenbroek 
623*00b67f09SDavid van Moolenbroek 	/*
624*00b67f09SDavid van Moolenbroek 	 * XXX  Need parameter to allow "delete rdatasets in nslab" merge,
625*00b67f09SDavid van Moolenbroek 	 * or perhaps another merge routine for this purpose.
626*00b67f09SDavid van Moolenbroek 	 */
627*00b67f09SDavid van Moolenbroek 
628*00b67f09SDavid van Moolenbroek 	REQUIRE(tslabp != NULL && *tslabp == NULL);
629*00b67f09SDavid van Moolenbroek 	REQUIRE(oslab != NULL && nslab != NULL);
630*00b67f09SDavid van Moolenbroek 
631*00b67f09SDavid van Moolenbroek 	ocurrent = oslab + reservelen;
632*00b67f09SDavid van Moolenbroek 	ocount = *ocurrent++ * 256;
633*00b67f09SDavid van Moolenbroek 	ocount += *ocurrent++;
634*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
635*00b67f09SDavid van Moolenbroek 	ocurrent += (4 * ocount);
636*00b67f09SDavid van Moolenbroek #endif
637*00b67f09SDavid van Moolenbroek 	ostart = ocurrent;
638*00b67f09SDavid van Moolenbroek 	ncurrent = nslab + reservelen;
639*00b67f09SDavid van Moolenbroek 	ncount = *ncurrent++ * 256;
640*00b67f09SDavid van Moolenbroek 	ncount += *ncurrent++;
641*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
642*00b67f09SDavid van Moolenbroek 	ncurrent += (4 * ncount);
643*00b67f09SDavid van Moolenbroek #endif
644*00b67f09SDavid van Moolenbroek 	INSIST(ocount > 0 && ncount > 0);
645*00b67f09SDavid van Moolenbroek 
646*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
647*00b67f09SDavid van Moolenbroek 	oncount = ncount;
648*00b67f09SDavid van Moolenbroek #endif
649*00b67f09SDavid van Moolenbroek 
650*00b67f09SDavid van Moolenbroek 	/*
651*00b67f09SDavid van Moolenbroek 	 * Yes, this is inefficient!
652*00b67f09SDavid van Moolenbroek 	 */
653*00b67f09SDavid van Moolenbroek 
654*00b67f09SDavid van Moolenbroek 	/*
655*00b67f09SDavid van Moolenbroek 	 * Figure out the length of the old slab's data.
656*00b67f09SDavid van Moolenbroek 	 */
657*00b67f09SDavid van Moolenbroek 	olength = 0;
658*00b67f09SDavid van Moolenbroek 	for (count = 0; count < ocount; count++) {
659*00b67f09SDavid van Moolenbroek 		length = *ocurrent++ * 256;
660*00b67f09SDavid van Moolenbroek 		length += *ocurrent++;
661*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
662*00b67f09SDavid van Moolenbroek 		olength += length + 8;
663*00b67f09SDavid van Moolenbroek 		ocurrent += length + 2;
664*00b67f09SDavid van Moolenbroek #else
665*00b67f09SDavid van Moolenbroek 		olength += length + 2;
666*00b67f09SDavid van Moolenbroek 		ocurrent += length;
667*00b67f09SDavid van Moolenbroek #endif
668*00b67f09SDavid van Moolenbroek 	}
669*00b67f09SDavid van Moolenbroek 
670*00b67f09SDavid van Moolenbroek 	/*
671*00b67f09SDavid van Moolenbroek 	 * Start figuring out the target length and count.
672*00b67f09SDavid van Moolenbroek 	 */
673*00b67f09SDavid van Moolenbroek 	tlength = reservelen + 2 + olength;
674*00b67f09SDavid van Moolenbroek 	tcount = ocount;
675*00b67f09SDavid van Moolenbroek 
676*00b67f09SDavid van Moolenbroek 	/*
677*00b67f09SDavid van Moolenbroek 	 * Add in the length of rdata in the new slab that aren't in
678*00b67f09SDavid van Moolenbroek 	 * the old slab.
679*00b67f09SDavid van Moolenbroek 	 */
680*00b67f09SDavid van Moolenbroek 	do {
681*00b67f09SDavid van Moolenbroek 		dns_rdata_init(&nrdata);
682*00b67f09SDavid van Moolenbroek 		rdata_from_slab(&ncurrent, rdclass, type, &nrdata);
683*00b67f09SDavid van Moolenbroek 		if (!rdata_in_slab(oslab, reservelen, rdclass, type, &nrdata))
684*00b67f09SDavid van Moolenbroek 		{
685*00b67f09SDavid van Moolenbroek 			/*
686*00b67f09SDavid van Moolenbroek 			 * This rdata isn't in the old slab.
687*00b67f09SDavid van Moolenbroek 			 */
688*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
689*00b67f09SDavid van Moolenbroek 			tlength += nrdata.length + 8;
690*00b67f09SDavid van Moolenbroek #else
691*00b67f09SDavid van Moolenbroek 			tlength += nrdata.length + 2;
692*00b67f09SDavid van Moolenbroek #endif
693*00b67f09SDavid van Moolenbroek 			if (type == dns_rdatatype_rrsig)
694*00b67f09SDavid van Moolenbroek 				tlength++;
695*00b67f09SDavid van Moolenbroek 			tcount++;
696*00b67f09SDavid van Moolenbroek 			nncount++;
697*00b67f09SDavid van Moolenbroek 			added_something = ISC_TRUE;
698*00b67f09SDavid van Moolenbroek 		}
699*00b67f09SDavid van Moolenbroek 		ncount--;
700*00b67f09SDavid van Moolenbroek 	} while (ncount > 0);
701*00b67f09SDavid van Moolenbroek 	ncount = nncount;
702*00b67f09SDavid van Moolenbroek 
703*00b67f09SDavid van Moolenbroek 	if (((flags & DNS_RDATASLAB_EXACT) != 0) &&
704*00b67f09SDavid van Moolenbroek 	    (tcount != ncount + ocount))
705*00b67f09SDavid van Moolenbroek 		return (DNS_R_NOTEXACT);
706*00b67f09SDavid van Moolenbroek 
707*00b67f09SDavid van Moolenbroek 	if (!added_something && (flags & DNS_RDATASLAB_FORCE) == 0)
708*00b67f09SDavid van Moolenbroek 		return (DNS_R_UNCHANGED);
709*00b67f09SDavid van Moolenbroek 
710*00b67f09SDavid van Moolenbroek 	/*
711*00b67f09SDavid van Moolenbroek 	 * Ensure that singleton types are actually singletons.
712*00b67f09SDavid van Moolenbroek 	 */
713*00b67f09SDavid van Moolenbroek 	if (tcount > 1 && dns_rdatatype_issingleton(type)) {
714*00b67f09SDavid van Moolenbroek 		/*
715*00b67f09SDavid van Moolenbroek 		 * We have a singleton type, but there's more than one
716*00b67f09SDavid van Moolenbroek 		 * RR in the rdataset.
717*00b67f09SDavid van Moolenbroek 		 */
718*00b67f09SDavid van Moolenbroek 		return (DNS_R_SINGLETON);
719*00b67f09SDavid van Moolenbroek 	}
720*00b67f09SDavid van Moolenbroek 
721*00b67f09SDavid van Moolenbroek 	if (tcount > 0xffff)
722*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOSPACE);
723*00b67f09SDavid van Moolenbroek 
724*00b67f09SDavid van Moolenbroek 	/*
725*00b67f09SDavid van Moolenbroek 	 * Copy the reserved area from the new slab.
726*00b67f09SDavid van Moolenbroek 	 */
727*00b67f09SDavid van Moolenbroek 	tstart = isc_mem_get(mctx, tlength);
728*00b67f09SDavid van Moolenbroek 	if (tstart == NULL)
729*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOMEMORY);
730*00b67f09SDavid van Moolenbroek 	memmove(tstart, nslab, reservelen);
731*00b67f09SDavid van Moolenbroek 	tcurrent = tstart + reservelen;
732*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
733*00b67f09SDavid van Moolenbroek 	offsetbase = tcurrent;
734*00b67f09SDavid van Moolenbroek #endif
735*00b67f09SDavid van Moolenbroek 
736*00b67f09SDavid van Moolenbroek 	/*
737*00b67f09SDavid van Moolenbroek 	 * Write the new count.
738*00b67f09SDavid van Moolenbroek 	 */
739*00b67f09SDavid van Moolenbroek 	*tcurrent++ = (tcount & 0xff00) >> 8;
740*00b67f09SDavid van Moolenbroek 	*tcurrent++ = (tcount & 0x00ff);
741*00b67f09SDavid van Moolenbroek 
742*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
743*00b67f09SDavid van Moolenbroek 	/*
744*00b67f09SDavid van Moolenbroek 	 * Skip offset table.
745*00b67f09SDavid van Moolenbroek 	 */
746*00b67f09SDavid van Moolenbroek 	tcurrent += (tcount * 4);
747*00b67f09SDavid van Moolenbroek 
748*00b67f09SDavid van Moolenbroek 	offsettable = isc_mem_get(mctx,
749*00b67f09SDavid van Moolenbroek 				  (ocount + oncount) * sizeof(unsigned int));
750*00b67f09SDavid van Moolenbroek 	if (offsettable == NULL) {
751*00b67f09SDavid van Moolenbroek 		isc_mem_put(mctx, tstart, tlength);
752*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOMEMORY);
753*00b67f09SDavid van Moolenbroek 	}
754*00b67f09SDavid van Moolenbroek 	memset(offsettable, 0, (ocount + oncount) * sizeof(unsigned int));
755*00b67f09SDavid van Moolenbroek #endif
756*00b67f09SDavid van Moolenbroek 
757*00b67f09SDavid van Moolenbroek 	/*
758*00b67f09SDavid van Moolenbroek 	 * Merge the two slabs.
759*00b67f09SDavid van Moolenbroek 	 */
760*00b67f09SDavid van Moolenbroek 	ocurrent = ostart;
761*00b67f09SDavid van Moolenbroek 	INSIST(ocount != 0);
762*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
763*00b67f09SDavid van Moolenbroek 	oorder = ocurrent[2] * 256 + ocurrent[3];
764*00b67f09SDavid van Moolenbroek 	INSIST(oorder < ocount);
765*00b67f09SDavid van Moolenbroek #endif
766*00b67f09SDavid van Moolenbroek 	rdata_from_slab(&ocurrent, rdclass, type, &ordata);
767*00b67f09SDavid van Moolenbroek 
768*00b67f09SDavid van Moolenbroek 	ncurrent = nslab + reservelen + 2;
769*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
770*00b67f09SDavid van Moolenbroek 	ncurrent += (4 * oncount);
771*00b67f09SDavid van Moolenbroek #endif
772*00b67f09SDavid van Moolenbroek 
773*00b67f09SDavid van Moolenbroek 	if (ncount > 0) {
774*00b67f09SDavid van Moolenbroek 		do {
775*00b67f09SDavid van Moolenbroek 			dns_rdata_reset(&nrdata);
776*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
777*00b67f09SDavid van Moolenbroek 			norder = ncurrent[2] * 256 + ncurrent[3];
778*00b67f09SDavid van Moolenbroek 
779*00b67f09SDavid van Moolenbroek 			INSIST(norder < oncount);
780*00b67f09SDavid van Moolenbroek #endif
781*00b67f09SDavid van Moolenbroek 			rdata_from_slab(&ncurrent, rdclass, type, &nrdata);
782*00b67f09SDavid van Moolenbroek 		} while (rdata_in_slab(oslab, reservelen, rdclass,
783*00b67f09SDavid van Moolenbroek 				       type, &nrdata));
784*00b67f09SDavid van Moolenbroek 	}
785*00b67f09SDavid van Moolenbroek 
786*00b67f09SDavid van Moolenbroek 	while (oadded < ocount || nadded < ncount) {
787*00b67f09SDavid van Moolenbroek 		isc_boolean_t fromold;
788*00b67f09SDavid van Moolenbroek 		if (oadded == ocount)
789*00b67f09SDavid van Moolenbroek 			fromold = ISC_FALSE;
790*00b67f09SDavid van Moolenbroek 		else if (nadded == ncount)
791*00b67f09SDavid van Moolenbroek 			fromold = ISC_TRUE;
792*00b67f09SDavid van Moolenbroek 		else
793*00b67f09SDavid van Moolenbroek 			fromold = ISC_TF(compare_rdata(&ordata, &nrdata) < 0);
794*00b67f09SDavid van Moolenbroek 		if (fromold) {
795*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
796*00b67f09SDavid van Moolenbroek 			offsettable[oorder] = tcurrent - offsetbase;
797*00b67f09SDavid van Moolenbroek #endif
798*00b67f09SDavid van Moolenbroek 			length = ordata.length;
799*00b67f09SDavid van Moolenbroek 			data = ordata.data;
800*00b67f09SDavid van Moolenbroek 			if (type == dns_rdatatype_rrsig) {
801*00b67f09SDavid van Moolenbroek 				length++;
802*00b67f09SDavid van Moolenbroek 				data--;
803*00b67f09SDavid van Moolenbroek 			}
804*00b67f09SDavid van Moolenbroek 			*tcurrent++ = (length & 0xff00) >> 8;
805*00b67f09SDavid van Moolenbroek 			*tcurrent++ = (length & 0x00ff);
806*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
807*00b67f09SDavid van Moolenbroek 			tcurrent += 2;	/* fill in later */
808*00b67f09SDavid van Moolenbroek #endif
809*00b67f09SDavid van Moolenbroek 			memmove(tcurrent, data, length);
810*00b67f09SDavid van Moolenbroek 			tcurrent += length;
811*00b67f09SDavid van Moolenbroek 			oadded++;
812*00b67f09SDavid van Moolenbroek 			if (oadded < ocount) {
813*00b67f09SDavid van Moolenbroek 				dns_rdata_reset(&ordata);
814*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
815*00b67f09SDavid van Moolenbroek 				oorder = ocurrent[2] * 256 + ocurrent[3];
816*00b67f09SDavid van Moolenbroek 				INSIST(oorder < ocount);
817*00b67f09SDavid van Moolenbroek #endif
818*00b67f09SDavid van Moolenbroek 				rdata_from_slab(&ocurrent, rdclass, type,
819*00b67f09SDavid van Moolenbroek 						&ordata);
820*00b67f09SDavid van Moolenbroek 			}
821*00b67f09SDavid van Moolenbroek 		} else {
822*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
823*00b67f09SDavid van Moolenbroek 			offsettable[ocount + norder] = tcurrent - offsetbase;
824*00b67f09SDavid van Moolenbroek #endif
825*00b67f09SDavid van Moolenbroek 			length = nrdata.length;
826*00b67f09SDavid van Moolenbroek 			data = nrdata.data;
827*00b67f09SDavid van Moolenbroek 			if (type == dns_rdatatype_rrsig) {
828*00b67f09SDavid van Moolenbroek 				length++;
829*00b67f09SDavid van Moolenbroek 				data--;
830*00b67f09SDavid van Moolenbroek 			}
831*00b67f09SDavid van Moolenbroek 			*tcurrent++ = (length & 0xff00) >> 8;
832*00b67f09SDavid van Moolenbroek 			*tcurrent++ = (length & 0x00ff);
833*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
834*00b67f09SDavid van Moolenbroek 			tcurrent += 2;	/* fill in later */
835*00b67f09SDavid van Moolenbroek #endif
836*00b67f09SDavid van Moolenbroek 			memmove(tcurrent, data, length);
837*00b67f09SDavid van Moolenbroek 			tcurrent += length;
838*00b67f09SDavid van Moolenbroek 			nadded++;
839*00b67f09SDavid van Moolenbroek 			if (nadded < ncount) {
840*00b67f09SDavid van Moolenbroek 				do {
841*00b67f09SDavid van Moolenbroek 					dns_rdata_reset(&nrdata);
842*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
843*00b67f09SDavid van Moolenbroek 					norder = ncurrent[2] * 256 + ncurrent[3];
844*00b67f09SDavid van Moolenbroek 					INSIST(norder < oncount);
845*00b67f09SDavid van Moolenbroek #endif
846*00b67f09SDavid van Moolenbroek 					rdata_from_slab(&ncurrent, rdclass,
847*00b67f09SDavid van Moolenbroek 							type, &nrdata);
848*00b67f09SDavid van Moolenbroek 				} while (rdata_in_slab(oslab, reservelen,
849*00b67f09SDavid van Moolenbroek 						       rdclass, type,
850*00b67f09SDavid van Moolenbroek 						       &nrdata));
851*00b67f09SDavid van Moolenbroek 			}
852*00b67f09SDavid van Moolenbroek 		}
853*00b67f09SDavid van Moolenbroek 	}
854*00b67f09SDavid van Moolenbroek 
855*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
856*00b67f09SDavid van Moolenbroek 	fillin_offsets(offsetbase, offsettable, ocount + oncount);
857*00b67f09SDavid van Moolenbroek 
858*00b67f09SDavid van Moolenbroek 	isc_mem_put(mctx, offsettable,
859*00b67f09SDavid van Moolenbroek 		    (ocount + oncount) * sizeof(unsigned int));
860*00b67f09SDavid van Moolenbroek #endif
861*00b67f09SDavid van Moolenbroek 
862*00b67f09SDavid van Moolenbroek 	INSIST(tcurrent == tstart + tlength);
863*00b67f09SDavid van Moolenbroek 
864*00b67f09SDavid van Moolenbroek 	*tslabp = tstart;
865*00b67f09SDavid van Moolenbroek 
866*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
867*00b67f09SDavid van Moolenbroek }
868*00b67f09SDavid van Moolenbroek 
869*00b67f09SDavid van Moolenbroek isc_result_t
dns_rdataslab_subtract(unsigned char * mslab,unsigned char * sslab,unsigned int reservelen,isc_mem_t * mctx,dns_rdataclass_t rdclass,dns_rdatatype_t type,unsigned int flags,unsigned char ** tslabp)870*00b67f09SDavid van Moolenbroek dns_rdataslab_subtract(unsigned char *mslab, unsigned char *sslab,
871*00b67f09SDavid van Moolenbroek 		       unsigned int reservelen, isc_mem_t *mctx,
872*00b67f09SDavid van Moolenbroek 		       dns_rdataclass_t rdclass, dns_rdatatype_t type,
873*00b67f09SDavid van Moolenbroek 		       unsigned int flags, unsigned char **tslabp)
874*00b67f09SDavid van Moolenbroek {
875*00b67f09SDavid van Moolenbroek 	unsigned char *mcurrent, *sstart, *scurrent, *tstart, *tcurrent;
876*00b67f09SDavid van Moolenbroek 	unsigned int mcount, scount, rcount ,count, tlength, tcount, i;
877*00b67f09SDavid van Moolenbroek 	dns_rdata_t srdata = DNS_RDATA_INIT;
878*00b67f09SDavid van Moolenbroek 	dns_rdata_t mrdata = DNS_RDATA_INIT;
879*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
880*00b67f09SDavid van Moolenbroek 	unsigned char *offsetbase;
881*00b67f09SDavid van Moolenbroek 	unsigned int *offsettable;
882*00b67f09SDavid van Moolenbroek 	unsigned int order;
883*00b67f09SDavid van Moolenbroek #endif
884*00b67f09SDavid van Moolenbroek 
885*00b67f09SDavid van Moolenbroek 	REQUIRE(tslabp != NULL && *tslabp == NULL);
886*00b67f09SDavid van Moolenbroek 	REQUIRE(mslab != NULL && sslab != NULL);
887*00b67f09SDavid van Moolenbroek 
888*00b67f09SDavid van Moolenbroek 	mcurrent = mslab + reservelen;
889*00b67f09SDavid van Moolenbroek 	mcount = *mcurrent++ * 256;
890*00b67f09SDavid van Moolenbroek 	mcount += *mcurrent++;
891*00b67f09SDavid van Moolenbroek 	scurrent = sslab + reservelen;
892*00b67f09SDavid van Moolenbroek 	scount = *scurrent++ * 256;
893*00b67f09SDavid van Moolenbroek 	scount += *scurrent++;
894*00b67f09SDavid van Moolenbroek 	INSIST(mcount > 0 && scount > 0);
895*00b67f09SDavid van Moolenbroek 
896*00b67f09SDavid van Moolenbroek 	/*
897*00b67f09SDavid van Moolenbroek 	 * Yes, this is inefficient!
898*00b67f09SDavid van Moolenbroek 	 */
899*00b67f09SDavid van Moolenbroek 
900*00b67f09SDavid van Moolenbroek 	/*
901*00b67f09SDavid van Moolenbroek 	 * Start figuring out the target length and count.
902*00b67f09SDavid van Moolenbroek 	 */
903*00b67f09SDavid van Moolenbroek 	tlength = reservelen + 2;
904*00b67f09SDavid van Moolenbroek 	tcount = 0;
905*00b67f09SDavid van Moolenbroek 	rcount = 0;
906*00b67f09SDavid van Moolenbroek 
907*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
908*00b67f09SDavid van Moolenbroek 	mcurrent += 4 * mcount;
909*00b67f09SDavid van Moolenbroek 	scurrent += 4 * scount;
910*00b67f09SDavid van Moolenbroek #endif
911*00b67f09SDavid van Moolenbroek 	sstart = scurrent;
912*00b67f09SDavid van Moolenbroek 
913*00b67f09SDavid van Moolenbroek 	/*
914*00b67f09SDavid van Moolenbroek 	 * Add in the length of rdata in the mslab that aren't in
915*00b67f09SDavid van Moolenbroek 	 * the sslab.
916*00b67f09SDavid van Moolenbroek 	 */
917*00b67f09SDavid van Moolenbroek 	for (i = 0; i < mcount; i++) {
918*00b67f09SDavid van Moolenbroek 		unsigned char *mrdatabegin = mcurrent;
919*00b67f09SDavid van Moolenbroek 		rdata_from_slab(&mcurrent, rdclass, type, &mrdata);
920*00b67f09SDavid van Moolenbroek 		scurrent = sstart;
921*00b67f09SDavid van Moolenbroek 		for (count = 0; count < scount; count++) {
922*00b67f09SDavid van Moolenbroek 			dns_rdata_reset(&srdata);
923*00b67f09SDavid van Moolenbroek 			rdata_from_slab(&scurrent, rdclass, type, &srdata);
924*00b67f09SDavid van Moolenbroek 			if (dns_rdata_compare(&mrdata, &srdata) == 0)
925*00b67f09SDavid van Moolenbroek 				break;
926*00b67f09SDavid van Moolenbroek 		}
927*00b67f09SDavid van Moolenbroek 		if (count == scount) {
928*00b67f09SDavid van Moolenbroek 			/*
929*00b67f09SDavid van Moolenbroek 			 * This rdata isn't in the sslab, and thus isn't
930*00b67f09SDavid van Moolenbroek 			 * being subtracted.
931*00b67f09SDavid van Moolenbroek 			 */
932*00b67f09SDavid van Moolenbroek 			tlength += (unsigned int)(mcurrent - mrdatabegin);
933*00b67f09SDavid van Moolenbroek 			tcount++;
934*00b67f09SDavid van Moolenbroek 		} else
935*00b67f09SDavid van Moolenbroek 			rcount++;
936*00b67f09SDavid van Moolenbroek 		dns_rdata_reset(&mrdata);
937*00b67f09SDavid van Moolenbroek 	}
938*00b67f09SDavid van Moolenbroek 
939*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
940*00b67f09SDavid van Moolenbroek 	tlength += (4 * tcount);
941*00b67f09SDavid van Moolenbroek #endif
942*00b67f09SDavid van Moolenbroek 
943*00b67f09SDavid van Moolenbroek 	/*
944*00b67f09SDavid van Moolenbroek 	 * Check that all the records originally existed.  The numeric
945*00b67f09SDavid van Moolenbroek 	 * check only works as rdataslabs do not contain duplicates.
946*00b67f09SDavid van Moolenbroek 	 */
947*00b67f09SDavid van Moolenbroek 	if (((flags & DNS_RDATASLAB_EXACT) != 0) && (rcount != scount))
948*00b67f09SDavid van Moolenbroek 		return (DNS_R_NOTEXACT);
949*00b67f09SDavid van Moolenbroek 
950*00b67f09SDavid van Moolenbroek 	/*
951*00b67f09SDavid van Moolenbroek 	 * Don't continue if the new rdataslab would be empty.
952*00b67f09SDavid van Moolenbroek 	 */
953*00b67f09SDavid van Moolenbroek 	if (tcount == 0)
954*00b67f09SDavid van Moolenbroek 		return (DNS_R_NXRRSET);
955*00b67f09SDavid van Moolenbroek 
956*00b67f09SDavid van Moolenbroek 	/*
957*00b67f09SDavid van Moolenbroek 	 * If nothing is going to change, we can stop.
958*00b67f09SDavid van Moolenbroek 	 */
959*00b67f09SDavid van Moolenbroek 	if (rcount == 0)
960*00b67f09SDavid van Moolenbroek 		return (DNS_R_UNCHANGED);
961*00b67f09SDavid van Moolenbroek 
962*00b67f09SDavid van Moolenbroek 	/*
963*00b67f09SDavid van Moolenbroek 	 * Copy the reserved area from the mslab.
964*00b67f09SDavid van Moolenbroek 	 */
965*00b67f09SDavid van Moolenbroek 	tstart = isc_mem_get(mctx, tlength);
966*00b67f09SDavid van Moolenbroek 	if (tstart == NULL)
967*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOMEMORY);
968*00b67f09SDavid van Moolenbroek 	memmove(tstart, mslab, reservelen);
969*00b67f09SDavid van Moolenbroek 	tcurrent = tstart + reservelen;
970*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
971*00b67f09SDavid van Moolenbroek 	offsetbase = tcurrent;
972*00b67f09SDavid van Moolenbroek 
973*00b67f09SDavid van Moolenbroek 	offsettable = isc_mem_get(mctx, mcount * sizeof(unsigned int));
974*00b67f09SDavid van Moolenbroek 	if (offsettable == NULL) {
975*00b67f09SDavid van Moolenbroek 		isc_mem_put(mctx, tstart, tlength);
976*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOMEMORY);
977*00b67f09SDavid van Moolenbroek 	}
978*00b67f09SDavid van Moolenbroek 	memset(offsettable, 0, mcount * sizeof(unsigned int));
979*00b67f09SDavid van Moolenbroek #endif
980*00b67f09SDavid van Moolenbroek 
981*00b67f09SDavid van Moolenbroek 	/*
982*00b67f09SDavid van Moolenbroek 	 * Write the new count.
983*00b67f09SDavid van Moolenbroek 	 */
984*00b67f09SDavid van Moolenbroek 	*tcurrent++ = (tcount & 0xff00) >> 8;
985*00b67f09SDavid van Moolenbroek 	*tcurrent++ = (tcount & 0x00ff);
986*00b67f09SDavid van Moolenbroek 
987*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
988*00b67f09SDavid van Moolenbroek 	tcurrent += (4 * tcount);
989*00b67f09SDavid van Moolenbroek #endif
990*00b67f09SDavid van Moolenbroek 
991*00b67f09SDavid van Moolenbroek 	/*
992*00b67f09SDavid van Moolenbroek 	 * Copy the parts of mslab not in sslab.
993*00b67f09SDavid van Moolenbroek 	 */
994*00b67f09SDavid van Moolenbroek 	mcurrent = mslab + reservelen;
995*00b67f09SDavid van Moolenbroek 	mcount = *mcurrent++ * 256;
996*00b67f09SDavid van Moolenbroek 	mcount += *mcurrent++;
997*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
998*00b67f09SDavid van Moolenbroek 	mcurrent += (4 * mcount);
999*00b67f09SDavid van Moolenbroek #endif
1000*00b67f09SDavid van Moolenbroek 	for (i = 0; i < mcount; i++) {
1001*00b67f09SDavid van Moolenbroek 		unsigned char *mrdatabegin = mcurrent;
1002*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
1003*00b67f09SDavid van Moolenbroek 		order = mcurrent[2] * 256 + mcurrent[3];
1004*00b67f09SDavid van Moolenbroek 		INSIST(order < mcount);
1005*00b67f09SDavid van Moolenbroek #endif
1006*00b67f09SDavid van Moolenbroek 		rdata_from_slab(&mcurrent, rdclass, type, &mrdata);
1007*00b67f09SDavid van Moolenbroek 		scurrent = sstart;
1008*00b67f09SDavid van Moolenbroek 		for (count = 0; count < scount; count++) {
1009*00b67f09SDavid van Moolenbroek 			dns_rdata_reset(&srdata);
1010*00b67f09SDavid van Moolenbroek 			rdata_from_slab(&scurrent, rdclass, type, &srdata);
1011*00b67f09SDavid van Moolenbroek 			if (dns_rdata_compare(&mrdata, &srdata) == 0)
1012*00b67f09SDavid van Moolenbroek 				break;
1013*00b67f09SDavid van Moolenbroek 		}
1014*00b67f09SDavid van Moolenbroek 		if (count == scount) {
1015*00b67f09SDavid van Moolenbroek 			/*
1016*00b67f09SDavid van Moolenbroek 			 * This rdata isn't in the sslab, and thus should be
1017*00b67f09SDavid van Moolenbroek 			 * copied to the tslab.
1018*00b67f09SDavid van Moolenbroek 			 */
1019*00b67f09SDavid van Moolenbroek 			unsigned int length;
1020*00b67f09SDavid van Moolenbroek 			length = (unsigned int)(mcurrent - mrdatabegin);
1021*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
1022*00b67f09SDavid van Moolenbroek 			offsettable[order] = tcurrent - offsetbase;
1023*00b67f09SDavid van Moolenbroek #endif
1024*00b67f09SDavid van Moolenbroek 			memmove(tcurrent, mrdatabegin, length);
1025*00b67f09SDavid van Moolenbroek 			tcurrent += length;
1026*00b67f09SDavid van Moolenbroek 		}
1027*00b67f09SDavid van Moolenbroek 		dns_rdata_reset(&mrdata);
1028*00b67f09SDavid van Moolenbroek 	}
1029*00b67f09SDavid van Moolenbroek 
1030*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
1031*00b67f09SDavid van Moolenbroek 	fillin_offsets(offsetbase, offsettable, mcount);
1032*00b67f09SDavid van Moolenbroek 
1033*00b67f09SDavid van Moolenbroek 	isc_mem_put(mctx, offsettable, mcount * sizeof(unsigned int));
1034*00b67f09SDavid van Moolenbroek #endif
1035*00b67f09SDavid van Moolenbroek 
1036*00b67f09SDavid van Moolenbroek 	INSIST(tcurrent == tstart + tlength);
1037*00b67f09SDavid van Moolenbroek 
1038*00b67f09SDavid van Moolenbroek 	*tslabp = tstart;
1039*00b67f09SDavid van Moolenbroek 
1040*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
1041*00b67f09SDavid van Moolenbroek }
1042*00b67f09SDavid van Moolenbroek 
1043*00b67f09SDavid van Moolenbroek isc_boolean_t
dns_rdataslab_equal(unsigned char * slab1,unsigned char * slab2,unsigned int reservelen)1044*00b67f09SDavid van Moolenbroek dns_rdataslab_equal(unsigned char *slab1, unsigned char *slab2,
1045*00b67f09SDavid van Moolenbroek 		    unsigned int reservelen)
1046*00b67f09SDavid van Moolenbroek {
1047*00b67f09SDavid van Moolenbroek 	unsigned char *current1, *current2;
1048*00b67f09SDavid van Moolenbroek 	unsigned int count1, count2;
1049*00b67f09SDavid van Moolenbroek 	unsigned int length1, length2;
1050*00b67f09SDavid van Moolenbroek 
1051*00b67f09SDavid van Moolenbroek 	current1 = slab1 + reservelen;
1052*00b67f09SDavid van Moolenbroek 	count1 = *current1++ * 256;
1053*00b67f09SDavid van Moolenbroek 	count1 += *current1++;
1054*00b67f09SDavid van Moolenbroek 
1055*00b67f09SDavid van Moolenbroek 	current2 = slab2 + reservelen;
1056*00b67f09SDavid van Moolenbroek 	count2 = *current2++ * 256;
1057*00b67f09SDavid van Moolenbroek 	count2 += *current2++;
1058*00b67f09SDavid van Moolenbroek 
1059*00b67f09SDavid van Moolenbroek 	if (count1 != count2)
1060*00b67f09SDavid van Moolenbroek 		return (ISC_FALSE);
1061*00b67f09SDavid van Moolenbroek 
1062*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
1063*00b67f09SDavid van Moolenbroek 	current1 += (4 * count1);
1064*00b67f09SDavid van Moolenbroek 	current2 += (4 * count2);
1065*00b67f09SDavid van Moolenbroek #endif
1066*00b67f09SDavid van Moolenbroek 
1067*00b67f09SDavid van Moolenbroek 	while (count1 > 0) {
1068*00b67f09SDavid van Moolenbroek 		length1 = *current1++ * 256;
1069*00b67f09SDavid van Moolenbroek 		length1 += *current1++;
1070*00b67f09SDavid van Moolenbroek 
1071*00b67f09SDavid van Moolenbroek 		length2 = *current2++ * 256;
1072*00b67f09SDavid van Moolenbroek 		length2 += *current2++;
1073*00b67f09SDavid van Moolenbroek 
1074*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
1075*00b67f09SDavid van Moolenbroek 		current1 += 2;
1076*00b67f09SDavid van Moolenbroek 		current2 += 2;
1077*00b67f09SDavid van Moolenbroek #endif
1078*00b67f09SDavid van Moolenbroek 
1079*00b67f09SDavid van Moolenbroek 		if (length1 != length2 ||
1080*00b67f09SDavid van Moolenbroek 		    memcmp(current1, current2, length1) != 0)
1081*00b67f09SDavid van Moolenbroek 			return (ISC_FALSE);
1082*00b67f09SDavid van Moolenbroek 
1083*00b67f09SDavid van Moolenbroek 		current1 += length1;
1084*00b67f09SDavid van Moolenbroek 		current2 += length1;
1085*00b67f09SDavid van Moolenbroek 
1086*00b67f09SDavid van Moolenbroek 		count1--;
1087*00b67f09SDavid van Moolenbroek 	}
1088*00b67f09SDavid van Moolenbroek 	return (ISC_TRUE);
1089*00b67f09SDavid van Moolenbroek }
1090*00b67f09SDavid van Moolenbroek 
1091*00b67f09SDavid van Moolenbroek isc_boolean_t
dns_rdataslab_equalx(unsigned char * slab1,unsigned char * slab2,unsigned int reservelen,dns_rdataclass_t rdclass,dns_rdatatype_t type)1092*00b67f09SDavid van Moolenbroek dns_rdataslab_equalx(unsigned char *slab1, unsigned char *slab2,
1093*00b67f09SDavid van Moolenbroek 		     unsigned int reservelen, dns_rdataclass_t rdclass,
1094*00b67f09SDavid van Moolenbroek 		     dns_rdatatype_t type)
1095*00b67f09SDavid van Moolenbroek {
1096*00b67f09SDavid van Moolenbroek 	unsigned char *current1, *current2;
1097*00b67f09SDavid van Moolenbroek 	unsigned int count1, count2;
1098*00b67f09SDavid van Moolenbroek 	dns_rdata_t rdata1 = DNS_RDATA_INIT;
1099*00b67f09SDavid van Moolenbroek 	dns_rdata_t rdata2 = DNS_RDATA_INIT;
1100*00b67f09SDavid van Moolenbroek 
1101*00b67f09SDavid van Moolenbroek 	current1 = slab1 + reservelen;
1102*00b67f09SDavid van Moolenbroek 	count1 = *current1++ * 256;
1103*00b67f09SDavid van Moolenbroek 	count1 += *current1++;
1104*00b67f09SDavid van Moolenbroek 
1105*00b67f09SDavid van Moolenbroek 	current2 = slab2 + reservelen;
1106*00b67f09SDavid van Moolenbroek 	count2 = *current2++ * 256;
1107*00b67f09SDavid van Moolenbroek 	count2 += *current2++;
1108*00b67f09SDavid van Moolenbroek 
1109*00b67f09SDavid van Moolenbroek 	if (count1 != count2)
1110*00b67f09SDavid van Moolenbroek 		return (ISC_FALSE);
1111*00b67f09SDavid van Moolenbroek 
1112*00b67f09SDavid van Moolenbroek #if DNS_RDATASET_FIXED
1113*00b67f09SDavid van Moolenbroek 	current1 += (4 * count1);
1114*00b67f09SDavid van Moolenbroek 	current2 += (4 * count2);
1115*00b67f09SDavid van Moolenbroek #endif
1116*00b67f09SDavid van Moolenbroek 
1117*00b67f09SDavid van Moolenbroek 	while (count1-- > 0) {
1118*00b67f09SDavid van Moolenbroek 		rdata_from_slab(&current1, rdclass, type, &rdata1);
1119*00b67f09SDavid van Moolenbroek 		rdata_from_slab(&current2, rdclass, type, &rdata2);
1120*00b67f09SDavid van Moolenbroek 		if (dns_rdata_compare(&rdata1, &rdata2) != 0)
1121*00b67f09SDavid van Moolenbroek 			return (ISC_FALSE);
1122*00b67f09SDavid van Moolenbroek 		dns_rdata_reset(&rdata1);
1123*00b67f09SDavid van Moolenbroek 		dns_rdata_reset(&rdata2);
1124*00b67f09SDavid van Moolenbroek 	}
1125*00b67f09SDavid van Moolenbroek 	return (ISC_TRUE);
1126*00b67f09SDavid van Moolenbroek }
1127