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