xref: /freebsd/contrib/unbound/sldns/wire2str.c (revision 790c6b245151d6d5a26b84e5f34fee61453e2e60)
109a3aaf3SDag-Erling Smørgrav /*
209a3aaf3SDag-Erling Smørgrav  * wire2str.c
309a3aaf3SDag-Erling Smørgrav  *
409a3aaf3SDag-Erling Smørgrav  * conversion routines from the wire format
509a3aaf3SDag-Erling Smørgrav  * to the presentation format (strings)
609a3aaf3SDag-Erling Smørgrav  *
709a3aaf3SDag-Erling Smørgrav  * (c) NLnet Labs, 2004-2006
809a3aaf3SDag-Erling Smørgrav  *
909a3aaf3SDag-Erling Smørgrav  * See the file LICENSE for the license
1009a3aaf3SDag-Erling Smørgrav  */
1109a3aaf3SDag-Erling Smørgrav /**
1209a3aaf3SDag-Erling Smørgrav  * \file
1309a3aaf3SDag-Erling Smørgrav  *
1409a3aaf3SDag-Erling Smørgrav  * Contains functions to translate the wireformat to text
1509a3aaf3SDag-Erling Smørgrav  * representation, as well as functions to print them.
1609a3aaf3SDag-Erling Smørgrav  */
1709a3aaf3SDag-Erling Smørgrav #include "config.h"
1809a3aaf3SDag-Erling Smørgrav #include "sldns/wire2str.h"
1909a3aaf3SDag-Erling Smørgrav #include "sldns/str2wire.h"
2009a3aaf3SDag-Erling Smørgrav #include "sldns/rrdef.h"
2109a3aaf3SDag-Erling Smørgrav #include "sldns/pkthdr.h"
2209a3aaf3SDag-Erling Smørgrav #include "sldns/parseutil.h"
2309a3aaf3SDag-Erling Smørgrav #include "sldns/sbuffer.h"
2409a3aaf3SDag-Erling Smørgrav #include "sldns/keyraw.h"
250eefd307SCy Schubert #include "util/data/dname.h"
2609a3aaf3SDag-Erling Smørgrav #ifdef HAVE_TIME_H
2709a3aaf3SDag-Erling Smørgrav #include <time.h>
2809a3aaf3SDag-Erling Smørgrav #endif
2909a3aaf3SDag-Erling Smørgrav #include <sys/time.h>
3009a3aaf3SDag-Erling Smørgrav #include <stdarg.h>
3109a3aaf3SDag-Erling Smørgrav #include <ctype.h>
3209a3aaf3SDag-Erling Smørgrav #ifdef HAVE_NETDB_H
3309a3aaf3SDag-Erling Smørgrav #include <netdb.h>
3409a3aaf3SDag-Erling Smørgrav #endif
3509a3aaf3SDag-Erling Smørgrav 
3609a3aaf3SDag-Erling Smørgrav /* lookup tables for standard DNS stuff  */
3709a3aaf3SDag-Erling Smørgrav /* Taken from RFC 2535, section 7.  */
3809a3aaf3SDag-Erling Smørgrav static sldns_lookup_table sldns_algorithms_data[] = {
3909a3aaf3SDag-Erling Smørgrav 	{ LDNS_RSAMD5, "RSAMD5" },
4009a3aaf3SDag-Erling Smørgrav 	{ LDNS_DH, "DH" },
4109a3aaf3SDag-Erling Smørgrav 	{ LDNS_DSA, "DSA" },
4209a3aaf3SDag-Erling Smørgrav 	{ LDNS_ECC, "ECC" },
4309a3aaf3SDag-Erling Smørgrav 	{ LDNS_RSASHA1, "RSASHA1" },
4409a3aaf3SDag-Erling Smørgrav 	{ LDNS_DSA_NSEC3, "DSA-NSEC3-SHA1" },
4509a3aaf3SDag-Erling Smørgrav 	{ LDNS_RSASHA1_NSEC3, "RSASHA1-NSEC3-SHA1" },
4609a3aaf3SDag-Erling Smørgrav 	{ LDNS_RSASHA256, "RSASHA256"},
4709a3aaf3SDag-Erling Smørgrav 	{ LDNS_RSASHA512, "RSASHA512"},
4809a3aaf3SDag-Erling Smørgrav 	{ LDNS_ECC_GOST, "ECC-GOST"},
4909a3aaf3SDag-Erling Smørgrav 	{ LDNS_ECDSAP256SHA256, "ECDSAP256SHA256"},
5009a3aaf3SDag-Erling Smørgrav 	{ LDNS_ECDSAP384SHA384, "ECDSAP384SHA384"},
5165b390aaSDag-Erling Smørgrav 	{ LDNS_ED25519, "ED25519"},
5265b390aaSDag-Erling Smørgrav 	{ LDNS_ED448, "ED448"},
5309a3aaf3SDag-Erling Smørgrav 	{ LDNS_INDIRECT, "INDIRECT" },
5409a3aaf3SDag-Erling Smørgrav 	{ LDNS_PRIVATEDNS, "PRIVATEDNS" },
5509a3aaf3SDag-Erling Smørgrav 	{ LDNS_PRIVATEOID, "PRIVATEOID" },
5609a3aaf3SDag-Erling Smørgrav 	{ 0, NULL }
5709a3aaf3SDag-Erling Smørgrav };
5809a3aaf3SDag-Erling Smørgrav sldns_lookup_table* sldns_algorithms = sldns_algorithms_data;
5909a3aaf3SDag-Erling Smørgrav 
6009a3aaf3SDag-Erling Smørgrav /* hash algorithms in DS record */
6109a3aaf3SDag-Erling Smørgrav static sldns_lookup_table sldns_hashes_data[] = {
6209a3aaf3SDag-Erling Smørgrav 	{ LDNS_SHA1, "SHA1" },
6309a3aaf3SDag-Erling Smørgrav 	{ LDNS_SHA256, "SHA256" },
6409a3aaf3SDag-Erling Smørgrav 	{ LDNS_HASH_GOST, "HASH-GOST" },
6509a3aaf3SDag-Erling Smørgrav 	{ LDNS_SHA384, "SHA384" },
6609a3aaf3SDag-Erling Smørgrav 	{ 0, NULL }
6709a3aaf3SDag-Erling Smørgrav };
6809a3aaf3SDag-Erling Smørgrav sldns_lookup_table* sldns_hashes = sldns_hashes_data;
6909a3aaf3SDag-Erling Smørgrav 
7009a3aaf3SDag-Erling Smørgrav /* Taken from RFC 4398  */
7109a3aaf3SDag-Erling Smørgrav static sldns_lookup_table sldns_cert_algorithms_data[] = {
7209a3aaf3SDag-Erling Smørgrav 	{ LDNS_CERT_PKIX, "PKIX" },
7309a3aaf3SDag-Erling Smørgrav 	{ LDNS_CERT_SPKI, "SPKI" },
7409a3aaf3SDag-Erling Smørgrav 	{ LDNS_CERT_PGP, "PGP" },
7509a3aaf3SDag-Erling Smørgrav 	{ LDNS_CERT_IPKIX, "IPKIX" },
7609a3aaf3SDag-Erling Smørgrav 	{ LDNS_CERT_ISPKI, "ISPKI" },
7709a3aaf3SDag-Erling Smørgrav 	{ LDNS_CERT_IPGP, "IPGP" },
7809a3aaf3SDag-Erling Smørgrav 	{ LDNS_CERT_ACPKIX, "ACPKIX" },
7909a3aaf3SDag-Erling Smørgrav 	{ LDNS_CERT_IACPKIX, "IACPKIX" },
8009a3aaf3SDag-Erling Smørgrav 	{ LDNS_CERT_URI, "URI" },
8109a3aaf3SDag-Erling Smørgrav 	{ LDNS_CERT_OID, "OID" },
8209a3aaf3SDag-Erling Smørgrav 	{ 0, NULL }
8309a3aaf3SDag-Erling Smørgrav };
8409a3aaf3SDag-Erling Smørgrav sldns_lookup_table* sldns_cert_algorithms = sldns_cert_algorithms_data;
8509a3aaf3SDag-Erling Smørgrav 
8609a3aaf3SDag-Erling Smørgrav /* if these are used elsewhere */
8709a3aaf3SDag-Erling Smørgrav static sldns_lookup_table sldns_rcodes_data[] = {
8809a3aaf3SDag-Erling Smørgrav 	{ LDNS_RCODE_NOERROR, "NOERROR" },
8909a3aaf3SDag-Erling Smørgrav 	{ LDNS_RCODE_FORMERR, "FORMERR" },
9009a3aaf3SDag-Erling Smørgrav 	{ LDNS_RCODE_SERVFAIL, "SERVFAIL" },
9109a3aaf3SDag-Erling Smørgrav 	{ LDNS_RCODE_NXDOMAIN, "NXDOMAIN" },
9209a3aaf3SDag-Erling Smørgrav 	{ LDNS_RCODE_NOTIMPL, "NOTIMPL" },
9309a3aaf3SDag-Erling Smørgrav 	{ LDNS_RCODE_REFUSED, "REFUSED" },
9409a3aaf3SDag-Erling Smørgrav 	{ LDNS_RCODE_YXDOMAIN, "YXDOMAIN" },
9509a3aaf3SDag-Erling Smørgrav 	{ LDNS_RCODE_YXRRSET, "YXRRSET" },
9609a3aaf3SDag-Erling Smørgrav 	{ LDNS_RCODE_NXRRSET, "NXRRSET" },
9709a3aaf3SDag-Erling Smørgrav 	{ LDNS_RCODE_NOTAUTH, "NOTAUTH" },
9809a3aaf3SDag-Erling Smørgrav 	{ LDNS_RCODE_NOTZONE, "NOTZONE" },
9909a3aaf3SDag-Erling Smørgrav 	{ 0, NULL }
10009a3aaf3SDag-Erling Smørgrav };
10109a3aaf3SDag-Erling Smørgrav sldns_lookup_table* sldns_rcodes = sldns_rcodes_data;
10209a3aaf3SDag-Erling Smørgrav 
10309a3aaf3SDag-Erling Smørgrav static sldns_lookup_table sldns_opcodes_data[] = {
10409a3aaf3SDag-Erling Smørgrav 	{ LDNS_PACKET_QUERY, "QUERY" },
10509a3aaf3SDag-Erling Smørgrav 	{ LDNS_PACKET_IQUERY, "IQUERY" },
10609a3aaf3SDag-Erling Smørgrav 	{ LDNS_PACKET_STATUS, "STATUS" },
10709a3aaf3SDag-Erling Smørgrav 	{ LDNS_PACKET_NOTIFY, "NOTIFY" },
10809a3aaf3SDag-Erling Smørgrav 	{ LDNS_PACKET_UPDATE, "UPDATE" },
10909a3aaf3SDag-Erling Smørgrav 	{ 0, NULL }
11009a3aaf3SDag-Erling Smørgrav };
11109a3aaf3SDag-Erling Smørgrav sldns_lookup_table* sldns_opcodes = sldns_opcodes_data;
11209a3aaf3SDag-Erling Smørgrav 
11309a3aaf3SDag-Erling Smørgrav static sldns_lookup_table sldns_wireparse_errors_data[] = {
11409a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_OK, "no parse error" },
11509a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_GENERAL, "parse error" },
11609a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_DOMAINNAME_OVERFLOW, "Domainname length overflow" },
11709a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_DOMAINNAME_UNDERFLOW, "Domainname length underflow (zero length)" },
11809a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_BUFFER_TOO_SMALL, "buffer too small" },
11909a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_LABEL_OVERFLOW, "Label length overflow" },
12009a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_EMPTY_LABEL, "Empty label" },
12109a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_BAD_ESCAPE, "Syntax error, bad escape sequence" },
12209a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX, "Syntax error, could not parse the RR" },
12309a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_TTL, "Syntax error, could not parse the RR's TTL" },
12409a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_TYPE, "Syntax error, could not parse the RR's type" },
12509a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_CLASS, "Syntax error, could not parse the RR's class" },
12609a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_RDATA, "Syntax error, could not parse the RR's rdata" },
12709a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_MISSING_VALUE, "Syntax error, value expected" },
12809a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_INVALID_STR, "Conversion error, string expected" },
12909a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_B64, "Conversion error, b64 encoding expected" },
13009a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_B32_EXT, "Conversion error, b32 ext encoding expected" },
13109a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_HEX, "Conversion error, hex encoding expected" },
13209a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_CERT_BAD_ALGORITHM, "Bad algorithm type for CERT record" },
13309a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_TIME, "Conversion error, time encoding expected" },
13409a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_PERIOD, "Conversion error, time period encoding expected" },
13509a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_ILNP64, "Conversion error, 4 colon separated hex numbers expected" },
13609a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_EUI48,
13709a3aaf3SDag-Erling Smørgrav 		"Conversion error, 6 two character hex numbers "
13809a3aaf3SDag-Erling Smørgrav 		"separated by dashes expected (i.e. xx-xx-xx-xx-xx-xx" },
13909a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_EUI64,
14009a3aaf3SDag-Erling Smørgrav 		"Conversion error, 8 two character hex numbers "
14109a3aaf3SDag-Erling Smørgrav 		"separated by dashes expected (i.e. xx-xx-xx-xx-xx-xx-xx-xx" },
14209a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_TAG,
14309a3aaf3SDag-Erling Smørgrav 		"Conversion error, a non-zero sequence of US-ASCII letters "
14409a3aaf3SDag-Erling Smørgrav 		"and numbers in lower case expected" },
14509a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_NOT_IMPL, "not implemented" },
14609a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_INT, "Conversion error, integer expected" },
14709a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_IP4, "Conversion error, ip4 addr expected" },
14809a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_IP6, "Conversion error, ip6 addr expected" },
14909a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_SYNTAX_INTEGER_OVERFLOW, "Syntax error, integer overflow" },
15009a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_INCLUDE, "$INCLUDE directive was seen in the zone" },
15109a3aaf3SDag-Erling Smørgrav 	{ LDNS_WIREPARSE_ERR_PARENTHESIS, "Parse error, parenthesis mismatch" },
1525469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCB_UNKNOWN_KEY, "Unknown SvcParamKey"},
1535469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCB_MISSING_PARAM, "SvcParam is missing a SvcParamValue"},
1545469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCB_DUPLICATE_KEYS, "Duplicate SVCB key found"},
1555469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCB_MANDATORY_TOO_MANY_KEYS, "Too many keys in mandatory" },
1565469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCB_TOO_MANY_PARAMS,
1575469a995SCy Schubert 		"Too many SvcParams. Unbound only allows 63 entries" },
1585469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCB_MANDATORY_MISSING_PARAM,
1595469a995SCy Schubert 		"Mandatory SvcParamKey is missing"},
1605469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCB_MANDATORY_DUPLICATE_KEY,
1615469a995SCy Schubert 		"Keys in SvcParam mandatory MUST be unique" },
1625469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCB_MANDATORY_IN_MANDATORY,
1635469a995SCy Schubert 		"mandatory MUST not be included as mandatory parameter" },
1645469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCB_PORT_VALUE_SYNTAX,
1655469a995SCy Schubert 		"Could not parse port SvcParamValue" },
1665469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCB_IPV4_TOO_MANY_ADDRESSES,
1675469a995SCy Schubert 		"Too many IPv4 addresses in ipv4hint" },
1685469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCB_IPV6_TOO_MANY_ADDRESSES,
1695469a995SCy Schubert 		"Too many IPv6 addresses in ipv6hint" },
1705469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCB_ALPN_KEY_TOO_LARGE,
1715469a995SCy Schubert 		"Alpn strings need to be smaller than 255 chars"},
1725469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCB_NO_DEFAULT_ALPN_VALUE,
1735469a995SCy Schubert 		"No-default-alpn should not have a value" },
1745469a995SCy Schubert 	{ LDNS_WIREPARSE_ERR_SVCPARAM_BROKEN_RDATA,
1755469a995SCy Schubert 		"General SVCParam error" },
17609a3aaf3SDag-Erling Smørgrav 	{ 0, NULL }
17709a3aaf3SDag-Erling Smørgrav };
17809a3aaf3SDag-Erling Smørgrav sldns_lookup_table* sldns_wireparse_errors = sldns_wireparse_errors_data;
17909a3aaf3SDag-Erling Smørgrav 
18009a3aaf3SDag-Erling Smørgrav static sldns_lookup_table sldns_edns_flags_data[] = {
18109a3aaf3SDag-Erling Smørgrav 	{ 3600, "do"},
18209a3aaf3SDag-Erling Smørgrav 	{ 0, NULL}
18309a3aaf3SDag-Erling Smørgrav };
18409a3aaf3SDag-Erling Smørgrav sldns_lookup_table* sldns_edns_flags = sldns_edns_flags_data;
18509a3aaf3SDag-Erling Smørgrav 
18609a3aaf3SDag-Erling Smørgrav static sldns_lookup_table sldns_edns_options_data[] = {
18709a3aaf3SDag-Erling Smørgrav 	{ 1, "LLQ" },
18809a3aaf3SDag-Erling Smørgrav 	{ 2, "UL" },
18909a3aaf3SDag-Erling Smørgrav 	{ 3, "NSID" },
19009a3aaf3SDag-Erling Smørgrav 	/* 4 draft-cheshire-edns0-owner-option */
19109a3aaf3SDag-Erling Smørgrav 	{ 5, "DAU" },
19209a3aaf3SDag-Erling Smørgrav 	{ 6, "DHU" },
19309a3aaf3SDag-Erling Smørgrav 	{ 7, "N3U" },
19409a3aaf3SDag-Erling Smørgrav 	{ 8, "edns-client-subnet" },
19565b390aaSDag-Erling Smørgrav 	{ 11, "edns-tcp-keepalive"},
196e2d15004SDag-Erling Smørgrav 	{ 12, "Padding" },
197a39a5a69SCy Schubert 	{ 15, "EDE"},
19809a3aaf3SDag-Erling Smørgrav 	{ 0, NULL}
19909a3aaf3SDag-Erling Smørgrav };
20009a3aaf3SDag-Erling Smørgrav sldns_lookup_table* sldns_edns_options = sldns_edns_options_data;
20109a3aaf3SDag-Erling Smørgrav 
202c7f4d7adSDag-Erling Smørgrav static sldns_lookup_table sldns_tsig_errors_data[] = {
203c7f4d7adSDag-Erling Smørgrav 	{ LDNS_TSIG_ERROR_NOERROR, "NOERROR" },
204c7f4d7adSDag-Erling Smørgrav 	{ LDNS_RCODE_FORMERR, "FORMERR" },
205c7f4d7adSDag-Erling Smørgrav 	{ LDNS_RCODE_SERVFAIL, "SERVFAIL" },
206c7f4d7adSDag-Erling Smørgrav 	{ LDNS_RCODE_NXDOMAIN, "NXDOMAIN" },
207c7f4d7adSDag-Erling Smørgrav 	{ LDNS_RCODE_NOTIMPL, "NOTIMPL" },
208c7f4d7adSDag-Erling Smørgrav 	{ LDNS_RCODE_REFUSED, "REFUSED" },
209c7f4d7adSDag-Erling Smørgrav 	{ LDNS_RCODE_YXDOMAIN, "YXDOMAIN" },
210c7f4d7adSDag-Erling Smørgrav 	{ LDNS_RCODE_YXRRSET, "YXRRSET" },
211c7f4d7adSDag-Erling Smørgrav 	{ LDNS_RCODE_NXRRSET, "NXRRSET" },
212c7f4d7adSDag-Erling Smørgrav 	{ LDNS_RCODE_NOTAUTH, "NOTAUTH" },
213c7f4d7adSDag-Erling Smørgrav 	{ LDNS_RCODE_NOTZONE, "NOTZONE" },
214c7f4d7adSDag-Erling Smørgrav 	{ LDNS_TSIG_ERROR_BADSIG, "BADSIG" },
215c7f4d7adSDag-Erling Smørgrav 	{ LDNS_TSIG_ERROR_BADKEY, "BADKEY" },
216c7f4d7adSDag-Erling Smørgrav 	{ LDNS_TSIG_ERROR_BADTIME, "BADTIME" },
217c7f4d7adSDag-Erling Smørgrav 	{ LDNS_TSIG_ERROR_BADMODE, "BADMODE" },
218c7f4d7adSDag-Erling Smørgrav 	{ LDNS_TSIG_ERROR_BADNAME, "BADNAME" },
219c7f4d7adSDag-Erling Smørgrav 	{ LDNS_TSIG_ERROR_BADALG, "BADALG" },
220c7f4d7adSDag-Erling Smørgrav 	{ 0, NULL }
221c7f4d7adSDag-Erling Smørgrav };
222c7f4d7adSDag-Erling Smørgrav sldns_lookup_table* sldns_tsig_errors = sldns_tsig_errors_data;
223c7f4d7adSDag-Erling Smørgrav 
2245469a995SCy Schubert /* draft-ietf-dnsop-svcb-https-06: 6. Initial SvcParamKeys */
2255469a995SCy Schubert const char *svcparamkey_strs[] = {
2265469a995SCy Schubert 	"mandatory", "alpn", "no-default-alpn", "port",
2275469a995SCy Schubert 	"ipv4hint", "ech", "ipv6hint"
2285469a995SCy Schubert };
2295469a995SCy Schubert 
23009a3aaf3SDag-Erling Smørgrav char* sldns_wire2str_pkt(uint8_t* data, size_t len)
23109a3aaf3SDag-Erling Smørgrav {
23209a3aaf3SDag-Erling Smørgrav 	size_t slen = (size_t)sldns_wire2str_pkt_buf(data, len, NULL, 0);
23309a3aaf3SDag-Erling Smørgrav 	char* result = (char*)malloc(slen+1);
23409a3aaf3SDag-Erling Smørgrav 	if(!result) return NULL;
23509a3aaf3SDag-Erling Smørgrav 	sldns_wire2str_pkt_buf(data, len, result, slen+1);
23609a3aaf3SDag-Erling Smørgrav 	return result;
23709a3aaf3SDag-Erling Smørgrav }
23809a3aaf3SDag-Erling Smørgrav 
23909a3aaf3SDag-Erling Smørgrav char* sldns_wire2str_rr(uint8_t* rr, size_t len)
24009a3aaf3SDag-Erling Smørgrav {
24109a3aaf3SDag-Erling Smørgrav 	size_t slen = (size_t)sldns_wire2str_rr_buf(rr, len, NULL, 0);
24209a3aaf3SDag-Erling Smørgrav 	char* result = (char*)malloc(slen+1);
24309a3aaf3SDag-Erling Smørgrav 	if(!result) return NULL;
24409a3aaf3SDag-Erling Smørgrav 	sldns_wire2str_rr_buf(rr, len, result, slen+1);
24509a3aaf3SDag-Erling Smørgrav 	return result;
24609a3aaf3SDag-Erling Smørgrav }
24709a3aaf3SDag-Erling Smørgrav 
24809a3aaf3SDag-Erling Smørgrav char* sldns_wire2str_type(uint16_t rrtype)
24909a3aaf3SDag-Erling Smørgrav {
25009a3aaf3SDag-Erling Smørgrav 	char buf[16];
25109a3aaf3SDag-Erling Smørgrav 	sldns_wire2str_type_buf(rrtype, buf, sizeof(buf));
25209a3aaf3SDag-Erling Smørgrav 	return strdup(buf);
25309a3aaf3SDag-Erling Smørgrav }
25409a3aaf3SDag-Erling Smørgrav 
25509a3aaf3SDag-Erling Smørgrav char* sldns_wire2str_class(uint16_t rrclass)
25609a3aaf3SDag-Erling Smørgrav {
25709a3aaf3SDag-Erling Smørgrav 	char buf[16];
25809a3aaf3SDag-Erling Smørgrav 	sldns_wire2str_class_buf(rrclass, buf, sizeof(buf));
25909a3aaf3SDag-Erling Smørgrav 	return strdup(buf);
26009a3aaf3SDag-Erling Smørgrav }
26109a3aaf3SDag-Erling Smørgrav 
26209a3aaf3SDag-Erling Smørgrav char* sldns_wire2str_dname(uint8_t* dname, size_t dname_len)
26309a3aaf3SDag-Erling Smørgrav {
26409a3aaf3SDag-Erling Smørgrav 	size_t slen=(size_t)sldns_wire2str_dname_buf(dname, dname_len, NULL, 0);
26509a3aaf3SDag-Erling Smørgrav 	char* result = (char*)malloc(slen+1);
26609a3aaf3SDag-Erling Smørgrav 	if(!result) return NULL;
26709a3aaf3SDag-Erling Smørgrav 	sldns_wire2str_dname_buf(dname, dname_len, result, slen+1);
26809a3aaf3SDag-Erling Smørgrav 	return result;
26909a3aaf3SDag-Erling Smørgrav }
27009a3aaf3SDag-Erling Smørgrav 
27109a3aaf3SDag-Erling Smørgrav char* sldns_wire2str_rcode(int rcode)
27209a3aaf3SDag-Erling Smørgrav {
27309a3aaf3SDag-Erling Smørgrav 	char buf[16];
27409a3aaf3SDag-Erling Smørgrav 	sldns_wire2str_rcode_buf(rcode, buf, sizeof(buf));
27509a3aaf3SDag-Erling Smørgrav 	return strdup(buf);
27609a3aaf3SDag-Erling Smørgrav }
27709a3aaf3SDag-Erling Smørgrav 
27809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_pkt_buf(uint8_t* d, size_t dlen, char* s, size_t slen)
27909a3aaf3SDag-Erling Smørgrav {
28009a3aaf3SDag-Erling Smørgrav 	/* use arguments as temporary variables */
28109a3aaf3SDag-Erling Smørgrav 	return sldns_wire2str_pkt_scan(&d, &dlen, &s, &slen);
28209a3aaf3SDag-Erling Smørgrav }
28309a3aaf3SDag-Erling Smørgrav 
28409a3aaf3SDag-Erling Smørgrav int sldns_wire2str_rr_buf(uint8_t* d, size_t dlen, char* s, size_t slen)
28509a3aaf3SDag-Erling Smørgrav {
28609a3aaf3SDag-Erling Smørgrav 	/* use arguments as temporary variables */
2870eefd307SCy Schubert 	return sldns_wire2str_rr_scan(&d, &dlen, &s, &slen, NULL, 0, NULL);
28809a3aaf3SDag-Erling Smørgrav }
28909a3aaf3SDag-Erling Smørgrav 
29057bddd21SDag-Erling Smørgrav int sldns_wire2str_rrquestion_buf(uint8_t* d, size_t dlen, char* s, size_t slen)
29157bddd21SDag-Erling Smørgrav {
29257bddd21SDag-Erling Smørgrav 	/* use arguments as temporary variables */
2930eefd307SCy Schubert 	return sldns_wire2str_rrquestion_scan(&d, &dlen, &s, &slen, NULL, 0, NULL);
29457bddd21SDag-Erling Smørgrav }
29557bddd21SDag-Erling Smørgrav 
29609a3aaf3SDag-Erling Smørgrav int sldns_wire2str_rdata_buf(uint8_t* rdata, size_t rdata_len, char* str,
29709a3aaf3SDag-Erling Smørgrav 	size_t str_len, uint16_t rrtype)
29809a3aaf3SDag-Erling Smørgrav {
29909a3aaf3SDag-Erling Smørgrav 	/* use arguments as temporary variables */
30009a3aaf3SDag-Erling Smørgrav 	return sldns_wire2str_rdata_scan(&rdata, &rdata_len, &str, &str_len,
3010eefd307SCy Schubert 		rrtype, NULL, 0, NULL);
30209a3aaf3SDag-Erling Smørgrav }
30309a3aaf3SDag-Erling Smørgrav 
30409a3aaf3SDag-Erling Smørgrav int sldns_wire2str_rr_unknown_buf(uint8_t* d, size_t dlen, char* s, size_t slen)
30509a3aaf3SDag-Erling Smørgrav {
30609a3aaf3SDag-Erling Smørgrav 	/* use arguments as temporary variables */
3070eefd307SCy Schubert 	return sldns_wire2str_rr_unknown_scan(&d, &dlen, &s, &slen, NULL, 0, NULL);
30809a3aaf3SDag-Erling Smørgrav }
30909a3aaf3SDag-Erling Smørgrav 
31009a3aaf3SDag-Erling Smørgrav int sldns_wire2str_rr_comment_buf(uint8_t* rr, size_t rrlen, size_t dname_len,
31109a3aaf3SDag-Erling Smørgrav 	char* s, size_t slen)
31209a3aaf3SDag-Erling Smørgrav {
31309a3aaf3SDag-Erling Smørgrav 	uint16_t rrtype = sldns_wirerr_get_type(rr, rrlen, dname_len);
31409a3aaf3SDag-Erling Smørgrav 	return sldns_wire2str_rr_comment_print(&s, &slen, rr, rrlen, dname_len,
31509a3aaf3SDag-Erling Smørgrav 		rrtype);
31609a3aaf3SDag-Erling Smørgrav }
31709a3aaf3SDag-Erling Smørgrav 
31809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_type_buf(uint16_t rrtype, char* s, size_t slen)
31909a3aaf3SDag-Erling Smørgrav {
32009a3aaf3SDag-Erling Smørgrav 	/* use arguments as temporary variables */
32109a3aaf3SDag-Erling Smørgrav 	return sldns_wire2str_type_print(&s, &slen, rrtype);
32209a3aaf3SDag-Erling Smørgrav }
32309a3aaf3SDag-Erling Smørgrav 
32409a3aaf3SDag-Erling Smørgrav int sldns_wire2str_class_buf(uint16_t rrclass, char* s, size_t slen)
32509a3aaf3SDag-Erling Smørgrav {
32609a3aaf3SDag-Erling Smørgrav 	/* use arguments as temporary variables */
32709a3aaf3SDag-Erling Smørgrav 	return sldns_wire2str_class_print(&s, &slen, rrclass);
32809a3aaf3SDag-Erling Smørgrav }
32909a3aaf3SDag-Erling Smørgrav 
33009a3aaf3SDag-Erling Smørgrav int sldns_wire2str_rcode_buf(int rcode, char* s, size_t slen)
33109a3aaf3SDag-Erling Smørgrav {
33209a3aaf3SDag-Erling Smørgrav 	/* use arguments as temporary variables */
33309a3aaf3SDag-Erling Smørgrav 	return sldns_wire2str_rcode_print(&s, &slen, rcode);
33409a3aaf3SDag-Erling Smørgrav }
33509a3aaf3SDag-Erling Smørgrav 
33665b390aaSDag-Erling Smørgrav int sldns_wire2str_opcode_buf(int opcode, char* s, size_t slen)
33765b390aaSDag-Erling Smørgrav {
33865b390aaSDag-Erling Smørgrav 	/* use arguments as temporary variables */
33965b390aaSDag-Erling Smørgrav 	return sldns_wire2str_opcode_print(&s, &slen, opcode);
34065b390aaSDag-Erling Smørgrav }
34165b390aaSDag-Erling Smørgrav 
34209a3aaf3SDag-Erling Smørgrav int sldns_wire2str_dname_buf(uint8_t* d, size_t dlen, char* s, size_t slen)
34309a3aaf3SDag-Erling Smørgrav {
34409a3aaf3SDag-Erling Smørgrav 	/* use arguments as temporary variables */
3450eefd307SCy Schubert 	return sldns_wire2str_dname_scan(&d, &dlen, &s, &slen, NULL, 0, NULL);
34609a3aaf3SDag-Erling Smørgrav }
34709a3aaf3SDag-Erling Smørgrav 
34809a3aaf3SDag-Erling Smørgrav int sldns_str_vprint(char** str, size_t* slen, const char* format, va_list args)
34909a3aaf3SDag-Erling Smørgrav {
35009a3aaf3SDag-Erling Smørgrav 	int w = vsnprintf(*str, *slen, format, args);
35109a3aaf3SDag-Erling Smørgrav 	if(w < 0) {
35209a3aaf3SDag-Erling Smørgrav 		/* error in printout */
35309a3aaf3SDag-Erling Smørgrav 		return 0;
35409a3aaf3SDag-Erling Smørgrav 	} else if((size_t)w >= *slen) {
35509a3aaf3SDag-Erling Smørgrav 		*str = NULL; /* we do not want str to point outside of buffer*/
35609a3aaf3SDag-Erling Smørgrav 		*slen = 0;
35709a3aaf3SDag-Erling Smørgrav 	} else {
35809a3aaf3SDag-Erling Smørgrav 		*str += w;
35909a3aaf3SDag-Erling Smørgrav 		*slen -= w;
36009a3aaf3SDag-Erling Smørgrav 	}
36109a3aaf3SDag-Erling Smørgrav 	return w;
36209a3aaf3SDag-Erling Smørgrav }
36309a3aaf3SDag-Erling Smørgrav 
36409a3aaf3SDag-Erling Smørgrav int sldns_str_print(char** str, size_t* slen, const char* format, ...)
36509a3aaf3SDag-Erling Smørgrav {
36609a3aaf3SDag-Erling Smørgrav 	int w;
36709a3aaf3SDag-Erling Smørgrav 	va_list args;
36809a3aaf3SDag-Erling Smørgrav 	va_start(args, format);
36909a3aaf3SDag-Erling Smørgrav 	w = sldns_str_vprint(str, slen, format, args);
37009a3aaf3SDag-Erling Smørgrav 	va_end(args);
37109a3aaf3SDag-Erling Smørgrav 	return w;
37209a3aaf3SDag-Erling Smørgrav }
37309a3aaf3SDag-Erling Smørgrav 
37409a3aaf3SDag-Erling Smørgrav /** print hex format into text buffer for specified length */
37509a3aaf3SDag-Erling Smørgrav static int print_hex_buf(char** s, size_t* slen, uint8_t* buf, size_t len)
37609a3aaf3SDag-Erling Smørgrav {
37709a3aaf3SDag-Erling Smørgrav 	const char* hex = "0123456789ABCDEF";
37809a3aaf3SDag-Erling Smørgrav 	size_t i;
37909a3aaf3SDag-Erling Smørgrav 	for(i=0; i<len; i++) {
38009a3aaf3SDag-Erling Smørgrav 		(void)sldns_str_print(s, slen, "%c%c", hex[(buf[i]&0xf0)>>4],
38109a3aaf3SDag-Erling Smørgrav 			hex[buf[i]&0x0f]);
38209a3aaf3SDag-Erling Smørgrav 	}
38309a3aaf3SDag-Erling Smørgrav 	return (int)len*2;
38409a3aaf3SDag-Erling Smørgrav }
38509a3aaf3SDag-Erling Smørgrav 
38609a3aaf3SDag-Erling Smørgrav /** print remainder of buffer in hex format with prefixed text */
38709a3aaf3SDag-Erling Smørgrav static int print_remainder_hex(const char* pref, uint8_t** d, size_t* dlen,
38809a3aaf3SDag-Erling Smørgrav 	char** s, size_t* slen)
38909a3aaf3SDag-Erling Smørgrav {
39009a3aaf3SDag-Erling Smørgrav 	int w = 0;
39109a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "%s", pref);
39209a3aaf3SDag-Erling Smørgrav 	w += print_hex_buf(s, slen, *d, *dlen);
39309a3aaf3SDag-Erling Smørgrav 	*d += *dlen;
39409a3aaf3SDag-Erling Smørgrav 	*dlen = 0;
39509a3aaf3SDag-Erling Smørgrav 	return w;
39609a3aaf3SDag-Erling Smørgrav }
39709a3aaf3SDag-Erling Smørgrav 
39809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_pkt_scan(uint8_t** d, size_t* dlen, char** s, size_t* slen)
39909a3aaf3SDag-Erling Smørgrav {
4000eefd307SCy Schubert 	int w = 0, comprloop = 0;
40109a3aaf3SDag-Erling Smørgrav 	unsigned qdcount, ancount, nscount, arcount, i;
40209a3aaf3SDag-Erling Smørgrav 	uint8_t* pkt = *d;
40309a3aaf3SDag-Erling Smørgrav 	size_t pktlen = *dlen;
40409a3aaf3SDag-Erling Smørgrav 	if(*dlen >= LDNS_HEADER_SIZE) {
40509a3aaf3SDag-Erling Smørgrav 		qdcount = (unsigned)LDNS_QDCOUNT(*d);
40609a3aaf3SDag-Erling Smørgrav 		ancount = (unsigned)LDNS_ANCOUNT(*d);
40709a3aaf3SDag-Erling Smørgrav 		nscount = (unsigned)LDNS_NSCOUNT(*d);
40809a3aaf3SDag-Erling Smørgrav 		arcount = (unsigned)LDNS_ARCOUNT(*d);
40909a3aaf3SDag-Erling Smørgrav 	} else {
41009a3aaf3SDag-Erling Smørgrav 		qdcount = ancount = nscount = arcount = 0;
41109a3aaf3SDag-Erling Smørgrav 	}
41209a3aaf3SDag-Erling Smørgrav 	w += sldns_wire2str_header_scan(d, dlen, s, slen);
41309a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\n");
41409a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, ";; QUESTION SECTION:\n");
41509a3aaf3SDag-Erling Smørgrav 	for(i=0; i<qdcount; i++) {
41609a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_rrquestion_scan(d, dlen, s, slen,
4170eefd307SCy Schubert 			pkt, pktlen, &comprloop);
41809a3aaf3SDag-Erling Smørgrav 		if(!*dlen) break;
41909a3aaf3SDag-Erling Smørgrav 	}
42009a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\n");
42109a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, ";; ANSWER SECTION:\n");
42209a3aaf3SDag-Erling Smørgrav 	for(i=0; i<ancount; i++) {
4230eefd307SCy Schubert 		w += sldns_wire2str_rr_scan(d, dlen, s, slen, pkt, pktlen, &comprloop);
42409a3aaf3SDag-Erling Smørgrav 		if(!*dlen) break;
42509a3aaf3SDag-Erling Smørgrav 	}
42609a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\n");
42709a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, ";; AUTHORITY SECTION:\n");
42809a3aaf3SDag-Erling Smørgrav 	for(i=0; i<nscount; i++) {
4290eefd307SCy Schubert 		w += sldns_wire2str_rr_scan(d, dlen, s, slen, pkt, pktlen, &comprloop);
43009a3aaf3SDag-Erling Smørgrav 		if(!*dlen) break;
43109a3aaf3SDag-Erling Smørgrav 	}
43209a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\n");
43309a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, ";; ADDITIONAL SECTION:\n");
43409a3aaf3SDag-Erling Smørgrav 	for(i=0; i<arcount; i++) {
4350eefd307SCy Schubert 		w += sldns_wire2str_rr_scan(d, dlen, s, slen, pkt, pktlen, &comprloop);
43609a3aaf3SDag-Erling Smørgrav 		if(!*dlen) break;
43709a3aaf3SDag-Erling Smørgrav 	}
43809a3aaf3SDag-Erling Smørgrav 	/* other fields: WHEN(time), SERVER(IP) not available here. */
43909a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, ";; MSG SIZE  rcvd: %d\n", (int)pktlen);
44009a3aaf3SDag-Erling Smørgrav 	if(*dlen > 0) {
44109a3aaf3SDag-Erling Smørgrav 		w += print_remainder_hex(";; trailing garbage 0x",
44209a3aaf3SDag-Erling Smørgrav 			d, dlen, s, slen);
44309a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, slen, "\n");
44409a3aaf3SDag-Erling Smørgrav 	}
44509a3aaf3SDag-Erling Smørgrav 	return w;
44609a3aaf3SDag-Erling Smørgrav }
44709a3aaf3SDag-Erling Smørgrav 
44809a3aaf3SDag-Erling Smørgrav /** scan type, class and ttl and printout, for rr */
44909a3aaf3SDag-Erling Smørgrav static int sldns_rr_tcttl_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
45009a3aaf3SDag-Erling Smørgrav {
45109a3aaf3SDag-Erling Smørgrav 	int w = 0;
45209a3aaf3SDag-Erling Smørgrav 	uint16_t t, c;
45309a3aaf3SDag-Erling Smørgrav 	uint32_t ttl;
45409a3aaf3SDag-Erling Smørgrav 	if(*dl < 8) {
45509a3aaf3SDag-Erling Smørgrav 		if(*dl < 4)
45609a3aaf3SDag-Erling Smørgrav 			return w + print_remainder_hex("; Error malformed 0x",
45709a3aaf3SDag-Erling Smørgrav 				d, dl, s, sl);
45809a3aaf3SDag-Erling Smørgrav 		/* these print values or 0x.. if none left */
45909a3aaf3SDag-Erling Smørgrav 		t = sldns_read_uint16(*d);
46009a3aaf3SDag-Erling Smørgrav 		c = sldns_read_uint16((*d)+2);
46109a3aaf3SDag-Erling Smørgrav 		(*d)+=4;
46209a3aaf3SDag-Erling Smørgrav 		(*dl)-=4;
46309a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_class_print(s, sl, c);
46409a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "\t");
46509a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_type_print(s, sl, t);
46609a3aaf3SDag-Erling Smørgrav 		if(*dl == 0)
46709a3aaf3SDag-Erling Smørgrav 			return w + sldns_str_print(s, sl, "; Error no ttl");
46809a3aaf3SDag-Erling Smørgrav 		return w + print_remainder_hex(
46909a3aaf3SDag-Erling Smørgrav 			"; Error malformed ttl 0x", d, dl, s, sl);
47009a3aaf3SDag-Erling Smørgrav 	}
47109a3aaf3SDag-Erling Smørgrav 	t = sldns_read_uint16(*d);
47209a3aaf3SDag-Erling Smørgrav 	c = sldns_read_uint16((*d)+2);
47309a3aaf3SDag-Erling Smørgrav 	ttl = sldns_read_uint32((*d)+4);
47409a3aaf3SDag-Erling Smørgrav 	(*d)+=8;
47509a3aaf3SDag-Erling Smørgrav 	(*dl)-=8;
47609a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, "%lu\t", (unsigned long)ttl);
47709a3aaf3SDag-Erling Smørgrav 	w += sldns_wire2str_class_print(s, sl, c);
47809a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, "\t");
47909a3aaf3SDag-Erling Smørgrav 	w += sldns_wire2str_type_print(s, sl, t);
48009a3aaf3SDag-Erling Smørgrav 	return w;
48109a3aaf3SDag-Erling Smørgrav }
48209a3aaf3SDag-Erling Smørgrav 
48309a3aaf3SDag-Erling Smørgrav int sldns_wire2str_rr_scan(uint8_t** d, size_t* dlen, char** s, size_t* slen,
4840eefd307SCy Schubert 	uint8_t* pkt, size_t pktlen, int* comprloop)
48509a3aaf3SDag-Erling Smørgrav {
48609a3aaf3SDag-Erling Smørgrav 	int w = 0;
48709a3aaf3SDag-Erling Smørgrav 	uint8_t* rr = *d;
48809a3aaf3SDag-Erling Smørgrav 	size_t rrlen = *dlen, dname_off, rdlen, ordlen;
48909a3aaf3SDag-Erling Smørgrav 	uint16_t rrtype = 0;
49009a3aaf3SDag-Erling Smørgrav 
49109a3aaf3SDag-Erling Smørgrav 	if(*dlen >= 3 && (*d)[0]==0 &&
49209a3aaf3SDag-Erling Smørgrav 		sldns_read_uint16((*d)+1)==LDNS_RR_TYPE_OPT) {
49309a3aaf3SDag-Erling Smørgrav 		/* perform EDNS OPT processing */
49409a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_edns_scan(d, dlen, s, slen, pkt, pktlen);
49509a3aaf3SDag-Erling Smørgrav 	}
49609a3aaf3SDag-Erling Smørgrav 
49709a3aaf3SDag-Erling Smørgrav 	/* try to scan the rdata with pretty-printing, but if that fails, then
49809a3aaf3SDag-Erling Smørgrav 	 * scan the rdata as an unknown RR type */
4990eefd307SCy Schubert 	w += sldns_wire2str_dname_scan(d, dlen, s, slen, pkt, pktlen, comprloop);
50009a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\t");
50109a3aaf3SDag-Erling Smørgrav 	dname_off = rrlen-(*dlen);
50209a3aaf3SDag-Erling Smørgrav 	if(*dlen == 4) {
50309a3aaf3SDag-Erling Smørgrav 		/* like a question-RR */
50409a3aaf3SDag-Erling Smørgrav 		uint16_t t = sldns_read_uint16(*d);
50509a3aaf3SDag-Erling Smørgrav 		uint16_t c = sldns_read_uint16((*d)+2);
50609a3aaf3SDag-Erling Smørgrav 		(*d)+=4;
50709a3aaf3SDag-Erling Smørgrav 		(*dlen)-=4;
50809a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_class_print(s, slen, c);
50909a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, slen, "\t");
51009a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_type_print(s, slen, t);
51109a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, slen, " ; Error no ttl,rdata\n");
51209a3aaf3SDag-Erling Smørgrav 		return w;
51309a3aaf3SDag-Erling Smørgrav 	}
51409a3aaf3SDag-Erling Smørgrav 	if(*dlen < 8) {
51509a3aaf3SDag-Erling Smørgrav 		if(*dlen == 0)
51609a3aaf3SDag-Erling Smørgrav 			return w + sldns_str_print(s, slen, ";Error missing RR\n");
51709a3aaf3SDag-Erling Smørgrav 		w += print_remainder_hex(";Error partial RR 0x", d, dlen, s, slen);
51809a3aaf3SDag-Erling Smørgrav 		return w + sldns_str_print(s, slen, "\n");
51909a3aaf3SDag-Erling Smørgrav 	}
52009a3aaf3SDag-Erling Smørgrav 	rrtype = sldns_read_uint16(*d);
52109a3aaf3SDag-Erling Smørgrav 	w += sldns_rr_tcttl_scan(d, dlen, s, slen);
52209a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\t");
52309a3aaf3SDag-Erling Smørgrav 
52409a3aaf3SDag-Erling Smørgrav 	/* rdata */
52509a3aaf3SDag-Erling Smørgrav 	if(*dlen < 2) {
52609a3aaf3SDag-Erling Smørgrav 		if(*dlen == 0)
52709a3aaf3SDag-Erling Smørgrav 			return w + sldns_str_print(s, slen, ";Error missing rdatalen\n");
52809a3aaf3SDag-Erling Smørgrav 		w += print_remainder_hex(";Error missing rdatalen 0x",
52909a3aaf3SDag-Erling Smørgrav 			d, dlen, s, slen);
53009a3aaf3SDag-Erling Smørgrav 		return w + sldns_str_print(s, slen, "\n");
53109a3aaf3SDag-Erling Smørgrav 	}
53209a3aaf3SDag-Erling Smørgrav 	rdlen = sldns_read_uint16(*d);
53309a3aaf3SDag-Erling Smørgrav 	ordlen = rdlen;
53409a3aaf3SDag-Erling Smørgrav 	(*d)+=2;
53509a3aaf3SDag-Erling Smørgrav 	(*dlen)-=2;
53609a3aaf3SDag-Erling Smørgrav 	if(*dlen < rdlen) {
53709a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, slen, "\\# %u ", (unsigned)rdlen);
53809a3aaf3SDag-Erling Smørgrav 		if(*dlen == 0)
53909a3aaf3SDag-Erling Smørgrav 			return w + sldns_str_print(s, slen, ";Error missing rdata\n");
54009a3aaf3SDag-Erling Smørgrav 		w += print_remainder_hex(";Error partial rdata 0x", d, dlen, s, slen);
54109a3aaf3SDag-Erling Smørgrav 		return w + sldns_str_print(s, slen, "\n");
54209a3aaf3SDag-Erling Smørgrav 	}
5430eefd307SCy Schubert 	w += sldns_wire2str_rdata_scan(d, &rdlen, s, slen, rrtype, pkt, pktlen,
5440eefd307SCy Schubert 		comprloop);
54509a3aaf3SDag-Erling Smørgrav 	(*dlen) -= (ordlen-rdlen);
54609a3aaf3SDag-Erling Smørgrav 
54709a3aaf3SDag-Erling Smørgrav 	/* default comment */
54809a3aaf3SDag-Erling Smørgrav 	w += sldns_wire2str_rr_comment_print(s, slen, rr, rrlen, dname_off,
54909a3aaf3SDag-Erling Smørgrav 		rrtype);
55009a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\n");
55109a3aaf3SDag-Erling Smørgrav 	return w;
55209a3aaf3SDag-Erling Smørgrav }
55309a3aaf3SDag-Erling Smørgrav 
55409a3aaf3SDag-Erling Smørgrav int sldns_wire2str_rrquestion_scan(uint8_t** d, size_t* dlen, char** s,
5550eefd307SCy Schubert 	size_t* slen, uint8_t* pkt, size_t pktlen, int* comprloop)
55609a3aaf3SDag-Erling Smørgrav {
55709a3aaf3SDag-Erling Smørgrav 	int w = 0;
55809a3aaf3SDag-Erling Smørgrav 	uint16_t t, c;
5590eefd307SCy Schubert 	w += sldns_wire2str_dname_scan(d, dlen, s, slen, pkt, pktlen, comprloop);
56009a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\t");
56109a3aaf3SDag-Erling Smørgrav 	if(*dlen < 4) {
56209a3aaf3SDag-Erling Smørgrav 		if(*dlen == 0)
56309a3aaf3SDag-Erling Smørgrav 			return w + sldns_str_print(s, slen, "Error malformed\n");
56409a3aaf3SDag-Erling Smørgrav 		w += print_remainder_hex("Error malformed 0x", d, dlen, s, slen);
56509a3aaf3SDag-Erling Smørgrav 		return w + sldns_str_print(s, slen, "\n");
56609a3aaf3SDag-Erling Smørgrav 	}
56709a3aaf3SDag-Erling Smørgrav 	t = sldns_read_uint16(*d);
56809a3aaf3SDag-Erling Smørgrav 	c = sldns_read_uint16((*d)+2);
56909a3aaf3SDag-Erling Smørgrav 	(*d)+=4;
57009a3aaf3SDag-Erling Smørgrav 	(*dlen)-=4;
57109a3aaf3SDag-Erling Smørgrav 	w += sldns_wire2str_class_print(s, slen, c);
57209a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\t");
57309a3aaf3SDag-Erling Smørgrav 	w += sldns_wire2str_type_print(s, slen, t);
57409a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\n");
57509a3aaf3SDag-Erling Smørgrav 	return w;
57609a3aaf3SDag-Erling Smørgrav }
57709a3aaf3SDag-Erling Smørgrav 
57809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_rr_unknown_scan(uint8_t** d, size_t* dlen, char** s,
5790eefd307SCy Schubert 	size_t* slen, uint8_t* pkt, size_t pktlen, int* comprloop)
58009a3aaf3SDag-Erling Smørgrav {
58109a3aaf3SDag-Erling Smørgrav 	size_t rdlen, ordlen;
58209a3aaf3SDag-Erling Smørgrav 	int w = 0;
5830eefd307SCy Schubert 	w += sldns_wire2str_dname_scan(d, dlen, s, slen, pkt, pktlen, comprloop);
58409a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\t");
58509a3aaf3SDag-Erling Smørgrav 	w += sldns_rr_tcttl_scan(d, dlen, s, slen);
58609a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\t");
58709a3aaf3SDag-Erling Smørgrav 	if(*dlen < 2) {
58809a3aaf3SDag-Erling Smørgrav 		if(*dlen == 0)
58909a3aaf3SDag-Erling Smørgrav 			return w + sldns_str_print(s, slen, ";Error missing rdatalen\n");
59009a3aaf3SDag-Erling Smørgrav 		w += print_remainder_hex(";Error missing rdatalen 0x",
59109a3aaf3SDag-Erling Smørgrav 			d, dlen, s, slen);
59209a3aaf3SDag-Erling Smørgrav 		return w + sldns_str_print(s, slen, "\n");
59309a3aaf3SDag-Erling Smørgrav 	}
59409a3aaf3SDag-Erling Smørgrav 	rdlen = sldns_read_uint16(*d);
59509a3aaf3SDag-Erling Smørgrav 	ordlen = rdlen;
59609a3aaf3SDag-Erling Smørgrav 	(*d) += 2;
59709a3aaf3SDag-Erling Smørgrav 	(*dlen) -= 2;
59809a3aaf3SDag-Erling Smørgrav 	if(*dlen < rdlen) {
59909a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, slen, "\\# %u ", (unsigned)rdlen);
60009a3aaf3SDag-Erling Smørgrav 		if(*dlen == 0)
60109a3aaf3SDag-Erling Smørgrav 			return w + sldns_str_print(s, slen, ";Error missing rdata\n");
60209a3aaf3SDag-Erling Smørgrav 		w += print_remainder_hex(";Error partial rdata 0x", d, dlen, s, slen);
60309a3aaf3SDag-Erling Smørgrav 		return w + sldns_str_print(s, slen, "\n");
60409a3aaf3SDag-Erling Smørgrav 	}
60509a3aaf3SDag-Erling Smørgrav 	w += sldns_wire2str_rdata_unknown_scan(d, &rdlen, s, slen);
60609a3aaf3SDag-Erling Smørgrav 	(*dlen) -= (ordlen-rdlen);
60709a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\n");
60809a3aaf3SDag-Erling Smørgrav 	return w;
60909a3aaf3SDag-Erling Smørgrav }
61009a3aaf3SDag-Erling Smørgrav 
61109a3aaf3SDag-Erling Smørgrav /** print rr comment for type DNSKEY */
61209a3aaf3SDag-Erling Smørgrav static int rr_comment_dnskey(char** s, size_t* slen, uint8_t* rr,
61309a3aaf3SDag-Erling Smørgrav 	size_t rrlen, size_t dname_off)
61409a3aaf3SDag-Erling Smørgrav {
61509a3aaf3SDag-Erling Smørgrav 	size_t rdlen;
61609a3aaf3SDag-Erling Smørgrav 	uint8_t* rdata;
61709a3aaf3SDag-Erling Smørgrav 	int flags, w = 0;
61809a3aaf3SDag-Erling Smørgrav 	if(rrlen < dname_off + 10) return 0;
61909a3aaf3SDag-Erling Smørgrav 	rdlen = sldns_read_uint16(rr+dname_off+8);
62009a3aaf3SDag-Erling Smørgrav 	if(rrlen < dname_off + 10 + rdlen) return 0;
6210eefd307SCy Schubert 	if(rdlen < 2) return 0;
62209a3aaf3SDag-Erling Smørgrav 	rdata = rr + dname_off + 10;
62309a3aaf3SDag-Erling Smørgrav 	flags = (int)sldns_read_uint16(rdata);
62409a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, " ;{");
62509a3aaf3SDag-Erling Smørgrav 
62609a3aaf3SDag-Erling Smørgrav 	/* id */
62709a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "id = %u",
62809a3aaf3SDag-Erling Smørgrav 		sldns_calc_keytag_raw(rdata, rdlen));
62909a3aaf3SDag-Erling Smørgrav 
63009a3aaf3SDag-Erling Smørgrav 	/* flags */
63109a3aaf3SDag-Erling Smørgrav 	if((flags&LDNS_KEY_ZONE_KEY)) {
63209a3aaf3SDag-Erling Smørgrav 		if((flags&LDNS_KEY_SEP_KEY))
63309a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, slen, " (ksk)");
63409a3aaf3SDag-Erling Smørgrav 		else 	w += sldns_str_print(s, slen, " (zsk)");
63509a3aaf3SDag-Erling Smørgrav 	}
63609a3aaf3SDag-Erling Smørgrav 
63709a3aaf3SDag-Erling Smørgrav 	/* keysize */
63809a3aaf3SDag-Erling Smørgrav 	if(rdlen > 4) {
63909a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, slen, ", ");
64009a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, slen, "size = %db",
64109a3aaf3SDag-Erling Smørgrav 			(int)sldns_rr_dnskey_key_size_raw(
64209a3aaf3SDag-Erling Smørgrav 			(unsigned char*)rdata+4, rdlen-4, (int)(rdata[3])));
64309a3aaf3SDag-Erling Smørgrav 	}
64409a3aaf3SDag-Erling Smørgrav 
64509a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "}");
64609a3aaf3SDag-Erling Smørgrav 	return w;
64709a3aaf3SDag-Erling Smørgrav }
64809a3aaf3SDag-Erling Smørgrav 
64909a3aaf3SDag-Erling Smørgrav /** print rr comment for type RRSIG */
65009a3aaf3SDag-Erling Smørgrav static int rr_comment_rrsig(char** s, size_t* slen, uint8_t* rr,
65109a3aaf3SDag-Erling Smørgrav 	size_t rrlen, size_t dname_off)
65209a3aaf3SDag-Erling Smørgrav {
65309a3aaf3SDag-Erling Smørgrav 	size_t rdlen;
65409a3aaf3SDag-Erling Smørgrav 	uint8_t* rdata;
65509a3aaf3SDag-Erling Smørgrav 	if(rrlen < dname_off + 10) return 0;
65609a3aaf3SDag-Erling Smørgrav 	rdlen = sldns_read_uint16(rr+dname_off+8);
65709a3aaf3SDag-Erling Smørgrav 	if(rrlen < dname_off + 10 + rdlen) return 0;
65809a3aaf3SDag-Erling Smørgrav 	rdata = rr + dname_off + 10;
65909a3aaf3SDag-Erling Smørgrav 	if(rdlen < 18) return 0;
66009a3aaf3SDag-Erling Smørgrav 	return sldns_str_print(s, slen, " ;{id = %d}",
66109a3aaf3SDag-Erling Smørgrav 		(int)sldns_read_uint16(rdata+16));
66209a3aaf3SDag-Erling Smørgrav }
66309a3aaf3SDag-Erling Smørgrav 
66409a3aaf3SDag-Erling Smørgrav /** print rr comment for type NSEC3 */
66509a3aaf3SDag-Erling Smørgrav static int rr_comment_nsec3(char** s, size_t* slen, uint8_t* rr,
66609a3aaf3SDag-Erling Smørgrav 	size_t rrlen, size_t dname_off)
66709a3aaf3SDag-Erling Smørgrav {
66809a3aaf3SDag-Erling Smørgrav 	size_t rdlen;
66909a3aaf3SDag-Erling Smørgrav 	uint8_t* rdata;
67009a3aaf3SDag-Erling Smørgrav 	int w = 0;
67109a3aaf3SDag-Erling Smørgrav 	if(rrlen < dname_off + 10) return 0;
67209a3aaf3SDag-Erling Smørgrav 	rdlen = sldns_read_uint16(rr+dname_off+8);
67309a3aaf3SDag-Erling Smørgrav 	if(rrlen < dname_off + 10 + rdlen) return 0;
67409a3aaf3SDag-Erling Smørgrav 	rdata = rr + dname_off + 10;
67509a3aaf3SDag-Erling Smørgrav 	if(rdlen < 2) return 0;
67609a3aaf3SDag-Erling Smørgrav 	if((rdata[1] & LDNS_NSEC3_VARS_OPTOUT_MASK))
67709a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, slen, " ;{flags: optout}");
67809a3aaf3SDag-Erling Smørgrav 	return w;
67909a3aaf3SDag-Erling Smørgrav }
68009a3aaf3SDag-Erling Smørgrav 
68109a3aaf3SDag-Erling Smørgrav int sldns_wire2str_rr_comment_print(char** s, size_t* slen, uint8_t* rr,
68209a3aaf3SDag-Erling Smørgrav 	size_t rrlen, size_t dname_off, uint16_t rrtype)
68309a3aaf3SDag-Erling Smørgrav {
68409a3aaf3SDag-Erling Smørgrav 	if(rrtype == LDNS_RR_TYPE_DNSKEY) {
68509a3aaf3SDag-Erling Smørgrav 		return rr_comment_dnskey(s, slen, rr, rrlen, dname_off);
68609a3aaf3SDag-Erling Smørgrav 	} else if(rrtype == LDNS_RR_TYPE_RRSIG) {
68709a3aaf3SDag-Erling Smørgrav 		return rr_comment_rrsig(s, slen, rr, rrlen, dname_off);
68809a3aaf3SDag-Erling Smørgrav 	} else if(rrtype == LDNS_RR_TYPE_NSEC3) {
68909a3aaf3SDag-Erling Smørgrav 		return rr_comment_nsec3(s, slen, rr, rrlen, dname_off);
69009a3aaf3SDag-Erling Smørgrav 	}
69109a3aaf3SDag-Erling Smørgrav 	return 0;
69209a3aaf3SDag-Erling Smørgrav }
69309a3aaf3SDag-Erling Smørgrav 
69409a3aaf3SDag-Erling Smørgrav int sldns_wire2str_header_scan(uint8_t** d, size_t* dlen, char** s,
69509a3aaf3SDag-Erling Smørgrav 	size_t* slen)
69609a3aaf3SDag-Erling Smørgrav {
69709a3aaf3SDag-Erling Smørgrav 	int w = 0;
69809a3aaf3SDag-Erling Smørgrav 	int opcode, rcode;
69909a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, ";; ->>HEADER<<- ");
70009a3aaf3SDag-Erling Smørgrav 	if(*dlen == 0)
70109a3aaf3SDag-Erling Smørgrav 		return w+sldns_str_print(s, slen, "Error empty packet");
70209a3aaf3SDag-Erling Smørgrav 	if(*dlen < 4)
70309a3aaf3SDag-Erling Smørgrav 		return w+print_remainder_hex("Error header too short 0x", d, dlen, s, slen);
70409a3aaf3SDag-Erling Smørgrav 	opcode = (int)LDNS_OPCODE_WIRE(*d);
70509a3aaf3SDag-Erling Smørgrav 	rcode = (int)LDNS_RCODE_WIRE(*d);
70609a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "opcode: ");
70709a3aaf3SDag-Erling Smørgrav 	w += sldns_wire2str_opcode_print(s, slen, opcode);
70809a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, ", ");
70909a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "rcode: ");
71009a3aaf3SDag-Erling Smørgrav 	w += sldns_wire2str_rcode_print(s, slen, rcode);
71109a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, ", ");
71209a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "id: %d\n", (int)LDNS_ID_WIRE(*d));
71309a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, ";; flags:");
71409a3aaf3SDag-Erling Smørgrav 	if(LDNS_QR_WIRE(*d)) w += sldns_str_print(s, slen, " qr");
71509a3aaf3SDag-Erling Smørgrav 	if(LDNS_AA_WIRE(*d)) w += sldns_str_print(s, slen, " aa");
71609a3aaf3SDag-Erling Smørgrav 	if(LDNS_TC_WIRE(*d)) w += sldns_str_print(s, slen, " tc");
71709a3aaf3SDag-Erling Smørgrav 	if(LDNS_RD_WIRE(*d)) w += sldns_str_print(s, slen, " rd");
71809a3aaf3SDag-Erling Smørgrav 	if(LDNS_CD_WIRE(*d)) w += sldns_str_print(s, slen, " cd");
71909a3aaf3SDag-Erling Smørgrav 	if(LDNS_RA_WIRE(*d)) w += sldns_str_print(s, slen, " ra");
72009a3aaf3SDag-Erling Smørgrav 	if(LDNS_AD_WIRE(*d)) w += sldns_str_print(s, slen, " ad");
72109a3aaf3SDag-Erling Smørgrav 	if(LDNS_Z_WIRE(*d))  w += sldns_str_print(s, slen, " z");
72209a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, " ; ");
72309a3aaf3SDag-Erling Smørgrav 	if(*dlen < LDNS_HEADER_SIZE)
72409a3aaf3SDag-Erling Smørgrav 		return w+print_remainder_hex("Error header too short 0x", d, dlen, s, slen);
72509a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "QUERY: %d, ", (int)LDNS_QDCOUNT(*d));
72609a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "ANSWER: %d, ", (int)LDNS_ANCOUNT(*d));
72709a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "AUTHORITY: %d, ", (int)LDNS_NSCOUNT(*d));
72809a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "ADDITIONAL: %d ", (int)LDNS_ARCOUNT(*d));
72909a3aaf3SDag-Erling Smørgrav 	*d += LDNS_HEADER_SIZE;
73009a3aaf3SDag-Erling Smørgrav 	*dlen -= LDNS_HEADER_SIZE;
73109a3aaf3SDag-Erling Smørgrav 	return w;
73209a3aaf3SDag-Erling Smørgrav }
73309a3aaf3SDag-Erling Smørgrav 
73409a3aaf3SDag-Erling Smørgrav int sldns_wire2str_rdata_scan(uint8_t** d, size_t* dlen, char** s,
7350eefd307SCy Schubert 	size_t* slen, uint16_t rrtype, uint8_t* pkt, size_t pktlen,
7360eefd307SCy Schubert 	int* comprloop)
73709a3aaf3SDag-Erling Smørgrav {
73809a3aaf3SDag-Erling Smørgrav 	/* try to prettyprint, but if that fails, use unknown format */
73909a3aaf3SDag-Erling Smørgrav 	uint8_t* origd = *d;
74009a3aaf3SDag-Erling Smørgrav 	char* origs = *s;
74109a3aaf3SDag-Erling Smørgrav 	size_t origdlen = *dlen, origslen = *slen;
742bc892140SDag-Erling Smørgrav 	size_t r_cnt, r_max;
74309a3aaf3SDag-Erling Smørgrav 	sldns_rdf_type rdftype;
74409a3aaf3SDag-Erling Smørgrav 	int w = 0, n;
74509a3aaf3SDag-Erling Smørgrav 
74609a3aaf3SDag-Erling Smørgrav 	const sldns_rr_descriptor *desc = sldns_rr_descript(rrtype);
74709a3aaf3SDag-Erling Smørgrav 	if(!desc) /* unknown format */
74809a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_rdata_unknown_scan(d, dlen, s, slen);
74909a3aaf3SDag-Erling Smørgrav 	/* dlen equals the rdatalen for the rdata */
75009a3aaf3SDag-Erling Smørgrav 
75109a3aaf3SDag-Erling Smørgrav 	r_max = sldns_rr_descriptor_maximum(desc);
75209a3aaf3SDag-Erling Smørgrav 	for(r_cnt=0; r_cnt < r_max; r_cnt++) {
75309a3aaf3SDag-Erling Smørgrav 		if(*dlen == 0) {
75409a3aaf3SDag-Erling Smørgrav 			if(r_cnt < sldns_rr_descriptor_minimum(desc))
75509a3aaf3SDag-Erling Smørgrav 				goto failed;
75609a3aaf3SDag-Erling Smørgrav 			break; /* nothing more to print */
75709a3aaf3SDag-Erling Smørgrav 		}
75809a3aaf3SDag-Erling Smørgrav 		rdftype = sldns_rr_descriptor_field_type(desc, r_cnt);
75909a3aaf3SDag-Erling Smørgrav 		if(r_cnt != 0)
76009a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, slen, " ");
76109a3aaf3SDag-Erling Smørgrav 		n = sldns_wire2str_rdf_scan(d, dlen, s, slen, rdftype,
7620eefd307SCy Schubert 			pkt, pktlen, comprloop);
76309a3aaf3SDag-Erling Smørgrav 		if(n == -1) {
76409a3aaf3SDag-Erling Smørgrav 		failed:
76509a3aaf3SDag-Erling Smørgrav 			/* failed, use unknown format */
76609a3aaf3SDag-Erling Smørgrav 			*d = origd; *s = origs;
76709a3aaf3SDag-Erling Smørgrav 			*dlen = origdlen; *slen = origslen;
76809a3aaf3SDag-Erling Smørgrav 			return sldns_wire2str_rdata_unknown_scan(d, dlen,
76909a3aaf3SDag-Erling Smørgrav 				s, slen);
77009a3aaf3SDag-Erling Smørgrav 		}
77109a3aaf3SDag-Erling Smørgrav 		w += n;
77209a3aaf3SDag-Erling Smørgrav 	}
77305ab2901SDag-Erling Smørgrav 	if(*dlen != 0) {
77405ab2901SDag-Erling Smørgrav 		goto failed;
77505ab2901SDag-Erling Smørgrav 	}
77609a3aaf3SDag-Erling Smørgrav 	return w;
77709a3aaf3SDag-Erling Smørgrav }
77809a3aaf3SDag-Erling Smørgrav 
77909a3aaf3SDag-Erling Smørgrav int sldns_wire2str_rdata_unknown_scan(uint8_t** d, size_t* dlen, char** s,
78009a3aaf3SDag-Erling Smørgrav 	size_t* slen)
78109a3aaf3SDag-Erling Smørgrav {
78209a3aaf3SDag-Erling Smørgrav 	int w = 0;
78309a3aaf3SDag-Erling Smørgrav 
78409a3aaf3SDag-Erling Smørgrav 	/* print length */
78509a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, slen, "\\# %u", (unsigned)*dlen);
78609a3aaf3SDag-Erling Smørgrav 
78709a3aaf3SDag-Erling Smørgrav 	/* print rdlen in hex */
78809a3aaf3SDag-Erling Smørgrav 	if(*dlen != 0)
78909a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, slen, " ");
79009a3aaf3SDag-Erling Smørgrav 	w += print_hex_buf(s, slen, *d, *dlen);
79109a3aaf3SDag-Erling Smørgrav 	(*d) += *dlen;
79209a3aaf3SDag-Erling Smørgrav 	(*dlen) = 0;
79309a3aaf3SDag-Erling Smørgrav 	return w;
79409a3aaf3SDag-Erling Smørgrav }
79509a3aaf3SDag-Erling Smørgrav 
79609a3aaf3SDag-Erling Smørgrav /** print and escape one character for a domain dname */
79709a3aaf3SDag-Erling Smørgrav static int dname_char_print(char** s, size_t* slen, uint8_t c)
79809a3aaf3SDag-Erling Smørgrav {
79909a3aaf3SDag-Erling Smørgrav 	if(c == '.' || c == ';' || c == '(' || c == ')' || c == '\\')
80009a3aaf3SDag-Erling Smørgrav 		return sldns_str_print(s, slen, "\\%c", c);
80109a3aaf3SDag-Erling Smørgrav 	else if(!(isascii((unsigned char)c) && isgraph((unsigned char)c)))
80209a3aaf3SDag-Erling Smørgrav 		return sldns_str_print(s, slen, "\\%03u", (unsigned)c);
80309a3aaf3SDag-Erling Smørgrav 	/* plain printout */
80409a3aaf3SDag-Erling Smørgrav 	if(*slen) {
80509a3aaf3SDag-Erling Smørgrav 		**s = (char)c;
80609a3aaf3SDag-Erling Smørgrav 		(*s)++;
80709a3aaf3SDag-Erling Smørgrav 		(*slen)--;
80809a3aaf3SDag-Erling Smørgrav 	}
80909a3aaf3SDag-Erling Smørgrav 	return 1;
81009a3aaf3SDag-Erling Smørgrav }
81109a3aaf3SDag-Erling Smørgrav 
81209a3aaf3SDag-Erling Smørgrav int sldns_wire2str_dname_scan(uint8_t** d, size_t* dlen, char** s, size_t* slen,
8130eefd307SCy Schubert 	uint8_t* pkt, size_t pktlen, int* comprloop)
81409a3aaf3SDag-Erling Smørgrav {
81509a3aaf3SDag-Erling Smørgrav 	int w = 0;
81609a3aaf3SDag-Erling Smørgrav 	/* spool labels onto the string, use compression if its there */
81709a3aaf3SDag-Erling Smørgrav 	uint8_t* pos = *d;
81809a3aaf3SDag-Erling Smørgrav 	unsigned i, counter=0;
8190eefd307SCy Schubert 	unsigned maxcompr = MAX_COMPRESS_PTRS; /* loop detection, max compr ptrs */
82009a3aaf3SDag-Erling Smørgrav 	int in_buf = 1;
8219cf5bc93SCy Schubert 	size_t dname_len = 0;
8220eefd307SCy Schubert 	if(comprloop) {
8230eefd307SCy Schubert 		if(*comprloop != 0)
8240eefd307SCy Schubert 			maxcompr = 30; /* for like ipv6 reverse name, per label */
8250eefd307SCy Schubert 		if(*comprloop > 4)
8260eefd307SCy Schubert 			maxcompr = 4; /* just don't want to spend time, any more */
8270eefd307SCy Schubert 	}
82809a3aaf3SDag-Erling Smørgrav 	if(*dlen == 0) return sldns_str_print(s, slen, "ErrorMissingDname");
82909a3aaf3SDag-Erling Smørgrav 	if(*pos == 0) {
83009a3aaf3SDag-Erling Smørgrav 		(*d)++;
83109a3aaf3SDag-Erling Smørgrav 		(*dlen)--;
83209a3aaf3SDag-Erling Smørgrav 		return sldns_str_print(s, slen, ".");
83309a3aaf3SDag-Erling Smørgrav 	}
8340eefd307SCy Schubert 	while((!pkt || pos < pkt+pktlen) && *pos) {
83509a3aaf3SDag-Erling Smørgrav 		/* read label length */
83609a3aaf3SDag-Erling Smørgrav 		uint8_t labellen = *pos++;
83709a3aaf3SDag-Erling Smørgrav 		if(in_buf) { (*d)++; (*dlen)--; }
83809a3aaf3SDag-Erling Smørgrav 
83909a3aaf3SDag-Erling Smørgrav 		/* find out what sort of label we have */
84009a3aaf3SDag-Erling Smørgrav 		if((labellen&0xc0) == 0xc0) {
84109a3aaf3SDag-Erling Smørgrav 			/* compressed */
84209a3aaf3SDag-Erling Smørgrav 			uint16_t target = 0;
84309a3aaf3SDag-Erling Smørgrav 			if(in_buf && *dlen == 0)
84409a3aaf3SDag-Erling Smørgrav 				return w + sldns_str_print(s, slen,
84509a3aaf3SDag-Erling Smørgrav 					"ErrorPartialDname");
84609a3aaf3SDag-Erling Smørgrav 			else if(!in_buf && pos+1 > pkt+pktlen)
84709a3aaf3SDag-Erling Smørgrav 				return w + sldns_str_print(s, slen,
84809a3aaf3SDag-Erling Smørgrav 					"ErrorPartialDname");
84909a3aaf3SDag-Erling Smørgrav 			target = ((labellen&0x3f)<<8) | *pos;
85009a3aaf3SDag-Erling Smørgrav 			if(in_buf) { (*d)++; (*dlen)--; }
85109a3aaf3SDag-Erling Smørgrav 			/* move to target, if possible */
85209a3aaf3SDag-Erling Smørgrav 			if(!pkt || target >= pktlen)
85309a3aaf3SDag-Erling Smørgrav 				return w + sldns_str_print(s, slen,
85409a3aaf3SDag-Erling Smørgrav 					"ErrorComprPtrOutOfBounds");
8550eefd307SCy Schubert 			if(counter++ > maxcompr) {
8560eefd307SCy Schubert 				if(comprloop && *comprloop < 10)
8570eefd307SCy Schubert 					(*comprloop)++;
85809a3aaf3SDag-Erling Smørgrav 				return w + sldns_str_print(s, slen,
85909a3aaf3SDag-Erling Smørgrav 					"ErrorComprPtrLooped");
8600eefd307SCy Schubert 			}
86109a3aaf3SDag-Erling Smørgrav 			in_buf = 0;
86209a3aaf3SDag-Erling Smørgrav 			pos = pkt+target;
86309a3aaf3SDag-Erling Smørgrav 			continue;
86409a3aaf3SDag-Erling Smørgrav 		} else if((labellen&0xc0)) {
86509a3aaf3SDag-Erling Smørgrav 			/* notimpl label type */
86609a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, slen,
86709a3aaf3SDag-Erling Smørgrav 				"ErrorLABELTYPE%xIsUnknown",
86809a3aaf3SDag-Erling Smørgrav 				(int)(labellen&0xc0));
86909a3aaf3SDag-Erling Smørgrav 			return w;
87009a3aaf3SDag-Erling Smørgrav 		}
87109a3aaf3SDag-Erling Smørgrav 
87209a3aaf3SDag-Erling Smørgrav 		/* spool label characters, end with '.' */
873bc892140SDag-Erling Smørgrav 		if(in_buf && *dlen < (size_t)labellen)
874bc892140SDag-Erling Smørgrav 			labellen = (uint8_t)*dlen;
875bc892140SDag-Erling Smørgrav 		else if(!in_buf && pos+(size_t)labellen > pkt+pktlen)
87609a3aaf3SDag-Erling Smørgrav 			labellen = (uint8_t)(pkt + pktlen - pos);
8779cf5bc93SCy Schubert 		dname_len += ((size_t)labellen)+1;
8789cf5bc93SCy Schubert 		if(dname_len > LDNS_MAX_DOMAINLEN) {
8799cf5bc93SCy Schubert 			/* dname_len counts the uncompressed length we have
8809cf5bc93SCy Schubert 			 * seen so far, and the domain name has become too
8819cf5bc93SCy Schubert 			 * long, prevent the loop from printing overly long
8829cf5bc93SCy Schubert 			 * content. */
8839cf5bc93SCy Schubert 			w += sldns_str_print(s, slen,
8849cf5bc93SCy Schubert 				"ErrorDomainNameTooLong");
8859cf5bc93SCy Schubert 			return w;
8869cf5bc93SCy Schubert 		}
88709a3aaf3SDag-Erling Smørgrav 		for(i=0; i<(unsigned)labellen; i++) {
88809a3aaf3SDag-Erling Smørgrav 			w += dname_char_print(s, slen, *pos++);
88909a3aaf3SDag-Erling Smørgrav 		}
89009a3aaf3SDag-Erling Smørgrav 		if(in_buf) {
89109a3aaf3SDag-Erling Smørgrav 			(*d) += labellen;
89209a3aaf3SDag-Erling Smørgrav 			(*dlen) -= labellen;
89309a3aaf3SDag-Erling Smørgrav 			if(*dlen == 0) break;
89409a3aaf3SDag-Erling Smørgrav 		}
89509a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, slen, ".");
89609a3aaf3SDag-Erling Smørgrav 	}
89709a3aaf3SDag-Erling Smørgrav 	/* skip over final root label */
89809a3aaf3SDag-Erling Smørgrav 	if(in_buf && *dlen > 0) { (*d)++; (*dlen)--; }
89909a3aaf3SDag-Erling Smørgrav 	/* in case we printed no labels, terminate dname */
90009a3aaf3SDag-Erling Smørgrav 	if(w == 0) w += sldns_str_print(s, slen, ".");
90109a3aaf3SDag-Erling Smørgrav 	return w;
90209a3aaf3SDag-Erling Smørgrav }
90309a3aaf3SDag-Erling Smørgrav 
90409a3aaf3SDag-Erling Smørgrav int sldns_wire2str_opcode_print(char** s, size_t* slen, int opcode)
90509a3aaf3SDag-Erling Smørgrav {
90609a3aaf3SDag-Erling Smørgrav 	sldns_lookup_table *lt = sldns_lookup_by_id(sldns_opcodes, opcode);
90709a3aaf3SDag-Erling Smørgrav 	if (lt && lt->name) {
90809a3aaf3SDag-Erling Smørgrav 		return sldns_str_print(s, slen, "%s", lt->name);
90909a3aaf3SDag-Erling Smørgrav 	}
91009a3aaf3SDag-Erling Smørgrav 	return sldns_str_print(s, slen, "OPCODE%u", (unsigned)opcode);
91109a3aaf3SDag-Erling Smørgrav }
91209a3aaf3SDag-Erling Smørgrav 
91309a3aaf3SDag-Erling Smørgrav int sldns_wire2str_rcode_print(char** s, size_t* slen, int rcode)
91409a3aaf3SDag-Erling Smørgrav {
91509a3aaf3SDag-Erling Smørgrav 	sldns_lookup_table *lt = sldns_lookup_by_id(sldns_rcodes, rcode);
91609a3aaf3SDag-Erling Smørgrav 	if (lt && lt->name) {
91709a3aaf3SDag-Erling Smørgrav 		return sldns_str_print(s, slen, "%s", lt->name);
91809a3aaf3SDag-Erling Smørgrav 	}
91909a3aaf3SDag-Erling Smørgrav 	return sldns_str_print(s, slen, "RCODE%u", (unsigned)rcode);
92009a3aaf3SDag-Erling Smørgrav }
92109a3aaf3SDag-Erling Smørgrav 
92209a3aaf3SDag-Erling Smørgrav int sldns_wire2str_class_print(char** s, size_t* slen, uint16_t rrclass)
92309a3aaf3SDag-Erling Smørgrav {
92409a3aaf3SDag-Erling Smørgrav 	sldns_lookup_table *lt = sldns_lookup_by_id(sldns_rr_classes,
92509a3aaf3SDag-Erling Smørgrav 		(int)rrclass);
92609a3aaf3SDag-Erling Smørgrav 	if (lt && lt->name) {
92709a3aaf3SDag-Erling Smørgrav 		return sldns_str_print(s, slen, "%s", lt->name);
92809a3aaf3SDag-Erling Smørgrav 	}
92909a3aaf3SDag-Erling Smørgrav 	return sldns_str_print(s, slen, "CLASS%u", (unsigned)rrclass);
93009a3aaf3SDag-Erling Smørgrav }
93109a3aaf3SDag-Erling Smørgrav 
93209a3aaf3SDag-Erling Smørgrav int sldns_wire2str_type_print(char** s, size_t* slen, uint16_t rrtype)
93309a3aaf3SDag-Erling Smørgrav {
93409a3aaf3SDag-Erling Smørgrav 	const sldns_rr_descriptor *descriptor = sldns_rr_descript(rrtype);
93509a3aaf3SDag-Erling Smørgrav 	if (descriptor && descriptor->_name) {
93609a3aaf3SDag-Erling Smørgrav 		return sldns_str_print(s, slen, "%s", descriptor->_name);
93709a3aaf3SDag-Erling Smørgrav 	}
93809a3aaf3SDag-Erling Smørgrav 	return sldns_str_print(s, slen, "TYPE%u", (unsigned)rrtype);
93909a3aaf3SDag-Erling Smørgrav }
94009a3aaf3SDag-Erling Smørgrav 
94109a3aaf3SDag-Erling Smørgrav int sldns_wire2str_edns_option_code_print(char** s, size_t* slen,
94209a3aaf3SDag-Erling Smørgrav 	uint16_t opcode)
94309a3aaf3SDag-Erling Smørgrav {
94409a3aaf3SDag-Erling Smørgrav 	sldns_lookup_table *lt = sldns_lookup_by_id(sldns_edns_options,
94509a3aaf3SDag-Erling Smørgrav 		(int)opcode);
94609a3aaf3SDag-Erling Smørgrav 	if (lt && lt->name) {
94709a3aaf3SDag-Erling Smørgrav 		return sldns_str_print(s, slen, "%s", lt->name);
94809a3aaf3SDag-Erling Smørgrav 	}
94909a3aaf3SDag-Erling Smørgrav 	return sldns_str_print(s, slen, "OPT%u", (unsigned)opcode);
95009a3aaf3SDag-Erling Smørgrav }
95109a3aaf3SDag-Erling Smørgrav 
95209a3aaf3SDag-Erling Smørgrav int sldns_wire2str_class_scan(uint8_t** d, size_t* dlen, char** s, size_t* slen)
95309a3aaf3SDag-Erling Smørgrav {
95409a3aaf3SDag-Erling Smørgrav 	uint16_t c;
95509a3aaf3SDag-Erling Smørgrav 	if(*dlen == 0) return 0;
95609a3aaf3SDag-Erling Smørgrav 	if(*dlen < 2) return print_remainder_hex("Error malformed 0x", d, dlen, s, slen);
95709a3aaf3SDag-Erling Smørgrav 	c = sldns_read_uint16(*d);
95809a3aaf3SDag-Erling Smørgrav 	(*d)+=2;
95909a3aaf3SDag-Erling Smørgrav 	(*dlen)-=2;
96009a3aaf3SDag-Erling Smørgrav 	return sldns_wire2str_class_print(s, slen, c);
96109a3aaf3SDag-Erling Smørgrav }
96209a3aaf3SDag-Erling Smørgrav 
96309a3aaf3SDag-Erling Smørgrav int sldns_wire2str_type_scan(uint8_t** d, size_t* dlen, char** s, size_t* slen)
96409a3aaf3SDag-Erling Smørgrav {
96509a3aaf3SDag-Erling Smørgrav 	uint16_t t;
96609a3aaf3SDag-Erling Smørgrav 	if(*dlen == 0) return 0;
96709a3aaf3SDag-Erling Smørgrav 	if(*dlen < 2) return print_remainder_hex("Error malformed 0x", d, dlen, s, slen);
96809a3aaf3SDag-Erling Smørgrav 	t = sldns_read_uint16(*d);
96909a3aaf3SDag-Erling Smørgrav 	(*d)+=2;
97009a3aaf3SDag-Erling Smørgrav 	(*dlen)-=2;
97109a3aaf3SDag-Erling Smørgrav 	return sldns_wire2str_type_print(s, slen, t);
97209a3aaf3SDag-Erling Smørgrav }
97309a3aaf3SDag-Erling Smørgrav 
97409a3aaf3SDag-Erling Smørgrav int sldns_wire2str_ttl_scan(uint8_t** d, size_t* dlen, char** s, size_t* slen)
97509a3aaf3SDag-Erling Smørgrav {
97609a3aaf3SDag-Erling Smørgrav 	uint32_t ttl;
97709a3aaf3SDag-Erling Smørgrav 	if(*dlen == 0) return 0;
97809a3aaf3SDag-Erling Smørgrav 	if(*dlen < 4) return print_remainder_hex("Error malformed 0x", d, dlen, s, slen);
97909a3aaf3SDag-Erling Smørgrav 	ttl = sldns_read_uint32(*d);
98009a3aaf3SDag-Erling Smørgrav 	(*d)+=4;
98109a3aaf3SDag-Erling Smørgrav 	(*dlen)-=4;
98209a3aaf3SDag-Erling Smørgrav 	return sldns_str_print(s, slen, "%u", (unsigned)ttl);
98309a3aaf3SDag-Erling Smørgrav }
98409a3aaf3SDag-Erling Smørgrav 
9855469a995SCy Schubert static int
9865469a995SCy Schubert sldns_print_svcparamkey(char** s, size_t* slen, uint16_t svcparamkey)
9875469a995SCy Schubert {
9885469a995SCy Schubert 	if (svcparamkey < SVCPARAMKEY_COUNT) {
9895469a995SCy Schubert 		return sldns_str_print(s, slen, "%s", svcparamkey_strs[svcparamkey]);
9905469a995SCy Schubert 	}
9915469a995SCy Schubert 	else {
9925469a995SCy Schubert 		return sldns_str_print(s, slen, "key%d", (int)svcparamkey);
9935469a995SCy Schubert 	}
9945469a995SCy Schubert }
9955469a995SCy Schubert 
9965469a995SCy Schubert static int sldns_wire2str_svcparam_port2str(char** s,
9975469a995SCy Schubert 	size_t* slen, uint16_t data_len, uint8_t* data)
9985469a995SCy Schubert {
9995469a995SCy Schubert 	int w = 0;
10005469a995SCy Schubert 
10015469a995SCy Schubert 	if (data_len != 2)
10025469a995SCy Schubert 		return -1; /* wireformat error, a short is 2 bytes */
10035469a995SCy Schubert 	w = sldns_str_print(s, slen, "=%d", (int)sldns_read_uint16(data));
10045469a995SCy Schubert 
10055469a995SCy Schubert 	return w;
10065469a995SCy Schubert }
10075469a995SCy Schubert 
10085469a995SCy Schubert static int sldns_wire2str_svcparam_ipv4hint2str(char** s,
10095469a995SCy Schubert 	size_t* slen, uint16_t data_len, uint8_t* data)
10105469a995SCy Schubert {
10115469a995SCy Schubert 	char ip_str[INET_ADDRSTRLEN + 1];
10125469a995SCy Schubert 
10135469a995SCy Schubert 	int w = 0;
10145469a995SCy Schubert 
10155469a995SCy Schubert 	assert(data_len > 0);
10165469a995SCy Schubert 
10175469a995SCy Schubert 	if ((data_len % LDNS_IP4ADDRLEN) == 0) {
10185469a995SCy Schubert 		if (inet_ntop(AF_INET, data, ip_str, sizeof(ip_str)) == NULL)
10195469a995SCy Schubert 			return -1; /* wireformat error, incorrect size or inet family */
10205469a995SCy Schubert 
10215469a995SCy Schubert 		w += sldns_str_print(s, slen, "=%s", ip_str);
10225469a995SCy Schubert 		data += LDNS_IP4ADDRLEN;
10235469a995SCy Schubert 
10245469a995SCy Schubert 		while ((data_len -= LDNS_IP4ADDRLEN) > 0) {
10255469a995SCy Schubert 			if (inet_ntop(AF_INET, data, ip_str, sizeof(ip_str)) == NULL)
10265469a995SCy Schubert 				return -1; /* wireformat error, incorrect size or inet family */
10275469a995SCy Schubert 
10285469a995SCy Schubert 			w += sldns_str_print(s, slen, ",%s", ip_str);
10295469a995SCy Schubert 			data += LDNS_IP4ADDRLEN;
10305469a995SCy Schubert 		}
10315469a995SCy Schubert 	} else
10325469a995SCy Schubert 		return -1;
10335469a995SCy Schubert 
10345469a995SCy Schubert 	return w;
10355469a995SCy Schubert }
10365469a995SCy Schubert 
10375469a995SCy Schubert static int sldns_wire2str_svcparam_ipv6hint2str(char** s,
10385469a995SCy Schubert 	size_t* slen, uint16_t data_len, uint8_t* data)
10395469a995SCy Schubert {
10405469a995SCy Schubert 	char ip_str[INET6_ADDRSTRLEN + 1];
10415469a995SCy Schubert 
10425469a995SCy Schubert 	int w = 0;
10435469a995SCy Schubert 
10445469a995SCy Schubert 	assert(data_len > 0);
10455469a995SCy Schubert 
10465469a995SCy Schubert 	if ((data_len % LDNS_IP6ADDRLEN) == 0) {
10475469a995SCy Schubert 		if (inet_ntop(AF_INET6, data, ip_str, sizeof(ip_str)) == NULL)
10485469a995SCy Schubert 			return -1; /* wireformat error, incorrect size or inet family */
10495469a995SCy Schubert 
10505469a995SCy Schubert 		w += sldns_str_print(s, slen, "=%s", ip_str);
10515469a995SCy Schubert 		data += LDNS_IP6ADDRLEN;
10525469a995SCy Schubert 
10535469a995SCy Schubert 		while ((data_len -= LDNS_IP6ADDRLEN) > 0) {
10545469a995SCy Schubert 			if (inet_ntop(AF_INET6, data, ip_str, sizeof(ip_str)) == NULL)
10555469a995SCy Schubert 				return -1; /* wireformat error, incorrect size or inet family */
10565469a995SCy Schubert 
10575469a995SCy Schubert 			w += sldns_str_print(s, slen, ",%s", ip_str);
10585469a995SCy Schubert 			data += LDNS_IP6ADDRLEN;
10595469a995SCy Schubert 		}
10605469a995SCy Schubert 	} else
10615469a995SCy Schubert 		return -1;
10625469a995SCy Schubert 
10635469a995SCy Schubert 	return w;
10645469a995SCy Schubert }
10655469a995SCy Schubert 
10665469a995SCy Schubert static int sldns_wire2str_svcparam_mandatory2str(char** s,
10675469a995SCy Schubert 	size_t* slen, uint16_t data_len, uint8_t* data)
10685469a995SCy Schubert {
10695469a995SCy Schubert 	int w = 0;
10705469a995SCy Schubert 
10715469a995SCy Schubert 	assert(data_len > 0);
10725469a995SCy Schubert 
10735469a995SCy Schubert 	if (data_len % sizeof(uint16_t))
1074*790c6b24SCy Schubert 		return -1; /* wireformat error, data_len must be multiple of shorts */
10755469a995SCy Schubert 	w += sldns_str_print(s, slen, "=");
10765469a995SCy Schubert 	w += sldns_print_svcparamkey(s, slen, sldns_read_uint16(data));
10775469a995SCy Schubert 	data += 2;
10785469a995SCy Schubert 
10795469a995SCy Schubert 	while ((data_len -= sizeof(uint16_t))) {
10805469a995SCy Schubert 		w += sldns_str_print(s, slen, ",");
10815469a995SCy Schubert 		w += sldns_print_svcparamkey(s, slen, sldns_read_uint16(data));
10825469a995SCy Schubert 		data += 2;
10835469a995SCy Schubert 	}
10845469a995SCy Schubert 
10855469a995SCy Schubert 	return w;
10865469a995SCy Schubert }
10875469a995SCy Schubert 
10885469a995SCy Schubert static int sldns_wire2str_svcparam_alpn2str(char** s,
10895469a995SCy Schubert 	size_t* slen, uint16_t data_len, uint8_t* data)
10905469a995SCy Schubert {
10915469a995SCy Schubert 	uint8_t *dp = (void *)data;
10925469a995SCy Schubert 	int w = 0;
10935469a995SCy Schubert 
10945469a995SCy Schubert 	assert(data_len > 0); /* Guaranteed by sldns_wire2str_svcparam_scan */
10955469a995SCy Schubert 
10965469a995SCy Schubert 	w += sldns_str_print(s, slen, "=\"");
10975469a995SCy Schubert 	while (data_len) {
10985469a995SCy Schubert 		/* alpn is list of length byte (str_len) followed by a string of that size */
10995469a995SCy Schubert 		uint8_t i, str_len = *dp++;
11005469a995SCy Schubert 
11015469a995SCy Schubert 		if (str_len > --data_len)
11025469a995SCy Schubert 			return -1;
11035469a995SCy Schubert 
11045469a995SCy Schubert 		for (i = 0; i < str_len; i++) {
11055469a995SCy Schubert 			if (dp[i] == '"' || dp[i] == '\\')
11065469a995SCy Schubert 				w += sldns_str_print(s, slen, "\\\\\\%c", dp[i]);
11075469a995SCy Schubert 
11085469a995SCy Schubert 			else if (dp[i] == ',')
11095469a995SCy Schubert 				w += sldns_str_print(s, slen, "\\\\%c", dp[i]);
11105469a995SCy Schubert 
11115469a995SCy Schubert 			else if (!isprint(dp[i]))
11125469a995SCy Schubert 				w += sldns_str_print(s, slen, "\\%03u", (unsigned) dp[i]);
11135469a995SCy Schubert 
11145469a995SCy Schubert 			else
11155469a995SCy Schubert 				w += sldns_str_print(s, slen, "%c", dp[i]);
11165469a995SCy Schubert 		}
11175469a995SCy Schubert 		dp += str_len;
11185469a995SCy Schubert 		if ((data_len -= str_len))
11195469a995SCy Schubert 			w += sldns_str_print(s, slen, "%s", ",");
11205469a995SCy Schubert 	}
11215469a995SCy Schubert 	w += sldns_str_print(s, slen, "\"");
11225469a995SCy Schubert 
11235469a995SCy Schubert 	return w;
11245469a995SCy Schubert }
11255469a995SCy Schubert 
11265469a995SCy Schubert static int sldns_wire2str_svcparam_ech2str(char** s,
11275469a995SCy Schubert 	size_t* slen, uint16_t data_len, uint8_t* data)
11285469a995SCy Schubert {
11295469a995SCy Schubert 	int size;
11305469a995SCy Schubert 	int w = 0;
11315469a995SCy Schubert 
11325469a995SCy Schubert 	assert(data_len > 0); /* Guaranteed by sldns_wire2str_svcparam_scan */
11335469a995SCy Schubert 
11345469a995SCy Schubert 	w += sldns_str_print(s, slen, "=\"");
11355469a995SCy Schubert 
11365469a995SCy Schubert 	if ((size = sldns_b64_ntop(data, data_len, *s, *slen)) < 0)
11375469a995SCy Schubert 		return -1;
11385469a995SCy Schubert 
11395469a995SCy Schubert 	(*s) += size;
11405469a995SCy Schubert 	(*slen) -= size;
11415469a995SCy Schubert 
11425469a995SCy Schubert 	w += sldns_str_print(s, slen, "\"");
11435469a995SCy Schubert 
11445469a995SCy Schubert 	return w + size;
11455469a995SCy Schubert }
11465469a995SCy Schubert 
11475469a995SCy Schubert int sldns_wire2str_svcparam_scan(uint8_t** d, size_t* dlen, char** s, size_t* slen)
11485469a995SCy Schubert {
11495469a995SCy Schubert 	uint8_t ch;
11505469a995SCy Schubert 	uint16_t svcparamkey, data_len;
11515469a995SCy Schubert 	int written_chars = 0;
11525469a995SCy Schubert 	int r, i;
11535469a995SCy Schubert 
11545469a995SCy Schubert 	/* verify that we have enough data to read svcparamkey and data_len */
11555469a995SCy Schubert 	if(*dlen < 4)
11565469a995SCy Schubert 		return -1;
11575469a995SCy Schubert 
11585469a995SCy Schubert 	svcparamkey = sldns_read_uint16(*d);
11595469a995SCy Schubert 	data_len = sldns_read_uint16(*d+2);
11605469a995SCy Schubert 	*d    += 4;
11615469a995SCy Schubert 	*dlen -= 4;
11625469a995SCy Schubert 
11635469a995SCy Schubert 	/* verify that we have data_len data */
11645469a995SCy Schubert 	if (data_len > *dlen)
11655469a995SCy Schubert 		return -1;
11665469a995SCy Schubert 
11675469a995SCy Schubert 	written_chars += sldns_print_svcparamkey(s, slen, svcparamkey);
11685469a995SCy Schubert 	if (!data_len) {
11695469a995SCy Schubert 
11705469a995SCy Schubert 	 	/* Some SvcParams MUST have values */
11715469a995SCy Schubert 	 	switch (svcparamkey) {
11725469a995SCy Schubert 	 	case SVCB_KEY_ALPN:
11735469a995SCy Schubert 	 	case SVCB_KEY_PORT:
11745469a995SCy Schubert 	 	case SVCB_KEY_IPV4HINT:
11755469a995SCy Schubert 	 	case SVCB_KEY_IPV6HINT:
11765469a995SCy Schubert 	 	case SVCB_KEY_MANDATORY:
11775469a995SCy Schubert 	 		return -1;
11785469a995SCy Schubert 	 	default:
11795469a995SCy Schubert 	 		return written_chars;
11805469a995SCy Schubert 	 	}
11815469a995SCy Schubert 	}
11825469a995SCy Schubert 
11835469a995SCy Schubert 	switch (svcparamkey) {
11845469a995SCy Schubert 	case SVCB_KEY_PORT:
11855469a995SCy Schubert 		r = sldns_wire2str_svcparam_port2str(s, slen, data_len, *d);
11865469a995SCy Schubert 		break;
11875469a995SCy Schubert 	case SVCB_KEY_IPV4HINT:
11885469a995SCy Schubert 		r = sldns_wire2str_svcparam_ipv4hint2str(s, slen, data_len, *d);
11895469a995SCy Schubert 		break;
11905469a995SCy Schubert 	case SVCB_KEY_IPV6HINT:
11915469a995SCy Schubert 		r = sldns_wire2str_svcparam_ipv6hint2str(s, slen, data_len, *d);
11925469a995SCy Schubert 		break;
11935469a995SCy Schubert 	case SVCB_KEY_MANDATORY:
11945469a995SCy Schubert 		r = sldns_wire2str_svcparam_mandatory2str(s, slen, data_len, *d);
11955469a995SCy Schubert 		break;
11965469a995SCy Schubert 	case SVCB_KEY_NO_DEFAULT_ALPN:
11975469a995SCy Schubert 		return -1;  /* wireformat error, should not have a value */
11985469a995SCy Schubert 	case SVCB_KEY_ALPN:
11995469a995SCy Schubert 		r = sldns_wire2str_svcparam_alpn2str(s, slen, data_len, *d);
12005469a995SCy Schubert 		break;
12015469a995SCy Schubert 	case SVCB_KEY_ECH:
12025469a995SCy Schubert 		r = sldns_wire2str_svcparam_ech2str(s, slen, data_len, *d);
12035469a995SCy Schubert 		break;
12045469a995SCy Schubert 	default:
12055469a995SCy Schubert 		r = sldns_str_print(s, slen, "=\"");
12065469a995SCy Schubert 
12075469a995SCy Schubert 		for (i = 0; i < data_len; i++) {
12085469a995SCy Schubert 			ch = (*d)[i];
12095469a995SCy Schubert 
12105469a995SCy Schubert 			if (ch == '"' || ch == '\\')
12115469a995SCy Schubert 				r += sldns_str_print(s, slen, "\\%c", ch);
12125469a995SCy Schubert 
12135469a995SCy Schubert 			else if (!isprint(ch))
12145469a995SCy Schubert 				r += sldns_str_print(s, slen, "\\%03u", (unsigned) ch);
12155469a995SCy Schubert 
12165469a995SCy Schubert 			else
12175469a995SCy Schubert 				r += sldns_str_print(s, slen, "%c", ch);
12185469a995SCy Schubert 
12195469a995SCy Schubert 		}
12205469a995SCy Schubert 		r += sldns_str_print(s, slen, "\"");
12215469a995SCy Schubert 		break;
12225469a995SCy Schubert 	}
12235469a995SCy Schubert 	if (r <= 0)
12245469a995SCy Schubert 		return -1; /* wireformat error */
12255469a995SCy Schubert 
12265469a995SCy Schubert 	written_chars += r;
12275469a995SCy Schubert 	*d    += data_len;
12285469a995SCy Schubert 	*dlen -= data_len;
12295469a995SCy Schubert 	return written_chars;
12305469a995SCy Schubert }
12315469a995SCy Schubert 
123209a3aaf3SDag-Erling Smørgrav int sldns_wire2str_rdf_scan(uint8_t** d, size_t* dlen, char** s, size_t* slen,
12330eefd307SCy Schubert 	int rdftype, uint8_t* pkt, size_t pktlen, int* comprloop)
123409a3aaf3SDag-Erling Smørgrav {
123509a3aaf3SDag-Erling Smørgrav 	if(*dlen == 0) return 0;
123609a3aaf3SDag-Erling Smørgrav 	switch(rdftype) {
123709a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_NONE:
123809a3aaf3SDag-Erling Smørgrav 		return 0;
123909a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_DNAME:
12400eefd307SCy Schubert 		return sldns_wire2str_dname_scan(d, dlen, s, slen, pkt, pktlen, comprloop);
124109a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_INT8:
124209a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_int8_scan(d, dlen, s, slen);
124309a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_INT16:
124409a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_int16_scan(d, dlen, s, slen);
124509a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_INT32:
124609a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_int32_scan(d, dlen, s, slen);
124709a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_PERIOD:
124809a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_period_scan(d, dlen, s, slen);
124909a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_TSIGTIME:
125009a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_tsigtime_scan(d, dlen, s, slen);
125109a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_A:
125209a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_a_scan(d, dlen, s, slen);
125309a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_AAAA:
125409a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_aaaa_scan(d, dlen, s, slen);
125509a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_STR:
125609a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_str_scan(d, dlen, s, slen);
125709a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_APL:
125809a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_apl_scan(d, dlen, s, slen);
125909a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_B32_EXT:
126009a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_b32_ext_scan(d, dlen, s, slen);
126109a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_B64:
126209a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_b64_scan(d, dlen, s, slen);
126309a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_HEX:
126409a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_hex_scan(d, dlen, s, slen);
126509a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_NSEC:
126609a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_nsec_scan(d, dlen, s, slen);
126709a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_NSEC3_SALT:
126809a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_nsec3_salt_scan(d, dlen, s, slen);
126909a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_TYPE:
127009a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_type_scan(d, dlen, s, slen);
127109a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_CLASS:
127209a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_class_scan(d, dlen, s, slen);
127309a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_CERT_ALG:
127409a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_cert_alg_scan(d, dlen, s, slen);
127509a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_ALG:
127609a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_alg_scan(d, dlen, s, slen);
127709a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_UNKNOWN:
127809a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_unknown_scan(d, dlen, s, slen);
127909a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_TIME:
128009a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_time_scan(d, dlen, s, slen);
128109a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_LOC:
128209a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_loc_scan(d, dlen, s, slen);
128309a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_WKS:
128409a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_SERVICE:
128509a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_wks_scan(d, dlen, s, slen);
128609a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_NSAP:
128709a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_nsap_scan(d, dlen, s, slen);
128809a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_ATMA:
128909a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_atma_scan(d, dlen, s, slen);
129009a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_IPSECKEY:
129109a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_ipseckey_scan(d, dlen, s, slen, pkt,
12920eefd307SCy Schubert 			pktlen, comprloop);
129309a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_HIP:
129409a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_hip_scan(d, dlen, s, slen);
129509a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_INT16_DATA:
129609a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_int16_data_scan(d, dlen, s, slen);
129709a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_NSEC3_NEXT_OWNER:
129809a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_b32_ext_scan(d, dlen, s, slen);
129909a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_ILNP64:
130009a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_ilnp64_scan(d, dlen, s, slen);
130109a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_EUI48:
130209a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_eui48_scan(d, dlen, s, slen);
130309a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_EUI64:
130409a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_eui64_scan(d, dlen, s, slen);
130509a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_TAG:
130609a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_tag_scan(d, dlen, s, slen);
130709a3aaf3SDag-Erling Smørgrav 	case LDNS_RDF_TYPE_LONG_STR:
130809a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_long_str_scan(d, dlen, s, slen);
13095469a995SCy Schubert 	case LDNS_RDF_TYPE_SVCPARAM:
13105469a995SCy Schubert 		return sldns_wire2str_svcparam_scan(d, dlen, s, slen);
1311c7f4d7adSDag-Erling Smørgrav 	case LDNS_RDF_TYPE_TSIGERROR:
1312c7f4d7adSDag-Erling Smørgrav 		return sldns_wire2str_tsigerror_scan(d, dlen, s, slen);
131309a3aaf3SDag-Erling Smørgrav 	}
131409a3aaf3SDag-Erling Smørgrav 	/* unknown rdf type */
131509a3aaf3SDag-Erling Smørgrav 	return -1;
131609a3aaf3SDag-Erling Smørgrav }
131709a3aaf3SDag-Erling Smørgrav 
131809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_int8_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
131909a3aaf3SDag-Erling Smørgrav {
132009a3aaf3SDag-Erling Smørgrav 	int w;
132109a3aaf3SDag-Erling Smørgrav 	if(*dl < 1) return -1;
132209a3aaf3SDag-Erling Smørgrav 	w = sldns_str_print(s, sl, "%u", (unsigned)**d);
132309a3aaf3SDag-Erling Smørgrav 	(*d)++;
132409a3aaf3SDag-Erling Smørgrav 	(*dl)--;
132509a3aaf3SDag-Erling Smørgrav 	return w;
132609a3aaf3SDag-Erling Smørgrav }
132709a3aaf3SDag-Erling Smørgrav 
132809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_int16_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
132909a3aaf3SDag-Erling Smørgrav {
133009a3aaf3SDag-Erling Smørgrav 	int w;
133109a3aaf3SDag-Erling Smørgrav 	if(*dl < 2) return -1;
133209a3aaf3SDag-Erling Smørgrav 	w = sldns_str_print(s, sl, "%lu", (unsigned long)sldns_read_uint16(*d));
133309a3aaf3SDag-Erling Smørgrav 	(*d)+=2;
133409a3aaf3SDag-Erling Smørgrav 	(*dl)-=2;
133509a3aaf3SDag-Erling Smørgrav 	return w;
133609a3aaf3SDag-Erling Smørgrav }
133709a3aaf3SDag-Erling Smørgrav 
133809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_int32_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
133909a3aaf3SDag-Erling Smørgrav {
134009a3aaf3SDag-Erling Smørgrav 	int w;
134109a3aaf3SDag-Erling Smørgrav 	if(*dl < 4) return -1;
134209a3aaf3SDag-Erling Smørgrav 	w = sldns_str_print(s, sl, "%lu", (unsigned long)sldns_read_uint32(*d));
134309a3aaf3SDag-Erling Smørgrav 	(*d)+=4;
134409a3aaf3SDag-Erling Smørgrav 	(*dl)-=4;
134509a3aaf3SDag-Erling Smørgrav 	return w;
134609a3aaf3SDag-Erling Smørgrav }
134709a3aaf3SDag-Erling Smørgrav 
134809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_period_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
134909a3aaf3SDag-Erling Smørgrav {
135009a3aaf3SDag-Erling Smørgrav 	int w;
135109a3aaf3SDag-Erling Smørgrav 	if(*dl < 4) return -1;
135209a3aaf3SDag-Erling Smørgrav 	w = sldns_str_print(s, sl, "%u", (unsigned)sldns_read_uint32(*d));
135309a3aaf3SDag-Erling Smørgrav 	(*d)+=4;
135409a3aaf3SDag-Erling Smørgrav 	(*dl)-=4;
135509a3aaf3SDag-Erling Smørgrav 	return w;
135609a3aaf3SDag-Erling Smørgrav }
135709a3aaf3SDag-Erling Smørgrav 
135809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_tsigtime_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
135909a3aaf3SDag-Erling Smørgrav {
136009a3aaf3SDag-Erling Smørgrav 	/* tsigtime is 48 bits network order unsigned integer */
136109a3aaf3SDag-Erling Smørgrav 	int w;
136209a3aaf3SDag-Erling Smørgrav 	uint64_t tsigtime = 0;
136309a3aaf3SDag-Erling Smørgrav 	uint64_t d0, d1, d2, d3, d4, d5;
136409a3aaf3SDag-Erling Smørgrav 	if(*dl < 6) return -1;
136509a3aaf3SDag-Erling Smørgrav 	d0 = (*d)[0]; /* cast to uint64 for shift operations */
136609a3aaf3SDag-Erling Smørgrav 	d1 = (*d)[1];
136709a3aaf3SDag-Erling Smørgrav 	d2 = (*d)[2];
136809a3aaf3SDag-Erling Smørgrav 	d3 = (*d)[3];
136909a3aaf3SDag-Erling Smørgrav 	d4 = (*d)[4];
137009a3aaf3SDag-Erling Smørgrav 	d5 = (*d)[5];
137109a3aaf3SDag-Erling Smørgrav 	tsigtime = (d0<<40) | (d1<<32) | (d2<<24) | (d3<<16) | (d4<<8) | d5;
137209a3aaf3SDag-Erling Smørgrav #ifndef USE_WINSOCK
137309a3aaf3SDag-Erling Smørgrav 	w = sldns_str_print(s, sl, "%llu", (long long)tsigtime);
137409a3aaf3SDag-Erling Smørgrav #else
137509a3aaf3SDag-Erling Smørgrav 	w = sldns_str_print(s, sl, "%I64u", (long long)tsigtime);
137609a3aaf3SDag-Erling Smørgrav #endif
137709a3aaf3SDag-Erling Smørgrav 	(*d)+=6;
137809a3aaf3SDag-Erling Smørgrav 	(*dl)-=6;
137909a3aaf3SDag-Erling Smørgrav 	return w;
138009a3aaf3SDag-Erling Smørgrav }
138109a3aaf3SDag-Erling Smørgrav 
138209a3aaf3SDag-Erling Smørgrav int sldns_wire2str_a_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
138309a3aaf3SDag-Erling Smørgrav {
138409a3aaf3SDag-Erling Smørgrav 	char buf[32];
138509a3aaf3SDag-Erling Smørgrav 	int w;
138609a3aaf3SDag-Erling Smørgrav 	if(*dl < 4) return -1;
138709a3aaf3SDag-Erling Smørgrav 	if(!inet_ntop(AF_INET, *d, buf, (socklen_t)sizeof(buf)))
138809a3aaf3SDag-Erling Smørgrav 		return -1;
138909a3aaf3SDag-Erling Smørgrav 	w = sldns_str_print(s, sl, "%s", buf);
139009a3aaf3SDag-Erling Smørgrav 	(*d)+=4;
139109a3aaf3SDag-Erling Smørgrav 	(*dl)-=4;
139209a3aaf3SDag-Erling Smørgrav 	return w;
139309a3aaf3SDag-Erling Smørgrav }
139409a3aaf3SDag-Erling Smørgrav 
139509a3aaf3SDag-Erling Smørgrav int sldns_wire2str_aaaa_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
139609a3aaf3SDag-Erling Smørgrav {
139709a3aaf3SDag-Erling Smørgrav #ifdef AF_INET6
139809a3aaf3SDag-Erling Smørgrav 	char buf[64];
139909a3aaf3SDag-Erling Smørgrav 	int w;
140009a3aaf3SDag-Erling Smørgrav 	if(*dl < 16) return -1;
140109a3aaf3SDag-Erling Smørgrav 	if(!inet_ntop(AF_INET6, *d, buf, (socklen_t)sizeof(buf)))
140209a3aaf3SDag-Erling Smørgrav 		return -1;
140309a3aaf3SDag-Erling Smørgrav 	w = sldns_str_print(s, sl, "%s", buf);
140409a3aaf3SDag-Erling Smørgrav 	(*d)+=16;
140509a3aaf3SDag-Erling Smørgrav 	(*dl)-=16;
140609a3aaf3SDag-Erling Smørgrav 	return w;
140709a3aaf3SDag-Erling Smørgrav #else
140809a3aaf3SDag-Erling Smørgrav 	return -1;
140909a3aaf3SDag-Erling Smørgrav #endif
141009a3aaf3SDag-Erling Smørgrav }
141109a3aaf3SDag-Erling Smørgrav 
141209a3aaf3SDag-Erling Smørgrav /** printout escaped TYPE_STR character */
141309a3aaf3SDag-Erling Smørgrav static int str_char_print(char** s, size_t* sl, uint8_t c)
141409a3aaf3SDag-Erling Smørgrav {
141509a3aaf3SDag-Erling Smørgrav 	if(isprint((unsigned char)c) || c == '\t') {
141609a3aaf3SDag-Erling Smørgrav 		if(c == '\"' || c == '\\')
141709a3aaf3SDag-Erling Smørgrav 			return sldns_str_print(s, sl, "\\%c", c);
141809a3aaf3SDag-Erling Smørgrav 		if(*sl) {
141909a3aaf3SDag-Erling Smørgrav 			**s = (char)c;
142009a3aaf3SDag-Erling Smørgrav 			(*s)++;
142109a3aaf3SDag-Erling Smørgrav 			(*sl)--;
142209a3aaf3SDag-Erling Smørgrav 		}
142309a3aaf3SDag-Erling Smørgrav 		return 1;
142409a3aaf3SDag-Erling Smørgrav 	}
142509a3aaf3SDag-Erling Smørgrav 	return sldns_str_print(s, sl, "\\%03u", (unsigned)c);
142609a3aaf3SDag-Erling Smørgrav }
142709a3aaf3SDag-Erling Smørgrav 
142809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_str_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
142909a3aaf3SDag-Erling Smørgrav {
143009a3aaf3SDag-Erling Smørgrav 	int w = 0;
143109a3aaf3SDag-Erling Smørgrav 	size_t i, len;
143209a3aaf3SDag-Erling Smørgrav 	if(*dl < 1) return -1;
143309a3aaf3SDag-Erling Smørgrav 	len = **d;
143409a3aaf3SDag-Erling Smørgrav 	if(*dl < 1+len) return -1;
143509a3aaf3SDag-Erling Smørgrav 	(*d)++;
143609a3aaf3SDag-Erling Smørgrav 	(*dl)--;
143709a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, "\"");
143809a3aaf3SDag-Erling Smørgrav 	for(i=0; i<len; i++)
143909a3aaf3SDag-Erling Smørgrav 		w += str_char_print(s, sl, (*d)[i]);
144009a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, "\"");
144109a3aaf3SDag-Erling Smørgrav 	(*d)+=len;
144209a3aaf3SDag-Erling Smørgrav 	(*dl)-=len;
144309a3aaf3SDag-Erling Smørgrav 	return w;
144409a3aaf3SDag-Erling Smørgrav }
144509a3aaf3SDag-Erling Smørgrav 
144609a3aaf3SDag-Erling Smørgrav int sldns_wire2str_apl_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
144709a3aaf3SDag-Erling Smørgrav {
144809a3aaf3SDag-Erling Smørgrav 	int i, w = 0;
144909a3aaf3SDag-Erling Smørgrav 	uint16_t family;
145009a3aaf3SDag-Erling Smørgrav 	uint8_t negation, prefix, adflength;
145109a3aaf3SDag-Erling Smørgrav 	if(*dl < 4) return -1;
145209a3aaf3SDag-Erling Smørgrav 	family = sldns_read_uint16(*d);
145309a3aaf3SDag-Erling Smørgrav 	prefix = (*d)[2];
145409a3aaf3SDag-Erling Smørgrav 	negation = ((*d)[3] & LDNS_APL_NEGATION);
145509a3aaf3SDag-Erling Smørgrav 	adflength = ((*d)[3] & LDNS_APL_MASK);
145609a3aaf3SDag-Erling Smørgrav 	if(*dl < 4+(size_t)adflength) return -1;
145709a3aaf3SDag-Erling Smørgrav 	if(family != LDNS_APL_IP4 && family != LDNS_APL_IP6)
145809a3aaf3SDag-Erling Smørgrav 		return -1; /* unknown address family */
145909a3aaf3SDag-Erling Smørgrav 	if(negation)
146009a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "!");
146109a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, "%u:", (unsigned)family);
146209a3aaf3SDag-Erling Smørgrav 	if(family == LDNS_APL_IP4) {
146309a3aaf3SDag-Erling Smørgrav 		/* check if prefix <32 ? */
146409a3aaf3SDag-Erling Smørgrav 		/* address is variable length 0 - 4 */
146509a3aaf3SDag-Erling Smørgrav 		for(i=0; i<4; i++) {
146609a3aaf3SDag-Erling Smørgrav 			if(i > 0)
146709a3aaf3SDag-Erling Smørgrav 				w += sldns_str_print(s, sl, ".");
146809a3aaf3SDag-Erling Smørgrav 			if(i < (int)adflength)
146909a3aaf3SDag-Erling Smørgrav 				w += sldns_str_print(s, sl, "%d", (*d)[4+i]);
147009a3aaf3SDag-Erling Smørgrav 			else	w += sldns_str_print(s, sl, "0");
147109a3aaf3SDag-Erling Smørgrav 		}
147209a3aaf3SDag-Erling Smørgrav 	} else if(family == LDNS_APL_IP6) {
147309a3aaf3SDag-Erling Smørgrav 		/* check if prefix <128 ? */
147409a3aaf3SDag-Erling Smørgrav 		/* address is variable length 0 - 16 */
147509a3aaf3SDag-Erling Smørgrav 		for(i=0; i<16; i++) {
147609a3aaf3SDag-Erling Smørgrav 			if(i%2 == 0 && i>0)
147709a3aaf3SDag-Erling Smørgrav 				w += sldns_str_print(s, sl, ":");
147809a3aaf3SDag-Erling Smørgrav 			if(i < (int)adflength)
147909a3aaf3SDag-Erling Smørgrav 				w += sldns_str_print(s, sl, "%02x", (*d)[4+i]);
148009a3aaf3SDag-Erling Smørgrav 			else	w += sldns_str_print(s, sl, "00");
148109a3aaf3SDag-Erling Smørgrav 		}
148209a3aaf3SDag-Erling Smørgrav 	}
148309a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, "/%u", (unsigned)prefix);
148409a3aaf3SDag-Erling Smørgrav 	(*d) += 4+adflength;
148509a3aaf3SDag-Erling Smørgrav 	(*dl) -= 4+adflength;
148609a3aaf3SDag-Erling Smørgrav 	return w;
148709a3aaf3SDag-Erling Smørgrav }
148809a3aaf3SDag-Erling Smørgrav 
148909a3aaf3SDag-Erling Smørgrav int sldns_wire2str_b32_ext_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
149009a3aaf3SDag-Erling Smørgrav {
149109a3aaf3SDag-Erling Smørgrav 	size_t datalen;
149209a3aaf3SDag-Erling Smørgrav 	size_t sz;
149309a3aaf3SDag-Erling Smørgrav 	if(*dl < 1) return -1;
149409a3aaf3SDag-Erling Smørgrav 	datalen = (*d)[0];
149509a3aaf3SDag-Erling Smørgrav 	if(*dl < 1+datalen) return -1;
149609a3aaf3SDag-Erling Smørgrav 	sz = sldns_b32_ntop_calculate_size(datalen);
149709a3aaf3SDag-Erling Smørgrav 	if(*sl < sz+1) {
149809a3aaf3SDag-Erling Smørgrav 		(*d) += datalen+1;
149909a3aaf3SDag-Erling Smørgrav 		(*dl) -= (datalen+1);
150009a3aaf3SDag-Erling Smørgrav 		return (int)sz; /* out of space really, but would need buffer
150109a3aaf3SDag-Erling Smørgrav 			in order to truncate the output */
150209a3aaf3SDag-Erling Smørgrav 	}
150309a3aaf3SDag-Erling Smørgrav 	sldns_b32_ntop_extended_hex((*d)+1, datalen, *s, *sl);
150409a3aaf3SDag-Erling Smørgrav 	(*d) += datalen+1;
150509a3aaf3SDag-Erling Smørgrav 	(*dl) -= (datalen+1);
150609a3aaf3SDag-Erling Smørgrav 	(*s) += sz;
150709a3aaf3SDag-Erling Smørgrav 	(*sl) -= sz;
150809a3aaf3SDag-Erling Smørgrav 	return (int)sz;
150909a3aaf3SDag-Erling Smørgrav }
151009a3aaf3SDag-Erling Smørgrav 
151109a3aaf3SDag-Erling Smørgrav /** scan number of bytes from wire into b64 presentation format */
151209a3aaf3SDag-Erling Smørgrav static int sldns_wire2str_b64_scan_num(uint8_t** d, size_t* dl, char** s,
151309a3aaf3SDag-Erling Smørgrav 	size_t* sl, size_t num)
151409a3aaf3SDag-Erling Smørgrav {
151509a3aaf3SDag-Erling Smørgrav 	/* b64_ntop_calculate size includes null at the end */
151609a3aaf3SDag-Erling Smørgrav 	size_t sz = sldns_b64_ntop_calculate_size(num)-1;
151709a3aaf3SDag-Erling Smørgrav 	if(*sl < sz+1) {
151809a3aaf3SDag-Erling Smørgrav 		(*d) += num;
151909a3aaf3SDag-Erling Smørgrav 		(*dl) -= num;
152009a3aaf3SDag-Erling Smørgrav 		return (int)sz; /* out of space really, but would need buffer
152109a3aaf3SDag-Erling Smørgrav 			in order to truncate the output */
152209a3aaf3SDag-Erling Smørgrav 	}
152309a3aaf3SDag-Erling Smørgrav 	sldns_b64_ntop(*d, num, *s, *sl);
152409a3aaf3SDag-Erling Smørgrav 	(*d) += num;
152509a3aaf3SDag-Erling Smørgrav 	(*dl) -= num;
152609a3aaf3SDag-Erling Smørgrav 	(*s) += sz;
152709a3aaf3SDag-Erling Smørgrav 	(*sl) -= sz;
152809a3aaf3SDag-Erling Smørgrav 	return (int)sz;
152909a3aaf3SDag-Erling Smørgrav }
153009a3aaf3SDag-Erling Smørgrav 
153109a3aaf3SDag-Erling Smørgrav int sldns_wire2str_b64_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
153209a3aaf3SDag-Erling Smørgrav {
1533971980c3SDag-Erling Smørgrav 	if(*dl == 0) {
1534971980c3SDag-Erling Smørgrav 		return sldns_str_print(s, sl, "0");
1535971980c3SDag-Erling Smørgrav 	}
153609a3aaf3SDag-Erling Smørgrav 	return sldns_wire2str_b64_scan_num(d, dl, s, sl, *dl);
153709a3aaf3SDag-Erling Smørgrav }
153809a3aaf3SDag-Erling Smørgrav 
153909a3aaf3SDag-Erling Smørgrav int sldns_wire2str_hex_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
154009a3aaf3SDag-Erling Smørgrav {
1541971980c3SDag-Erling Smørgrav 	if(*dl == 0) {
1542971980c3SDag-Erling Smørgrav 		return sldns_str_print(s, sl, "0");
1543971980c3SDag-Erling Smørgrav 	}
154409a3aaf3SDag-Erling Smørgrav 	return print_remainder_hex("", d, dl, s, sl);
154509a3aaf3SDag-Erling Smørgrav }
154609a3aaf3SDag-Erling Smørgrav 
154709a3aaf3SDag-Erling Smørgrav int sldns_wire2str_nsec_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
154809a3aaf3SDag-Erling Smørgrav {
154909a3aaf3SDag-Erling Smørgrav 	uint8_t* p = *d;
155009a3aaf3SDag-Erling Smørgrav 	size_t pl = *dl;
155109a3aaf3SDag-Erling Smørgrav 	unsigned i, bit, window, block_len;
155209a3aaf3SDag-Erling Smørgrav 	uint16_t t;
155309a3aaf3SDag-Erling Smørgrav 	int w = 0;
155409a3aaf3SDag-Erling Smørgrav 
155509a3aaf3SDag-Erling Smørgrav 	/* check for errors */
155609a3aaf3SDag-Erling Smørgrav 	while(pl) {
155709a3aaf3SDag-Erling Smørgrav 		if(pl < 2) return -1;
155809a3aaf3SDag-Erling Smørgrav 		block_len = (unsigned)p[1];
155909a3aaf3SDag-Erling Smørgrav 		if(pl < 2+block_len) return -1;
156009a3aaf3SDag-Erling Smørgrav 		p += block_len+2;
156109a3aaf3SDag-Erling Smørgrav 		pl -= block_len+2;
156209a3aaf3SDag-Erling Smørgrav 	}
156309a3aaf3SDag-Erling Smørgrav 
156409a3aaf3SDag-Erling Smørgrav 	/* do it */
156509a3aaf3SDag-Erling Smørgrav 	p = *d;
156609a3aaf3SDag-Erling Smørgrav 	pl = *dl;
156709a3aaf3SDag-Erling Smørgrav 	while(pl) {
156809a3aaf3SDag-Erling Smørgrav 		if(pl < 2) return -1; /* cannot happen */
156909a3aaf3SDag-Erling Smørgrav 		window = (unsigned)p[0];
157009a3aaf3SDag-Erling Smørgrav 		block_len = (unsigned)p[1];
157109a3aaf3SDag-Erling Smørgrav 		if(pl < 2+block_len) return -1; /* cannot happen */
157209a3aaf3SDag-Erling Smørgrav 		p += 2;
157309a3aaf3SDag-Erling Smørgrav 		for(i=0; i<block_len; i++) {
157409a3aaf3SDag-Erling Smørgrav 			if(p[i] == 0) continue;
157509a3aaf3SDag-Erling Smørgrav 			/* base type number for this octet */
157609a3aaf3SDag-Erling Smørgrav 			t = ((window)<<8) | (i << 3);
157709a3aaf3SDag-Erling Smørgrav 			for(bit=0; bit<8; bit++) {
157809a3aaf3SDag-Erling Smørgrav 				if((p[i]&(0x80>>bit))) {
157909a3aaf3SDag-Erling Smørgrav 					if(w) w += sldns_str_print(s, sl, " ");
158009a3aaf3SDag-Erling Smørgrav 					w += sldns_wire2str_type_print(s, sl,
158109a3aaf3SDag-Erling Smørgrav 						t+bit);
158209a3aaf3SDag-Erling Smørgrav 				}
158309a3aaf3SDag-Erling Smørgrav 			}
158409a3aaf3SDag-Erling Smørgrav 		}
158509a3aaf3SDag-Erling Smørgrav 		p += block_len;
158609a3aaf3SDag-Erling Smørgrav 		pl -= block_len+2;
158709a3aaf3SDag-Erling Smørgrav 	}
158809a3aaf3SDag-Erling Smørgrav 	(*d) += *dl;
158909a3aaf3SDag-Erling Smørgrav 	(*dl) = 0;
159009a3aaf3SDag-Erling Smørgrav 	return w;
159109a3aaf3SDag-Erling Smørgrav }
159209a3aaf3SDag-Erling Smørgrav 
159309a3aaf3SDag-Erling Smørgrav int sldns_wire2str_nsec3_salt_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
159409a3aaf3SDag-Erling Smørgrav {
159509a3aaf3SDag-Erling Smørgrav 	size_t salt_len;
159609a3aaf3SDag-Erling Smørgrav 	int w;
159709a3aaf3SDag-Erling Smørgrav 	if(*dl < 1) return -1;
159809a3aaf3SDag-Erling Smørgrav 	salt_len = (size_t)(*d)[0];
159909a3aaf3SDag-Erling Smørgrav 	if(*dl < 1+salt_len) return -1;
160009a3aaf3SDag-Erling Smørgrav 	(*d)++;
160109a3aaf3SDag-Erling Smørgrav 	(*dl)--;
160209a3aaf3SDag-Erling Smørgrav 	if(salt_len == 0) {
160309a3aaf3SDag-Erling Smørgrav 		return sldns_str_print(s, sl, "-");
160409a3aaf3SDag-Erling Smørgrav 	}
160509a3aaf3SDag-Erling Smørgrav 	w = print_hex_buf(s, sl, *d, salt_len);
160609a3aaf3SDag-Erling Smørgrav 	(*dl)-=salt_len;
160709a3aaf3SDag-Erling Smørgrav 	(*d)+=salt_len;
160809a3aaf3SDag-Erling Smørgrav 	return w;
160909a3aaf3SDag-Erling Smørgrav }
161009a3aaf3SDag-Erling Smørgrav 
161109a3aaf3SDag-Erling Smørgrav int sldns_wire2str_cert_alg_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
161209a3aaf3SDag-Erling Smørgrav {
161309a3aaf3SDag-Erling Smørgrav 	sldns_lookup_table *lt;
161409a3aaf3SDag-Erling Smørgrav 	int data, w;
161509a3aaf3SDag-Erling Smørgrav 	if(*dl < 2) return -1;
161609a3aaf3SDag-Erling Smørgrav 	data = (int)sldns_read_uint16(*d);
161709a3aaf3SDag-Erling Smørgrav 	lt = sldns_lookup_by_id(sldns_cert_algorithms, data);
161809a3aaf3SDag-Erling Smørgrav 	if(lt && lt->name)
161909a3aaf3SDag-Erling Smørgrav 		w = sldns_str_print(s, sl, "%s", lt->name);
162009a3aaf3SDag-Erling Smørgrav 	else 	w = sldns_str_print(s, sl, "%d", data);
162109a3aaf3SDag-Erling Smørgrav 	(*dl)-=2;
162209a3aaf3SDag-Erling Smørgrav 	(*d)+=2;
162309a3aaf3SDag-Erling Smørgrav 	return w;
162409a3aaf3SDag-Erling Smørgrav }
162509a3aaf3SDag-Erling Smørgrav 
162609a3aaf3SDag-Erling Smørgrav int sldns_wire2str_alg_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
162709a3aaf3SDag-Erling Smørgrav {
162809a3aaf3SDag-Erling Smørgrav 	/* don't use algorithm mnemonics in the presentation format
162909a3aaf3SDag-Erling Smørgrav 	 * this kind of got sneaked into the rfc's */
163009a3aaf3SDag-Erling Smørgrav 	return sldns_wire2str_int8_scan(d, dl, s, sl);
163109a3aaf3SDag-Erling Smørgrav }
163209a3aaf3SDag-Erling Smørgrav 
163309a3aaf3SDag-Erling Smørgrav int sldns_wire2str_unknown_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
163409a3aaf3SDag-Erling Smørgrav {
163509a3aaf3SDag-Erling Smørgrav 	return sldns_wire2str_rdata_unknown_scan(d, dl, s, sl);
163609a3aaf3SDag-Erling Smørgrav }
163709a3aaf3SDag-Erling Smørgrav 
163809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_time_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
163909a3aaf3SDag-Erling Smørgrav {
164009a3aaf3SDag-Erling Smørgrav 	/* create a YYYYMMDDHHMMSS string if possible */
164109a3aaf3SDag-Erling Smørgrav 	struct tm tm;
164209a3aaf3SDag-Erling Smørgrav 	char date_buf[16];
164309a3aaf3SDag-Erling Smørgrav 	uint32_t t;
164409a3aaf3SDag-Erling Smørgrav 	memset(&tm, 0, sizeof(tm));
164509a3aaf3SDag-Erling Smørgrav 	if(*dl < 4) return -1;
164609a3aaf3SDag-Erling Smørgrav 	t = sldns_read_uint32(*d);
164709a3aaf3SDag-Erling Smørgrav 	date_buf[15]=0;
16488a384985SDag-Erling Smørgrav 	if(sldns_serial_arithmetics_gmtime_r(t, time(NULL), &tm) &&
164909a3aaf3SDag-Erling Smørgrav 		strftime(date_buf, 15, "%Y%m%d%H%M%S", &tm)) {
165009a3aaf3SDag-Erling Smørgrav 		(*d) += 4;
165109a3aaf3SDag-Erling Smørgrav 		(*dl) -= 4;
165209a3aaf3SDag-Erling Smørgrav 		return sldns_str_print(s, sl, "%s", date_buf);
165309a3aaf3SDag-Erling Smørgrav 	}
165409a3aaf3SDag-Erling Smørgrav 	return -1;
165509a3aaf3SDag-Erling Smørgrav }
165609a3aaf3SDag-Erling Smørgrav 
165709a3aaf3SDag-Erling Smørgrav static int
165809a3aaf3SDag-Erling Smørgrav loc_cm_print(char** str, size_t* sl, uint8_t mantissa, uint8_t exponent)
165909a3aaf3SDag-Erling Smørgrav {
166009a3aaf3SDag-Erling Smørgrav 	int w = 0;
166109a3aaf3SDag-Erling Smørgrav 	uint8_t i;
166209a3aaf3SDag-Erling Smørgrav 	/* is it 0.<two digits> ? */
166309a3aaf3SDag-Erling Smørgrav 	if(exponent < 2) {
166409a3aaf3SDag-Erling Smørgrav 		if(exponent == 1)
166509a3aaf3SDag-Erling Smørgrav 			mantissa *= 10;
166609a3aaf3SDag-Erling Smørgrav 		return sldns_str_print(str, sl, "0.%02ld", (long)mantissa);
166709a3aaf3SDag-Erling Smørgrav 	}
166809a3aaf3SDag-Erling Smørgrav 	/* always <digit><string of zeros> */
166909a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(str, sl, "%d", (int)mantissa);
167009a3aaf3SDag-Erling Smørgrav 	for(i=0; i<exponent-2; i++)
167109a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(str, sl, "0");
167209a3aaf3SDag-Erling Smørgrav 	return w;
167309a3aaf3SDag-Erling Smørgrav }
167409a3aaf3SDag-Erling Smørgrav 
167509a3aaf3SDag-Erling Smørgrav int sldns_wire2str_loc_scan(uint8_t** d, size_t* dl, char** str, size_t* sl)
167609a3aaf3SDag-Erling Smørgrav {
167709a3aaf3SDag-Erling Smørgrav 	/* we could do checking (ie degrees < 90 etc)? */
167809a3aaf3SDag-Erling Smørgrav 	uint8_t version;
167909a3aaf3SDag-Erling Smørgrav 	uint8_t size;
168009a3aaf3SDag-Erling Smørgrav 	uint8_t horizontal_precision;
168109a3aaf3SDag-Erling Smørgrav 	uint8_t vertical_precision;
168209a3aaf3SDag-Erling Smørgrav 	uint32_t longitude;
168309a3aaf3SDag-Erling Smørgrav 	uint32_t latitude;
168409a3aaf3SDag-Erling Smørgrav 	uint32_t altitude;
168509a3aaf3SDag-Erling Smørgrav 	char northerness;
168609a3aaf3SDag-Erling Smørgrav 	char easterness;
168709a3aaf3SDag-Erling Smørgrav 	uint32_t h;
168809a3aaf3SDag-Erling Smørgrav 	uint32_t m;
168909a3aaf3SDag-Erling Smørgrav 	double s;
169009a3aaf3SDag-Erling Smørgrav 	uint32_t equator = (uint32_t)1 << 31; /* 2**31 */
169109a3aaf3SDag-Erling Smørgrav 	int w = 0;
169209a3aaf3SDag-Erling Smørgrav 
169309a3aaf3SDag-Erling Smørgrav 	if(*dl < 16) return -1;
169409a3aaf3SDag-Erling Smørgrav 	version = (*d)[0];
169509a3aaf3SDag-Erling Smørgrav 	if(version != 0)
169609a3aaf3SDag-Erling Smørgrav 		return sldns_wire2str_hex_scan(d, dl, str, sl);
169709a3aaf3SDag-Erling Smørgrav 	size = (*d)[1];
169809a3aaf3SDag-Erling Smørgrav 	horizontal_precision = (*d)[2];
169909a3aaf3SDag-Erling Smørgrav 	vertical_precision = (*d)[3];
170009a3aaf3SDag-Erling Smørgrav 
170109a3aaf3SDag-Erling Smørgrav 	latitude = sldns_read_uint32((*d)+4);
170209a3aaf3SDag-Erling Smørgrav 	longitude = sldns_read_uint32((*d)+8);
170309a3aaf3SDag-Erling Smørgrav 	altitude = sldns_read_uint32((*d)+12);
170409a3aaf3SDag-Erling Smørgrav 
170509a3aaf3SDag-Erling Smørgrav 	if (latitude > equator) {
170609a3aaf3SDag-Erling Smørgrav 		northerness = 'N';
170709a3aaf3SDag-Erling Smørgrav 		latitude = latitude - equator;
170809a3aaf3SDag-Erling Smørgrav 	} else {
170909a3aaf3SDag-Erling Smørgrav 		northerness = 'S';
171009a3aaf3SDag-Erling Smørgrav 		latitude = equator - latitude;
171109a3aaf3SDag-Erling Smørgrav 	}
171209a3aaf3SDag-Erling Smørgrav 	h = latitude / (1000 * 60 * 60);
171309a3aaf3SDag-Erling Smørgrav 	latitude = latitude % (1000 * 60 * 60);
171409a3aaf3SDag-Erling Smørgrav 	m = latitude / (1000 * 60);
171509a3aaf3SDag-Erling Smørgrav 	latitude = latitude % (1000 * 60);
171609a3aaf3SDag-Erling Smørgrav 	s = (double) latitude / 1000.0;
171709a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(str, sl, "%02u %02u %06.3f %c ",
171809a3aaf3SDag-Erling Smørgrav 		h, m, s, northerness);
171909a3aaf3SDag-Erling Smørgrav 
172009a3aaf3SDag-Erling Smørgrav 	if (longitude > equator) {
172109a3aaf3SDag-Erling Smørgrav 		easterness = 'E';
172209a3aaf3SDag-Erling Smørgrav 		longitude = longitude - equator;
172309a3aaf3SDag-Erling Smørgrav 	} else {
172409a3aaf3SDag-Erling Smørgrav 		easterness = 'W';
172509a3aaf3SDag-Erling Smørgrav 		longitude = equator - longitude;
172609a3aaf3SDag-Erling Smørgrav 	}
172709a3aaf3SDag-Erling Smørgrav 	h = longitude / (1000 * 60 * 60);
172809a3aaf3SDag-Erling Smørgrav 	longitude = longitude % (1000 * 60 * 60);
172909a3aaf3SDag-Erling Smørgrav 	m = longitude / (1000 * 60);
173009a3aaf3SDag-Erling Smørgrav 	longitude = longitude % (1000 * 60);
173109a3aaf3SDag-Erling Smørgrav 	s = (double) longitude / (1000.0);
173209a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(str, sl, "%02u %02u %06.3f %c ",
173309a3aaf3SDag-Erling Smørgrav 		h, m, s, easterness);
173409a3aaf3SDag-Erling Smørgrav 
173509a3aaf3SDag-Erling Smørgrav 	s = ((double) altitude) / 100;
173609a3aaf3SDag-Erling Smørgrav 	s -= 100000;
173709a3aaf3SDag-Erling Smørgrav 
173809a3aaf3SDag-Erling Smørgrav 	if(altitude%100 != 0)
173909a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(str, sl, "%.2f", s);
174009a3aaf3SDag-Erling Smørgrav 	else
174109a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(str, sl, "%.0f", s);
174209a3aaf3SDag-Erling Smørgrav 
174309a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(str, sl, "m ");
174409a3aaf3SDag-Erling Smørgrav 
174509a3aaf3SDag-Erling Smørgrav 	w += loc_cm_print(str, sl, (size & 0xf0) >> 4, size & 0x0f);
174609a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(str, sl, "m ");
174709a3aaf3SDag-Erling Smørgrav 
174809a3aaf3SDag-Erling Smørgrav 	w += loc_cm_print(str, sl, (horizontal_precision & 0xf0) >> 4,
174909a3aaf3SDag-Erling Smørgrav 		horizontal_precision & 0x0f);
175009a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(str, sl, "m ");
175109a3aaf3SDag-Erling Smørgrav 
175209a3aaf3SDag-Erling Smørgrav 	w += loc_cm_print(str, sl, (vertical_precision & 0xf0) >> 4,
175309a3aaf3SDag-Erling Smørgrav 		vertical_precision & 0x0f);
175409a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(str, sl, "m");
175509a3aaf3SDag-Erling Smørgrav 
175609a3aaf3SDag-Erling Smørgrav 	(*d)+=16;
175709a3aaf3SDag-Erling Smørgrav 	(*dl)-=16;
175809a3aaf3SDag-Erling Smørgrav 	return w;
175909a3aaf3SDag-Erling Smørgrav }
176009a3aaf3SDag-Erling Smørgrav 
176109a3aaf3SDag-Erling Smørgrav int sldns_wire2str_wks_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
176209a3aaf3SDag-Erling Smørgrav {
176309a3aaf3SDag-Erling Smørgrav 	/* protocol, followed by bitmap of services */
176409a3aaf3SDag-Erling Smørgrav 	const char* proto_name = NULL;
176509a3aaf3SDag-Erling Smørgrav 	struct protoent *protocol;
176609a3aaf3SDag-Erling Smørgrav 	struct servent *service;
176709a3aaf3SDag-Erling Smørgrav 	uint8_t protocol_nr;
176809a3aaf3SDag-Erling Smørgrav 	int bit, port, w = 0;
176909a3aaf3SDag-Erling Smørgrav 	size_t i;
177009a3aaf3SDag-Erling Smørgrav 	/* we cannot print with strings because they
177109a3aaf3SDag-Erling Smørgrav 	 * are not portable, the presentation format may
177209a3aaf3SDag-Erling Smørgrav 	 * not be able to be read in on another computer.  */
177309a3aaf3SDag-Erling Smørgrav 	int print_symbols = 0;
177409a3aaf3SDag-Erling Smørgrav 
177509a3aaf3SDag-Erling Smørgrav 	/* protocol */
177609a3aaf3SDag-Erling Smørgrav 	if(*dl < 1) return -1;
177709a3aaf3SDag-Erling Smørgrav 	protocol_nr = (*d)[0];
177809a3aaf3SDag-Erling Smørgrav 	(*d)++;
177909a3aaf3SDag-Erling Smørgrav 	(*dl)--;
178009a3aaf3SDag-Erling Smørgrav 	protocol = getprotobynumber((int)protocol_nr);
178109a3aaf3SDag-Erling Smørgrav 	if(protocol && (protocol->p_name != NULL)) {
178209a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "%s", protocol->p_name);
178309a3aaf3SDag-Erling Smørgrav 		proto_name = protocol->p_name;
1784971980c3SDag-Erling Smørgrav 	} else if(protocol_nr == 6) {
1785971980c3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "tcp");
1786971980c3SDag-Erling Smørgrav 	} else if(protocol_nr == 17) {
1787971980c3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "udp");
178809a3aaf3SDag-Erling Smørgrav 	} else	{
178909a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "%u", (unsigned)protocol_nr);
179009a3aaf3SDag-Erling Smørgrav 	}
179109a3aaf3SDag-Erling Smørgrav 
179209a3aaf3SDag-Erling Smørgrav 	for(i=0; i<*dl; i++) {
179309a3aaf3SDag-Erling Smørgrav 		if((*d)[i] == 0)
179409a3aaf3SDag-Erling Smørgrav 			continue;
179509a3aaf3SDag-Erling Smørgrav 		for(bit=0; bit<8; bit++) {
179609a3aaf3SDag-Erling Smørgrav 			if(!(((*d)[i])&(0x80>>bit)))
179709a3aaf3SDag-Erling Smørgrav 				continue;
179809a3aaf3SDag-Erling Smørgrav 			port = (int)i*8 + bit;
179909a3aaf3SDag-Erling Smørgrav 
180009a3aaf3SDag-Erling Smørgrav 			if(!print_symbols)
180109a3aaf3SDag-Erling Smørgrav 				service = NULL;
180209a3aaf3SDag-Erling Smørgrav 			else
180309a3aaf3SDag-Erling Smørgrav 				service = getservbyport(
180409a3aaf3SDag-Erling Smørgrav 					(int)htons((uint16_t)port), proto_name);
180509a3aaf3SDag-Erling Smørgrav 			if(service && service->s_name)
180609a3aaf3SDag-Erling Smørgrav 				w += sldns_str_print(s, sl, " %s",
180709a3aaf3SDag-Erling Smørgrav 					service->s_name);
180809a3aaf3SDag-Erling Smørgrav 			else 	w += sldns_str_print(s, sl, " %u",
180909a3aaf3SDag-Erling Smørgrav 					(unsigned)port);
181009a3aaf3SDag-Erling Smørgrav 		}
181109a3aaf3SDag-Erling Smørgrav 	}
181209a3aaf3SDag-Erling Smørgrav 
181309a3aaf3SDag-Erling Smørgrav #ifdef HAVE_ENDSERVENT
181409a3aaf3SDag-Erling Smørgrav 	endservent();
181509a3aaf3SDag-Erling Smørgrav #endif
181609a3aaf3SDag-Erling Smørgrav #ifdef HAVE_ENDPROTOENT
181709a3aaf3SDag-Erling Smørgrav         endprotoent();
181809a3aaf3SDag-Erling Smørgrav #endif
181909a3aaf3SDag-Erling Smørgrav 	(*d) += *dl;
182009a3aaf3SDag-Erling Smørgrav 	(*dl) = 0;
182109a3aaf3SDag-Erling Smørgrav 	return w;
182209a3aaf3SDag-Erling Smørgrav }
182309a3aaf3SDag-Erling Smørgrav 
182409a3aaf3SDag-Erling Smørgrav int sldns_wire2str_nsap_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
182509a3aaf3SDag-Erling Smørgrav {
182609a3aaf3SDag-Erling Smørgrav 	return print_remainder_hex("0x", d, dl, s, sl);
182709a3aaf3SDag-Erling Smørgrav }
182809a3aaf3SDag-Erling Smørgrav 
182909a3aaf3SDag-Erling Smørgrav int sldns_wire2str_atma_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
183009a3aaf3SDag-Erling Smørgrav {
183109a3aaf3SDag-Erling Smørgrav 	return print_remainder_hex("", d, dl, s, sl);
183209a3aaf3SDag-Erling Smørgrav }
183309a3aaf3SDag-Erling Smørgrav 
183409a3aaf3SDag-Erling Smørgrav /* internal scan routine that can modify arguments on failure */
183509a3aaf3SDag-Erling Smørgrav static int sldns_wire2str_ipseckey_scan_internal(uint8_t** d, size_t* dl,
18360eefd307SCy Schubert 	char** s, size_t* sl, uint8_t* pkt, size_t pktlen, int* comprloop)
183709a3aaf3SDag-Erling Smørgrav {
183809a3aaf3SDag-Erling Smørgrav 	/* http://www.ietf.org/internet-drafts/draft-ietf-ipseckey-rr-12.txt*/
183909a3aaf3SDag-Erling Smørgrav 	uint8_t precedence, gateway_type, algorithm;
184009a3aaf3SDag-Erling Smørgrav 	int w = 0;
184109a3aaf3SDag-Erling Smørgrav 
184209a3aaf3SDag-Erling Smørgrav 	if(*dl < 3) return -1;
184309a3aaf3SDag-Erling Smørgrav 	precedence = (*d)[0];
184409a3aaf3SDag-Erling Smørgrav 	gateway_type = (*d)[1];
184509a3aaf3SDag-Erling Smørgrav 	algorithm = (*d)[2];
184609a3aaf3SDag-Erling Smørgrav 	if(gateway_type > 3)
184709a3aaf3SDag-Erling Smørgrav 		return -1; /* unknown */
184809a3aaf3SDag-Erling Smørgrav 	(*d)+=3;
184909a3aaf3SDag-Erling Smørgrav 	(*dl)-=3;
185009a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, "%d %d %d ",
185109a3aaf3SDag-Erling Smørgrav 		(int)precedence, (int)gateway_type, (int)algorithm);
185209a3aaf3SDag-Erling Smørgrav 
185309a3aaf3SDag-Erling Smørgrav 	switch(gateway_type) {
185409a3aaf3SDag-Erling Smørgrav 	case 0: /* no gateway */
185509a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, ".");
185609a3aaf3SDag-Erling Smørgrav 		break;
185709a3aaf3SDag-Erling Smørgrav 	case 1: /* ip4 */
185809a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_a_scan(d, dl, s, sl);
185909a3aaf3SDag-Erling Smørgrav 		break;
186009a3aaf3SDag-Erling Smørgrav 	case 2: /* ip6 */
186109a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_aaaa_scan(d, dl, s, sl);
186209a3aaf3SDag-Erling Smørgrav 		break;
186309a3aaf3SDag-Erling Smørgrav 	case 3: /* dname */
18640eefd307SCy Schubert 		w += sldns_wire2str_dname_scan(d, dl, s, sl, pkt, pktlen, comprloop);
186509a3aaf3SDag-Erling Smørgrav 		break;
186609a3aaf3SDag-Erling Smørgrav 	default: /* unknown */
186709a3aaf3SDag-Erling Smørgrav 		return -1;
186809a3aaf3SDag-Erling Smørgrav 	}
186909a3aaf3SDag-Erling Smørgrav 
187009a3aaf3SDag-Erling Smørgrav 	if(*dl < 1)
187109a3aaf3SDag-Erling Smørgrav 		return -1;
187209a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, " ");
187309a3aaf3SDag-Erling Smørgrav 	w += sldns_wire2str_b64_scan_num(d, dl, s, sl, *dl);
187409a3aaf3SDag-Erling Smørgrav 	return w;
187509a3aaf3SDag-Erling Smørgrav }
187609a3aaf3SDag-Erling Smørgrav 
187709a3aaf3SDag-Erling Smørgrav int sldns_wire2str_ipseckey_scan(uint8_t** d, size_t* dl, char** s, size_t* sl,
18780eefd307SCy Schubert 	uint8_t* pkt, size_t pktlen, int* comprloop)
187909a3aaf3SDag-Erling Smørgrav {
188009a3aaf3SDag-Erling Smørgrav 	uint8_t* od = *d;
188109a3aaf3SDag-Erling Smørgrav 	char* os = *s;
188209a3aaf3SDag-Erling Smørgrav 	size_t odl = *dl, osl = *sl;
18830eefd307SCy Schubert 	int w=sldns_wire2str_ipseckey_scan_internal(d, dl, s, sl, pkt, pktlen, comprloop);
188409a3aaf3SDag-Erling Smørgrav 	if(w == -1) {
188509a3aaf3SDag-Erling Smørgrav 		*d = od;
188609a3aaf3SDag-Erling Smørgrav 		*s = os;
188709a3aaf3SDag-Erling Smørgrav 		*dl = odl;
188809a3aaf3SDag-Erling Smørgrav 		*sl = osl;
188909a3aaf3SDag-Erling Smørgrav 		return -1;
189009a3aaf3SDag-Erling Smørgrav 	}
189109a3aaf3SDag-Erling Smørgrav 	return w;
189209a3aaf3SDag-Erling Smørgrav }
189309a3aaf3SDag-Erling Smørgrav 
189409a3aaf3SDag-Erling Smørgrav int sldns_wire2str_hip_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
189509a3aaf3SDag-Erling Smørgrav {
189609a3aaf3SDag-Erling Smørgrav 	int w;
189709a3aaf3SDag-Erling Smørgrav 	uint8_t algo, hitlen;
189809a3aaf3SDag-Erling Smørgrav 	uint16_t pklen;
189909a3aaf3SDag-Erling Smørgrav 
190009a3aaf3SDag-Erling Smørgrav 	/* read lengths */
190109a3aaf3SDag-Erling Smørgrav 	if(*dl < 4)
190209a3aaf3SDag-Erling Smørgrav 		return -1;
190309a3aaf3SDag-Erling Smørgrav 	hitlen = (*d)[0];
190409a3aaf3SDag-Erling Smørgrav 	algo = (*d)[1];
190509a3aaf3SDag-Erling Smørgrav 	pklen = sldns_read_uint16((*d)+2);
190609a3aaf3SDag-Erling Smørgrav 	if(*dl < (size_t)4 + (size_t)hitlen + (size_t)pklen)
190709a3aaf3SDag-Erling Smørgrav 		return -1;
190809a3aaf3SDag-Erling Smørgrav 
190909a3aaf3SDag-Erling Smørgrav 	/* write: algo hit pubkey */
191009a3aaf3SDag-Erling Smørgrav 	w = sldns_str_print(s, sl, "%u ", (unsigned)algo);
191109a3aaf3SDag-Erling Smørgrav 	w += print_hex_buf(s, sl, (*d)+4, hitlen);
191209a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, " ");
191309a3aaf3SDag-Erling Smørgrav 	(*d)+=4+hitlen;
191409a3aaf3SDag-Erling Smørgrav 	(*dl)-= (4+hitlen);
191509a3aaf3SDag-Erling Smørgrav 	w += sldns_wire2str_b64_scan_num(d, dl, s, sl, pklen);
191609a3aaf3SDag-Erling Smørgrav 	return w;
191709a3aaf3SDag-Erling Smørgrav }
191809a3aaf3SDag-Erling Smørgrav 
191909a3aaf3SDag-Erling Smørgrav int sldns_wire2str_int16_data_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
192009a3aaf3SDag-Erling Smørgrav {
1921c7f4d7adSDag-Erling Smørgrav 	int w;
192209a3aaf3SDag-Erling Smørgrav 	uint16_t n;
192309a3aaf3SDag-Erling Smørgrav 	if(*dl < 2)
192409a3aaf3SDag-Erling Smørgrav 		return -1;
192509a3aaf3SDag-Erling Smørgrav 	n = sldns_read_uint16(*d);
192609a3aaf3SDag-Erling Smørgrav 	if(*dl < 2+(size_t)n)
192709a3aaf3SDag-Erling Smørgrav 		return -1;
192809a3aaf3SDag-Erling Smørgrav 	(*d)+=2;
192909a3aaf3SDag-Erling Smørgrav 	(*dl)-=2;
1930c7f4d7adSDag-Erling Smørgrav 	if(n == 0) {
1931c7f4d7adSDag-Erling Smørgrav 		return sldns_str_print(s, sl, "0");
1932c7f4d7adSDag-Erling Smørgrav 	}
1933c7f4d7adSDag-Erling Smørgrav 	w = sldns_str_print(s, sl, "%u ", (unsigned)n);
1934c7f4d7adSDag-Erling Smørgrav 	w += sldns_wire2str_b64_scan_num(d, dl, s, sl, n);
1935c7f4d7adSDag-Erling Smørgrav 	return w;
193609a3aaf3SDag-Erling Smørgrav }
193709a3aaf3SDag-Erling Smørgrav 
193809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_nsec3_next_owner_scan(uint8_t** d, size_t* dl, char** s,
193909a3aaf3SDag-Erling Smørgrav 	size_t* sl)
194009a3aaf3SDag-Erling Smørgrav {
194109a3aaf3SDag-Erling Smørgrav 	return sldns_wire2str_b32_ext_scan(d, dl, s, sl);
194209a3aaf3SDag-Erling Smørgrav }
194309a3aaf3SDag-Erling Smørgrav 
194409a3aaf3SDag-Erling Smørgrav int sldns_wire2str_ilnp64_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
194509a3aaf3SDag-Erling Smørgrav {
194609a3aaf3SDag-Erling Smørgrav 	int w;
194709a3aaf3SDag-Erling Smørgrav 	if(*dl < 8)
194809a3aaf3SDag-Erling Smørgrav 		return -1;
194909a3aaf3SDag-Erling Smørgrav 	w = sldns_str_print(s, sl, "%.4x:%.4x:%.4x:%.4x",
195009a3aaf3SDag-Erling Smørgrav 		sldns_read_uint16(*d), sldns_read_uint16((*d)+2),
195109a3aaf3SDag-Erling Smørgrav 		sldns_read_uint16((*d)+4), sldns_read_uint16((*d)+6));
195209a3aaf3SDag-Erling Smørgrav 	(*d)+=8;
195309a3aaf3SDag-Erling Smørgrav 	(*dl)-=8;
195409a3aaf3SDag-Erling Smørgrav 	return w;
195509a3aaf3SDag-Erling Smørgrav }
195609a3aaf3SDag-Erling Smørgrav 
195709a3aaf3SDag-Erling Smørgrav int sldns_wire2str_eui48_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
195809a3aaf3SDag-Erling Smørgrav {
195909a3aaf3SDag-Erling Smørgrav 	int w;
196009a3aaf3SDag-Erling Smørgrav 	if(*dl < 6)
196109a3aaf3SDag-Erling Smørgrav 		return -1;
196209a3aaf3SDag-Erling Smørgrav 	w = sldns_str_print(s, sl, "%.2x-%.2x-%.2x-%.2x-%.2x-%.2x",
196309a3aaf3SDag-Erling Smørgrav 		(*d)[0], (*d)[1], (*d)[2], (*d)[3], (*d)[4], (*d)[5]);
196409a3aaf3SDag-Erling Smørgrav 	(*d)+=6;
196509a3aaf3SDag-Erling Smørgrav 	(*dl)-=6;
196609a3aaf3SDag-Erling Smørgrav 	return w;
196709a3aaf3SDag-Erling Smørgrav }
196809a3aaf3SDag-Erling Smørgrav 
196909a3aaf3SDag-Erling Smørgrav int sldns_wire2str_eui64_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
197009a3aaf3SDag-Erling Smørgrav {
197109a3aaf3SDag-Erling Smørgrav 	int w;
197209a3aaf3SDag-Erling Smørgrav 	if(*dl < 8)
197309a3aaf3SDag-Erling Smørgrav 		return -1;
197409a3aaf3SDag-Erling Smørgrav 	w = sldns_str_print(s, sl, "%.2x-%.2x-%.2x-%.2x-%.2x-%.2x-%.2x-%.2x",
197509a3aaf3SDag-Erling Smørgrav 		(*d)[0], (*d)[1], (*d)[2], (*d)[3], (*d)[4], (*d)[5],
197609a3aaf3SDag-Erling Smørgrav 		(*d)[6], (*d)[7]);
197709a3aaf3SDag-Erling Smørgrav 	(*d)+=8;
197809a3aaf3SDag-Erling Smørgrav 	(*dl)-=8;
197909a3aaf3SDag-Erling Smørgrav 	return w;
198009a3aaf3SDag-Erling Smørgrav }
198109a3aaf3SDag-Erling Smørgrav 
198209a3aaf3SDag-Erling Smørgrav int sldns_wire2str_tag_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
198309a3aaf3SDag-Erling Smørgrav {
198409a3aaf3SDag-Erling Smørgrav 	size_t i, n;
198509a3aaf3SDag-Erling Smørgrav 	int w = 0;
198609a3aaf3SDag-Erling Smørgrav 	if(*dl < 1)
198709a3aaf3SDag-Erling Smørgrav 		return -1;
198809a3aaf3SDag-Erling Smørgrav 	n = (size_t)((*d)[0]);
198909a3aaf3SDag-Erling Smørgrav 	if(*dl < 1+n)
199009a3aaf3SDag-Erling Smørgrav 		return -1;
199109a3aaf3SDag-Erling Smørgrav 	for(i=0; i<n; i++)
1992c7f4d7adSDag-Erling Smørgrav 		if(!isalnum((unsigned char)(*d)[i+1]))
199309a3aaf3SDag-Erling Smørgrav 			return -1;
199409a3aaf3SDag-Erling Smørgrav 	for(i=0; i<n; i++)
1995c7f4d7adSDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "%c", (char)(*d)[i+1]);
199609a3aaf3SDag-Erling Smørgrav 	(*d)+=n+1;
199709a3aaf3SDag-Erling Smørgrav 	(*dl)-=(n+1);
199809a3aaf3SDag-Erling Smørgrav 	return w;
199909a3aaf3SDag-Erling Smørgrav }
200009a3aaf3SDag-Erling Smørgrav 
200109a3aaf3SDag-Erling Smørgrav int sldns_wire2str_long_str_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
200209a3aaf3SDag-Erling Smørgrav {
200309a3aaf3SDag-Erling Smørgrav 	size_t i;
200409a3aaf3SDag-Erling Smørgrav 	int w = 0;
200509a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, "\"");
200609a3aaf3SDag-Erling Smørgrav 	for(i=0; i<*dl; i++)
200709a3aaf3SDag-Erling Smørgrav 		w += str_char_print(s, sl, (*d)[i]);
200809a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, "\"");
200909a3aaf3SDag-Erling Smørgrav 	(*d)+=*dl;
201009a3aaf3SDag-Erling Smørgrav 	(*dl)=0;
201109a3aaf3SDag-Erling Smørgrav 	return w;
201209a3aaf3SDag-Erling Smørgrav }
201309a3aaf3SDag-Erling Smørgrav 
2014c7f4d7adSDag-Erling Smørgrav int sldns_wire2str_tsigerror_scan(uint8_t** d, size_t* dl, char** s, size_t* sl)
2015c7f4d7adSDag-Erling Smørgrav {
2016c7f4d7adSDag-Erling Smørgrav 	sldns_lookup_table *lt;
2017c7f4d7adSDag-Erling Smørgrav 	int data, w;
2018c7f4d7adSDag-Erling Smørgrav 	if(*dl < 2) return -1;
2019c7f4d7adSDag-Erling Smørgrav 	data = (int)sldns_read_uint16(*d);
2020c7f4d7adSDag-Erling Smørgrav 	lt = sldns_lookup_by_id(sldns_tsig_errors, data);
2021c7f4d7adSDag-Erling Smørgrav 	if(lt && lt->name)
2022c7f4d7adSDag-Erling Smørgrav 		w = sldns_str_print(s, sl, "%s", lt->name);
2023c7f4d7adSDag-Erling Smørgrav 	else 	w = sldns_str_print(s, sl, "%d", data);
2024c7f4d7adSDag-Erling Smørgrav 	(*dl)-=2;
2025c7f4d7adSDag-Erling Smørgrav 	(*d)+=2;
2026c7f4d7adSDag-Erling Smørgrav 	return w;
2027c7f4d7adSDag-Erling Smørgrav }
2028c7f4d7adSDag-Erling Smørgrav 
202909a3aaf3SDag-Erling Smørgrav int sldns_wire2str_edns_llq_print(char** s, size_t* sl, uint8_t* data,
203009a3aaf3SDag-Erling Smørgrav 	size_t len)
203109a3aaf3SDag-Erling Smørgrav {
203209a3aaf3SDag-Erling Smørgrav 	/* LLQ constants */
203309a3aaf3SDag-Erling Smørgrav 	const char* llq_errors[] = {"NO-ERROR", "SERV-FULL", "STATIC",
203409a3aaf3SDag-Erling Smørgrav 		"FORMAT-ERR", "NO-SUCH-LLQ", "BAD-VERS", "UNKNOWN_ERR"};
203509a3aaf3SDag-Erling Smørgrav 	const unsigned int llq_errors_num = 7;
203609a3aaf3SDag-Erling Smørgrav 	const char* llq_opcodes[] = {"LLQ-SETUP", "LLQ-REFRESH", "LLQ-EVENT"};
203709a3aaf3SDag-Erling Smørgrav 	const unsigned int llq_opcodes_num = 3;
203809a3aaf3SDag-Erling Smørgrav 	uint16_t version, llq_opcode, error_code;
203909a3aaf3SDag-Erling Smørgrav 	uint64_t llq_id;
204009a3aaf3SDag-Erling Smørgrav 	uint32_t lease_life; /* Requested or granted life of LLQ, in seconds */
204109a3aaf3SDag-Erling Smørgrav 	int w = 0;
204209a3aaf3SDag-Erling Smørgrav 
204309a3aaf3SDag-Erling Smørgrav 	/* read the record */
204409a3aaf3SDag-Erling Smørgrav 	if(len != 18) {
204509a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "malformed LLQ ");
204609a3aaf3SDag-Erling Smørgrav 		w += print_hex_buf(s, sl, data, len);
204709a3aaf3SDag-Erling Smørgrav 		return w;
204809a3aaf3SDag-Erling Smørgrav 	}
204909a3aaf3SDag-Erling Smørgrav 	version = sldns_read_uint16(data);
205009a3aaf3SDag-Erling Smørgrav 	llq_opcode = sldns_read_uint16(data+2);
205109a3aaf3SDag-Erling Smørgrav 	error_code = sldns_read_uint16(data+4);
205209a3aaf3SDag-Erling Smørgrav 	memmove(&llq_id, data+6, sizeof(llq_id));
205309a3aaf3SDag-Erling Smørgrav 	lease_life = sldns_read_uint32(data+14);
205409a3aaf3SDag-Erling Smørgrav 
205509a3aaf3SDag-Erling Smørgrav 	/* print it */
205609a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, "v%d ", (int)version);
205709a3aaf3SDag-Erling Smørgrav 	if(llq_opcode < llq_opcodes_num)
205809a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "%s", llq_opcodes[llq_opcode]);
205909a3aaf3SDag-Erling Smørgrav 	else	w += sldns_str_print(s, sl, "opcode %d", (int)llq_opcode);
206009a3aaf3SDag-Erling Smørgrav 	if(error_code < llq_errors_num)
206109a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, " %s", llq_errors[error_code]);
206209a3aaf3SDag-Erling Smørgrav 	else	w += sldns_str_print(s, sl, " error %d", (int)error_code);
206309a3aaf3SDag-Erling Smørgrav #ifndef USE_WINSOCK
206409a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, " id %llx lease-life %lu",
206509a3aaf3SDag-Erling Smørgrav 		(unsigned long long)llq_id, (unsigned long)lease_life);
206609a3aaf3SDag-Erling Smørgrav #else
206709a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, " id %I64x lease-life %lu",
206809a3aaf3SDag-Erling Smørgrav 		(unsigned long long)llq_id, (unsigned long)lease_life);
206909a3aaf3SDag-Erling Smørgrav #endif
207009a3aaf3SDag-Erling Smørgrav 	return w;
207109a3aaf3SDag-Erling Smørgrav }
207209a3aaf3SDag-Erling Smørgrav 
207309a3aaf3SDag-Erling Smørgrav int sldns_wire2str_edns_ul_print(char** s, size_t* sl, uint8_t* data,
207409a3aaf3SDag-Erling Smørgrav 	size_t len)
207509a3aaf3SDag-Erling Smørgrav {
207609a3aaf3SDag-Erling Smørgrav 	uint32_t lease;
207709a3aaf3SDag-Erling Smørgrav 	int w = 0;
207809a3aaf3SDag-Erling Smørgrav 	if(len != 4) {
207909a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "malformed UL ");
208009a3aaf3SDag-Erling Smørgrav 		w += print_hex_buf(s, sl, data, len);
208109a3aaf3SDag-Erling Smørgrav 		return w;
208209a3aaf3SDag-Erling Smørgrav 	}
208309a3aaf3SDag-Erling Smørgrav 	lease = sldns_read_uint32(data);
208409a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, "lease %lu", (unsigned long)lease);
208509a3aaf3SDag-Erling Smørgrav 	return w;
208609a3aaf3SDag-Erling Smørgrav }
208709a3aaf3SDag-Erling Smørgrav 
208809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_edns_nsid_print(char** s, size_t* sl, uint8_t* data,
208909a3aaf3SDag-Erling Smørgrav 	size_t len)
209009a3aaf3SDag-Erling Smørgrav {
209109a3aaf3SDag-Erling Smørgrav 	int w = 0;
209209a3aaf3SDag-Erling Smørgrav 	size_t i, printed=0;
209309a3aaf3SDag-Erling Smørgrav 	w += print_hex_buf(s, sl, data, len);
209409a3aaf3SDag-Erling Smørgrav 	for(i=0; i<len; i++) {
209509a3aaf3SDag-Erling Smørgrav 		if(isprint((unsigned char)data[i]) || data[i] == '\t') {
209609a3aaf3SDag-Erling Smørgrav 			if(!printed) {
209709a3aaf3SDag-Erling Smørgrav 				w += sldns_str_print(s, sl, " (");
209809a3aaf3SDag-Erling Smørgrav 				printed = 1;
209909a3aaf3SDag-Erling Smørgrav 			}
210009a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, "%c", (char)data[i]);
210109a3aaf3SDag-Erling Smørgrav 		}
210209a3aaf3SDag-Erling Smørgrav 	}
210309a3aaf3SDag-Erling Smørgrav 	if(printed)
210409a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, ")");
210509a3aaf3SDag-Erling Smørgrav 	return w;
210609a3aaf3SDag-Erling Smørgrav }
210709a3aaf3SDag-Erling Smørgrav 
210809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_edns_dau_print(char** s, size_t* sl, uint8_t* data,
210909a3aaf3SDag-Erling Smørgrav 	size_t len)
211009a3aaf3SDag-Erling Smørgrav {
211109a3aaf3SDag-Erling Smørgrav 	sldns_lookup_table *lt;
211209a3aaf3SDag-Erling Smørgrav 	size_t i;
211309a3aaf3SDag-Erling Smørgrav 	int w = 0;
211409a3aaf3SDag-Erling Smørgrav 	for(i=0; i<len; i++) {
211509a3aaf3SDag-Erling Smørgrav 		lt = sldns_lookup_by_id(sldns_algorithms, (int)data[i]);
211609a3aaf3SDag-Erling Smørgrav 		if(lt && lt->name)
211709a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, " %s", lt->name);
211809a3aaf3SDag-Erling Smørgrav 		else 	w += sldns_str_print(s, sl, " %d", (int)data[i]);
211909a3aaf3SDag-Erling Smørgrav 	}
212009a3aaf3SDag-Erling Smørgrav 	return w;
212109a3aaf3SDag-Erling Smørgrav }
212209a3aaf3SDag-Erling Smørgrav 
212309a3aaf3SDag-Erling Smørgrav int sldns_wire2str_edns_dhu_print(char** s, size_t* sl, uint8_t* data,
212409a3aaf3SDag-Erling Smørgrav 	size_t len)
212509a3aaf3SDag-Erling Smørgrav {
212609a3aaf3SDag-Erling Smørgrav 	sldns_lookup_table *lt;
212709a3aaf3SDag-Erling Smørgrav 	size_t i;
212809a3aaf3SDag-Erling Smørgrav 	int w = 0;
212909a3aaf3SDag-Erling Smørgrav 	for(i=0; i<len; i++) {
213009a3aaf3SDag-Erling Smørgrav 		lt = sldns_lookup_by_id(sldns_hashes, (int)data[i]);
213109a3aaf3SDag-Erling Smørgrav 		if(lt && lt->name)
213209a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, " %s", lt->name);
213309a3aaf3SDag-Erling Smørgrav 		else 	w += sldns_str_print(s, sl, " %d", (int)data[i]);
213409a3aaf3SDag-Erling Smørgrav 	}
213509a3aaf3SDag-Erling Smørgrav 	return w;
213609a3aaf3SDag-Erling Smørgrav }
213709a3aaf3SDag-Erling Smørgrav 
213809a3aaf3SDag-Erling Smørgrav int sldns_wire2str_edns_n3u_print(char** s, size_t* sl, uint8_t* data,
213909a3aaf3SDag-Erling Smørgrav 	size_t len)
214009a3aaf3SDag-Erling Smørgrav {
214109a3aaf3SDag-Erling Smørgrav 	size_t i;
214209a3aaf3SDag-Erling Smørgrav 	int w = 0;
214309a3aaf3SDag-Erling Smørgrav 	for(i=0; i<len; i++) {
214409a3aaf3SDag-Erling Smørgrav 		if(data[i] == 1)
214509a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, " SHA1");
214609a3aaf3SDag-Erling Smørgrav 		else 	w += sldns_str_print(s, sl, " %d", (int)data[i]);
214709a3aaf3SDag-Erling Smørgrav 	}
214809a3aaf3SDag-Erling Smørgrav 	return w;
214909a3aaf3SDag-Erling Smørgrav }
215009a3aaf3SDag-Erling Smørgrav 
215109a3aaf3SDag-Erling Smørgrav int sldns_wire2str_edns_subnet_print(char** s, size_t* sl, uint8_t* data,
215209a3aaf3SDag-Erling Smørgrav 	size_t len)
215309a3aaf3SDag-Erling Smørgrav {
215409a3aaf3SDag-Erling Smørgrav 	int w = 0;
215509a3aaf3SDag-Erling Smørgrav 	uint16_t family;
215609a3aaf3SDag-Erling Smørgrav 	uint8_t source, scope;
215709a3aaf3SDag-Erling Smørgrav 	if(len < 4) {
215809a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "malformed subnet ");
215909a3aaf3SDag-Erling Smørgrav 		w += print_hex_buf(s, sl, data, len);
216009a3aaf3SDag-Erling Smørgrav 		return w;
216109a3aaf3SDag-Erling Smørgrav 	}
216209a3aaf3SDag-Erling Smørgrav 	family = sldns_read_uint16(data);
216309a3aaf3SDag-Erling Smørgrav 	source = data[2];
216409a3aaf3SDag-Erling Smørgrav 	scope = data[3];
216509a3aaf3SDag-Erling Smørgrav 	if(family == 1) {
216609a3aaf3SDag-Erling Smørgrav 		/* IP4 */
216709a3aaf3SDag-Erling Smørgrav 		char buf[64];
216809a3aaf3SDag-Erling Smørgrav 		uint8_t ip4[4];
216909a3aaf3SDag-Erling Smørgrav 		memset(ip4, 0, sizeof(ip4));
217009a3aaf3SDag-Erling Smørgrav 		if(len-4 > 4) {
217109a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, "trailingdata:");
217209a3aaf3SDag-Erling Smørgrav 			w += print_hex_buf(s, sl, data+4+4, len-4-4);
217309a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, " ");
217409a3aaf3SDag-Erling Smørgrav 			len = 4+4;
217509a3aaf3SDag-Erling Smørgrav 		}
217609a3aaf3SDag-Erling Smørgrav 		memmove(ip4, data+4, len-4);
217709a3aaf3SDag-Erling Smørgrav 		if(!inet_ntop(AF_INET, ip4, buf, (socklen_t)sizeof(buf))) {
217809a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, "ip4ntoperror ");
217909a3aaf3SDag-Erling Smørgrav 			w += print_hex_buf(s, sl, data+4+4, len-4-4);
218009a3aaf3SDag-Erling Smørgrav 		} else {
218109a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, "%s", buf);
218209a3aaf3SDag-Erling Smørgrav 		}
218309a3aaf3SDag-Erling Smørgrav 	} else if(family == 2) {
218409a3aaf3SDag-Erling Smørgrav 		/* IP6 */
218509a3aaf3SDag-Erling Smørgrav 		char buf[64];
218609a3aaf3SDag-Erling Smørgrav 		uint8_t ip6[16];
218709a3aaf3SDag-Erling Smørgrav 		memset(ip6, 0, sizeof(ip6));
218809a3aaf3SDag-Erling Smørgrav 		if(len-4 > 16) {
218909a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, "trailingdata:");
219009a3aaf3SDag-Erling Smørgrav 			w += print_hex_buf(s, sl, data+4+16, len-4-16);
219109a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, " ");
219209a3aaf3SDag-Erling Smørgrav 			len = 4+16;
219309a3aaf3SDag-Erling Smørgrav 		}
219409a3aaf3SDag-Erling Smørgrav 		memmove(ip6, data+4, len-4);
219509a3aaf3SDag-Erling Smørgrav #ifdef AF_INET6
219609a3aaf3SDag-Erling Smørgrav 		if(!inet_ntop(AF_INET6, ip6, buf, (socklen_t)sizeof(buf))) {
219709a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, "ip6ntoperror ");
219809a3aaf3SDag-Erling Smørgrav 			w += print_hex_buf(s, sl, data+4+4, len-4-4);
219909a3aaf3SDag-Erling Smørgrav 		} else {
220009a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, "%s", buf);
220109a3aaf3SDag-Erling Smørgrav 		}
220209a3aaf3SDag-Erling Smørgrav #else
220309a3aaf3SDag-Erling Smørgrav 		w += print_hex_buf(s, sl, data+4+4, len-4-4);
220409a3aaf3SDag-Erling Smørgrav #endif
220509a3aaf3SDag-Erling Smørgrav 	} else {
220609a3aaf3SDag-Erling Smørgrav 		/* unknown */
220709a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "family %d ",
220809a3aaf3SDag-Erling Smørgrav 			(int)family);
220909a3aaf3SDag-Erling Smørgrav 		w += print_hex_buf(s, sl, data, len);
221009a3aaf3SDag-Erling Smørgrav 	}
221109a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, "/%d scope /%d", (int)source, (int)scope);
221209a3aaf3SDag-Erling Smørgrav 	return w;
221309a3aaf3SDag-Erling Smørgrav }
221409a3aaf3SDag-Erling Smørgrav 
2215e86b9096SDag-Erling Smørgrav static int sldns_wire2str_edns_keepalive_print(char** s, size_t* sl,
2216e86b9096SDag-Erling Smørgrav 	uint8_t* data, size_t len)
221765b390aaSDag-Erling Smørgrav {
221865b390aaSDag-Erling Smørgrav 	int w = 0;
221965b390aaSDag-Erling Smørgrav 	uint16_t timeout;
222065b390aaSDag-Erling Smørgrav 	if(!(len == 0 || len == 2)) {
222165b390aaSDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "malformed keepalive ");
222265b390aaSDag-Erling Smørgrav 		w += print_hex_buf(s, sl, data, len);
222365b390aaSDag-Erling Smørgrav 		return w;
222465b390aaSDag-Erling Smørgrav 	}
222565b390aaSDag-Erling Smørgrav 	if(len == 0 ) {
222665b390aaSDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "no timeout value (only valid for client option) ");
222765b390aaSDag-Erling Smørgrav 	} else {
222865b390aaSDag-Erling Smørgrav 		timeout = sldns_read_uint16(data);
222965b390aaSDag-Erling Smørgrav 		w += sldns_str_print(s, sl, "timeout value in units of 100ms %u", (int)timeout);
223065b390aaSDag-Erling Smørgrav 	}
223165b390aaSDag-Erling Smørgrav 	return w;
223265b390aaSDag-Erling Smørgrav }
223365b390aaSDag-Erling Smørgrav 
223409a3aaf3SDag-Erling Smørgrav int sldns_wire2str_edns_option_print(char** s, size_t* sl,
223509a3aaf3SDag-Erling Smørgrav 	uint16_t option_code, uint8_t* optdata, size_t optlen)
223609a3aaf3SDag-Erling Smørgrav {
223709a3aaf3SDag-Erling Smørgrav 	int w = 0;
223809a3aaf3SDag-Erling Smørgrav 	w += sldns_wire2str_edns_option_code_print(s, sl, option_code);
223909a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(s, sl, ": ");
224009a3aaf3SDag-Erling Smørgrav 	switch(option_code) {
224109a3aaf3SDag-Erling Smørgrav 	case LDNS_EDNS_LLQ:
224209a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_edns_llq_print(s, sl, optdata, optlen);
224309a3aaf3SDag-Erling Smørgrav 		break;
224409a3aaf3SDag-Erling Smørgrav 	case LDNS_EDNS_UL:
224509a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_edns_ul_print(s, sl, optdata, optlen);
224609a3aaf3SDag-Erling Smørgrav 		break;
224709a3aaf3SDag-Erling Smørgrav 	case LDNS_EDNS_NSID:
224809a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_edns_nsid_print(s, sl, optdata, optlen);
224909a3aaf3SDag-Erling Smørgrav 		break;
225009a3aaf3SDag-Erling Smørgrav 	case LDNS_EDNS_DAU:
225109a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_edns_dau_print(s, sl, optdata, optlen);
225209a3aaf3SDag-Erling Smørgrav 		break;
225309a3aaf3SDag-Erling Smørgrav 	case LDNS_EDNS_DHU:
225409a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_edns_dhu_print(s, sl, optdata, optlen);
225509a3aaf3SDag-Erling Smørgrav 		break;
225609a3aaf3SDag-Erling Smørgrav 	case LDNS_EDNS_N3U:
225709a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_edns_n3u_print(s, sl, optdata, optlen);
225809a3aaf3SDag-Erling Smørgrav 		break;
225909a3aaf3SDag-Erling Smørgrav 	case LDNS_EDNS_CLIENT_SUBNET:
226009a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_edns_subnet_print(s, sl, optdata, optlen);
226109a3aaf3SDag-Erling Smørgrav 		break;
226265b390aaSDag-Erling Smørgrav 	 case LDNS_EDNS_KEEPALIVE:
226365b390aaSDag-Erling Smørgrav 		w += sldns_wire2str_edns_keepalive_print(s, sl, optdata, optlen);
226465b390aaSDag-Erling Smørgrav 		break;
2265e2d15004SDag-Erling Smørgrav 	case LDNS_EDNS_PADDING:
2266e2d15004SDag-Erling Smørgrav 		w += print_hex_buf(s, sl, optdata, optlen);
2267e2d15004SDag-Erling Smørgrav 		break;
226809a3aaf3SDag-Erling Smørgrav 	default:
226909a3aaf3SDag-Erling Smørgrav 		/* unknown option code */
227009a3aaf3SDag-Erling Smørgrav 		w += print_hex_buf(s, sl, optdata, optlen);
227109a3aaf3SDag-Erling Smørgrav 		break;
227209a3aaf3SDag-Erling Smørgrav 	}
227309a3aaf3SDag-Erling Smørgrav 	return w;
227409a3aaf3SDag-Erling Smørgrav }
227509a3aaf3SDag-Erling Smørgrav 
227609a3aaf3SDag-Erling Smørgrav /** print the edns options to string */
227709a3aaf3SDag-Erling Smørgrav static int
227809a3aaf3SDag-Erling Smørgrav print_edns_opts(char** s, size_t* sl, uint8_t* rdata, size_t rdatalen)
227909a3aaf3SDag-Erling Smørgrav {
228009a3aaf3SDag-Erling Smørgrav 	uint16_t option_code, option_len;
228109a3aaf3SDag-Erling Smørgrav 	int w = 0;
228209a3aaf3SDag-Erling Smørgrav 	while(rdatalen > 0) {
228309a3aaf3SDag-Erling Smørgrav 		/* option name */
228409a3aaf3SDag-Erling Smørgrav 		if(rdatalen < 4) {
228509a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, " ; malformed: ");
228609a3aaf3SDag-Erling Smørgrav 			w += print_hex_buf(s, sl, rdata, rdatalen);
228709a3aaf3SDag-Erling Smørgrav 			return w;
228809a3aaf3SDag-Erling Smørgrav 		}
228909a3aaf3SDag-Erling Smørgrav 		option_code = sldns_read_uint16(rdata);
229009a3aaf3SDag-Erling Smørgrav 		option_len = sldns_read_uint16(rdata+2);
229109a3aaf3SDag-Erling Smørgrav 		rdata += 4;
229209a3aaf3SDag-Erling Smørgrav 		rdatalen -= 4;
229309a3aaf3SDag-Erling Smørgrav 
229409a3aaf3SDag-Erling Smørgrav 		/* option value */
229509a3aaf3SDag-Erling Smørgrav 		if(rdatalen < (size_t)option_len) {
229609a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, " ; malformed ");
229709a3aaf3SDag-Erling Smørgrav 			w += sldns_wire2str_edns_option_code_print(s, sl,
229809a3aaf3SDag-Erling Smørgrav 				option_code);
229909a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(s, sl, ": ");
230009a3aaf3SDag-Erling Smørgrav 			w += print_hex_buf(s, sl, rdata, rdatalen);
230109a3aaf3SDag-Erling Smørgrav 			return w;
230209a3aaf3SDag-Erling Smørgrav 		}
230309a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(s, sl, " ; ");
230409a3aaf3SDag-Erling Smørgrav 		w += sldns_wire2str_edns_option_print(s, sl, option_code,
230509a3aaf3SDag-Erling Smørgrav 			rdata, option_len);
230609a3aaf3SDag-Erling Smørgrav 		rdata += option_len;
230709a3aaf3SDag-Erling Smørgrav 		rdatalen -= option_len;
230809a3aaf3SDag-Erling Smørgrav 	}
230909a3aaf3SDag-Erling Smørgrav 	return w;
231009a3aaf3SDag-Erling Smørgrav }
231109a3aaf3SDag-Erling Smørgrav 
231209a3aaf3SDag-Erling Smørgrav int sldns_wire2str_edns_scan(uint8_t** data, size_t* data_len, char** str,
231309a3aaf3SDag-Erling Smørgrav         size_t* str_len, uint8_t* pkt, size_t pktlen)
231409a3aaf3SDag-Erling Smørgrav {
231509a3aaf3SDag-Erling Smørgrav 	int w = 0;
231609a3aaf3SDag-Erling Smørgrav 	uint8_t ext_rcode, edns_version;
231709a3aaf3SDag-Erling Smørgrav 	uint16_t udpsize, edns_bits, rdatalen;
231809a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(str, str_len, "; EDNS:");
231909a3aaf3SDag-Erling Smørgrav 
232009a3aaf3SDag-Erling Smørgrav 	/* some input checks, domain name */
232109a3aaf3SDag-Erling Smørgrav 	if(*data_len < 1+10)
232209a3aaf3SDag-Erling Smørgrav 		return w + print_remainder_hex("Error malformed 0x",
232309a3aaf3SDag-Erling Smørgrav 			data, data_len, str, str_len);
232409a3aaf3SDag-Erling Smørgrav 	if(*data[0] != 0) {
232509a3aaf3SDag-Erling Smørgrav 		return w + print_remainder_hex("Error nonrootdname 0x",
232609a3aaf3SDag-Erling Smørgrav 			data, data_len, str, str_len);
232709a3aaf3SDag-Erling Smørgrav 	}
232809a3aaf3SDag-Erling Smørgrav 	(*data)++;
232909a3aaf3SDag-Erling Smørgrav 	(*data_len)--;
233009a3aaf3SDag-Erling Smørgrav 
233109a3aaf3SDag-Erling Smørgrav 	/* check type and read fixed contents */
233209a3aaf3SDag-Erling Smørgrav 	if(sldns_read_uint16((*data)) != LDNS_RR_TYPE_OPT) {
233309a3aaf3SDag-Erling Smørgrav 		return w + print_remainder_hex("Error nottypeOPT 0x",
233409a3aaf3SDag-Erling Smørgrav 			data, data_len, str, str_len);
233509a3aaf3SDag-Erling Smørgrav 	}
233609a3aaf3SDag-Erling Smørgrav 	udpsize = sldns_read_uint16((*data)+2);
233709a3aaf3SDag-Erling Smørgrav 	ext_rcode = (*data)[4];
233809a3aaf3SDag-Erling Smørgrav 	edns_version = (*data)[5];
233909a3aaf3SDag-Erling Smørgrav 	edns_bits = sldns_read_uint16((*data)+6);
234009a3aaf3SDag-Erling Smørgrav 	rdatalen = sldns_read_uint16((*data)+8);
234109a3aaf3SDag-Erling Smørgrav 	(*data)+=10;
234209a3aaf3SDag-Erling Smørgrav 	(*data_len)-=10;
234309a3aaf3SDag-Erling Smørgrav 
234409a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(str, str_len, " version: %u;",
234509a3aaf3SDag-Erling Smørgrav 		(unsigned)edns_version);
234609a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(str, str_len, " flags:");
234709a3aaf3SDag-Erling Smørgrav 	if((edns_bits & LDNS_EDNS_MASK_DO_BIT))
234809a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(str, str_len, " do");
234909a3aaf3SDag-Erling Smørgrav 	/* the extended rcode is the value set, shifted four bits,
235009a3aaf3SDag-Erling Smørgrav 	 * and or'd with the original rcode */
235109a3aaf3SDag-Erling Smørgrav 	if(ext_rcode) {
235209a3aaf3SDag-Erling Smørgrav 		int rc = ((int)ext_rcode)<<4;
235309a3aaf3SDag-Erling Smørgrav 		if(pkt && pktlen >= LDNS_HEADER_SIZE)
235409a3aaf3SDag-Erling Smørgrav 			rc |= LDNS_RCODE_WIRE(pkt);
235509a3aaf3SDag-Erling Smørgrav 		w += sldns_str_print(str, str_len, " ; ext-rcode: %d", rc);
235609a3aaf3SDag-Erling Smørgrav 	}
235709a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(str, str_len, " ; udp: %u", (unsigned)udpsize);
235809a3aaf3SDag-Erling Smørgrav 
235909a3aaf3SDag-Erling Smørgrav 	if(rdatalen) {
2360bc892140SDag-Erling Smørgrav 		if((size_t)*data_len < rdatalen) {
236109a3aaf3SDag-Erling Smørgrav 			w += sldns_str_print(str, str_len,
236209a3aaf3SDag-Erling Smørgrav 				" ; Error EDNS rdata too short; ");
2363bc892140SDag-Erling Smørgrav 			rdatalen = (uint16_t)*data_len;
236409a3aaf3SDag-Erling Smørgrav 		}
236509a3aaf3SDag-Erling Smørgrav 		w += print_edns_opts(str, str_len, *data, rdatalen);
236609a3aaf3SDag-Erling Smørgrav 		(*data) += rdatalen;
236709a3aaf3SDag-Erling Smørgrav 		(*data_len) -= rdatalen;
236809a3aaf3SDag-Erling Smørgrav 	}
236909a3aaf3SDag-Erling Smørgrav 	w += sldns_str_print(str, str_len, "\n");
237009a3aaf3SDag-Erling Smørgrav 	return w;
237109a3aaf3SDag-Erling Smørgrav }
2372