1 /* Convert a DNS packet to a human-readable representation.
2 Copyright (C) 2016-2017 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, see
17 <http://www.gnu.org/licenses/>. */
19 #include <support/format_nss.h>
21 #include <arpa/inet.h>
23 #include <support/check.h>
24 #include <support/support.h>
25 #include <support/xmemstream.h>
29 const unsigned char *data
;
34 extract_8 (struct in_buffer
*in
, unsigned char *value
)
45 extract_16 (struct in_buffer
*in
, unsigned short *value
)
49 *value
= (in
->data
[0] << 8) | in
->data
[1];
56 extract_32 (struct in_buffer
*in
, unsigned *value
)
60 unsigned a
= in
->data
[0];
61 unsigned b
= in
->data
[1];
62 unsigned c
= in
->data
[2];
63 unsigned d
= in
->data
[3];
64 *value
= (a
<< 24) | (b
<< 16) | (c
<< 8) | d
;
71 extract_bytes (struct in_buffer
*in
, size_t length
, struct in_buffer
*value
)
73 if (in
->size
< length
)
75 *value
= (struct in_buffer
) {in
->data
, length
};
83 char name
[MAXDNAME
+ 1];
87 extract_name (struct in_buffer full
, struct in_buffer
*in
, struct dname
*value
)
89 const unsigned char *full_end
= full
.data
+ full
.size
;
90 /* Sanity checks; these indicate buffer misuse. */
92 (!(in
->data
< full
.data
|| in
->data
> full_end
93 || in
->size
> (size_t) (full_end
- in
->data
)));
94 int ret
= dn_expand (full
.data
, full_end
, in
->data
,
95 value
->name
, sizeof (value
->name
));
104 support_format_dns_packet (const unsigned char *buffer
, size_t length
)
106 struct in_buffer full
= { buffer
, length
};
107 struct in_buffer in
= full
;
108 struct xmemstream mem
;
109 xopen_memstream (&mem
);
111 unsigned short txnid
;
112 unsigned short flags
;
113 unsigned short qdcount
;
114 unsigned short ancount
;
115 unsigned short nscount
;
116 unsigned short adcount
;
117 if (!(extract_16 (&in
, &txnid
)
118 && extract_16 (&in
, &flags
)
119 && extract_16 (&in
, &qdcount
)
120 && extract_16 (&in
, &ancount
)
121 && extract_16 (&in
, &nscount
)
122 && extract_16 (&in
, &adcount
)))
124 fprintf (mem
.out
, "error: could not parse DNS header\n");
129 fprintf (mem
.out
, "error: question count is %d, not 1\n", qdcount
);
133 if (!extract_name (full
, &in
, &qname
))
135 fprintf (mem
.out
, "error: malformed QNAME\n");
138 unsigned short qtype
;
139 unsigned short qclass
;
140 if (!(extract_16 (&in
, &qtype
)
141 && extract_16 (&in
, &qclass
)))
143 fprintf (mem
.out
, "error: malformed question\n");
146 if (qtype
!= T_A
&& qtype
!= T_AAAA
&& qtype
!= T_PTR
)
148 fprintf (mem
.out
, "error: unsupported QTYPE %d\n", qtype
);
152 fprintf (mem
.out
, "name: %s\n", qname
.name
);
154 for (int i
= 0; i
< ancount
; ++i
)
157 if (!extract_name (full
, &in
, &rname
))
159 fprintf (mem
.out
, "error: malformed record name\n");
162 unsigned short rtype
;
163 unsigned short rclass
;
165 unsigned short rdlen
;
166 struct in_buffer rdata
;
167 if (!(extract_16 (&in
, &rtype
)
168 && extract_16 (&in
, &rclass
)
169 && extract_32 (&in
, &ttl
)
170 && extract_16 (&in
, &rdlen
)
171 && extract_bytes (&in
, rdlen
, &rdata
)))
173 fprintf (mem
.out
, "error: malformed record header\n");
176 /* Skip non-matching record types. */
177 if ((rtype
!= qtype
&& rtype
!= T_CNAME
) || rclass
!= qclass
)
183 fprintf (mem
.out
, "address: %d.%d.%d.%d\n",
189 fprintf (mem
.out
, "error: A record of size %d: %s\n",
197 if (inet_ntop (AF_INET6
, rdata
.data
, buf
, sizeof (buf
)) == NULL
)
198 fprintf (mem
.out
, "error: AAAA record decoding failed: %m\n");
200 fprintf (mem
.out
, "address: %s\n", buf
);
203 fprintf (mem
.out
, "error: AAAA record of size %d: %s\n",
211 if (extract_name (full
, &rdata
, &name
))
212 fprintf (mem
.out
, "name: %s\n", name
.name
);
214 fprintf (mem
.out
, "error: malformed CNAME/PTR record\n");
220 xfclose_memstream (&mem
);