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
|