xref: /openbsd/usr.bin/dig/lib/dns/rdata/generic/ds_43.c (revision 873f12b9)
15185a700Sflorian /*
25185a700Sflorian  * Copyright (C) Internet Systems Consortium, Inc. ("ISC")
35185a700Sflorian  *
45185a700Sflorian  * Permission to use, copy, modify, and/or distribute this software for any
55185a700Sflorian  * purpose with or without fee is hereby granted, provided that the above
65185a700Sflorian  * copyright notice and this permission notice appear in all copies.
75185a700Sflorian  *
85185a700Sflorian  * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
95185a700Sflorian  * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
105185a700Sflorian  * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
115185a700Sflorian  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
125185a700Sflorian  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
135185a700Sflorian  * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
145185a700Sflorian  * PERFORMANCE OF THIS SOFTWARE.
155185a700Sflorian  */
165185a700Sflorian 
17*873f12b9Sflorian /* $Id: ds_43.c,v 1.13 2020/02/26 18:47:59 florian Exp $ */
185185a700Sflorian 
195185a700Sflorian /* RFC3658 */
205185a700Sflorian 
215185a700Sflorian #ifndef RDATA_GENERIC_DS_43_C
225185a700Sflorian #define RDATA_GENERIC_DS_43_C
235185a700Sflorian 
245185a700Sflorian #include <isc/sha1.h>
255185a700Sflorian #include <isc/sha2.h>
265185a700Sflorian 
275185a700Sflorian #include <dns/ds.h>
285185a700Sflorian 
295185a700Sflorian static inline isc_result_t
generic_totext_ds(ARGS_TOTEXT)305185a700Sflorian generic_totext_ds(ARGS_TOTEXT) {
315185a700Sflorian 	isc_region_t sr;
325185a700Sflorian 	char buf[sizeof("64000 ")];
335185a700Sflorian 	unsigned int n;
345185a700Sflorian 
355185a700Sflorian 	REQUIRE(rdata->length != 0);
365185a700Sflorian 
375185a700Sflorian 	UNUSED(tctx);
385185a700Sflorian 
395185a700Sflorian 	dns_rdata_toregion(rdata, &sr);
405185a700Sflorian 
415185a700Sflorian 	/*
425185a700Sflorian 	 * Key tag.
435185a700Sflorian 	 */
445185a700Sflorian 	n = uint16_fromregion(&sr);
455185a700Sflorian 	isc_region_consume(&sr, 2);
465185a700Sflorian 	snprintf(buf, sizeof(buf), "%u ", n);
47*873f12b9Sflorian 	RETERR(isc_str_tobuffer(buf, target));
485185a700Sflorian 
495185a700Sflorian 	/*
505185a700Sflorian 	 * Algorithm.
515185a700Sflorian 	 */
525185a700Sflorian 	n = uint8_fromregion(&sr);
535185a700Sflorian 	isc_region_consume(&sr, 1);
545185a700Sflorian 	snprintf(buf, sizeof(buf), "%u ", n);
55*873f12b9Sflorian 	RETERR(isc_str_tobuffer(buf, target));
565185a700Sflorian 
575185a700Sflorian 	/*
585185a700Sflorian 	 * Digest type.
595185a700Sflorian 	 */
605185a700Sflorian 	n = uint8_fromregion(&sr);
615185a700Sflorian 	isc_region_consume(&sr, 1);
625185a700Sflorian 	snprintf(buf, sizeof(buf), "%u", n);
63*873f12b9Sflorian 	RETERR(isc_str_tobuffer(buf, target));
645185a700Sflorian 
655185a700Sflorian 	/*
665185a700Sflorian 	 * Digest.
675185a700Sflorian 	 */
685185a700Sflorian 	if ((tctx->flags & DNS_STYLEFLAG_MULTILINE) != 0)
69*873f12b9Sflorian 		RETERR(isc_str_tobuffer(" (", target));
70*873f12b9Sflorian 	RETERR(isc_str_tobuffer(tctx->linebreak, target));
715185a700Sflorian 	if ((tctx->flags & DNS_STYLEFLAG_NOCRYPTO) == 0) {
725185a700Sflorian 		if (tctx->width == 0) /* No splitting */
735185a700Sflorian 			RETERR(isc_hex_totext(&sr, 0, "", target));
745185a700Sflorian 		else
755185a700Sflorian 			RETERR(isc_hex_totext(&sr, tctx->width - 2,
765185a700Sflorian 					      tctx->linebreak, target));
775185a700Sflorian 	} else
78*873f12b9Sflorian 		RETERR(isc_str_tobuffer("[omitted]", target));
795185a700Sflorian 	if ((tctx->flags & DNS_STYLEFLAG_MULTILINE) != 0)
80*873f12b9Sflorian 		RETERR(isc_str_tobuffer(" )", target));
815185a700Sflorian 	return (ISC_R_SUCCESS);
825185a700Sflorian }
835185a700Sflorian 
845185a700Sflorian static inline isc_result_t
totext_ds(ARGS_TOTEXT)855185a700Sflorian totext_ds(ARGS_TOTEXT) {
865185a700Sflorian 
875185a700Sflorian 	REQUIRE(rdata->type == dns_rdatatype_ds);
885185a700Sflorian 
895185a700Sflorian 	return (generic_totext_ds(rdata, tctx, target));
905185a700Sflorian }
915185a700Sflorian 
925185a700Sflorian static inline isc_result_t
generic_fromwire_ds(ARGS_FROMWIRE)935185a700Sflorian generic_fromwire_ds(ARGS_FROMWIRE) {
945185a700Sflorian 	isc_region_t sr;
955185a700Sflorian 
965185a700Sflorian 	UNUSED(type);
975185a700Sflorian 	UNUSED(rdclass);
985185a700Sflorian 	UNUSED(dctx);
995185a700Sflorian 	UNUSED(options);
1005185a700Sflorian 
1015185a700Sflorian 	isc_buffer_activeregion(source, &sr);
1025185a700Sflorian 
1035185a700Sflorian 	/*
1045185a700Sflorian 	 * Check digest lengths if we know them.
1055185a700Sflorian 	 */
1065185a700Sflorian 	if (sr.length < 4 ||
1075185a700Sflorian 	    (sr.base[3] == DNS_DSDIGEST_SHA1 &&
1085185a700Sflorian 	     sr.length < 4 + ISC_SHA1_DIGESTLENGTH) ||
1095185a700Sflorian 	    (sr.base[3] == DNS_DSDIGEST_SHA256 &&
1105185a700Sflorian 	     sr.length < 4 + ISC_SHA256_DIGESTLENGTH) ||
1115185a700Sflorian 	    (sr.base[3] == DNS_DSDIGEST_SHA384 &&
1125185a700Sflorian 	     sr.length < 4 + ISC_SHA384_DIGESTLENGTH))
1135185a700Sflorian 		return (ISC_R_UNEXPECTEDEND);
1145185a700Sflorian 
1155185a700Sflorian 	/*
1165185a700Sflorian 	 * Only copy digest lengths if we know them.
1175185a700Sflorian 	 * If there is extra data dns_rdata_fromwire() will
1185185a700Sflorian 	 * detect that.
1195185a700Sflorian 	 */
1205185a700Sflorian 	if (sr.base[3] == DNS_DSDIGEST_SHA1)
1215185a700Sflorian 		sr.length = 4 + ISC_SHA1_DIGESTLENGTH;
1225185a700Sflorian 	else if (sr.base[3] == DNS_DSDIGEST_SHA256)
1235185a700Sflorian 		sr.length = 4 + ISC_SHA256_DIGESTLENGTH;
1245185a700Sflorian 	else if (sr.base[3] == DNS_DSDIGEST_SHA384)
1255185a700Sflorian 		sr.length = 4 + ISC_SHA384_DIGESTLENGTH;
1265185a700Sflorian 
1275185a700Sflorian 	isc_buffer_forward(source, sr.length);
128637d8eb6Sflorian 	return (isc_mem_tobuffer(target, sr.base, sr.length));
1295185a700Sflorian }
1305185a700Sflorian 
1315185a700Sflorian static inline isc_result_t
fromwire_ds(ARGS_FROMWIRE)1325185a700Sflorian fromwire_ds(ARGS_FROMWIRE) {
1335185a700Sflorian 
1345185a700Sflorian 	REQUIRE(type == dns_rdatatype_ds);
1355185a700Sflorian 
1365185a700Sflorian 	return (generic_fromwire_ds(rdclass, type, source, dctx, options,
1375185a700Sflorian 				    target));
1385185a700Sflorian }
1395185a700Sflorian 
1405185a700Sflorian static inline isc_result_t
towire_ds(ARGS_TOWIRE)1415185a700Sflorian towire_ds(ARGS_TOWIRE) {
1425185a700Sflorian 	isc_region_t sr;
1435185a700Sflorian 
1445185a700Sflorian 	REQUIRE(rdata->type == dns_rdatatype_ds);
1455185a700Sflorian 	REQUIRE(rdata->length != 0);
1465185a700Sflorian 
1475185a700Sflorian 	UNUSED(cctx);
1485185a700Sflorian 
1495185a700Sflorian 	dns_rdata_toregion(rdata, &sr);
150637d8eb6Sflorian 	return (isc_mem_tobuffer(target, sr.base, sr.length));
1515185a700Sflorian }
1525185a700Sflorian 
1535185a700Sflorian #endif	/* RDATA_GENERIC_DS_43_C */
154