aboutsummaryrefslogtreecommitdiff
path: root/include/arpa/nameser.h
blob: 70c634a69dfa59716f69a8238e522c52a7ce4950 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
#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;

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 (usually 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