2 * Copyright (C) 2004-2012 Internet Systems Consortium, Inc. ("ISC")
3 * Copyright (C) 1999-2003 Internet Software Consortium.
5 * Permission to use, copy, modify, and/or distribute this software for any
6 * purpose with or without fee is hereby granted, provided that the above
7 * copyright notice and this permission notice appear in all copies.
9 * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
10 * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
11 * AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
12 * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
13 * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
14 * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
15 * PERFORMANCE OF THIS SOFTWARE.
27 #include <isc/region.h>
28 #include <isc/string.h> /* Required for HP/UX (and others?) */
31 #include <dns/result.h>
32 #include <dns/rdata.h>
33 #include <dns/rdataset.h>
34 #include <dns/rdataslab.h>
37 * The rdataslab structure allows iteration to occur in both load order
38 * and DNSSEC order. The structure is as follows:
40 * header (reservelen bytes)
41 * record count (2 bytes)
42 * offset table (4 x record count bytes in load order)
44 * data length (2 bytes)
46 * meta data (1 byte for RRSIG's)
47 * data (data length bytes)
49 * If DNS_RDATASET_FIXED is defined to be zero (0) the format of a
50 * rdataslab is as follows:
52 * header (reservelen bytes)
53 * record count (2 bytes)
55 * data length (2 bytes)
56 * meta data (1 byte for RRSIG's)
57 * data (data length bytes)
59 * Offsets are from the end of the header.
61 * Load order traversal is performed by walking the offset table to find
62 * the start of the record (DNS_RDATASET_FIXED = 1).
64 * DNSSEC order traversal is performed by walking the data records.
66 * The order is stored with record to allow for efficient reconstruction
67 * of the offset table following a merge or subtraction.
69 * The iterator methods here currently only support DNSSEC order iteration.
71 * The iterator methods in rbtdb support both load order and DNSSEC order
75 * rbtdb.c directly interacts with the slab's raw structures. If the
76 * structure changes then rbtdb.c also needs to be updated to reflect
77 * the changes. See the areas tagged with "RDATASLAB".
85 /*% Note: the "const void *" are just to make qsort happy. */
87 compare_rdata(const void *p1, const void *p2) {
88 const struct xrdata *x1 = p1;
89 const struct xrdata *x2 = p2;
90 return (dns_rdata_compare(&x1->rdata, &x2->rdata));
93 #if DNS_RDATASET_FIXED
95 fillin_offsets(unsigned char *offsetbase, unsigned int *offsettable,
101 for (i = 0, j = 0; i < length; i++) {
103 if (offsettable[i] == 0)
107 * Fill in offset table.
109 raw = &offsetbase[j*4 + 2];
110 *raw++ = (offsettable[i] & 0xff000000) >> 24;
111 *raw++ = (offsettable[i] & 0xff0000) >> 16;
112 *raw++ = (offsettable[i] & 0xff00) >> 8;
113 *raw = offsettable[i] & 0xff;
116 * Fill in table index.
118 raw = offsetbase + offsettable[i] + 2;
119 *raw++ = (j & 0xff00) >> 8;
126 dns_rdataslab_fromrdataset(dns_rdataset_t *rdataset, isc_mem_t *mctx,
127 isc_region_t *region, unsigned int reservelen)
130 unsigned char *rawbuf;
131 #if DNS_RDATASET_FIXED
132 unsigned char *offsetbase;
139 #if DNS_RDATASET_FIXED
140 unsigned int *offsettable;
144 buflen = reservelen + 2;
146 nalloc = dns_rdataset_count(rdataset);
148 if (nitems == 0 && rdataset->type != 0)
149 return (ISC_R_FAILURE);
152 return (ISC_R_NOSPACE);
156 x = isc_mem_get(mctx, nalloc * sizeof(struct xrdata));
158 return (ISC_R_NOMEMORY);
163 * Save all of the rdata members into an array.
165 result = dns_rdataset_first(rdataset);
166 if (result != ISC_R_SUCCESS && result != ISC_R_NOMORE)
168 for (i = 0; i < nalloc && result == ISC_R_SUCCESS; i++) {
169 INSIST(result == ISC_R_SUCCESS);
170 dns_rdata_init(&x[i].rdata);
171 dns_rdataset_current(rdataset, &x[i].rdata);
172 #if DNS_RDATASET_FIXED
175 result = dns_rdataset_next(rdataset);
177 if (result != ISC_R_NOMORE)
181 * Somehow we iterated over fewer rdatas than
182 * dns_rdataset_count() said there were!
184 result = ISC_R_FAILURE;
189 * Put into DNSSEC order.
191 qsort(x, nalloc, sizeof(struct xrdata), compare_rdata);
194 * Remove duplicates and compute the total storage required.
196 * If an rdata is not a duplicate, accumulate the storage size
197 * required for the rdata. We do not store the class, type, etc,
198 * just the rdata, so our overhead is 2 bytes for the number of
199 * records, and 8 for each rdata, (length(2), offset(4) and order(2))
200 * and then the rdata itself.
202 for (i = 1; i < nalloc; i++) {
203 if (compare_rdata(&x[i-1].rdata, &x[i].rdata) == 0) {
204 x[i-1].rdata.data = NULL;
205 x[i-1].rdata.length = 0;
206 #if DNS_RDATASET_FIXED
208 * Preserve the least order so A, B, A -> A, B
209 * after duplicate removal.
211 if (x[i-1].order < x[i].order)
212 x[i].order = x[i-1].order;
216 #if DNS_RDATASET_FIXED
217 buflen += (8 + x[i-1].rdata.length);
219 buflen += (2 + x[i-1].rdata.length);
222 * Provide space to store the per RR meta data.
224 if (rdataset->type == dns_rdatatype_rrsig)
229 * Don't forget the last item!
232 #if DNS_RDATASET_FIXED
233 buflen += (8 + x[i-1].rdata.length);
235 buflen += (2 + x[i-1].rdata.length);
240 * Provide space to store the per RR meta data.
242 if (rdataset->type == dns_rdatatype_rrsig)
246 * Ensure that singleton types are actually singletons.
248 if (nitems > 1 && dns_rdatatype_issingleton(rdataset->type)) {
250 * We have a singleton type, but there's more than one
251 * RR in the rdataset.
253 result = DNS_R_SINGLETON;
258 * Allocate the memory, set up a buffer, start copying in
261 rawbuf = isc_mem_get(mctx, buflen);
262 if (rawbuf == NULL) {
263 result = ISC_R_NOMEMORY;
267 #if DNS_RDATASET_FIXED
268 /* Allocate temporary offset table. */
269 offsettable = isc_mem_get(mctx, nalloc * sizeof(unsigned int));
270 if (offsettable == NULL) {
271 isc_mem_put(mctx, rawbuf, buflen);
272 result = ISC_R_NOMEMORY;
275 memset(offsettable, 0, nalloc * sizeof(unsigned int));
278 region->base = rawbuf;
279 region->length = buflen;
281 rawbuf += reservelen;
282 #if DNS_RDATASET_FIXED
286 *rawbuf++ = (nitems & 0xff00) >> 8;
287 *rawbuf++ = (nitems & 0x00ff);
289 #if DNS_RDATASET_FIXED
290 /* Skip load order table. Filled in later. */
291 rawbuf += nitems * 4;
294 for (i = 0; i < nalloc; i++) {
295 if (x[i].rdata.data == NULL)
297 #if DNS_RDATASET_FIXED
298 offsettable[x[i].order] = rawbuf - offsetbase;
300 length = x[i].rdata.length;
301 if (rdataset->type == dns_rdatatype_rrsig)
303 *rawbuf++ = (length & 0xff00) >> 8;
304 *rawbuf++ = (length & 0x00ff);
305 #if DNS_RDATASET_FIXED
306 rawbuf += 2; /* filled in later */
309 * Store the per RR meta data.
311 if (rdataset->type == dns_rdatatype_rrsig) {
312 *rawbuf++ |= (x[i].rdata.flags & DNS_RDATA_OFFLINE) ?
313 DNS_RDATASLAB_OFFLINE : 0;
315 memcpy(rawbuf, x[i].rdata.data, x[i].rdata.length);
316 rawbuf += x[i].rdata.length;
319 #if DNS_RDATASET_FIXED
320 fillin_offsets(offsetbase, offsettable, nalloc);
321 isc_mem_put(mctx, offsettable, nalloc * sizeof(unsigned int));
324 result = ISC_R_SUCCESS;
328 isc_mem_put(mctx, x, nalloc * sizeof(struct xrdata));
333 rdataset_disassociate(dns_rdataset_t *rdataset) {
338 rdataset_first(dns_rdataset_t *rdataset) {
339 unsigned char *raw = rdataset->private3;
342 count = raw[0] * 256 + raw[1];
344 rdataset->private5 = NULL;
345 return (ISC_R_NOMORE);
347 #if DNS_RDATASET_FIXED
348 raw += 2 + (4 * count);
353 * The privateuint4 field is the number of rdata beyond the cursor
354 * position, so we decrement the total count by one before storing
358 rdataset->privateuint4 = count;
359 rdataset->private5 = raw;
361 return (ISC_R_SUCCESS);
365 rdataset_next(dns_rdataset_t *rdataset) {
370 count = rdataset->privateuint4;
372 return (ISC_R_NOMORE);
374 rdataset->privateuint4 = count;
375 raw = rdataset->private5;
376 length = raw[0] * 256 + raw[1];
377 #if DNS_RDATASET_FIXED
382 rdataset->private5 = raw;
384 return (ISC_R_SUCCESS);
388 rdataset_current(dns_rdataset_t *rdataset, dns_rdata_t *rdata) {
389 unsigned char *raw = rdataset->private5;
392 unsigned int flags = 0;
394 REQUIRE(raw != NULL);
396 length = raw[0] * 256 + raw[1];
397 #if DNS_RDATASET_FIXED
402 if (rdataset->type == dns_rdatatype_rrsig) {
403 if (*raw & DNS_RDATASLAB_OFFLINE)
404 flags |= DNS_RDATA_OFFLINE;
410 dns_rdata_fromregion(rdata, rdataset->rdclass, rdataset->type, &r);
411 rdata->flags |= flags;
415 rdataset_clone(dns_rdataset_t *source, dns_rdataset_t *target) {
419 * Reset iterator state.
421 target->privateuint4 = 0;
422 target->private5 = NULL;
426 rdataset_count(dns_rdataset_t *rdataset) {
427 unsigned char *raw = rdataset->private3;
430 count = raw[0] * 256 + raw[1];
435 static dns_rdatasetmethods_t rdataset_methods = {
436 rdataset_disassociate,
454 dns_rdataslab_tordataset(unsigned char *slab, unsigned int reservelen,
455 dns_rdataclass_t rdclass, dns_rdatatype_t rdtype,
456 dns_rdatatype_t covers, dns_ttl_t ttl,
457 dns_rdataset_t *rdataset)
459 REQUIRE(slab != NULL);
460 REQUIRE(!dns_rdataset_isassociated(rdataset));
462 rdataset->methods = &rdataset_methods;
463 rdataset->rdclass = rdclass;
464 rdataset->type = rdtype;
465 rdataset->covers = covers;
468 rdataset->private1 = NULL;
469 rdataset->private2 = NULL;
470 rdataset->private3 = slab + reservelen;
473 * Reset iterator state.
475 rdataset->privateuint4 = 0;
476 rdataset->private5 = NULL;
480 dns_rdataslab_size(unsigned char *slab, unsigned int reservelen) {
481 unsigned int count, length;
482 unsigned char *current;
484 REQUIRE(slab != NULL);
486 current = slab + reservelen;
487 count = *current++ * 256;
489 #if DNS_RDATASET_FIXED
490 current += (4 * count);
494 length = *current++ * 256;
495 length += *current++;
496 #if DNS_RDATASET_FIXED
497 current += length + 2;
503 return ((unsigned int)(current - slab));
507 * Make the dns_rdata_t 'rdata' refer to the slab item
508 * beginning at '*current', which is part of a slab of type
509 * 'type' and class 'rdclass', and advance '*current' to
510 * point to the next item in the slab.
513 rdata_from_slab(unsigned char **current,
514 dns_rdataclass_t rdclass, dns_rdatatype_t type,
517 unsigned char *tcurrent = *current;
520 isc_boolean_t offline = ISC_FALSE;
522 length = *tcurrent++ * 256;
523 length += *tcurrent++;
525 if (type == dns_rdatatype_rrsig) {
526 if ((*tcurrent & DNS_RDATASLAB_OFFLINE) != 0)
531 region.length = length;
532 #if DNS_RDATASET_FIXED
535 region.base = tcurrent;
536 tcurrent += region.length;
537 dns_rdata_fromregion(rdata, rdclass, type, ®ion);
539 rdata->flags |= DNS_RDATA_OFFLINE;
544 * Return true iff 'slab' (slab data of type 'type' and class 'rdclass')
545 * contains an rdata identical to 'rdata'. This does case insensitive
546 * comparisons per DNSSEC.
548 static inline isc_boolean_t
549 rdata_in_slab(unsigned char *slab, unsigned int reservelen,
550 dns_rdataclass_t rdclass, dns_rdatatype_t type,
553 unsigned int count, i;
554 unsigned char *current;
555 dns_rdata_t trdata = DNS_RDATA_INIT;
558 current = slab + reservelen;
559 count = *current++ * 256;
562 #if DNS_RDATASET_FIXED
563 current += (4 * count);
566 for (i = 0; i < count; i++) {
567 rdata_from_slab(¤t, rdclass, type, &trdata);
569 n = dns_rdata_compare(&trdata, rdata);
572 if (n > 0) /* In DNSSEC order. */
574 dns_rdata_reset(&trdata);
580 dns_rdataslab_merge(unsigned char *oslab, unsigned char *nslab,
581 unsigned int reservelen, isc_mem_t *mctx,
582 dns_rdataclass_t rdclass, dns_rdatatype_t type,
583 unsigned int flags, unsigned char **tslabp)
585 unsigned char *ocurrent, *ostart, *ncurrent, *tstart, *tcurrent, *data;
586 unsigned int ocount, ncount, count, olength, tlength, tcount, length;
587 dns_rdata_t ordata = DNS_RDATA_INIT;
588 dns_rdata_t nrdata = DNS_RDATA_INIT;
589 isc_boolean_t added_something = ISC_FALSE;
590 unsigned int oadded = 0;
591 unsigned int nadded = 0;
592 unsigned int nncount = 0;
593 #if DNS_RDATASET_FIXED
594 unsigned int oncount;
595 unsigned int norder = 0;
596 unsigned int oorder = 0;
597 unsigned char *offsetbase;
598 unsigned int *offsettable;
602 * XXX Need parameter to allow "delete rdatasets in nslab" merge,
603 * or perhaps another merge routine for this purpose.
606 REQUIRE(tslabp != NULL && *tslabp == NULL);
607 REQUIRE(oslab != NULL && nslab != NULL);
609 ocurrent = oslab + reservelen;
610 ocount = *ocurrent++ * 256;
611 ocount += *ocurrent++;
612 #if DNS_RDATASET_FIXED
613 ocurrent += (4 * ocount);
616 ncurrent = nslab + reservelen;
617 ncount = *ncurrent++ * 256;
618 ncount += *ncurrent++;
619 #if DNS_RDATASET_FIXED
620 ncurrent += (4 * ncount);
622 INSIST(ocount > 0 && ncount > 0);
624 #if DNS_RDATASET_FIXED
629 * Yes, this is inefficient!
633 * Figure out the length of the old slab's data.
636 for (count = 0; count < ocount; count++) {
637 length = *ocurrent++ * 256;
638 length += *ocurrent++;
639 #if DNS_RDATASET_FIXED
640 olength += length + 8;
641 ocurrent += length + 2;
643 olength += length + 2;
649 * Start figuring out the target length and count.
651 tlength = reservelen + 2 + olength;
655 * Add in the length of rdata in the new slab that aren't in
659 dns_rdata_init(&nrdata);
660 rdata_from_slab(&ncurrent, rdclass, type, &nrdata);
661 if (!rdata_in_slab(oslab, reservelen, rdclass, type, &nrdata))
664 * This rdata isn't in the old slab.
666 #if DNS_RDATASET_FIXED
667 tlength += nrdata.length + 8;
669 tlength += nrdata.length + 2;
671 if (type == dns_rdatatype_rrsig)
675 added_something = ISC_TRUE;
678 } while (ncount > 0);
681 if (((flags & DNS_RDATASLAB_EXACT) != 0) &&
682 (tcount != ncount + ocount))
683 return (DNS_R_NOTEXACT);
685 if (!added_something && (flags & DNS_RDATASLAB_FORCE) == 0)
686 return (DNS_R_UNCHANGED);
689 * Ensure that singleton types are actually singletons.
691 if (tcount > 1 && dns_rdatatype_issingleton(type)) {
693 * We have a singleton type, but there's more than one
694 * RR in the rdataset.
696 return (DNS_R_SINGLETON);
700 return (ISC_R_NOSPACE);
703 * Copy the reserved area from the new slab.
705 tstart = isc_mem_get(mctx, tlength);
707 return (ISC_R_NOMEMORY);
708 memcpy(tstart, nslab, reservelen);
709 tcurrent = tstart + reservelen;
710 #if DNS_RDATASET_FIXED
711 offsetbase = tcurrent;
715 * Write the new count.
717 *tcurrent++ = (tcount & 0xff00) >> 8;
718 *tcurrent++ = (tcount & 0x00ff);
720 #if DNS_RDATASET_FIXED
724 tcurrent += (tcount * 4);
726 offsettable = isc_mem_get(mctx,
727 (ocount + oncount) * sizeof(unsigned int));
728 if (offsettable == NULL) {
729 isc_mem_put(mctx, tstart, tlength);
730 return (ISC_R_NOMEMORY);
732 memset(offsettable, 0, (ocount + oncount) * sizeof(unsigned int));
736 * Merge the two slabs.
740 #if DNS_RDATASET_FIXED
741 oorder = ocurrent[2] * 256 + ocurrent[3];
742 INSIST(oorder < ocount);
744 rdata_from_slab(&ocurrent, rdclass, type, &ordata);
746 ncurrent = nslab + reservelen + 2;
747 #if DNS_RDATASET_FIXED
748 ncurrent += (4 * oncount);
753 dns_rdata_reset(&nrdata);
754 #if DNS_RDATASET_FIXED
755 norder = ncurrent[2] * 256 + ncurrent[3];
757 INSIST(norder < oncount);
759 rdata_from_slab(&ncurrent, rdclass, type, &nrdata);
760 } while (rdata_in_slab(oslab, reservelen, rdclass,
764 while (oadded < ocount || nadded < ncount) {
765 isc_boolean_t fromold;
766 if (oadded == ocount)
768 else if (nadded == ncount)
771 fromold = ISC_TF(compare_rdata(&ordata, &nrdata) < 0);
773 #if DNS_RDATASET_FIXED
774 offsettable[oorder] = tcurrent - offsetbase;
776 length = ordata.length;
778 if (type == dns_rdatatype_rrsig) {
782 *tcurrent++ = (length & 0xff00) >> 8;
783 *tcurrent++ = (length & 0x00ff);
784 #if DNS_RDATASET_FIXED
785 tcurrent += 2; /* fill in later */
787 memcpy(tcurrent, data, length);
790 if (oadded < ocount) {
791 dns_rdata_reset(&ordata);
792 #if DNS_RDATASET_FIXED
793 oorder = ocurrent[2] * 256 + ocurrent[3];
794 INSIST(oorder < ocount);
796 rdata_from_slab(&ocurrent, rdclass, type,
800 #if DNS_RDATASET_FIXED
801 offsettable[ocount + norder] = tcurrent - offsetbase;
803 length = nrdata.length;
805 if (type == dns_rdatatype_rrsig) {
809 *tcurrent++ = (length & 0xff00) >> 8;
810 *tcurrent++ = (length & 0x00ff);
811 #if DNS_RDATASET_FIXED
812 tcurrent += 2; /* fill in later */
814 memcpy(tcurrent, data, length);
817 if (nadded < ncount) {
819 dns_rdata_reset(&nrdata);
820 #if DNS_RDATASET_FIXED
821 norder = ncurrent[2] * 256 + ncurrent[3];
822 INSIST(norder < oncount);
824 rdata_from_slab(&ncurrent, rdclass,
826 } while (rdata_in_slab(oslab, reservelen,
833 #if DNS_RDATASET_FIXED
834 fillin_offsets(offsetbase, offsettable, ocount + oncount);
836 isc_mem_put(mctx, offsettable,
837 (ocount + oncount) * sizeof(unsigned int));
840 INSIST(tcurrent == tstart + tlength);
844 return (ISC_R_SUCCESS);
848 dns_rdataslab_subtract(unsigned char *mslab, unsigned char *sslab,
849 unsigned int reservelen, isc_mem_t *mctx,
850 dns_rdataclass_t rdclass, dns_rdatatype_t type,
851 unsigned int flags, unsigned char **tslabp)
853 unsigned char *mcurrent, *sstart, *scurrent, *tstart, *tcurrent;
854 unsigned int mcount, scount, rcount ,count, tlength, tcount, i;
855 dns_rdata_t srdata = DNS_RDATA_INIT;
856 dns_rdata_t mrdata = DNS_RDATA_INIT;
857 #if DNS_RDATASET_FIXED
858 unsigned char *offsetbase;
859 unsigned int *offsettable;
863 REQUIRE(tslabp != NULL && *tslabp == NULL);
864 REQUIRE(mslab != NULL && sslab != NULL);
866 mcurrent = mslab + reservelen;
867 mcount = *mcurrent++ * 256;
868 mcount += *mcurrent++;
869 scurrent = sslab + reservelen;
870 scount = *scurrent++ * 256;
871 scount += *scurrent++;
872 INSIST(mcount > 0 && scount > 0);
875 * Yes, this is inefficient!
879 * Start figuring out the target length and count.
881 tlength = reservelen + 2;
885 #if DNS_RDATASET_FIXED
886 mcurrent += 4 * mcount;
887 scurrent += 4 * scount;
892 * Add in the length of rdata in the mslab that aren't in
895 for (i = 0; i < mcount; i++) {
896 unsigned char *mrdatabegin = mcurrent;
897 rdata_from_slab(&mcurrent, rdclass, type, &mrdata);
899 for (count = 0; count < scount; count++) {
900 dns_rdata_reset(&srdata);
901 rdata_from_slab(&scurrent, rdclass, type, &srdata);
902 if (dns_rdata_compare(&mrdata, &srdata) == 0)
905 if (count == scount) {
907 * This rdata isn't in the sslab, and thus isn't
910 tlength += mcurrent - mrdatabegin;
914 dns_rdata_reset(&mrdata);
917 #if DNS_RDATASET_FIXED
918 tlength += (4 * tcount);
922 * Check that all the records originally existed. The numeric
923 * check only works as rdataslabs do not contain duplicates.
925 if (((flags & DNS_RDATASLAB_EXACT) != 0) && (rcount != scount))
926 return (DNS_R_NOTEXACT);
929 * Don't continue if the new rdataslab would be empty.
932 return (DNS_R_NXRRSET);
935 * If nothing is going to change, we can stop.
938 return (DNS_R_UNCHANGED);
941 * Copy the reserved area from the mslab.
943 tstart = isc_mem_get(mctx, tlength);
945 return (ISC_R_NOMEMORY);
946 memcpy(tstart, mslab, reservelen);
947 tcurrent = tstart + reservelen;
948 #if DNS_RDATASET_FIXED
949 offsetbase = tcurrent;
951 offsettable = isc_mem_get(mctx, mcount * sizeof(unsigned int));
952 if (offsettable == NULL) {
953 isc_mem_put(mctx, tstart, tlength);
954 return (ISC_R_NOMEMORY);
956 memset(offsettable, 0, mcount * sizeof(unsigned int));
960 * Write the new count.
962 *tcurrent++ = (tcount & 0xff00) >> 8;
963 *tcurrent++ = (tcount & 0x00ff);
965 #if DNS_RDATASET_FIXED
966 tcurrent += (4 * tcount);
970 * Copy the parts of mslab not in sslab.
972 mcurrent = mslab + reservelen;
973 mcount = *mcurrent++ * 256;
974 mcount += *mcurrent++;
975 #if DNS_RDATASET_FIXED
976 mcurrent += (4 * mcount);
978 for (i = 0; i < mcount; i++) {
979 unsigned char *mrdatabegin = mcurrent;
980 #if DNS_RDATASET_FIXED
981 order = mcurrent[2] * 256 + mcurrent[3];
982 INSIST(order < mcount);
984 rdata_from_slab(&mcurrent, rdclass, type, &mrdata);
986 for (count = 0; count < scount; count++) {
987 dns_rdata_reset(&srdata);
988 rdata_from_slab(&scurrent, rdclass, type, &srdata);
989 if (dns_rdata_compare(&mrdata, &srdata) == 0)
992 if (count == scount) {
994 * This rdata isn't in the sslab, and thus should be
995 * copied to the tslab.
997 unsigned int length = mcurrent - mrdatabegin;
998 #if DNS_RDATASET_FIXED
999 offsettable[order] = tcurrent - offsetbase;
1001 memcpy(tcurrent, mrdatabegin, length);
1004 dns_rdata_reset(&mrdata);
1007 #if DNS_RDATASET_FIXED
1008 fillin_offsets(offsetbase, offsettable, mcount);
1010 isc_mem_put(mctx, offsettable, mcount * sizeof(unsigned int));
1013 INSIST(tcurrent == tstart + tlength);
1017 return (ISC_R_SUCCESS);
1021 dns_rdataslab_equal(unsigned char *slab1, unsigned char *slab2,
1022 unsigned int reservelen)
1024 unsigned char *current1, *current2;
1025 unsigned int count1, count2;
1026 unsigned int length1, length2;
1028 current1 = slab1 + reservelen;
1029 count1 = *current1++ * 256;
1030 count1 += *current1++;
1032 current2 = slab2 + reservelen;
1033 count2 = *current2++ * 256;
1034 count2 += *current2++;
1036 if (count1 != count2)
1039 #if DNS_RDATASET_FIXED
1040 current1 += (4 * count1);
1041 current2 += (4 * count2);
1044 while (count1 > 0) {
1045 length1 = *current1++ * 256;
1046 length1 += *current1++;
1048 length2 = *current2++ * 256;
1049 length2 += *current2++;
1051 #if DNS_RDATASET_FIXED
1056 if (length1 != length2 ||
1057 memcmp(current1, current2, length1) != 0)
1060 current1 += length1;
1061 current2 += length1;
1069 dns_rdataslab_equalx(unsigned char *slab1, unsigned char *slab2,
1070 unsigned int reservelen, dns_rdataclass_t rdclass,
1071 dns_rdatatype_t type)
1073 unsigned char *current1, *current2;
1074 unsigned int count1, count2;
1075 dns_rdata_t rdata1 = DNS_RDATA_INIT;
1076 dns_rdata_t rdata2 = DNS_RDATA_INIT;
1078 current1 = slab1 + reservelen;
1079 count1 = *current1++ * 256;
1080 count1 += *current1++;
1082 current2 = slab2 + reservelen;
1083 count2 = *current2++ * 256;
1084 count2 += *current2++;
1086 if (count1 != count2)
1089 #if DNS_RDATASET_FIXED
1090 current1 += (4 * count1);
1091 current2 += (4 * count2);
1094 while (count1-- > 0) {
1095 rdata_from_slab(¤t1, rdclass, type, &rdata1);
1096 rdata_from_slab(¤t2, rdclass, type, &rdata2);
1097 if (dns_rdata_compare(&rdata1, &rdata2) != 0)
1099 dns_rdata_reset(&rdata1);
1100 dns_rdata_reset(&rdata2);