xref: /minix3/external/bsd/bind/dist/lib/dns/diff.c (revision 00b67f09dd46474d133c95011a48590a8e8f94c7)
1*00b67f09SDavid van Moolenbroek /*	$NetBSD: diff.c,v 1.9 2015/07/08 17:28:58 christos Exp $	*/
2*00b67f09SDavid van Moolenbroek 
3*00b67f09SDavid van Moolenbroek /*
4*00b67f09SDavid van Moolenbroek  * Copyright (C) 2004, 2005, 2007-2009, 2011, 2013-2015  Internet Systems Consortium, Inc. ("ISC")
5*00b67f09SDavid van Moolenbroek  * Copyright (C) 2000-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: diff.c,v 1.26 2011/03/25 23:53:02 each 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 <stdlib.h>
27*00b67f09SDavid van Moolenbroek 
28*00b67f09SDavid van Moolenbroek #include <isc/buffer.h>
29*00b67f09SDavid van Moolenbroek #include <isc/file.h>
30*00b67f09SDavid van Moolenbroek #include <isc/mem.h>
31*00b67f09SDavid van Moolenbroek #include <isc/string.h>
32*00b67f09SDavid van Moolenbroek #include <isc/util.h>
33*00b67f09SDavid van Moolenbroek 
34*00b67f09SDavid van Moolenbroek #include <dns/db.h>
35*00b67f09SDavid van Moolenbroek #include <dns/diff.h>
36*00b67f09SDavid van Moolenbroek #include <dns/log.h>
37*00b67f09SDavid van Moolenbroek #include <dns/rdataclass.h>
38*00b67f09SDavid van Moolenbroek #include <dns/rdatalist.h>
39*00b67f09SDavid van Moolenbroek #include <dns/rdataset.h>
40*00b67f09SDavid van Moolenbroek #include <dns/rdatastruct.h>
41*00b67f09SDavid van Moolenbroek #include <dns/rdatatype.h>
42*00b67f09SDavid van Moolenbroek #include <dns/result.h>
43*00b67f09SDavid van Moolenbroek 
44*00b67f09SDavid van Moolenbroek #define CHECK(op) \
45*00b67f09SDavid van Moolenbroek 	do { result = (op);					\
46*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS) goto failure;	\
47*00b67f09SDavid van Moolenbroek 	} while (/*CONSTCOND*/0)
48*00b67f09SDavid van Moolenbroek 
49*00b67f09SDavid van Moolenbroek #define DIFF_COMMON_LOGARGS \
50*00b67f09SDavid van Moolenbroek 	dns_lctx, DNS_LOGCATEGORY_GENERAL, DNS_LOGMODULE_DIFF
51*00b67f09SDavid van Moolenbroek 
52*00b67f09SDavid van Moolenbroek static dns_rdatatype_t
rdata_covers(dns_rdata_t * rdata)53*00b67f09SDavid van Moolenbroek rdata_covers(dns_rdata_t *rdata) {
54*00b67f09SDavid van Moolenbroek 	return (rdata->type == dns_rdatatype_rrsig ?
55*00b67f09SDavid van Moolenbroek 		dns_rdata_covers(rdata) : 0);
56*00b67f09SDavid van Moolenbroek }
57*00b67f09SDavid van Moolenbroek 
58*00b67f09SDavid van Moolenbroek isc_result_t
dns_difftuple_create(isc_mem_t * mctx,dns_diffop_t op,dns_name_t * name,dns_ttl_t ttl,dns_rdata_t * rdata,dns_difftuple_t ** tp)59*00b67f09SDavid van Moolenbroek dns_difftuple_create(isc_mem_t *mctx,
60*00b67f09SDavid van Moolenbroek 		     dns_diffop_t op, dns_name_t *name, dns_ttl_t ttl,
61*00b67f09SDavid van Moolenbroek 		     dns_rdata_t *rdata, dns_difftuple_t **tp)
62*00b67f09SDavid van Moolenbroek {
63*00b67f09SDavid van Moolenbroek 	dns_difftuple_t *t;
64*00b67f09SDavid van Moolenbroek 	unsigned int size;
65*00b67f09SDavid van Moolenbroek 	unsigned char *datap;
66*00b67f09SDavid van Moolenbroek 
67*00b67f09SDavid van Moolenbroek 	REQUIRE(tp != NULL && *tp == NULL);
68*00b67f09SDavid van Moolenbroek 
69*00b67f09SDavid van Moolenbroek 	/*
70*00b67f09SDavid van Moolenbroek 	 * Create a new tuple.  The variable-size wire-format name data and
71*00b67f09SDavid van Moolenbroek 	 * rdata immediately follow the dns_difftuple_t structure
72*00b67f09SDavid van Moolenbroek 	 * in memory.
73*00b67f09SDavid van Moolenbroek 	 */
74*00b67f09SDavid van Moolenbroek 	size = sizeof(*t) + name->length + rdata->length;
75*00b67f09SDavid van Moolenbroek 	t = isc_mem_allocate(mctx, size);
76*00b67f09SDavid van Moolenbroek 	if (t == NULL)
77*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOMEMORY);
78*00b67f09SDavid van Moolenbroek 	t->mctx = NULL;
79*00b67f09SDavid van Moolenbroek 	isc_mem_attach(mctx, &t->mctx);
80*00b67f09SDavid van Moolenbroek 	t->op = op;
81*00b67f09SDavid van Moolenbroek 
82*00b67f09SDavid van Moolenbroek 	datap = (unsigned char *)(t + 1);
83*00b67f09SDavid van Moolenbroek 
84*00b67f09SDavid van Moolenbroek 	memmove(datap, name->ndata, name->length);
85*00b67f09SDavid van Moolenbroek 	dns_name_init(&t->name, NULL);
86*00b67f09SDavid van Moolenbroek 	dns_name_clone(name, &t->name);
87*00b67f09SDavid van Moolenbroek 	t->name.ndata = datap;
88*00b67f09SDavid van Moolenbroek 	datap += name->length;
89*00b67f09SDavid van Moolenbroek 
90*00b67f09SDavid van Moolenbroek 	t->ttl = ttl;
91*00b67f09SDavid van Moolenbroek 
92*00b67f09SDavid van Moolenbroek 	memmove(datap, rdata->data, rdata->length);
93*00b67f09SDavid van Moolenbroek 	dns_rdata_init(&t->rdata);
94*00b67f09SDavid van Moolenbroek 	dns_rdata_clone(rdata, &t->rdata);
95*00b67f09SDavid van Moolenbroek 	t->rdata.data = datap;
96*00b67f09SDavid van Moolenbroek 	datap += rdata->length;
97*00b67f09SDavid van Moolenbroek 
98*00b67f09SDavid van Moolenbroek 	ISC_LINK_INIT(&t->rdata, link);
99*00b67f09SDavid van Moolenbroek 	ISC_LINK_INIT(t, link);
100*00b67f09SDavid van Moolenbroek 	t->magic = DNS_DIFFTUPLE_MAGIC;
101*00b67f09SDavid van Moolenbroek 
102*00b67f09SDavid van Moolenbroek 	INSIST(datap == (unsigned char *)t + size);
103*00b67f09SDavid van Moolenbroek 
104*00b67f09SDavid van Moolenbroek 	*tp = t;
105*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
106*00b67f09SDavid van Moolenbroek }
107*00b67f09SDavid van Moolenbroek 
108*00b67f09SDavid van Moolenbroek void
dns_difftuple_free(dns_difftuple_t ** tp)109*00b67f09SDavid van Moolenbroek dns_difftuple_free(dns_difftuple_t **tp) {
110*00b67f09SDavid van Moolenbroek 	dns_difftuple_t *t = *tp;
111*00b67f09SDavid van Moolenbroek 	isc_mem_t *mctx;
112*00b67f09SDavid van Moolenbroek 
113*00b67f09SDavid van Moolenbroek 	REQUIRE(DNS_DIFFTUPLE_VALID(t));
114*00b67f09SDavid van Moolenbroek 
115*00b67f09SDavid van Moolenbroek 	dns_name_invalidate(&t->name);
116*00b67f09SDavid van Moolenbroek 	t->magic = 0;
117*00b67f09SDavid van Moolenbroek 	mctx = t->mctx;
118*00b67f09SDavid van Moolenbroek 	isc_mem_free(mctx, t);
119*00b67f09SDavid van Moolenbroek 	isc_mem_detach(&mctx);
120*00b67f09SDavid van Moolenbroek 	*tp = NULL;
121*00b67f09SDavid van Moolenbroek }
122*00b67f09SDavid van Moolenbroek 
123*00b67f09SDavid van Moolenbroek isc_result_t
dns_difftuple_copy(dns_difftuple_t * orig,dns_difftuple_t ** copyp)124*00b67f09SDavid van Moolenbroek dns_difftuple_copy(dns_difftuple_t *orig, dns_difftuple_t **copyp) {
125*00b67f09SDavid van Moolenbroek 	return (dns_difftuple_create(orig->mctx, orig->op, &orig->name,
126*00b67f09SDavid van Moolenbroek 				     orig->ttl, &orig->rdata, copyp));
127*00b67f09SDavid van Moolenbroek }
128*00b67f09SDavid van Moolenbroek 
129*00b67f09SDavid van Moolenbroek void
dns_diff_init(isc_mem_t * mctx,dns_diff_t * diff)130*00b67f09SDavid van Moolenbroek dns_diff_init(isc_mem_t *mctx, dns_diff_t *diff) {
131*00b67f09SDavid van Moolenbroek 	diff->mctx = mctx;
132*00b67f09SDavid van Moolenbroek 	ISC_LIST_INIT(diff->tuples);
133*00b67f09SDavid van Moolenbroek 	diff->magic = DNS_DIFF_MAGIC;
134*00b67f09SDavid van Moolenbroek }
135*00b67f09SDavid van Moolenbroek 
136*00b67f09SDavid van Moolenbroek void
dns_diff_clear(dns_diff_t * diff)137*00b67f09SDavid van Moolenbroek dns_diff_clear(dns_diff_t *diff) {
138*00b67f09SDavid van Moolenbroek 	dns_difftuple_t *t;
139*00b67f09SDavid van Moolenbroek 	REQUIRE(DNS_DIFF_VALID(diff));
140*00b67f09SDavid van Moolenbroek 	while ((t = ISC_LIST_HEAD(diff->tuples)) != NULL) {
141*00b67f09SDavid van Moolenbroek 		ISC_LIST_UNLINK(diff->tuples, t, link);
142*00b67f09SDavid van Moolenbroek 		dns_difftuple_free(&t);
143*00b67f09SDavid van Moolenbroek 	}
144*00b67f09SDavid van Moolenbroek 	ENSURE(ISC_LIST_EMPTY(diff->tuples));
145*00b67f09SDavid van Moolenbroek }
146*00b67f09SDavid van Moolenbroek 
147*00b67f09SDavid van Moolenbroek void
dns_diff_append(dns_diff_t * diff,dns_difftuple_t ** tuplep)148*00b67f09SDavid van Moolenbroek dns_diff_append(dns_diff_t *diff, dns_difftuple_t **tuplep)
149*00b67f09SDavid van Moolenbroek {
150*00b67f09SDavid van Moolenbroek 	ISC_LIST_APPEND(diff->tuples, *tuplep, link);
151*00b67f09SDavid van Moolenbroek 	*tuplep = NULL;
152*00b67f09SDavid van Moolenbroek }
153*00b67f09SDavid van Moolenbroek 
154*00b67f09SDavid van Moolenbroek /* XXX this is O(N) */
155*00b67f09SDavid van Moolenbroek 
156*00b67f09SDavid van Moolenbroek void
dns_diff_appendminimal(dns_diff_t * diff,dns_difftuple_t ** tuplep)157*00b67f09SDavid van Moolenbroek dns_diff_appendminimal(dns_diff_t *diff, dns_difftuple_t **tuplep)
158*00b67f09SDavid van Moolenbroek {
159*00b67f09SDavid van Moolenbroek 	dns_difftuple_t *ot, *next_ot;
160*00b67f09SDavid van Moolenbroek 
161*00b67f09SDavid van Moolenbroek 	REQUIRE(DNS_DIFF_VALID(diff));
162*00b67f09SDavid van Moolenbroek 	REQUIRE(DNS_DIFFTUPLE_VALID(*tuplep));
163*00b67f09SDavid van Moolenbroek 
164*00b67f09SDavid van Moolenbroek 	/*
165*00b67f09SDavid van Moolenbroek 	 * Look for an existing tuple with the same owner name,
166*00b67f09SDavid van Moolenbroek 	 * rdata, and TTL.   If we are doing an addition and find a
167*00b67f09SDavid van Moolenbroek 	 * deletion or vice versa, remove both the old and the
168*00b67f09SDavid van Moolenbroek 	 * new tuple since they cancel each other out (assuming
169*00b67f09SDavid van Moolenbroek 	 * that we never delete nonexistent data or add existing
170*00b67f09SDavid van Moolenbroek 	 * data).
171*00b67f09SDavid van Moolenbroek 	 *
172*00b67f09SDavid van Moolenbroek 	 * If we find an old update of the same kind as
173*00b67f09SDavid van Moolenbroek 	 * the one we are doing, there must be a programming
174*00b67f09SDavid van Moolenbroek 	 * error.  We report it but try to continue anyway.
175*00b67f09SDavid van Moolenbroek 	 */
176*00b67f09SDavid van Moolenbroek 	for (ot = ISC_LIST_HEAD(diff->tuples); ot != NULL;
177*00b67f09SDavid van Moolenbroek 	     ot = next_ot)
178*00b67f09SDavid van Moolenbroek 	{
179*00b67f09SDavid van Moolenbroek 		next_ot = ISC_LIST_NEXT(ot, link);
180*00b67f09SDavid van Moolenbroek 		if (dns_name_equal(&ot->name, &(*tuplep)->name) &&
181*00b67f09SDavid van Moolenbroek 		    dns_rdata_compare(&ot->rdata, &(*tuplep)->rdata) == 0 &&
182*00b67f09SDavid van Moolenbroek 		    ot->ttl == (*tuplep)->ttl)
183*00b67f09SDavid van Moolenbroek 		{
184*00b67f09SDavid van Moolenbroek 			ISC_LIST_UNLINK(diff->tuples, ot, link);
185*00b67f09SDavid van Moolenbroek 			if ((*tuplep)->op == ot->op) {
186*00b67f09SDavid van Moolenbroek 				UNEXPECTED_ERROR(__FILE__, __LINE__,
187*00b67f09SDavid van Moolenbroek 					 "unexpected non-minimal diff");
188*00b67f09SDavid van Moolenbroek 			} else {
189*00b67f09SDavid van Moolenbroek 				dns_difftuple_free(tuplep);
190*00b67f09SDavid van Moolenbroek 			}
191*00b67f09SDavid van Moolenbroek 			dns_difftuple_free(&ot);
192*00b67f09SDavid van Moolenbroek 			break;
193*00b67f09SDavid van Moolenbroek 		}
194*00b67f09SDavid van Moolenbroek 	}
195*00b67f09SDavid van Moolenbroek 
196*00b67f09SDavid van Moolenbroek 	if (*tuplep != NULL) {
197*00b67f09SDavid van Moolenbroek 		ISC_LIST_APPEND(diff->tuples, *tuplep, link);
198*00b67f09SDavid van Moolenbroek 		*tuplep = NULL;
199*00b67f09SDavid van Moolenbroek 	}
200*00b67f09SDavid van Moolenbroek 
201*00b67f09SDavid van Moolenbroek 	ENSURE(*tuplep == NULL);
202*00b67f09SDavid van Moolenbroek }
203*00b67f09SDavid van Moolenbroek 
204*00b67f09SDavid van Moolenbroek static isc_stdtime_t
setresign(dns_rdataset_t * modified)205*00b67f09SDavid van Moolenbroek setresign(dns_rdataset_t *modified) {
206*00b67f09SDavid van Moolenbroek 	dns_rdata_t rdata = DNS_RDATA_INIT;
207*00b67f09SDavid van Moolenbroek 	dns_rdata_rrsig_t sig;
208*00b67f09SDavid van Moolenbroek 	isc_stdtime_t when;
209*00b67f09SDavid van Moolenbroek 	isc_result_t result;
210*00b67f09SDavid van Moolenbroek 
211*00b67f09SDavid van Moolenbroek 	result = dns_rdataset_first(modified);
212*00b67f09SDavid van Moolenbroek 	INSIST(result == ISC_R_SUCCESS);
213*00b67f09SDavid van Moolenbroek 	dns_rdataset_current(modified, &rdata);
214*00b67f09SDavid van Moolenbroek 	(void)dns_rdata_tostruct(&rdata, &sig, NULL);
215*00b67f09SDavid van Moolenbroek 	if ((rdata.flags & DNS_RDATA_OFFLINE) != 0)
216*00b67f09SDavid van Moolenbroek 		when = 0;
217*00b67f09SDavid van Moolenbroek 	else
218*00b67f09SDavid van Moolenbroek 		when = sig.timeexpire;
219*00b67f09SDavid van Moolenbroek 	dns_rdata_reset(&rdata);
220*00b67f09SDavid van Moolenbroek 
221*00b67f09SDavid van Moolenbroek 	result = dns_rdataset_next(modified);
222*00b67f09SDavid van Moolenbroek 	while (result == ISC_R_SUCCESS) {
223*00b67f09SDavid van Moolenbroek 		dns_rdataset_current(modified, &rdata);
224*00b67f09SDavid van Moolenbroek 		(void)dns_rdata_tostruct(&rdata, &sig, NULL);
225*00b67f09SDavid van Moolenbroek 		if ((rdata.flags & DNS_RDATA_OFFLINE) != 0) {
226*00b67f09SDavid van Moolenbroek 			goto next_rr;
227*00b67f09SDavid van Moolenbroek 		}
228*00b67f09SDavid van Moolenbroek 		if (when == 0 || sig.timeexpire < when)
229*00b67f09SDavid van Moolenbroek 			when = sig.timeexpire;
230*00b67f09SDavid van Moolenbroek  next_rr:
231*00b67f09SDavid van Moolenbroek 		dns_rdata_reset(&rdata);
232*00b67f09SDavid van Moolenbroek 		result = dns_rdataset_next(modified);
233*00b67f09SDavid van Moolenbroek 	}
234*00b67f09SDavid van Moolenbroek 	INSIST(result == ISC_R_NOMORE);
235*00b67f09SDavid van Moolenbroek 	return (when);
236*00b67f09SDavid van Moolenbroek }
237*00b67f09SDavid van Moolenbroek 
238*00b67f09SDavid van Moolenbroek static isc_result_t
diff_apply(dns_diff_t * diff,dns_db_t * db,dns_dbversion_t * ver,isc_boolean_t warn)239*00b67f09SDavid van Moolenbroek diff_apply(dns_diff_t *diff, dns_db_t *db, dns_dbversion_t *ver,
240*00b67f09SDavid van Moolenbroek 	   isc_boolean_t warn)
241*00b67f09SDavid van Moolenbroek {
242*00b67f09SDavid van Moolenbroek 	dns_difftuple_t *t;
243*00b67f09SDavid van Moolenbroek 	dns_dbnode_t *node = NULL;
244*00b67f09SDavid van Moolenbroek 	isc_result_t result;
245*00b67f09SDavid van Moolenbroek 	char namebuf[DNS_NAME_FORMATSIZE];
246*00b67f09SDavid van Moolenbroek 	char typebuf[DNS_RDATATYPE_FORMATSIZE];
247*00b67f09SDavid van Moolenbroek 	char classbuf[DNS_RDATACLASS_FORMATSIZE];
248*00b67f09SDavid van Moolenbroek 
249*00b67f09SDavid van Moolenbroek 	REQUIRE(DNS_DIFF_VALID(diff));
250*00b67f09SDavid van Moolenbroek 	REQUIRE(DNS_DB_VALID(db));
251*00b67f09SDavid van Moolenbroek 
252*00b67f09SDavid van Moolenbroek 	t = ISC_LIST_HEAD(diff->tuples);
253*00b67f09SDavid van Moolenbroek 	while (t != NULL) {
254*00b67f09SDavid van Moolenbroek 		dns_name_t *name;
255*00b67f09SDavid van Moolenbroek 
256*00b67f09SDavid van Moolenbroek 		INSIST(node == NULL);
257*00b67f09SDavid van Moolenbroek 		name = &t->name;
258*00b67f09SDavid van Moolenbroek 		/*
259*00b67f09SDavid van Moolenbroek 		 * Find the node.
260*00b67f09SDavid van Moolenbroek 		 * We create the node if it does not exist.
261*00b67f09SDavid van Moolenbroek 		 * This will cause an empty node to be created if the diff
262*00b67f09SDavid van Moolenbroek 		 * contains a deletion of an RR at a nonexistent name,
263*00b67f09SDavid van Moolenbroek 		 * but such diffs should never be created in the first
264*00b67f09SDavid van Moolenbroek 		 * place.
265*00b67f09SDavid van Moolenbroek 		 */
266*00b67f09SDavid van Moolenbroek 
267*00b67f09SDavid van Moolenbroek 		while (t != NULL && dns_name_equal(&t->name, name)) {
268*00b67f09SDavid van Moolenbroek 			dns_rdatatype_t type, covers;
269*00b67f09SDavid van Moolenbroek 			dns_diffop_t op;
270*00b67f09SDavid van Moolenbroek 			dns_rdatalist_t rdl;
271*00b67f09SDavid van Moolenbroek 			dns_rdataset_t rds;
272*00b67f09SDavid van Moolenbroek 			dns_rdataset_t ardataset;
273*00b67f09SDavid van Moolenbroek 			dns_rdataset_t *modified = NULL;
274*00b67f09SDavid van Moolenbroek 
275*00b67f09SDavid van Moolenbroek 			op = t->op;
276*00b67f09SDavid van Moolenbroek 			type = t->rdata.type;
277*00b67f09SDavid van Moolenbroek 			covers = rdata_covers(&t->rdata);
278*00b67f09SDavid van Moolenbroek 
279*00b67f09SDavid van Moolenbroek 			/*
280*00b67f09SDavid van Moolenbroek 			 * Collect a contiguous set of updates with
281*00b67f09SDavid van Moolenbroek 			 * the same operation (add/delete) and RR type
282*00b67f09SDavid van Moolenbroek 			 * into a single rdatalist so that the
283*00b67f09SDavid van Moolenbroek 			 * database rrset merging/subtraction code
284*00b67f09SDavid van Moolenbroek 			 * can work more efficiently than if each
285*00b67f09SDavid van Moolenbroek 			 * RR were merged into / subtracted from
286*00b67f09SDavid van Moolenbroek 			 * the database separately.
287*00b67f09SDavid van Moolenbroek 			 *
288*00b67f09SDavid van Moolenbroek 			 * This is done by linking rdata structures from the
289*00b67f09SDavid van Moolenbroek 			 * diff into "rdatalist".  This uses the rdata link
290*00b67f09SDavid van Moolenbroek 			 * field, not the diff link field, so the structure
291*00b67f09SDavid van Moolenbroek 			 * of the diff itself is not affected.
292*00b67f09SDavid van Moolenbroek 			 */
293*00b67f09SDavid van Moolenbroek 
294*00b67f09SDavid van Moolenbroek 			rdl.type = type;
295*00b67f09SDavid van Moolenbroek 			rdl.covers = covers;
296*00b67f09SDavid van Moolenbroek 			rdl.rdclass = t->rdata.rdclass;
297*00b67f09SDavid van Moolenbroek 			rdl.ttl = t->ttl;
298*00b67f09SDavid van Moolenbroek 			ISC_LIST_INIT(rdl.rdata);
299*00b67f09SDavid van Moolenbroek 			ISC_LINK_INIT(&rdl, link);
300*00b67f09SDavid van Moolenbroek 
301*00b67f09SDavid van Moolenbroek 			node = NULL;
302*00b67f09SDavid van Moolenbroek 			if (type != dns_rdatatype_nsec3 &&
303*00b67f09SDavid van Moolenbroek 			    covers != dns_rdatatype_nsec3)
304*00b67f09SDavid van Moolenbroek 				CHECK(dns_db_findnode(db, name, ISC_TRUE,
305*00b67f09SDavid van Moolenbroek 						      &node));
306*00b67f09SDavid van Moolenbroek 			else
307*00b67f09SDavid van Moolenbroek 				CHECK(dns_db_findnsec3node(db, name, ISC_TRUE,
308*00b67f09SDavid van Moolenbroek 							   &node));
309*00b67f09SDavid van Moolenbroek 
310*00b67f09SDavid van Moolenbroek 			while (t != NULL &&
311*00b67f09SDavid van Moolenbroek 			       dns_name_equal(&t->name, name) &&
312*00b67f09SDavid van Moolenbroek 			       t->op == op &&
313*00b67f09SDavid van Moolenbroek 			       t->rdata.type == type &&
314*00b67f09SDavid van Moolenbroek 			       rdata_covers(&t->rdata) == covers)
315*00b67f09SDavid van Moolenbroek 			{
316*00b67f09SDavid van Moolenbroek 				dns_name_format(name, namebuf, sizeof(namebuf));
317*00b67f09SDavid van Moolenbroek 				dns_rdatatype_format(t->rdata.type, typebuf,
318*00b67f09SDavid van Moolenbroek 						     sizeof(typebuf));
319*00b67f09SDavid van Moolenbroek 				dns_rdataclass_format(t->rdata.rdclass,
320*00b67f09SDavid van Moolenbroek 						      classbuf,
321*00b67f09SDavid van Moolenbroek 						      sizeof(classbuf));
322*00b67f09SDavid van Moolenbroek 				if (t->ttl != rdl.ttl && warn)
323*00b67f09SDavid van Moolenbroek 					isc_log_write(DIFF_COMMON_LOGARGS,
324*00b67f09SDavid van Moolenbroek 						ISC_LOG_WARNING,
325*00b67f09SDavid van Moolenbroek 						"'%s/%s/%s': TTL differs in "
326*00b67f09SDavid van Moolenbroek 						"rdataset, adjusting "
327*00b67f09SDavid van Moolenbroek 						"%lu -> %lu",
328*00b67f09SDavid van Moolenbroek 						namebuf, typebuf, classbuf,
329*00b67f09SDavid van Moolenbroek 						(unsigned long) t->ttl,
330*00b67f09SDavid van Moolenbroek 						(unsigned long) rdl.ttl);
331*00b67f09SDavid van Moolenbroek 				ISC_LIST_APPEND(rdl.rdata, &t->rdata, link);
332*00b67f09SDavid van Moolenbroek 				t = ISC_LIST_NEXT(t, link);
333*00b67f09SDavid van Moolenbroek 			}
334*00b67f09SDavid van Moolenbroek 
335*00b67f09SDavid van Moolenbroek 			/*
336*00b67f09SDavid van Moolenbroek 			 * Convert the rdatalist into a rdataset.
337*00b67f09SDavid van Moolenbroek 			 */
338*00b67f09SDavid van Moolenbroek 			dns_rdataset_init(&rds);
339*00b67f09SDavid van Moolenbroek 			CHECK(dns_rdatalist_tordataset(&rdl, &rds));
340*00b67f09SDavid van Moolenbroek 			if (rds.type == dns_rdatatype_rrsig)
341*00b67f09SDavid van Moolenbroek 				switch (op) {
342*00b67f09SDavid van Moolenbroek 				case DNS_DIFFOP_ADDRESIGN:
343*00b67f09SDavid van Moolenbroek 				case DNS_DIFFOP_DELRESIGN:
344*00b67f09SDavid van Moolenbroek 					modified = &ardataset;
345*00b67f09SDavid van Moolenbroek 					dns_rdataset_init(modified);
346*00b67f09SDavid van Moolenbroek 					break;
347*00b67f09SDavid van Moolenbroek 				default:
348*00b67f09SDavid van Moolenbroek 					break;
349*00b67f09SDavid van Moolenbroek 				}
350*00b67f09SDavid van Moolenbroek 			rds.trust = dns_trust_ultimate;
351*00b67f09SDavid van Moolenbroek 
352*00b67f09SDavid van Moolenbroek 			/*
353*00b67f09SDavid van Moolenbroek 			 * Merge the rdataset into the database.
354*00b67f09SDavid van Moolenbroek 			 */
355*00b67f09SDavid van Moolenbroek 			switch (op) {
356*00b67f09SDavid van Moolenbroek 			case DNS_DIFFOP_ADD:
357*00b67f09SDavid van Moolenbroek 			case DNS_DIFFOP_ADDRESIGN:
358*00b67f09SDavid van Moolenbroek 				result = dns_db_addrdataset(db, node, ver,
359*00b67f09SDavid van Moolenbroek 							    0, &rds,
360*00b67f09SDavid van Moolenbroek 							    DNS_DBADD_MERGE|
361*00b67f09SDavid van Moolenbroek 							    DNS_DBADD_EXACT|
362*00b67f09SDavid van Moolenbroek 							    DNS_DBADD_EXACTTTL,
363*00b67f09SDavid van Moolenbroek 							    modified);
364*00b67f09SDavid van Moolenbroek 				break;
365*00b67f09SDavid van Moolenbroek 			case DNS_DIFFOP_DEL:
366*00b67f09SDavid van Moolenbroek 			case DNS_DIFFOP_DELRESIGN:
367*00b67f09SDavid van Moolenbroek 				result = dns_db_subtractrdataset(db, node, ver,
368*00b67f09SDavid van Moolenbroek 							       &rds,
369*00b67f09SDavid van Moolenbroek 							       DNS_DBSUB_EXACT,
370*00b67f09SDavid van Moolenbroek 							       modified);
371*00b67f09SDavid van Moolenbroek 				break;
372*00b67f09SDavid van Moolenbroek 			default:
373*00b67f09SDavid van Moolenbroek 				INSIST(0);
374*00b67f09SDavid van Moolenbroek 			}
375*00b67f09SDavid van Moolenbroek 
376*00b67f09SDavid van Moolenbroek 			if (result == ISC_R_SUCCESS) {
377*00b67f09SDavid van Moolenbroek 				if (modified != NULL) {
378*00b67f09SDavid van Moolenbroek 					isc_stdtime_t resign;
379*00b67f09SDavid van Moolenbroek 					resign = setresign(modified);
380*00b67f09SDavid van Moolenbroek 					dns_db_setsigningtime(db, modified,
381*00b67f09SDavid van Moolenbroek 							      resign);
382*00b67f09SDavid van Moolenbroek 				}
383*00b67f09SDavid van Moolenbroek 			} else if (result == DNS_R_UNCHANGED) {
384*00b67f09SDavid van Moolenbroek 				/*
385*00b67f09SDavid van Moolenbroek 				 * This will not happen when executing a
386*00b67f09SDavid van Moolenbroek 				 * dynamic update, because that code will
387*00b67f09SDavid van Moolenbroek 				 * generate strictly minimal diffs.
388*00b67f09SDavid van Moolenbroek 				 * It may happen when receiving an IXFR
389*00b67f09SDavid van Moolenbroek 				 * from a server that is not as careful.
390*00b67f09SDavid van Moolenbroek 				 * Issue a warning and continue.
391*00b67f09SDavid van Moolenbroek 				 */
392*00b67f09SDavid van Moolenbroek 				if (warn) {
393*00b67f09SDavid van Moolenbroek 					dns_name_format(dns_db_origin(db),
394*00b67f09SDavid van Moolenbroek 							namebuf,
395*00b67f09SDavid van Moolenbroek 							sizeof(namebuf));
396*00b67f09SDavid van Moolenbroek 					dns_rdataclass_format(dns_db_class(db),
397*00b67f09SDavid van Moolenbroek 							      classbuf,
398*00b67f09SDavid van Moolenbroek 							      sizeof(classbuf));
399*00b67f09SDavid van Moolenbroek 					isc_log_write(DIFF_COMMON_LOGARGS,
400*00b67f09SDavid van Moolenbroek 						      ISC_LOG_WARNING,
401*00b67f09SDavid van Moolenbroek 						      "%s/%s: dns_diff_apply: "
402*00b67f09SDavid van Moolenbroek 						      "update with no effect",
403*00b67f09SDavid van Moolenbroek 						      namebuf, classbuf);
404*00b67f09SDavid van Moolenbroek 				}
405*00b67f09SDavid van Moolenbroek 			} else if (result == DNS_R_NXRRSET) {
406*00b67f09SDavid van Moolenbroek 				/*
407*00b67f09SDavid van Moolenbroek 				 * OK.
408*00b67f09SDavid van Moolenbroek 				 */
409*00b67f09SDavid van Moolenbroek 			} else {
410*00b67f09SDavid van Moolenbroek 				if (modified != NULL &&
411*00b67f09SDavid van Moolenbroek 				    dns_rdataset_isassociated(modified))
412*00b67f09SDavid van Moolenbroek 					dns_rdataset_disassociate(modified);
413*00b67f09SDavid van Moolenbroek 				CHECK(result);
414*00b67f09SDavid van Moolenbroek 			}
415*00b67f09SDavid van Moolenbroek 			dns_db_detachnode(db, &node);
416*00b67f09SDavid van Moolenbroek 			if (modified != NULL &&
417*00b67f09SDavid van Moolenbroek 			    dns_rdataset_isassociated(modified))
418*00b67f09SDavid van Moolenbroek 				dns_rdataset_disassociate(modified);
419*00b67f09SDavid van Moolenbroek 		}
420*00b67f09SDavid van Moolenbroek 	}
421*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
422*00b67f09SDavid van Moolenbroek 
423*00b67f09SDavid van Moolenbroek  failure:
424*00b67f09SDavid van Moolenbroek 	if (node != NULL)
425*00b67f09SDavid van Moolenbroek 		dns_db_detachnode(db, &node);
426*00b67f09SDavid van Moolenbroek 	return (result);
427*00b67f09SDavid van Moolenbroek }
428*00b67f09SDavid van Moolenbroek 
429*00b67f09SDavid van Moolenbroek isc_result_t
dns_diff_apply(dns_diff_t * diff,dns_db_t * db,dns_dbversion_t * ver)430*00b67f09SDavid van Moolenbroek dns_diff_apply(dns_diff_t *diff, dns_db_t *db, dns_dbversion_t *ver) {
431*00b67f09SDavid van Moolenbroek 	return (diff_apply(diff, db, ver, ISC_TRUE));
432*00b67f09SDavid van Moolenbroek }
433*00b67f09SDavid van Moolenbroek 
434*00b67f09SDavid van Moolenbroek isc_result_t
dns_diff_applysilently(dns_diff_t * diff,dns_db_t * db,dns_dbversion_t * ver)435*00b67f09SDavid van Moolenbroek dns_diff_applysilently(dns_diff_t *diff, dns_db_t *db, dns_dbversion_t *ver) {
436*00b67f09SDavid van Moolenbroek 	return (diff_apply(diff, db, ver, ISC_FALSE));
437*00b67f09SDavid van Moolenbroek }
438*00b67f09SDavid van Moolenbroek 
439*00b67f09SDavid van Moolenbroek /* XXX this duplicates lots of code in diff_apply(). */
440*00b67f09SDavid van Moolenbroek 
441*00b67f09SDavid van Moolenbroek isc_result_t
dns_diff_load(dns_diff_t * diff,dns_addrdatasetfunc_t addfunc,void * add_private)442*00b67f09SDavid van Moolenbroek dns_diff_load(dns_diff_t *diff, dns_addrdatasetfunc_t addfunc,
443*00b67f09SDavid van Moolenbroek 	      void *add_private)
444*00b67f09SDavid van Moolenbroek {
445*00b67f09SDavid van Moolenbroek 	dns_difftuple_t *t;
446*00b67f09SDavid van Moolenbroek 	isc_result_t result;
447*00b67f09SDavid van Moolenbroek 
448*00b67f09SDavid van Moolenbroek 	REQUIRE(DNS_DIFF_VALID(diff));
449*00b67f09SDavid van Moolenbroek 
450*00b67f09SDavid van Moolenbroek 	t = ISC_LIST_HEAD(diff->tuples);
451*00b67f09SDavid van Moolenbroek 	while (t != NULL) {
452*00b67f09SDavid van Moolenbroek 		dns_name_t *name;
453*00b67f09SDavid van Moolenbroek 
454*00b67f09SDavid van Moolenbroek 		name = &t->name;
455*00b67f09SDavid van Moolenbroek 		while (t != NULL && dns_name_equal(&t->name, name)) {
456*00b67f09SDavid van Moolenbroek 			dns_rdatatype_t type, covers;
457*00b67f09SDavid van Moolenbroek 			dns_diffop_t op;
458*00b67f09SDavid van Moolenbroek 			dns_rdatalist_t rdl;
459*00b67f09SDavid van Moolenbroek 			dns_rdataset_t rds;
460*00b67f09SDavid van Moolenbroek 
461*00b67f09SDavid van Moolenbroek 			op = t->op;
462*00b67f09SDavid van Moolenbroek 			type = t->rdata.type;
463*00b67f09SDavid van Moolenbroek 			covers = rdata_covers(&t->rdata);
464*00b67f09SDavid van Moolenbroek 
465*00b67f09SDavid van Moolenbroek 			rdl.type = type;
466*00b67f09SDavid van Moolenbroek 			rdl.covers = covers;
467*00b67f09SDavid van Moolenbroek 			rdl.rdclass = t->rdata.rdclass;
468*00b67f09SDavid van Moolenbroek 			rdl.ttl = t->ttl;
469*00b67f09SDavid van Moolenbroek 			ISC_LIST_INIT(rdl.rdata);
470*00b67f09SDavid van Moolenbroek 			ISC_LINK_INIT(&rdl, link);
471*00b67f09SDavid van Moolenbroek 
472*00b67f09SDavid van Moolenbroek 			while (t != NULL && dns_name_equal(&t->name, name) &&
473*00b67f09SDavid van Moolenbroek 			       t->op == op && t->rdata.type == type &&
474*00b67f09SDavid van Moolenbroek 			       rdata_covers(&t->rdata) == covers)
475*00b67f09SDavid van Moolenbroek 			{
476*00b67f09SDavid van Moolenbroek 				ISC_LIST_APPEND(rdl.rdata, &t->rdata, link);
477*00b67f09SDavid van Moolenbroek 				t = ISC_LIST_NEXT(t, link);
478*00b67f09SDavid van Moolenbroek 			}
479*00b67f09SDavid van Moolenbroek 
480*00b67f09SDavid van Moolenbroek 			/*
481*00b67f09SDavid van Moolenbroek 			 * Convert the rdatalist into a rdataset.
482*00b67f09SDavid van Moolenbroek 			 */
483*00b67f09SDavid van Moolenbroek 			dns_rdataset_init(&rds);
484*00b67f09SDavid van Moolenbroek 			CHECK(dns_rdatalist_tordataset(&rdl, &rds));
485*00b67f09SDavid van Moolenbroek 			rds.trust = dns_trust_ultimate;
486*00b67f09SDavid van Moolenbroek 
487*00b67f09SDavid van Moolenbroek 			INSIST(op == DNS_DIFFOP_ADD);
488*00b67f09SDavid van Moolenbroek 			result = (*addfunc)(add_private, name, &rds);
489*00b67f09SDavid van Moolenbroek 			if (result == DNS_R_UNCHANGED) {
490*00b67f09SDavid van Moolenbroek 				isc_log_write(DIFF_COMMON_LOGARGS,
491*00b67f09SDavid van Moolenbroek 					      ISC_LOG_WARNING,
492*00b67f09SDavid van Moolenbroek 					      "dns_diff_load: "
493*00b67f09SDavid van Moolenbroek 					      "update with no effect");
494*00b67f09SDavid van Moolenbroek 			} else if (result == ISC_R_SUCCESS ||
495*00b67f09SDavid van Moolenbroek 				   result == DNS_R_NXRRSET) {
496*00b67f09SDavid van Moolenbroek 				/*
497*00b67f09SDavid van Moolenbroek 				 * OK.
498*00b67f09SDavid van Moolenbroek 				 */
499*00b67f09SDavid van Moolenbroek 			} else {
500*00b67f09SDavid van Moolenbroek 				CHECK(result);
501*00b67f09SDavid van Moolenbroek 			}
502*00b67f09SDavid van Moolenbroek 		}
503*00b67f09SDavid van Moolenbroek 	}
504*00b67f09SDavid van Moolenbroek 	result = ISC_R_SUCCESS;
505*00b67f09SDavid van Moolenbroek  failure:
506*00b67f09SDavid van Moolenbroek 	return (result);
507*00b67f09SDavid van Moolenbroek }
508*00b67f09SDavid van Moolenbroek 
509*00b67f09SDavid van Moolenbroek /*
510*00b67f09SDavid van Moolenbroek  * XXX uses qsort(); a merge sort would be more natural for lists,
511*00b67f09SDavid van Moolenbroek  * and perhaps safer wrt thread stack overflow.
512*00b67f09SDavid van Moolenbroek  */
513*00b67f09SDavid van Moolenbroek isc_result_t
dns_diff_sort(dns_diff_t * diff,dns_diff_compare_func * compare)514*00b67f09SDavid van Moolenbroek dns_diff_sort(dns_diff_t *diff, dns_diff_compare_func *compare) {
515*00b67f09SDavid van Moolenbroek 	unsigned int length = 0;
516*00b67f09SDavid van Moolenbroek 	unsigned int i;
517*00b67f09SDavid van Moolenbroek 	dns_difftuple_t **v;
518*00b67f09SDavid van Moolenbroek 	dns_difftuple_t *p;
519*00b67f09SDavid van Moolenbroek 	REQUIRE(DNS_DIFF_VALID(diff));
520*00b67f09SDavid van Moolenbroek 
521*00b67f09SDavid van Moolenbroek 	for (p = ISC_LIST_HEAD(diff->tuples);
522*00b67f09SDavid van Moolenbroek 	     p != NULL;
523*00b67f09SDavid van Moolenbroek 	     p = ISC_LIST_NEXT(p, link))
524*00b67f09SDavid van Moolenbroek 		length++;
525*00b67f09SDavid van Moolenbroek 	if (length == 0)
526*00b67f09SDavid van Moolenbroek 		return (ISC_R_SUCCESS);
527*00b67f09SDavid van Moolenbroek 	v = isc_mem_get(diff->mctx, length * sizeof(dns_difftuple_t *));
528*00b67f09SDavid van Moolenbroek 	if (v == NULL)
529*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOMEMORY);
530*00b67f09SDavid van Moolenbroek 	for (i = 0; i < length; i++) {
531*00b67f09SDavid van Moolenbroek 		p = ISC_LIST_HEAD(diff->tuples);
532*00b67f09SDavid van Moolenbroek 		v[i] = p;
533*00b67f09SDavid van Moolenbroek 		ISC_LIST_UNLINK(diff->tuples, p, link);
534*00b67f09SDavid van Moolenbroek 	}
535*00b67f09SDavid van Moolenbroek 	INSIST(ISC_LIST_HEAD(diff->tuples) == NULL);
536*00b67f09SDavid van Moolenbroek 	qsort(v, length, sizeof(v[0]), compare);
537*00b67f09SDavid van Moolenbroek 	for (i = 0; i < length; i++) {
538*00b67f09SDavid van Moolenbroek 		ISC_LIST_APPEND(diff->tuples, v[i], link);
539*00b67f09SDavid van Moolenbroek 	}
540*00b67f09SDavid van Moolenbroek 	isc_mem_put(diff->mctx, v, length * sizeof(dns_difftuple_t *));
541*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
542*00b67f09SDavid van Moolenbroek }
543*00b67f09SDavid van Moolenbroek 
544*00b67f09SDavid van Moolenbroek 
545*00b67f09SDavid van Moolenbroek /*
546*00b67f09SDavid van Moolenbroek  * Create an rdataset containing the single RR of the given
547*00b67f09SDavid van Moolenbroek  * tuple.  The caller must allocate the rdata, rdataset and
548*00b67f09SDavid van Moolenbroek  * an rdatalist structure for it to refer to.
549*00b67f09SDavid van Moolenbroek  */
550*00b67f09SDavid van Moolenbroek 
551*00b67f09SDavid van Moolenbroek static isc_result_t
diff_tuple_tordataset(dns_difftuple_t * t,dns_rdata_t * rdata,dns_rdatalist_t * rdl,dns_rdataset_t * rds)552*00b67f09SDavid van Moolenbroek diff_tuple_tordataset(dns_difftuple_t *t, dns_rdata_t *rdata,
553*00b67f09SDavid van Moolenbroek 		      dns_rdatalist_t *rdl, dns_rdataset_t *rds)
554*00b67f09SDavid van Moolenbroek {
555*00b67f09SDavid van Moolenbroek 	REQUIRE(DNS_DIFFTUPLE_VALID(t));
556*00b67f09SDavid van Moolenbroek 	REQUIRE(rdl != NULL);
557*00b67f09SDavid van Moolenbroek 	REQUIRE(rds != NULL);
558*00b67f09SDavid van Moolenbroek 
559*00b67f09SDavid van Moolenbroek 	rdl->type = t->rdata.type;
560*00b67f09SDavid van Moolenbroek 	rdl->rdclass = t->rdata.rdclass;
561*00b67f09SDavid van Moolenbroek 	rdl->ttl = t->ttl;
562*00b67f09SDavid van Moolenbroek 	ISC_LIST_INIT(rdl->rdata);
563*00b67f09SDavid van Moolenbroek 	ISC_LINK_INIT(rdl, link);
564*00b67f09SDavid van Moolenbroek 	dns_rdataset_init(rds);
565*00b67f09SDavid van Moolenbroek 	ISC_LINK_INIT(rdata, link);
566*00b67f09SDavid van Moolenbroek 	dns_rdata_clone(&t->rdata, rdata);
567*00b67f09SDavid van Moolenbroek 	ISC_LIST_APPEND(rdl->rdata, rdata, link);
568*00b67f09SDavid van Moolenbroek 	return (dns_rdatalist_tordataset(rdl, rds));
569*00b67f09SDavid van Moolenbroek }
570*00b67f09SDavid van Moolenbroek 
571*00b67f09SDavid van Moolenbroek isc_result_t
dns_diff_print(dns_diff_t * diff,FILE * file)572*00b67f09SDavid van Moolenbroek dns_diff_print(dns_diff_t *diff, FILE *file) {
573*00b67f09SDavid van Moolenbroek 	isc_result_t result;
574*00b67f09SDavid van Moolenbroek 	dns_difftuple_t *t;
575*00b67f09SDavid van Moolenbroek 	char *mem = NULL;
576*00b67f09SDavid van Moolenbroek 	unsigned int size = 2048;
577*00b67f09SDavid van Moolenbroek 	const char *op = NULL;
578*00b67f09SDavid van Moolenbroek 
579*00b67f09SDavid van Moolenbroek 	REQUIRE(DNS_DIFF_VALID(diff));
580*00b67f09SDavid van Moolenbroek 
581*00b67f09SDavid van Moolenbroek 	mem = isc_mem_get(diff->mctx, size);
582*00b67f09SDavid van Moolenbroek 	if (mem == NULL)
583*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOMEMORY);
584*00b67f09SDavid van Moolenbroek 
585*00b67f09SDavid van Moolenbroek 	for (t = ISC_LIST_HEAD(diff->tuples); t != NULL;
586*00b67f09SDavid van Moolenbroek 	     t = ISC_LIST_NEXT(t, link))
587*00b67f09SDavid van Moolenbroek 	{
588*00b67f09SDavid van Moolenbroek 		isc_buffer_t buf;
589*00b67f09SDavid van Moolenbroek 		isc_region_t r;
590*00b67f09SDavid van Moolenbroek 
591*00b67f09SDavid van Moolenbroek 		dns_rdatalist_t rdl;
592*00b67f09SDavid van Moolenbroek 		dns_rdataset_t rds;
593*00b67f09SDavid van Moolenbroek 		dns_rdata_t rd = DNS_RDATA_INIT;
594*00b67f09SDavid van Moolenbroek 
595*00b67f09SDavid van Moolenbroek 		result = diff_tuple_tordataset(t, &rd, &rdl, &rds);
596*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS) {
597*00b67f09SDavid van Moolenbroek 			UNEXPECTED_ERROR(__FILE__, __LINE__,
598*00b67f09SDavid van Moolenbroek 					 "diff_tuple_tordataset failed: %s",
599*00b67f09SDavid van Moolenbroek 					 dns_result_totext(result));
600*00b67f09SDavid van Moolenbroek 			result =  ISC_R_UNEXPECTED;
601*00b67f09SDavid van Moolenbroek 			goto cleanup;
602*00b67f09SDavid van Moolenbroek 		}
603*00b67f09SDavid van Moolenbroek  again:
604*00b67f09SDavid van Moolenbroek 		isc_buffer_init(&buf, mem, size);
605*00b67f09SDavid van Moolenbroek 		result = dns_rdataset_totext(&rds, &t->name,
606*00b67f09SDavid van Moolenbroek 					     ISC_FALSE, ISC_FALSE, &buf);
607*00b67f09SDavid van Moolenbroek 
608*00b67f09SDavid van Moolenbroek 		if (result == ISC_R_NOSPACE) {
609*00b67f09SDavid van Moolenbroek 			isc_mem_put(diff->mctx, mem, size);
610*00b67f09SDavid van Moolenbroek 			size += 1024;
611*00b67f09SDavid van Moolenbroek 			mem = isc_mem_get(diff->mctx, size);
612*00b67f09SDavid van Moolenbroek 			if (mem == NULL) {
613*00b67f09SDavid van Moolenbroek 				result = ISC_R_NOMEMORY;
614*00b67f09SDavid van Moolenbroek 				goto cleanup;
615*00b67f09SDavid van Moolenbroek 			}
616*00b67f09SDavid van Moolenbroek 			goto again;
617*00b67f09SDavid van Moolenbroek 		}
618*00b67f09SDavid van Moolenbroek 
619*00b67f09SDavid van Moolenbroek 		if (result != ISC_R_SUCCESS)
620*00b67f09SDavid van Moolenbroek 			goto cleanup;
621*00b67f09SDavid van Moolenbroek 		/*
622*00b67f09SDavid van Moolenbroek 		 * Get rid of final newline.
623*00b67f09SDavid van Moolenbroek 		 */
624*00b67f09SDavid van Moolenbroek 		INSIST(buf.used >= 1 &&
625*00b67f09SDavid van Moolenbroek 		       ((char *) buf.base)[buf.used-1] == '\n');
626*00b67f09SDavid van Moolenbroek 		buf.used--;
627*00b67f09SDavid van Moolenbroek 
628*00b67f09SDavid van Moolenbroek 		isc_buffer_usedregion(&buf, &r);
629*00b67f09SDavid van Moolenbroek 		switch (t->op) {
630*00b67f09SDavid van Moolenbroek 		case DNS_DIFFOP_EXISTS: op = "exists"; break;
631*00b67f09SDavid van Moolenbroek 		case DNS_DIFFOP_ADD: op = "add"; break;
632*00b67f09SDavid van Moolenbroek 		case DNS_DIFFOP_DEL: op = "del"; break;
633*00b67f09SDavid van Moolenbroek 		case DNS_DIFFOP_ADDRESIGN: op = "add re-sign"; break;
634*00b67f09SDavid van Moolenbroek 		case DNS_DIFFOP_DELRESIGN: op = "del re-sign"; break;
635*00b67f09SDavid van Moolenbroek 		}
636*00b67f09SDavid van Moolenbroek 		if (file != NULL)
637*00b67f09SDavid van Moolenbroek 			fprintf(file, "%s %.*s\n", op, (int) r.length,
638*00b67f09SDavid van Moolenbroek 				(char *) r.base);
639*00b67f09SDavid van Moolenbroek 		else
640*00b67f09SDavid van Moolenbroek 			isc_log_write(DIFF_COMMON_LOGARGS, ISC_LOG_DEBUG(7),
641*00b67f09SDavid van Moolenbroek 				      "%s %.*s", op, (int) r.length,
642*00b67f09SDavid van Moolenbroek 				      (char *) r.base);
643*00b67f09SDavid van Moolenbroek 	}
644*00b67f09SDavid van Moolenbroek 	result = ISC_R_SUCCESS;
645*00b67f09SDavid van Moolenbroek  cleanup:
646*00b67f09SDavid van Moolenbroek 	if (mem != NULL)
647*00b67f09SDavid van Moolenbroek 		isc_mem_put(diff->mctx, mem, size);
648*00b67f09SDavid van Moolenbroek 	return (result);
649*00b67f09SDavid van Moolenbroek }
650