mirror of
https://sourceware.org/git/glibc.git
synced 2024-12-14 07:10:05 +00:00
857c890d9b
The public parser functions around the ns_rr record type produce textual domain names, but usually, this is not what we need while parsing DNS packets within glibc. This commit adds two new helper functions, __ns_rr_cursor_init and __ns_rr_cursor_next, for writing packet parsers, and struct ns_rr_cursor, struct ns_rr_wire as supporting types. In theory, it is possible to avoid copying the owner name into the rname field in __ns_rr_cursor_next, but this would need more functions that work on compressed names. Eventually, __res_context_send could be enhanced to preserve the result of the packet parsing that is necessary for matching the incoming UDP packets, so that this works does not have to be done twice. Reviewed-by: Siddhesh Poyarekar <siddhesh@sourceware.org>
200 lines
6.6 KiB
C
200 lines
6.6 KiB
C
#ifndef _ARPA_NAMESER_H_
|
|
|
|
#include <resolv/arpa/nameser.h>
|
|
|
|
# ifndef _ISOMAC
|
|
|
|
/* If the machine allows unaligned access we can do better than using
|
|
the NS_GET16, NS_GET32, NS_PUT16, and NS_PUT32 macros from the
|
|
installed header. */
|
|
#include <string.h>
|
|
#include <stdint.h>
|
|
#include <netinet/in.h>
|
|
|
|
extern const struct _ns_flagdata _ns_flagdata[] attribute_hidden;
|
|
|
|
#if _STRING_ARCH_unaligned
|
|
|
|
# undef NS_GET16
|
|
# define NS_GET16(s, cp) \
|
|
do { \
|
|
const uint16_t *t_cp = (const uint16_t *) (cp); \
|
|
(s) = ntohs (*t_cp); \
|
|
(cp) += NS_INT16SZ; \
|
|
} while (0)
|
|
|
|
# undef NS_GET32
|
|
# define NS_GET32(l, cp) \
|
|
do { \
|
|
const uint32_t *t_cp = (const uint32_t *) (cp); \
|
|
(l) = ntohl (*t_cp); \
|
|
(cp) += NS_INT32SZ; \
|
|
} while (0)
|
|
|
|
# undef NS_PUT16
|
|
# define NS_PUT16(s, cp) \
|
|
do { \
|
|
uint16_t *t_cp = (uint16_t *) (cp); \
|
|
*t_cp = htons (s); \
|
|
(cp) += NS_INT16SZ; \
|
|
} while (0)
|
|
|
|
# undef NS_PUT32
|
|
# define NS_PUT32(l, cp) \
|
|
do { \
|
|
uint32_t *t_cp = (uint32_t *) (cp); \
|
|
*t_cp = htonl (l); \
|
|
(cp) += NS_INT32SZ; \
|
|
} while (0)
|
|
|
|
#endif
|
|
|
|
extern unsigned int __ns_get16 (const unsigned char *) __THROW;
|
|
extern unsigned long __ns_get32 (const unsigned char *) __THROW;
|
|
int __ns_name_ntop (const unsigned char *, char *, size_t) __THROW;
|
|
int __ns_name_unpack (const unsigned char *, const unsigned char *,
|
|
const unsigned char *, unsigned char *, size_t) __THROW;
|
|
|
|
/* Like ns_samename, but for uncompressed binary names. Return true
|
|
if the two arguments compare are equal as case-insensitive domain
|
|
names. */
|
|
_Bool __ns_samebinaryname (const unsigned char *, const unsigned char *)
|
|
attribute_hidden;
|
|
|
|
#define ns_msg_getflag(handle, flag) \
|
|
(((handle)._flags & _ns_flagdata[flag].mask) >> _ns_flagdata[flag].shift)
|
|
|
|
libresolv_hidden_proto (ns_get16)
|
|
libresolv_hidden_proto (ns_get32)
|
|
libresolv_hidden_proto (ns_put16)
|
|
libresolv_hidden_proto (ns_put32)
|
|
libresolv_hidden_proto (ns_initparse)
|
|
libresolv_hidden_proto (ns_skiprr)
|
|
libresolv_hidden_proto (ns_parserr)
|
|
libresolv_hidden_proto (ns_sprintrr)
|
|
libresolv_hidden_proto (ns_sprintrrf)
|
|
libresolv_hidden_proto (ns_samedomain)
|
|
libresolv_hidden_proto (ns_format_ttl)
|
|
|
|
extern __typeof (ns_makecanon) __libc_ns_makecanon;
|
|
libc_hidden_proto (__libc_ns_makecanon)
|
|
extern __typeof (ns_name_compress) __ns_name_compress;
|
|
libc_hidden_proto (__ns_name_compress)
|
|
extern __typeof (ns_name_ntop) __ns_name_ntop;
|
|
libc_hidden_proto (__ns_name_ntop)
|
|
extern __typeof (ns_name_pack) __ns_name_pack;
|
|
libc_hidden_proto (__ns_name_pack)
|
|
extern __typeof (ns_name_pton) __ns_name_pton;
|
|
libc_hidden_proto (__ns_name_pton)
|
|
extern __typeof (ns_name_skip) __ns_name_skip;
|
|
libc_hidden_proto (__ns_name_skip)
|
|
extern __typeof (ns_name_uncompress) __ns_name_uncompress;
|
|
libc_hidden_proto (__ns_name_uncompress)
|
|
extern __typeof (ns_name_unpack) __ns_name_unpack;
|
|
libc_hidden_proto (__ns_name_unpack)
|
|
extern __typeof (ns_samename) __libc_ns_samename;
|
|
libc_hidden_proto (__libc_ns_samename)
|
|
|
|
/* Packet parser helper functions. */
|
|
|
|
/* Verify that P points to an uncompressed domain name in wire format.
|
|
On success, return the length of the encoded name, including the
|
|
terminating null byte. On failure, return -1 and set errno. EOM
|
|
must point one past the last byte in the packet. */
|
|
int __ns_name_length_uncompressed (const unsigned char *p,
|
|
const unsigned char *eom) attribute_hidden;
|
|
|
|
/* Iterator over the resource records in a DNS packet. */
|
|
struct ns_rr_cursor
|
|
{
|
|
/* These members are not changed after initialization. */
|
|
const unsigned char *begin; /* First byte of packet. */
|
|
const unsigned char *end; /* One past the last byte of the packet. */
|
|
const unsigned char *first_rr; /* First resource record (or packet end). */
|
|
|
|
/* Advanced towards the end while reading the packet. */
|
|
const unsigned char *current;
|
|
};
|
|
|
|
/* Returns the RCODE field from the DNS header. */
|
|
static inline int
|
|
ns_rr_cursor_rcode (const struct ns_rr_cursor *c)
|
|
{
|
|
return c->begin[3] & 0x0f; /* Lower 4 bits at offset 3. */
|
|
}
|
|
|
|
/* Returns the length of the answer section according to the DNS header. */
|
|
static inline int
|
|
ns_rr_cursor_ancount (const struct ns_rr_cursor *c)
|
|
{
|
|
return c->begin[6] * 256 + c->begin[7]; /* 16 bits at offset 6. */
|
|
}
|
|
|
|
/* Returns the length of the authority (name server) section according
|
|
to the DNS header. */
|
|
static inline int
|
|
ns_rr_cursor_nscount (const struct ns_rr_cursor *c)
|
|
{
|
|
return c->begin[8] * 256 + c->begin[9]; /* 16 bits at offset 8. */
|
|
}
|
|
|
|
/* Returns the length of the additional data section according to the
|
|
DNS header. */
|
|
static inline int
|
|
ns_rr_cursor_adcount (const struct ns_rr_cursor *c)
|
|
{
|
|
return c->begin[10] * 256 + c->begin[11]; /* 16 bits at offset 10. */
|
|
}
|
|
|
|
/* Returns a pointer to the uncompressed question name in wire
|
|
format. */
|
|
static inline const unsigned char *
|
|
ns_rr_cursor_qname (const struct ns_rr_cursor *c)
|
|
{
|
|
return c->begin + 12; /* QNAME starts right after the header. */
|
|
}
|
|
|
|
/* Returns the question type of the first and only question. */
|
|
static inline const int
|
|
ns_rr_cursor_qtype (const struct ns_rr_cursor *c)
|
|
{
|
|
/* 16 bits 4 bytes back from the first RR header start. */
|
|
return c->first_rr[-4] * 256 + c->first_rr[-3];
|
|
}
|
|
|
|
/* Returns the clss of the first and only question (usally C_IN). */
|
|
static inline const int
|
|
ns_rr_cursor_qclass (const struct ns_rr_cursor *c)
|
|
{
|
|
/* 16 bits 2 bytes back from the first RR header start. */
|
|
return c->first_rr[-2] * 256 + c->first_rr[-1];
|
|
}
|
|
|
|
/* Initializes *C to cover the packet [BUF, BUF+LEN). Returns false
|
|
if LEN is less than sizeof (*HD), if the packet does not contain a
|
|
full (uncompressed) question, or if the question count is not 1. */
|
|
_Bool __ns_rr_cursor_init (struct ns_rr_cursor *c,
|
|
const unsigned char *buf, size_t len)
|
|
attribute_hidden;
|
|
|
|
/* Like ns_rr, but the record owner name is not decoded into text format. */
|
|
struct ns_rr_wire
|
|
{
|
|
unsigned char rname[NS_MAXCDNAME]; /* Owner name of the record. */
|
|
uint16_t rtype; /* Resource record type (T_*). */
|
|
uint16_t rclass; /* Resource record class (C_*). */
|
|
uint32_t ttl; /* Time-to-live field. */
|
|
const unsigned char *rdata; /* Start of resource record data. */
|
|
uint16_t rdlength; /* Length of the data at rdata, in bytes. */
|
|
};
|
|
|
|
/* Attempts to parse the record at C into *RR. On success, return
|
|
true, and C is advanced past the record, and RR->rdata points to
|
|
the record data. On failure, errno is set to EMSGSIZE, and false
|
|
is returned. */
|
|
_Bool __ns_rr_cursor_next (struct ns_rr_cursor *c, struct ns_rr_wire *rr)
|
|
attribute_hidden;
|
|
|
|
# endif /* !_ISOMAC */
|
|
#endif
|