msgreply.c revision 1.1.1.1.2.2 1 1.1.1.1.2.2 pgoyette /*
2 1.1.1.1.2.2 pgoyette * util/data/msgreply.c - store message and reply data.
3 1.1.1.1.2.2 pgoyette *
4 1.1.1.1.2.2 pgoyette * Copyright (c) 2007, NLnet Labs. All rights reserved.
5 1.1.1.1.2.2 pgoyette *
6 1.1.1.1.2.2 pgoyette * This software is open source.
7 1.1.1.1.2.2 pgoyette *
8 1.1.1.1.2.2 pgoyette * Redistribution and use in source and binary forms, with or without
9 1.1.1.1.2.2 pgoyette * modification, are permitted provided that the following conditions
10 1.1.1.1.2.2 pgoyette * are met:
11 1.1.1.1.2.2 pgoyette *
12 1.1.1.1.2.2 pgoyette * Redistributions of source code must retain the above copyright notice,
13 1.1.1.1.2.2 pgoyette * this list of conditions and the following disclaimer.
14 1.1.1.1.2.2 pgoyette *
15 1.1.1.1.2.2 pgoyette * Redistributions in binary form must reproduce the above copyright notice,
16 1.1.1.1.2.2 pgoyette * this list of conditions and the following disclaimer in the documentation
17 1.1.1.1.2.2 pgoyette * and/or other materials provided with the distribution.
18 1.1.1.1.2.2 pgoyette *
19 1.1.1.1.2.2 pgoyette * Neither the name of the NLNET LABS nor the names of its contributors may
20 1.1.1.1.2.2 pgoyette * be used to endorse or promote products derived from this software without
21 1.1.1.1.2.2 pgoyette * specific prior written permission.
22 1.1.1.1.2.2 pgoyette *
23 1.1.1.1.2.2 pgoyette * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
24 1.1.1.1.2.2 pgoyette * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
25 1.1.1.1.2.2 pgoyette * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
26 1.1.1.1.2.2 pgoyette * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
27 1.1.1.1.2.2 pgoyette * HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
28 1.1.1.1.2.2 pgoyette * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
29 1.1.1.1.2.2 pgoyette * TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
30 1.1.1.1.2.2 pgoyette * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
31 1.1.1.1.2.2 pgoyette * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
32 1.1.1.1.2.2 pgoyette * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
33 1.1.1.1.2.2 pgoyette * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
34 1.1.1.1.2.2 pgoyette */
35 1.1.1.1.2.2 pgoyette
36 1.1.1.1.2.2 pgoyette /**
37 1.1.1.1.2.2 pgoyette * \file
38 1.1.1.1.2.2 pgoyette *
39 1.1.1.1.2.2 pgoyette * This file contains a data structure to store a message and its reply.
40 1.1.1.1.2.2 pgoyette */
41 1.1.1.1.2.2 pgoyette
42 1.1.1.1.2.2 pgoyette #include "config.h"
43 1.1.1.1.2.2 pgoyette #include "util/data/msgreply.h"
44 1.1.1.1.2.2 pgoyette #include "util/storage/lookup3.h"
45 1.1.1.1.2.2 pgoyette #include "util/log.h"
46 1.1.1.1.2.2 pgoyette #include "util/alloc.h"
47 1.1.1.1.2.2 pgoyette #include "util/netevent.h"
48 1.1.1.1.2.2 pgoyette #include "util/net_help.h"
49 1.1.1.1.2.2 pgoyette #include "util/data/dname.h"
50 1.1.1.1.2.2 pgoyette #include "util/regional.h"
51 1.1.1.1.2.2 pgoyette #include "util/data/msgparse.h"
52 1.1.1.1.2.2 pgoyette #include "util/data/msgencode.h"
53 1.1.1.1.2.2 pgoyette #include "sldns/sbuffer.h"
54 1.1.1.1.2.2 pgoyette #include "sldns/wire2str.h"
55 1.1.1.1.2.2 pgoyette
56 1.1.1.1.2.2 pgoyette /** MAX TTL default for messages and rrsets */
57 1.1.1.1.2.2 pgoyette time_t MAX_TTL = 3600 * 24 * 10; /* ten days */
58 1.1.1.1.2.2 pgoyette /** MIN TTL default for messages and rrsets */
59 1.1.1.1.2.2 pgoyette time_t MIN_TTL = 0;
60 1.1.1.1.2.2 pgoyette /** MAX Negative TTL, for SOA records in authority section */
61 1.1.1.1.2.2 pgoyette time_t MAX_NEG_TTL = 3600; /* one hour */
62 1.1.1.1.2.2 pgoyette
63 1.1.1.1.2.2 pgoyette /** allocate qinfo, return 0 on error */
64 1.1.1.1.2.2 pgoyette static int
65 1.1.1.1.2.2 pgoyette parse_create_qinfo(sldns_buffer* pkt, struct msg_parse* msg,
66 1.1.1.1.2.2 pgoyette struct query_info* qinf, struct regional* region)
67 1.1.1.1.2.2 pgoyette {
68 1.1.1.1.2.2 pgoyette if(msg->qname) {
69 1.1.1.1.2.2 pgoyette if(region)
70 1.1.1.1.2.2 pgoyette qinf->qname = (uint8_t*)regional_alloc(region,
71 1.1.1.1.2.2 pgoyette msg->qname_len);
72 1.1.1.1.2.2 pgoyette else qinf->qname = (uint8_t*)malloc(msg->qname_len);
73 1.1.1.1.2.2 pgoyette if(!qinf->qname) return 0;
74 1.1.1.1.2.2 pgoyette dname_pkt_copy(pkt, qinf->qname, msg->qname);
75 1.1.1.1.2.2 pgoyette } else qinf->qname = 0;
76 1.1.1.1.2.2 pgoyette qinf->qname_len = msg->qname_len;
77 1.1.1.1.2.2 pgoyette qinf->qtype = msg->qtype;
78 1.1.1.1.2.2 pgoyette qinf->qclass = msg->qclass;
79 1.1.1.1.2.2 pgoyette return 1;
80 1.1.1.1.2.2 pgoyette }
81 1.1.1.1.2.2 pgoyette
82 1.1.1.1.2.2 pgoyette /** constructor for replyinfo */
83 1.1.1.1.2.2 pgoyette struct reply_info*
84 1.1.1.1.2.2 pgoyette construct_reply_info_base(struct regional* region, uint16_t flags, size_t qd,
85 1.1.1.1.2.2 pgoyette time_t ttl, time_t prettl, size_t an, size_t ns, size_t ar,
86 1.1.1.1.2.2 pgoyette size_t total, enum sec_status sec)
87 1.1.1.1.2.2 pgoyette {
88 1.1.1.1.2.2 pgoyette struct reply_info* rep;
89 1.1.1.1.2.2 pgoyette /* rrset_count-1 because the first ref is part of the struct. */
90 1.1.1.1.2.2 pgoyette size_t s = sizeof(struct reply_info) - sizeof(struct rrset_ref) +
91 1.1.1.1.2.2 pgoyette sizeof(struct ub_packed_rrset_key*) * total;
92 1.1.1.1.2.2 pgoyette if(total >= RR_COUNT_MAX) return NULL; /* sanity check on numRRS*/
93 1.1.1.1.2.2 pgoyette if(region)
94 1.1.1.1.2.2 pgoyette rep = (struct reply_info*)regional_alloc(region, s);
95 1.1.1.1.2.2 pgoyette else rep = (struct reply_info*)malloc(s +
96 1.1.1.1.2.2 pgoyette sizeof(struct rrset_ref) * (total));
97 1.1.1.1.2.2 pgoyette if(!rep)
98 1.1.1.1.2.2 pgoyette return NULL;
99 1.1.1.1.2.2 pgoyette rep->flags = flags;
100 1.1.1.1.2.2 pgoyette rep->qdcount = qd;
101 1.1.1.1.2.2 pgoyette rep->ttl = ttl;
102 1.1.1.1.2.2 pgoyette rep->prefetch_ttl = prettl;
103 1.1.1.1.2.2 pgoyette rep->an_numrrsets = an;
104 1.1.1.1.2.2 pgoyette rep->ns_numrrsets = ns;
105 1.1.1.1.2.2 pgoyette rep->ar_numrrsets = ar;
106 1.1.1.1.2.2 pgoyette rep->rrset_count = total;
107 1.1.1.1.2.2 pgoyette rep->security = sec;
108 1.1.1.1.2.2 pgoyette rep->authoritative = 0;
109 1.1.1.1.2.2 pgoyette /* array starts after the refs */
110 1.1.1.1.2.2 pgoyette if(region)
111 1.1.1.1.2.2 pgoyette rep->rrsets = (struct ub_packed_rrset_key**)&(rep->ref[0]);
112 1.1.1.1.2.2 pgoyette else rep->rrsets = (struct ub_packed_rrset_key**)&(rep->ref[total]);
113 1.1.1.1.2.2 pgoyette /* zero the arrays to assist cleanup in case of malloc failure */
114 1.1.1.1.2.2 pgoyette memset( rep->rrsets, 0, sizeof(struct ub_packed_rrset_key*) * total);
115 1.1.1.1.2.2 pgoyette if(!region)
116 1.1.1.1.2.2 pgoyette memset( &rep->ref[0], 0, sizeof(struct rrset_ref) * total);
117 1.1.1.1.2.2 pgoyette return rep;
118 1.1.1.1.2.2 pgoyette }
119 1.1.1.1.2.2 pgoyette
120 1.1.1.1.2.2 pgoyette /** allocate replyinfo, return 0 on error */
121 1.1.1.1.2.2 pgoyette static int
122 1.1.1.1.2.2 pgoyette parse_create_repinfo(struct msg_parse* msg, struct reply_info** rep,
123 1.1.1.1.2.2 pgoyette struct regional* region)
124 1.1.1.1.2.2 pgoyette {
125 1.1.1.1.2.2 pgoyette *rep = construct_reply_info_base(region, msg->flags, msg->qdcount, 0,
126 1.1.1.1.2.2 pgoyette 0, msg->an_rrsets, msg->ns_rrsets, msg->ar_rrsets,
127 1.1.1.1.2.2 pgoyette msg->rrset_count, sec_status_unchecked);
128 1.1.1.1.2.2 pgoyette if(!*rep)
129 1.1.1.1.2.2 pgoyette return 0;
130 1.1.1.1.2.2 pgoyette return 1;
131 1.1.1.1.2.2 pgoyette }
132 1.1.1.1.2.2 pgoyette
133 1.1.1.1.2.2 pgoyette /** allocate (special) rrset keys, return 0 on error */
134 1.1.1.1.2.2 pgoyette static int
135 1.1.1.1.2.2 pgoyette repinfo_alloc_rrset_keys(struct reply_info* rep, struct alloc_cache* alloc,
136 1.1.1.1.2.2 pgoyette struct regional* region)
137 1.1.1.1.2.2 pgoyette {
138 1.1.1.1.2.2 pgoyette size_t i;
139 1.1.1.1.2.2 pgoyette for(i=0; i<rep->rrset_count; i++) {
140 1.1.1.1.2.2 pgoyette if(region) {
141 1.1.1.1.2.2 pgoyette rep->rrsets[i] = (struct ub_packed_rrset_key*)
142 1.1.1.1.2.2 pgoyette regional_alloc(region,
143 1.1.1.1.2.2 pgoyette sizeof(struct ub_packed_rrset_key));
144 1.1.1.1.2.2 pgoyette if(rep->rrsets[i]) {
145 1.1.1.1.2.2 pgoyette memset(rep->rrsets[i], 0,
146 1.1.1.1.2.2 pgoyette sizeof(struct ub_packed_rrset_key));
147 1.1.1.1.2.2 pgoyette rep->rrsets[i]->entry.key = rep->rrsets[i];
148 1.1.1.1.2.2 pgoyette }
149 1.1.1.1.2.2 pgoyette }
150 1.1.1.1.2.2 pgoyette else rep->rrsets[i] = alloc_special_obtain(alloc);
151 1.1.1.1.2.2 pgoyette if(!rep->rrsets[i])
152 1.1.1.1.2.2 pgoyette return 0;
153 1.1.1.1.2.2 pgoyette rep->rrsets[i]->entry.data = NULL;
154 1.1.1.1.2.2 pgoyette }
155 1.1.1.1.2.2 pgoyette return 1;
156 1.1.1.1.2.2 pgoyette }
157 1.1.1.1.2.2 pgoyette
158 1.1.1.1.2.2 pgoyette /** find the minimumttl in the rdata of SOA record */
159 1.1.1.1.2.2 pgoyette static time_t
160 1.1.1.1.2.2 pgoyette soa_find_minttl(struct rr_parse* rr)
161 1.1.1.1.2.2 pgoyette {
162 1.1.1.1.2.2 pgoyette uint16_t rlen = sldns_read_uint16(rr->ttl_data+4);
163 1.1.1.1.2.2 pgoyette if(rlen < 20)
164 1.1.1.1.2.2 pgoyette return 0; /* rdata too small for SOA (dname, dname, 5*32bit) */
165 1.1.1.1.2.2 pgoyette /* minimum TTL is the last 32bit value in the rdata of the record */
166 1.1.1.1.2.2 pgoyette /* at position ttl_data + 4(ttl) + 2(rdatalen) + rdatalen - 4(timeval)*/
167 1.1.1.1.2.2 pgoyette return (time_t)sldns_read_uint32(rr->ttl_data+6+rlen-4);
168 1.1.1.1.2.2 pgoyette }
169 1.1.1.1.2.2 pgoyette
170 1.1.1.1.2.2 pgoyette /** do the rdata copy */
171 1.1.1.1.2.2 pgoyette static int
172 1.1.1.1.2.2 pgoyette rdata_copy(sldns_buffer* pkt, struct packed_rrset_data* data, uint8_t* to,
173 1.1.1.1.2.2 pgoyette struct rr_parse* rr, time_t* rr_ttl, uint16_t type,
174 1.1.1.1.2.2 pgoyette sldns_pkt_section section)
175 1.1.1.1.2.2 pgoyette {
176 1.1.1.1.2.2 pgoyette uint16_t pkt_len;
177 1.1.1.1.2.2 pgoyette const sldns_rr_descriptor* desc;
178 1.1.1.1.2.2 pgoyette
179 1.1.1.1.2.2 pgoyette *rr_ttl = sldns_read_uint32(rr->ttl_data);
180 1.1.1.1.2.2 pgoyette /* RFC 2181 Section 8. if msb of ttl is set treat as if zero. */
181 1.1.1.1.2.2 pgoyette if(*rr_ttl & 0x80000000U)
182 1.1.1.1.2.2 pgoyette *rr_ttl = 0;
183 1.1.1.1.2.2 pgoyette if(type == LDNS_RR_TYPE_SOA && section == LDNS_SECTION_AUTHORITY) {
184 1.1.1.1.2.2 pgoyette /* negative response. see if TTL of SOA record larger than the
185 1.1.1.1.2.2 pgoyette * minimum-ttl in the rdata of the SOA record */
186 1.1.1.1.2.2 pgoyette if(*rr_ttl > soa_find_minttl(rr))
187 1.1.1.1.2.2 pgoyette *rr_ttl = soa_find_minttl(rr);
188 1.1.1.1.2.2 pgoyette if(*rr_ttl > MAX_NEG_TTL)
189 1.1.1.1.2.2 pgoyette *rr_ttl = MAX_NEG_TTL;
190 1.1.1.1.2.2 pgoyette }
191 1.1.1.1.2.2 pgoyette if(*rr_ttl < MIN_TTL)
192 1.1.1.1.2.2 pgoyette *rr_ttl = MIN_TTL;
193 1.1.1.1.2.2 pgoyette if(*rr_ttl < data->ttl)
194 1.1.1.1.2.2 pgoyette data->ttl = *rr_ttl;
195 1.1.1.1.2.2 pgoyette
196 1.1.1.1.2.2 pgoyette if(rr->outside_packet) {
197 1.1.1.1.2.2 pgoyette /* uncompressed already, only needs copy */
198 1.1.1.1.2.2 pgoyette memmove(to, rr->ttl_data+sizeof(uint32_t), rr->size);
199 1.1.1.1.2.2 pgoyette return 1;
200 1.1.1.1.2.2 pgoyette }
201 1.1.1.1.2.2 pgoyette
202 1.1.1.1.2.2 pgoyette sldns_buffer_set_position(pkt, (size_t)
203 1.1.1.1.2.2 pgoyette (rr->ttl_data - sldns_buffer_begin(pkt) + sizeof(uint32_t)));
204 1.1.1.1.2.2 pgoyette /* insert decompressed size into rdata len stored in memory */
205 1.1.1.1.2.2 pgoyette /* -2 because rdatalen bytes are not included. */
206 1.1.1.1.2.2 pgoyette pkt_len = htons(rr->size - 2);
207 1.1.1.1.2.2 pgoyette memmove(to, &pkt_len, sizeof(uint16_t));
208 1.1.1.1.2.2 pgoyette to += 2;
209 1.1.1.1.2.2 pgoyette /* read packet rdata len */
210 1.1.1.1.2.2 pgoyette pkt_len = sldns_buffer_read_u16(pkt);
211 1.1.1.1.2.2 pgoyette if(sldns_buffer_remaining(pkt) < pkt_len)
212 1.1.1.1.2.2 pgoyette return 0;
213 1.1.1.1.2.2 pgoyette desc = sldns_rr_descript(type);
214 1.1.1.1.2.2 pgoyette if(pkt_len > 0 && desc && desc->_dname_count > 0) {
215 1.1.1.1.2.2 pgoyette int count = (int)desc->_dname_count;
216 1.1.1.1.2.2 pgoyette int rdf = 0;
217 1.1.1.1.2.2 pgoyette size_t len;
218 1.1.1.1.2.2 pgoyette size_t oldpos;
219 1.1.1.1.2.2 pgoyette /* decompress dnames. */
220 1.1.1.1.2.2 pgoyette while(pkt_len > 0 && count) {
221 1.1.1.1.2.2 pgoyette switch(desc->_wireformat[rdf]) {
222 1.1.1.1.2.2 pgoyette case LDNS_RDF_TYPE_DNAME:
223 1.1.1.1.2.2 pgoyette oldpos = sldns_buffer_position(pkt);
224 1.1.1.1.2.2 pgoyette dname_pkt_copy(pkt, to,
225 1.1.1.1.2.2 pgoyette sldns_buffer_current(pkt));
226 1.1.1.1.2.2 pgoyette to += pkt_dname_len(pkt);
227 1.1.1.1.2.2 pgoyette pkt_len -= sldns_buffer_position(pkt)-oldpos;
228 1.1.1.1.2.2 pgoyette count--;
229 1.1.1.1.2.2 pgoyette len = 0;
230 1.1.1.1.2.2 pgoyette break;
231 1.1.1.1.2.2 pgoyette case LDNS_RDF_TYPE_STR:
232 1.1.1.1.2.2 pgoyette len = sldns_buffer_current(pkt)[0] + 1;
233 1.1.1.1.2.2 pgoyette break;
234 1.1.1.1.2.2 pgoyette default:
235 1.1.1.1.2.2 pgoyette len = get_rdf_size(desc->_wireformat[rdf]);
236 1.1.1.1.2.2 pgoyette break;
237 1.1.1.1.2.2 pgoyette }
238 1.1.1.1.2.2 pgoyette if(len) {
239 1.1.1.1.2.2 pgoyette memmove(to, sldns_buffer_current(pkt), len);
240 1.1.1.1.2.2 pgoyette to += len;
241 1.1.1.1.2.2 pgoyette sldns_buffer_skip(pkt, (ssize_t)len);
242 1.1.1.1.2.2 pgoyette log_assert(len <= pkt_len);
243 1.1.1.1.2.2 pgoyette pkt_len -= len;
244 1.1.1.1.2.2 pgoyette }
245 1.1.1.1.2.2 pgoyette rdf++;
246 1.1.1.1.2.2 pgoyette }
247 1.1.1.1.2.2 pgoyette }
248 1.1.1.1.2.2 pgoyette /* copy remaining rdata */
249 1.1.1.1.2.2 pgoyette if(pkt_len > 0)
250 1.1.1.1.2.2 pgoyette memmove(to, sldns_buffer_current(pkt), pkt_len);
251 1.1.1.1.2.2 pgoyette
252 1.1.1.1.2.2 pgoyette return 1;
253 1.1.1.1.2.2 pgoyette }
254 1.1.1.1.2.2 pgoyette
255 1.1.1.1.2.2 pgoyette /** copy over the data into packed rrset */
256 1.1.1.1.2.2 pgoyette static int
257 1.1.1.1.2.2 pgoyette parse_rr_copy(sldns_buffer* pkt, struct rrset_parse* pset,
258 1.1.1.1.2.2 pgoyette struct packed_rrset_data* data)
259 1.1.1.1.2.2 pgoyette {
260 1.1.1.1.2.2 pgoyette size_t i;
261 1.1.1.1.2.2 pgoyette struct rr_parse* rr = pset->rr_first;
262 1.1.1.1.2.2 pgoyette uint8_t* nextrdata;
263 1.1.1.1.2.2 pgoyette size_t total = pset->rr_count + pset->rrsig_count;
264 1.1.1.1.2.2 pgoyette data->ttl = MAX_TTL;
265 1.1.1.1.2.2 pgoyette data->count = pset->rr_count;
266 1.1.1.1.2.2 pgoyette data->rrsig_count = pset->rrsig_count;
267 1.1.1.1.2.2 pgoyette data->trust = rrset_trust_none;
268 1.1.1.1.2.2 pgoyette data->security = sec_status_unchecked;
269 1.1.1.1.2.2 pgoyette /* layout: struct - rr_len - rr_data - rr_ttl - rdata - rrsig */
270 1.1.1.1.2.2 pgoyette data->rr_len = (size_t*)((uint8_t*)data +
271 1.1.1.1.2.2 pgoyette sizeof(struct packed_rrset_data));
272 1.1.1.1.2.2 pgoyette data->rr_data = (uint8_t**)&(data->rr_len[total]);
273 1.1.1.1.2.2 pgoyette data->rr_ttl = (time_t*)&(data->rr_data[total]);
274 1.1.1.1.2.2 pgoyette nextrdata = (uint8_t*)&(data->rr_ttl[total]);
275 1.1.1.1.2.2 pgoyette for(i=0; i<data->count; i++) {
276 1.1.1.1.2.2 pgoyette data->rr_len[i] = rr->size;
277 1.1.1.1.2.2 pgoyette data->rr_data[i] = nextrdata;
278 1.1.1.1.2.2 pgoyette nextrdata += rr->size;
279 1.1.1.1.2.2 pgoyette if(!rdata_copy(pkt, data, data->rr_data[i], rr,
280 1.1.1.1.2.2 pgoyette &data->rr_ttl[i], pset->type, pset->section))
281 1.1.1.1.2.2 pgoyette return 0;
282 1.1.1.1.2.2 pgoyette rr = rr->next;
283 1.1.1.1.2.2 pgoyette }
284 1.1.1.1.2.2 pgoyette /* if rrsig, its rdata is at nextrdata */
285 1.1.1.1.2.2 pgoyette rr = pset->rrsig_first;
286 1.1.1.1.2.2 pgoyette for(i=data->count; i<total; i++) {
287 1.1.1.1.2.2 pgoyette data->rr_len[i] = rr->size;
288 1.1.1.1.2.2 pgoyette data->rr_data[i] = nextrdata;
289 1.1.1.1.2.2 pgoyette nextrdata += rr->size;
290 1.1.1.1.2.2 pgoyette if(!rdata_copy(pkt, data, data->rr_data[i], rr,
291 1.1.1.1.2.2 pgoyette &data->rr_ttl[i], LDNS_RR_TYPE_RRSIG, pset->section))
292 1.1.1.1.2.2 pgoyette return 0;
293 1.1.1.1.2.2 pgoyette rr = rr->next;
294 1.1.1.1.2.2 pgoyette }
295 1.1.1.1.2.2 pgoyette return 1;
296 1.1.1.1.2.2 pgoyette }
297 1.1.1.1.2.2 pgoyette
298 1.1.1.1.2.2 pgoyette /** create rrset return 0 on failure */
299 1.1.1.1.2.2 pgoyette static int
300 1.1.1.1.2.2 pgoyette parse_create_rrset(sldns_buffer* pkt, struct rrset_parse* pset,
301 1.1.1.1.2.2 pgoyette struct packed_rrset_data** data, struct regional* region)
302 1.1.1.1.2.2 pgoyette {
303 1.1.1.1.2.2 pgoyette /* allocate */
304 1.1.1.1.2.2 pgoyette size_t s;
305 1.1.1.1.2.2 pgoyette if(pset->rr_count > RR_COUNT_MAX || pset->rrsig_count > RR_COUNT_MAX ||
306 1.1.1.1.2.2 pgoyette pset->size > RR_COUNT_MAX)
307 1.1.1.1.2.2 pgoyette return 0; /* protect against integer overflow */
308 1.1.1.1.2.2 pgoyette s = sizeof(struct packed_rrset_data) +
309 1.1.1.1.2.2 pgoyette (pset->rr_count + pset->rrsig_count) *
310 1.1.1.1.2.2 pgoyette (sizeof(size_t)+sizeof(uint8_t*)+sizeof(time_t)) +
311 1.1.1.1.2.2 pgoyette pset->size;
312 1.1.1.1.2.2 pgoyette if(region)
313 1.1.1.1.2.2 pgoyette *data = regional_alloc(region, s);
314 1.1.1.1.2.2 pgoyette else *data = malloc(s);
315 1.1.1.1.2.2 pgoyette if(!*data)
316 1.1.1.1.2.2 pgoyette return 0;
317 1.1.1.1.2.2 pgoyette /* copy & decompress */
318 1.1.1.1.2.2 pgoyette if(!parse_rr_copy(pkt, pset, *data)) {
319 1.1.1.1.2.2 pgoyette if(!region) free(*data);
320 1.1.1.1.2.2 pgoyette return 0;
321 1.1.1.1.2.2 pgoyette }
322 1.1.1.1.2.2 pgoyette return 1;
323 1.1.1.1.2.2 pgoyette }
324 1.1.1.1.2.2 pgoyette
325 1.1.1.1.2.2 pgoyette /** get trust value for rrset */
326 1.1.1.1.2.2 pgoyette static enum rrset_trust
327 1.1.1.1.2.2 pgoyette get_rrset_trust(struct msg_parse* msg, struct rrset_parse* rrset)
328 1.1.1.1.2.2 pgoyette {
329 1.1.1.1.2.2 pgoyette uint16_t AA = msg->flags & BIT_AA;
330 1.1.1.1.2.2 pgoyette if(rrset->section == LDNS_SECTION_ANSWER) {
331 1.1.1.1.2.2 pgoyette if(AA) {
332 1.1.1.1.2.2 pgoyette /* RFC2181 says remainder of CNAME chain is nonauth*/
333 1.1.1.1.2.2 pgoyette if(msg->rrset_first &&
334 1.1.1.1.2.2 pgoyette msg->rrset_first->section==LDNS_SECTION_ANSWER
335 1.1.1.1.2.2 pgoyette && msg->rrset_first->type==LDNS_RR_TYPE_CNAME){
336 1.1.1.1.2.2 pgoyette if(rrset == msg->rrset_first)
337 1.1.1.1.2.2 pgoyette return rrset_trust_ans_AA;
338 1.1.1.1.2.2 pgoyette else return rrset_trust_ans_noAA;
339 1.1.1.1.2.2 pgoyette }
340 1.1.1.1.2.2 pgoyette if(msg->rrset_first &&
341 1.1.1.1.2.2 pgoyette msg->rrset_first->section==LDNS_SECTION_ANSWER
342 1.1.1.1.2.2 pgoyette && msg->rrset_first->type==LDNS_RR_TYPE_DNAME){
343 1.1.1.1.2.2 pgoyette if(rrset == msg->rrset_first ||
344 1.1.1.1.2.2 pgoyette rrset == msg->rrset_first->rrset_all_next)
345 1.1.1.1.2.2 pgoyette return rrset_trust_ans_AA;
346 1.1.1.1.2.2 pgoyette else return rrset_trust_ans_noAA;
347 1.1.1.1.2.2 pgoyette }
348 1.1.1.1.2.2 pgoyette return rrset_trust_ans_AA;
349 1.1.1.1.2.2 pgoyette }
350 1.1.1.1.2.2 pgoyette else return rrset_trust_ans_noAA;
351 1.1.1.1.2.2 pgoyette } else if(rrset->section == LDNS_SECTION_AUTHORITY) {
352 1.1.1.1.2.2 pgoyette if(AA) return rrset_trust_auth_AA;
353 1.1.1.1.2.2 pgoyette else return rrset_trust_auth_noAA;
354 1.1.1.1.2.2 pgoyette } else {
355 1.1.1.1.2.2 pgoyette /* addit section */
356 1.1.1.1.2.2 pgoyette if(AA) return rrset_trust_add_AA;
357 1.1.1.1.2.2 pgoyette else return rrset_trust_add_noAA;
358 1.1.1.1.2.2 pgoyette }
359 1.1.1.1.2.2 pgoyette /* NOTREACHED */
360 1.1.1.1.2.2 pgoyette return rrset_trust_none;
361 1.1.1.1.2.2 pgoyette }
362 1.1.1.1.2.2 pgoyette
363 1.1.1.1.2.2 pgoyette int
364 1.1.1.1.2.2 pgoyette parse_copy_decompress_rrset(sldns_buffer* pkt, struct msg_parse* msg,
365 1.1.1.1.2.2 pgoyette struct rrset_parse *pset, struct regional* region,
366 1.1.1.1.2.2 pgoyette struct ub_packed_rrset_key* pk)
367 1.1.1.1.2.2 pgoyette {
368 1.1.1.1.2.2 pgoyette struct packed_rrset_data* data;
369 1.1.1.1.2.2 pgoyette pk->rk.flags = pset->flags;
370 1.1.1.1.2.2 pgoyette pk->rk.dname_len = pset->dname_len;
371 1.1.1.1.2.2 pgoyette if(region)
372 1.1.1.1.2.2 pgoyette pk->rk.dname = (uint8_t*)regional_alloc(
373 1.1.1.1.2.2 pgoyette region, pset->dname_len);
374 1.1.1.1.2.2 pgoyette else pk->rk.dname =
375 1.1.1.1.2.2 pgoyette (uint8_t*)malloc(pset->dname_len);
376 1.1.1.1.2.2 pgoyette if(!pk->rk.dname)
377 1.1.1.1.2.2 pgoyette return 0;
378 1.1.1.1.2.2 pgoyette /** copy & decompress dname */
379 1.1.1.1.2.2 pgoyette dname_pkt_copy(pkt, pk->rk.dname, pset->dname);
380 1.1.1.1.2.2 pgoyette /** copy over type and class */
381 1.1.1.1.2.2 pgoyette pk->rk.type = htons(pset->type);
382 1.1.1.1.2.2 pgoyette pk->rk.rrset_class = pset->rrset_class;
383 1.1.1.1.2.2 pgoyette /** read data part. */
384 1.1.1.1.2.2 pgoyette if(!parse_create_rrset(pkt, pset, &data, region))
385 1.1.1.1.2.2 pgoyette return 0;
386 1.1.1.1.2.2 pgoyette pk->entry.data = (void*)data;
387 1.1.1.1.2.2 pgoyette pk->entry.key = (void*)pk;
388 1.1.1.1.2.2 pgoyette pk->entry.hash = pset->hash;
389 1.1.1.1.2.2 pgoyette data->trust = get_rrset_trust(msg, pset);
390 1.1.1.1.2.2 pgoyette return 1;
391 1.1.1.1.2.2 pgoyette }
392 1.1.1.1.2.2 pgoyette
393 1.1.1.1.2.2 pgoyette /**
394 1.1.1.1.2.2 pgoyette * Copy and decompress rrs
395 1.1.1.1.2.2 pgoyette * @param pkt: the packet for compression pointer resolution.
396 1.1.1.1.2.2 pgoyette * @param msg: the parsed message
397 1.1.1.1.2.2 pgoyette * @param rep: reply info to put rrs into.
398 1.1.1.1.2.2 pgoyette * @param region: if not NULL, used for allocation.
399 1.1.1.1.2.2 pgoyette * @return 0 on failure.
400 1.1.1.1.2.2 pgoyette */
401 1.1.1.1.2.2 pgoyette static int
402 1.1.1.1.2.2 pgoyette parse_copy_decompress(sldns_buffer* pkt, struct msg_parse* msg,
403 1.1.1.1.2.2 pgoyette struct reply_info* rep, struct regional* region)
404 1.1.1.1.2.2 pgoyette {
405 1.1.1.1.2.2 pgoyette size_t i;
406 1.1.1.1.2.2 pgoyette struct rrset_parse *pset = msg->rrset_first;
407 1.1.1.1.2.2 pgoyette struct packed_rrset_data* data;
408 1.1.1.1.2.2 pgoyette log_assert(rep);
409 1.1.1.1.2.2 pgoyette rep->ttl = MAX_TTL;
410 1.1.1.1.2.2 pgoyette rep->security = sec_status_unchecked;
411 1.1.1.1.2.2 pgoyette if(rep->rrset_count == 0)
412 1.1.1.1.2.2 pgoyette rep->ttl = NORR_TTL;
413 1.1.1.1.2.2 pgoyette
414 1.1.1.1.2.2 pgoyette for(i=0; i<rep->rrset_count; i++) {
415 1.1.1.1.2.2 pgoyette if(!parse_copy_decompress_rrset(pkt, msg, pset, region,
416 1.1.1.1.2.2 pgoyette rep->rrsets[i]))
417 1.1.1.1.2.2 pgoyette return 0;
418 1.1.1.1.2.2 pgoyette data = (struct packed_rrset_data*)rep->rrsets[i]->entry.data;
419 1.1.1.1.2.2 pgoyette if(data->ttl < rep->ttl)
420 1.1.1.1.2.2 pgoyette rep->ttl = data->ttl;
421 1.1.1.1.2.2 pgoyette
422 1.1.1.1.2.2 pgoyette pset = pset->rrset_all_next;
423 1.1.1.1.2.2 pgoyette }
424 1.1.1.1.2.2 pgoyette rep->prefetch_ttl = PREFETCH_TTL_CALC(rep->ttl);
425 1.1.1.1.2.2 pgoyette return 1;
426 1.1.1.1.2.2 pgoyette }
427 1.1.1.1.2.2 pgoyette
428 1.1.1.1.2.2 pgoyette int
429 1.1.1.1.2.2 pgoyette parse_create_msg(sldns_buffer* pkt, struct msg_parse* msg,
430 1.1.1.1.2.2 pgoyette struct alloc_cache* alloc, struct query_info* qinf,
431 1.1.1.1.2.2 pgoyette struct reply_info** rep, struct regional* region)
432 1.1.1.1.2.2 pgoyette {
433 1.1.1.1.2.2 pgoyette log_assert(pkt && msg);
434 1.1.1.1.2.2 pgoyette if(!parse_create_qinfo(pkt, msg, qinf, region))
435 1.1.1.1.2.2 pgoyette return 0;
436 1.1.1.1.2.2 pgoyette if(!parse_create_repinfo(msg, rep, region))
437 1.1.1.1.2.2 pgoyette return 0;
438 1.1.1.1.2.2 pgoyette if(!repinfo_alloc_rrset_keys(*rep, alloc, region))
439 1.1.1.1.2.2 pgoyette return 0;
440 1.1.1.1.2.2 pgoyette if(!parse_copy_decompress(pkt, msg, *rep, region))
441 1.1.1.1.2.2 pgoyette return 0;
442 1.1.1.1.2.2 pgoyette return 1;
443 1.1.1.1.2.2 pgoyette }
444 1.1.1.1.2.2 pgoyette
445 1.1.1.1.2.2 pgoyette int reply_info_parse(sldns_buffer* pkt, struct alloc_cache* alloc,
446 1.1.1.1.2.2 pgoyette struct query_info* qinf, struct reply_info** rep,
447 1.1.1.1.2.2 pgoyette struct regional* region, struct edns_data* edns)
448 1.1.1.1.2.2 pgoyette {
449 1.1.1.1.2.2 pgoyette /* use scratch pad region-allocator during parsing. */
450 1.1.1.1.2.2 pgoyette struct msg_parse* msg;
451 1.1.1.1.2.2 pgoyette int ret;
452 1.1.1.1.2.2 pgoyette
453 1.1.1.1.2.2 pgoyette qinf->qname = NULL;
454 1.1.1.1.2.2 pgoyette *rep = NULL;
455 1.1.1.1.2.2 pgoyette if(!(msg = regional_alloc(region, sizeof(*msg)))) {
456 1.1.1.1.2.2 pgoyette return LDNS_RCODE_SERVFAIL;
457 1.1.1.1.2.2 pgoyette }
458 1.1.1.1.2.2 pgoyette memset(msg, 0, sizeof(*msg));
459 1.1.1.1.2.2 pgoyette
460 1.1.1.1.2.2 pgoyette sldns_buffer_set_position(pkt, 0);
461 1.1.1.1.2.2 pgoyette if((ret = parse_packet(pkt, msg, region)) != 0) {
462 1.1.1.1.2.2 pgoyette return ret;
463 1.1.1.1.2.2 pgoyette }
464 1.1.1.1.2.2 pgoyette if((ret = parse_extract_edns(msg, edns, region)) != 0)
465 1.1.1.1.2.2 pgoyette return ret;
466 1.1.1.1.2.2 pgoyette
467 1.1.1.1.2.2 pgoyette /* parse OK, allocate return structures */
468 1.1.1.1.2.2 pgoyette /* this also performs dname decompression */
469 1.1.1.1.2.2 pgoyette if(!parse_create_msg(pkt, msg, alloc, qinf, rep, NULL)) {
470 1.1.1.1.2.2 pgoyette query_info_clear(qinf);
471 1.1.1.1.2.2 pgoyette reply_info_parsedelete(*rep, alloc);
472 1.1.1.1.2.2 pgoyette *rep = NULL;
473 1.1.1.1.2.2 pgoyette return LDNS_RCODE_SERVFAIL;
474 1.1.1.1.2.2 pgoyette }
475 1.1.1.1.2.2 pgoyette return 0;
476 1.1.1.1.2.2 pgoyette }
477 1.1.1.1.2.2 pgoyette
478 1.1.1.1.2.2 pgoyette /** helper compare function to sort in lock order */
479 1.1.1.1.2.2 pgoyette static int
480 1.1.1.1.2.2 pgoyette reply_info_sortref_cmp(const void* a, const void* b)
481 1.1.1.1.2.2 pgoyette {
482 1.1.1.1.2.2 pgoyette struct rrset_ref* x = (struct rrset_ref*)a;
483 1.1.1.1.2.2 pgoyette struct rrset_ref* y = (struct rrset_ref*)b;
484 1.1.1.1.2.2 pgoyette if(x->key < y->key) return -1;
485 1.1.1.1.2.2 pgoyette if(x->key > y->key) return 1;
486 1.1.1.1.2.2 pgoyette return 0;
487 1.1.1.1.2.2 pgoyette }
488 1.1.1.1.2.2 pgoyette
489 1.1.1.1.2.2 pgoyette void
490 1.1.1.1.2.2 pgoyette reply_info_sortref(struct reply_info* rep)
491 1.1.1.1.2.2 pgoyette {
492 1.1.1.1.2.2 pgoyette qsort(&rep->ref[0], rep->rrset_count, sizeof(struct rrset_ref),
493 1.1.1.1.2.2 pgoyette reply_info_sortref_cmp);
494 1.1.1.1.2.2 pgoyette }
495 1.1.1.1.2.2 pgoyette
496 1.1.1.1.2.2 pgoyette void
497 1.1.1.1.2.2 pgoyette reply_info_set_ttls(struct reply_info* rep, time_t timenow)
498 1.1.1.1.2.2 pgoyette {
499 1.1.1.1.2.2 pgoyette size_t i, j;
500 1.1.1.1.2.2 pgoyette rep->ttl += timenow;
501 1.1.1.1.2.2 pgoyette rep->prefetch_ttl += timenow;
502 1.1.1.1.2.2 pgoyette for(i=0; i<rep->rrset_count; i++) {
503 1.1.1.1.2.2 pgoyette struct packed_rrset_data* data = (struct packed_rrset_data*)
504 1.1.1.1.2.2 pgoyette rep->ref[i].key->entry.data;
505 1.1.1.1.2.2 pgoyette if(i>0 && rep->ref[i].key == rep->ref[i-1].key)
506 1.1.1.1.2.2 pgoyette continue;
507 1.1.1.1.2.2 pgoyette data->ttl += timenow;
508 1.1.1.1.2.2 pgoyette for(j=0; j<data->count + data->rrsig_count; j++) {
509 1.1.1.1.2.2 pgoyette data->rr_ttl[j] += timenow;
510 1.1.1.1.2.2 pgoyette }
511 1.1.1.1.2.2 pgoyette }
512 1.1.1.1.2.2 pgoyette }
513 1.1.1.1.2.2 pgoyette
514 1.1.1.1.2.2 pgoyette void
515 1.1.1.1.2.2 pgoyette reply_info_parsedelete(struct reply_info* rep, struct alloc_cache* alloc)
516 1.1.1.1.2.2 pgoyette {
517 1.1.1.1.2.2 pgoyette size_t i;
518 1.1.1.1.2.2 pgoyette if(!rep)
519 1.1.1.1.2.2 pgoyette return;
520 1.1.1.1.2.2 pgoyette /* no need to lock, since not shared in hashtables. */
521 1.1.1.1.2.2 pgoyette for(i=0; i<rep->rrset_count; i++) {
522 1.1.1.1.2.2 pgoyette ub_packed_rrset_parsedelete(rep->rrsets[i], alloc);
523 1.1.1.1.2.2 pgoyette }
524 1.1.1.1.2.2 pgoyette free(rep);
525 1.1.1.1.2.2 pgoyette }
526 1.1.1.1.2.2 pgoyette
527 1.1.1.1.2.2 pgoyette int
528 1.1.1.1.2.2 pgoyette query_info_parse(struct query_info* m, sldns_buffer* query)
529 1.1.1.1.2.2 pgoyette {
530 1.1.1.1.2.2 pgoyette uint8_t* q = sldns_buffer_begin(query);
531 1.1.1.1.2.2 pgoyette /* minimum size: header + \0 + qtype + qclass */
532 1.1.1.1.2.2 pgoyette if(sldns_buffer_limit(query) < LDNS_HEADER_SIZE + 5)
533 1.1.1.1.2.2 pgoyette return 0;
534 1.1.1.1.2.2 pgoyette if(LDNS_OPCODE_WIRE(q) != LDNS_PACKET_QUERY ||
535 1.1.1.1.2.2 pgoyette LDNS_QDCOUNT(q) != 1 || sldns_buffer_position(query) != 0)
536 1.1.1.1.2.2 pgoyette return 0;
537 1.1.1.1.2.2 pgoyette sldns_buffer_skip(query, LDNS_HEADER_SIZE);
538 1.1.1.1.2.2 pgoyette m->qname = sldns_buffer_current(query);
539 1.1.1.1.2.2 pgoyette if((m->qname_len = query_dname_len(query)) == 0)
540 1.1.1.1.2.2 pgoyette return 0; /* parse error */
541 1.1.1.1.2.2 pgoyette if(sldns_buffer_remaining(query) < 4)
542 1.1.1.1.2.2 pgoyette return 0; /* need qtype, qclass */
543 1.1.1.1.2.2 pgoyette m->qtype = sldns_buffer_read_u16(query);
544 1.1.1.1.2.2 pgoyette m->qclass = sldns_buffer_read_u16(query);
545 1.1.1.1.2.2 pgoyette return 1;
546 1.1.1.1.2.2 pgoyette }
547 1.1.1.1.2.2 pgoyette
548 1.1.1.1.2.2 pgoyette /** tiny subroutine for msgreply_compare */
549 1.1.1.1.2.2 pgoyette #define COMPARE_IT(x, y) \
550 1.1.1.1.2.2 pgoyette if( (x) < (y) ) return -1; \
551 1.1.1.1.2.2 pgoyette else if( (x) > (y) ) return +1; \
552 1.1.1.1.2.2 pgoyette log_assert( (x) == (y) );
553 1.1.1.1.2.2 pgoyette
554 1.1.1.1.2.2 pgoyette int
555 1.1.1.1.2.2 pgoyette query_info_compare(void* m1, void* m2)
556 1.1.1.1.2.2 pgoyette {
557 1.1.1.1.2.2 pgoyette struct query_info* msg1 = (struct query_info*)m1;
558 1.1.1.1.2.2 pgoyette struct query_info* msg2 = (struct query_info*)m2;
559 1.1.1.1.2.2 pgoyette int mc;
560 1.1.1.1.2.2 pgoyette /* from most different to least different for speed */
561 1.1.1.1.2.2 pgoyette COMPARE_IT(msg1->qtype, msg2->qtype);
562 1.1.1.1.2.2 pgoyette if((mc = query_dname_compare(msg1->qname, msg2->qname)) != 0)
563 1.1.1.1.2.2 pgoyette return mc;
564 1.1.1.1.2.2 pgoyette log_assert(msg1->qname_len == msg2->qname_len);
565 1.1.1.1.2.2 pgoyette COMPARE_IT(msg1->qclass, msg2->qclass);
566 1.1.1.1.2.2 pgoyette return 0;
567 1.1.1.1.2.2 pgoyette #undef COMPARE_IT
568 1.1.1.1.2.2 pgoyette }
569 1.1.1.1.2.2 pgoyette
570 1.1.1.1.2.2 pgoyette void
571 1.1.1.1.2.2 pgoyette query_info_clear(struct query_info* m)
572 1.1.1.1.2.2 pgoyette {
573 1.1.1.1.2.2 pgoyette free(m->qname);
574 1.1.1.1.2.2 pgoyette m->qname = NULL;
575 1.1.1.1.2.2 pgoyette }
576 1.1.1.1.2.2 pgoyette
577 1.1.1.1.2.2 pgoyette size_t
578 1.1.1.1.2.2 pgoyette msgreply_sizefunc(void* k, void* d)
579 1.1.1.1.2.2 pgoyette {
580 1.1.1.1.2.2 pgoyette struct msgreply_entry* q = (struct msgreply_entry*)k;
581 1.1.1.1.2.2 pgoyette struct reply_info* r = (struct reply_info*)d;
582 1.1.1.1.2.2 pgoyette size_t s = sizeof(struct msgreply_entry) + sizeof(struct reply_info)
583 1.1.1.1.2.2 pgoyette + q->key.qname_len + lock_get_mem(&q->entry.lock)
584 1.1.1.1.2.2 pgoyette - sizeof(struct rrset_ref);
585 1.1.1.1.2.2 pgoyette s += r->rrset_count * sizeof(struct rrset_ref);
586 1.1.1.1.2.2 pgoyette s += r->rrset_count * sizeof(struct ub_packed_rrset_key*);
587 1.1.1.1.2.2 pgoyette return s;
588 1.1.1.1.2.2 pgoyette }
589 1.1.1.1.2.2 pgoyette
590 1.1.1.1.2.2 pgoyette void
591 1.1.1.1.2.2 pgoyette query_entry_delete(void *k, void* ATTR_UNUSED(arg))
592 1.1.1.1.2.2 pgoyette {
593 1.1.1.1.2.2 pgoyette struct msgreply_entry* q = (struct msgreply_entry*)k;
594 1.1.1.1.2.2 pgoyette lock_rw_destroy(&q->entry.lock);
595 1.1.1.1.2.2 pgoyette query_info_clear(&q->key);
596 1.1.1.1.2.2 pgoyette free(q);
597 1.1.1.1.2.2 pgoyette }
598 1.1.1.1.2.2 pgoyette
599 1.1.1.1.2.2 pgoyette void
600 1.1.1.1.2.2 pgoyette reply_info_delete(void* d, void* ATTR_UNUSED(arg))
601 1.1.1.1.2.2 pgoyette {
602 1.1.1.1.2.2 pgoyette struct reply_info* r = (struct reply_info*)d;
603 1.1.1.1.2.2 pgoyette free(r);
604 1.1.1.1.2.2 pgoyette }
605 1.1.1.1.2.2 pgoyette
606 1.1.1.1.2.2 pgoyette hashvalue_t
607 1.1.1.1.2.2 pgoyette query_info_hash(struct query_info *q, uint16_t flags)
608 1.1.1.1.2.2 pgoyette {
609 1.1.1.1.2.2 pgoyette hashvalue_t h = 0xab;
610 1.1.1.1.2.2 pgoyette h = hashlittle(&q->qtype, sizeof(q->qtype), h);
611 1.1.1.1.2.2 pgoyette if(q->qtype == LDNS_RR_TYPE_AAAA && (flags&BIT_CD))
612 1.1.1.1.2.2 pgoyette h++;
613 1.1.1.1.2.2 pgoyette h = hashlittle(&q->qclass, sizeof(q->qclass), h);
614 1.1.1.1.2.2 pgoyette h = dname_query_hash(q->qname, h);
615 1.1.1.1.2.2 pgoyette return h;
616 1.1.1.1.2.2 pgoyette }
617 1.1.1.1.2.2 pgoyette
618 1.1.1.1.2.2 pgoyette struct msgreply_entry*
619 1.1.1.1.2.2 pgoyette query_info_entrysetup(struct query_info* q, struct reply_info* r,
620 1.1.1.1.2.2 pgoyette hashvalue_t h)
621 1.1.1.1.2.2 pgoyette {
622 1.1.1.1.2.2 pgoyette struct msgreply_entry* e = (struct msgreply_entry*)malloc(
623 1.1.1.1.2.2 pgoyette sizeof(struct msgreply_entry));
624 1.1.1.1.2.2 pgoyette if(!e) return NULL;
625 1.1.1.1.2.2 pgoyette memcpy(&e->key, q, sizeof(*q));
626 1.1.1.1.2.2 pgoyette e->entry.hash = h;
627 1.1.1.1.2.2 pgoyette e->entry.key = e;
628 1.1.1.1.2.2 pgoyette e->entry.data = r;
629 1.1.1.1.2.2 pgoyette lock_rw_init(&e->entry.lock);
630 1.1.1.1.2.2 pgoyette lock_protect(&e->entry.lock, &e->key, sizeof(e->key));
631 1.1.1.1.2.2 pgoyette lock_protect(&e->entry.lock, &e->entry.hash, sizeof(e->entry.hash) +
632 1.1.1.1.2.2 pgoyette sizeof(e->entry.key) + sizeof(e->entry.data));
633 1.1.1.1.2.2 pgoyette lock_protect(&e->entry.lock, e->key.qname, e->key.qname_len);
634 1.1.1.1.2.2 pgoyette q->qname = NULL;
635 1.1.1.1.2.2 pgoyette return e;
636 1.1.1.1.2.2 pgoyette }
637 1.1.1.1.2.2 pgoyette
638 1.1.1.1.2.2 pgoyette /** copy rrsets from replyinfo to dest replyinfo */
639 1.1.1.1.2.2 pgoyette static int
640 1.1.1.1.2.2 pgoyette repinfo_copy_rrsets(struct reply_info* dest, struct reply_info* from,
641 1.1.1.1.2.2 pgoyette struct regional* region)
642 1.1.1.1.2.2 pgoyette {
643 1.1.1.1.2.2 pgoyette size_t i, s;
644 1.1.1.1.2.2 pgoyette struct packed_rrset_data* fd, *dd;
645 1.1.1.1.2.2 pgoyette struct ub_packed_rrset_key* fk, *dk;
646 1.1.1.1.2.2 pgoyette for(i=0; i<dest->rrset_count; i++) {
647 1.1.1.1.2.2 pgoyette fk = from->rrsets[i];
648 1.1.1.1.2.2 pgoyette dk = dest->rrsets[i];
649 1.1.1.1.2.2 pgoyette fd = (struct packed_rrset_data*)fk->entry.data;
650 1.1.1.1.2.2 pgoyette dk->entry.hash = fk->entry.hash;
651 1.1.1.1.2.2 pgoyette dk->rk = fk->rk;
652 1.1.1.1.2.2 pgoyette if(region) {
653 1.1.1.1.2.2 pgoyette dk->id = fk->id;
654 1.1.1.1.2.2 pgoyette dk->rk.dname = (uint8_t*)regional_alloc_init(region,
655 1.1.1.1.2.2 pgoyette fk->rk.dname, fk->rk.dname_len);
656 1.1.1.1.2.2 pgoyette } else
657 1.1.1.1.2.2 pgoyette dk->rk.dname = (uint8_t*)memdup(fk->rk.dname,
658 1.1.1.1.2.2 pgoyette fk->rk.dname_len);
659 1.1.1.1.2.2 pgoyette if(!dk->rk.dname)
660 1.1.1.1.2.2 pgoyette return 0;
661 1.1.1.1.2.2 pgoyette s = packed_rrset_sizeof(fd);
662 1.1.1.1.2.2 pgoyette if(region)
663 1.1.1.1.2.2 pgoyette dd = (struct packed_rrset_data*)regional_alloc_init(
664 1.1.1.1.2.2 pgoyette region, fd, s);
665 1.1.1.1.2.2 pgoyette else dd = (struct packed_rrset_data*)memdup(fd, s);
666 1.1.1.1.2.2 pgoyette if(!dd)
667 1.1.1.1.2.2 pgoyette return 0;
668 1.1.1.1.2.2 pgoyette packed_rrset_ptr_fixup(dd);
669 1.1.1.1.2.2 pgoyette dk->entry.data = (void*)dd;
670 1.1.1.1.2.2 pgoyette }
671 1.1.1.1.2.2 pgoyette return 1;
672 1.1.1.1.2.2 pgoyette }
673 1.1.1.1.2.2 pgoyette
674 1.1.1.1.2.2 pgoyette struct reply_info*
675 1.1.1.1.2.2 pgoyette reply_info_copy(struct reply_info* rep, struct alloc_cache* alloc,
676 1.1.1.1.2.2 pgoyette struct regional* region)
677 1.1.1.1.2.2 pgoyette {
678 1.1.1.1.2.2 pgoyette struct reply_info* cp;
679 1.1.1.1.2.2 pgoyette cp = construct_reply_info_base(region, rep->flags, rep->qdcount,
680 1.1.1.1.2.2 pgoyette rep->ttl, rep->prefetch_ttl, rep->an_numrrsets,
681 1.1.1.1.2.2 pgoyette rep->ns_numrrsets, rep->ar_numrrsets, rep->rrset_count,
682 1.1.1.1.2.2 pgoyette rep->security);
683 1.1.1.1.2.2 pgoyette if(!cp)
684 1.1.1.1.2.2 pgoyette return NULL;
685 1.1.1.1.2.2 pgoyette /* allocate ub_key structures special or not */
686 1.1.1.1.2.2 pgoyette if(!repinfo_alloc_rrset_keys(cp, alloc, region)) {
687 1.1.1.1.2.2 pgoyette if(!region)
688 1.1.1.1.2.2 pgoyette reply_info_parsedelete(cp, alloc);
689 1.1.1.1.2.2 pgoyette return NULL;
690 1.1.1.1.2.2 pgoyette }
691 1.1.1.1.2.2 pgoyette if(!repinfo_copy_rrsets(cp, rep, region)) {
692 1.1.1.1.2.2 pgoyette if(!region)
693 1.1.1.1.2.2 pgoyette reply_info_parsedelete(cp, alloc);
694 1.1.1.1.2.2 pgoyette return NULL;
695 1.1.1.1.2.2 pgoyette }
696 1.1.1.1.2.2 pgoyette return cp;
697 1.1.1.1.2.2 pgoyette }
698 1.1.1.1.2.2 pgoyette
699 1.1.1.1.2.2 pgoyette uint8_t*
700 1.1.1.1.2.2 pgoyette reply_find_final_cname_target(struct query_info* qinfo, struct reply_info* rep)
701 1.1.1.1.2.2 pgoyette {
702 1.1.1.1.2.2 pgoyette uint8_t* sname = qinfo->qname;
703 1.1.1.1.2.2 pgoyette size_t snamelen = qinfo->qname_len;
704 1.1.1.1.2.2 pgoyette size_t i;
705 1.1.1.1.2.2 pgoyette for(i=0; i<rep->an_numrrsets; i++) {
706 1.1.1.1.2.2 pgoyette struct ub_packed_rrset_key* s = rep->rrsets[i];
707 1.1.1.1.2.2 pgoyette /* follow CNAME chain (if any) */
708 1.1.1.1.2.2 pgoyette if(ntohs(s->rk.type) == LDNS_RR_TYPE_CNAME &&
709 1.1.1.1.2.2 pgoyette ntohs(s->rk.rrset_class) == qinfo->qclass &&
710 1.1.1.1.2.2 pgoyette snamelen == s->rk.dname_len &&
711 1.1.1.1.2.2 pgoyette query_dname_compare(sname, s->rk.dname) == 0) {
712 1.1.1.1.2.2 pgoyette get_cname_target(s, &sname, &snamelen);
713 1.1.1.1.2.2 pgoyette }
714 1.1.1.1.2.2 pgoyette }
715 1.1.1.1.2.2 pgoyette if(sname != qinfo->qname)
716 1.1.1.1.2.2 pgoyette return sname;
717 1.1.1.1.2.2 pgoyette return NULL;
718 1.1.1.1.2.2 pgoyette }
719 1.1.1.1.2.2 pgoyette
720 1.1.1.1.2.2 pgoyette struct ub_packed_rrset_key*
721 1.1.1.1.2.2 pgoyette reply_find_answer_rrset(struct query_info* qinfo, struct reply_info* rep)
722 1.1.1.1.2.2 pgoyette {
723 1.1.1.1.2.2 pgoyette uint8_t* sname = qinfo->qname;
724 1.1.1.1.2.2 pgoyette size_t snamelen = qinfo->qname_len;
725 1.1.1.1.2.2 pgoyette size_t i;
726 1.1.1.1.2.2 pgoyette for(i=0; i<rep->an_numrrsets; i++) {
727 1.1.1.1.2.2 pgoyette struct ub_packed_rrset_key* s = rep->rrsets[i];
728 1.1.1.1.2.2 pgoyette /* first match type, for query of qtype cname */
729 1.1.1.1.2.2 pgoyette if(ntohs(s->rk.type) == qinfo->qtype &&
730 1.1.1.1.2.2 pgoyette ntohs(s->rk.rrset_class) == qinfo->qclass &&
731 1.1.1.1.2.2 pgoyette snamelen == s->rk.dname_len &&
732 1.1.1.1.2.2 pgoyette query_dname_compare(sname, s->rk.dname) == 0) {
733 1.1.1.1.2.2 pgoyette return s;
734 1.1.1.1.2.2 pgoyette }
735 1.1.1.1.2.2 pgoyette /* follow CNAME chain (if any) */
736 1.1.1.1.2.2 pgoyette if(ntohs(s->rk.type) == LDNS_RR_TYPE_CNAME &&
737 1.1.1.1.2.2 pgoyette ntohs(s->rk.rrset_class) == qinfo->qclass &&
738 1.1.1.1.2.2 pgoyette snamelen == s->rk.dname_len &&
739 1.1.1.1.2.2 pgoyette query_dname_compare(sname, s->rk.dname) == 0) {
740 1.1.1.1.2.2 pgoyette get_cname_target(s, &sname, &snamelen);
741 1.1.1.1.2.2 pgoyette }
742 1.1.1.1.2.2 pgoyette }
743 1.1.1.1.2.2 pgoyette return NULL;
744 1.1.1.1.2.2 pgoyette }
745 1.1.1.1.2.2 pgoyette
746 1.1.1.1.2.2 pgoyette struct ub_packed_rrset_key* reply_find_rrset_section_an(struct reply_info* rep,
747 1.1.1.1.2.2 pgoyette uint8_t* name, size_t namelen, uint16_t type, uint16_t dclass)
748 1.1.1.1.2.2 pgoyette {
749 1.1.1.1.2.2 pgoyette size_t i;
750 1.1.1.1.2.2 pgoyette for(i=0; i<rep->an_numrrsets; i++) {
751 1.1.1.1.2.2 pgoyette struct ub_packed_rrset_key* s = rep->rrsets[i];
752 1.1.1.1.2.2 pgoyette if(ntohs(s->rk.type) == type &&
753 1.1.1.1.2.2 pgoyette ntohs(s->rk.rrset_class) == dclass &&
754 1.1.1.1.2.2 pgoyette namelen == s->rk.dname_len &&
755 1.1.1.1.2.2 pgoyette query_dname_compare(name, s->rk.dname) == 0) {
756 1.1.1.1.2.2 pgoyette return s;
757 1.1.1.1.2.2 pgoyette }
758 1.1.1.1.2.2 pgoyette }
759 1.1.1.1.2.2 pgoyette return NULL;
760 1.1.1.1.2.2 pgoyette }
761 1.1.1.1.2.2 pgoyette
762 1.1.1.1.2.2 pgoyette struct ub_packed_rrset_key* reply_find_rrset_section_ns(struct reply_info* rep,
763 1.1.1.1.2.2 pgoyette uint8_t* name, size_t namelen, uint16_t type, uint16_t dclass)
764 1.1.1.1.2.2 pgoyette {
765 1.1.1.1.2.2 pgoyette size_t i;
766 1.1.1.1.2.2 pgoyette for(i=rep->an_numrrsets; i<rep->an_numrrsets+rep->ns_numrrsets; i++) {
767 1.1.1.1.2.2 pgoyette struct ub_packed_rrset_key* s = rep->rrsets[i];
768 1.1.1.1.2.2 pgoyette if(ntohs(s->rk.type) == type &&
769 1.1.1.1.2.2 pgoyette ntohs(s->rk.rrset_class) == dclass &&
770 1.1.1.1.2.2 pgoyette namelen == s->rk.dname_len &&
771 1.1.1.1.2.2 pgoyette query_dname_compare(name, s->rk.dname) == 0) {
772 1.1.1.1.2.2 pgoyette return s;
773 1.1.1.1.2.2 pgoyette }
774 1.1.1.1.2.2 pgoyette }
775 1.1.1.1.2.2 pgoyette return NULL;
776 1.1.1.1.2.2 pgoyette }
777 1.1.1.1.2.2 pgoyette
778 1.1.1.1.2.2 pgoyette struct ub_packed_rrset_key* reply_find_rrset(struct reply_info* rep,
779 1.1.1.1.2.2 pgoyette uint8_t* name, size_t namelen, uint16_t type, uint16_t dclass)
780 1.1.1.1.2.2 pgoyette {
781 1.1.1.1.2.2 pgoyette size_t i;
782 1.1.1.1.2.2 pgoyette for(i=0; i<rep->rrset_count; i++) {
783 1.1.1.1.2.2 pgoyette struct ub_packed_rrset_key* s = rep->rrsets[i];
784 1.1.1.1.2.2 pgoyette if(ntohs(s->rk.type) == type &&
785 1.1.1.1.2.2 pgoyette ntohs(s->rk.rrset_class) == dclass &&
786 1.1.1.1.2.2 pgoyette namelen == s->rk.dname_len &&
787 1.1.1.1.2.2 pgoyette query_dname_compare(name, s->rk.dname) == 0) {
788 1.1.1.1.2.2 pgoyette return s;
789 1.1.1.1.2.2 pgoyette }
790 1.1.1.1.2.2 pgoyette }
791 1.1.1.1.2.2 pgoyette return NULL;
792 1.1.1.1.2.2 pgoyette }
793 1.1.1.1.2.2 pgoyette
794 1.1.1.1.2.2 pgoyette void
795 1.1.1.1.2.2 pgoyette log_dns_msg(const char* str, struct query_info* qinfo, struct reply_info* rep)
796 1.1.1.1.2.2 pgoyette {
797 1.1.1.1.2.2 pgoyette /* not particularly fast but flexible, make wireformat and print */
798 1.1.1.1.2.2 pgoyette sldns_buffer* buf = sldns_buffer_new(65535);
799 1.1.1.1.2.2 pgoyette struct regional* region = regional_create();
800 1.1.1.1.2.2 pgoyette if(!reply_info_encode(qinfo, rep, 0, rep->flags, buf, 0,
801 1.1.1.1.2.2 pgoyette region, 65535, 1)) {
802 1.1.1.1.2.2 pgoyette log_info("%s: log_dns_msg: out of memory", str);
803 1.1.1.1.2.2 pgoyette } else {
804 1.1.1.1.2.2 pgoyette char* s = sldns_wire2str_pkt(sldns_buffer_begin(buf),
805 1.1.1.1.2.2 pgoyette sldns_buffer_limit(buf));
806 1.1.1.1.2.2 pgoyette if(!s) {
807 1.1.1.1.2.2 pgoyette log_info("%s: log_dns_msg: ldns tostr failed", str);
808 1.1.1.1.2.2 pgoyette } else {
809 1.1.1.1.2.2 pgoyette log_info("%s %s", str, s);
810 1.1.1.1.2.2 pgoyette }
811 1.1.1.1.2.2 pgoyette free(s);
812 1.1.1.1.2.2 pgoyette }
813 1.1.1.1.2.2 pgoyette sldns_buffer_free(buf);
814 1.1.1.1.2.2 pgoyette regional_destroy(region);
815 1.1.1.1.2.2 pgoyette }
816 1.1.1.1.2.2 pgoyette
817 1.1.1.1.2.2 pgoyette void
818 1.1.1.1.2.2 pgoyette log_query_info(enum verbosity_value v, const char* str,
819 1.1.1.1.2.2 pgoyette struct query_info* qinf)
820 1.1.1.1.2.2 pgoyette {
821 1.1.1.1.2.2 pgoyette log_nametypeclass(v, str, qinf->qname, qinf->qtype, qinf->qclass);
822 1.1.1.1.2.2 pgoyette }
823 1.1.1.1.2.2 pgoyette
824 1.1.1.1.2.2 pgoyette int
825 1.1.1.1.2.2 pgoyette reply_check_cname_chain(struct query_info* qinfo, struct reply_info* rep)
826 1.1.1.1.2.2 pgoyette {
827 1.1.1.1.2.2 pgoyette /* check only answer section rrs for matching cname chain.
828 1.1.1.1.2.2 pgoyette * the cache may return changed rdata, but owner names are untouched.*/
829 1.1.1.1.2.2 pgoyette size_t i;
830 1.1.1.1.2.2 pgoyette uint8_t* sname = qinfo->qname;
831 1.1.1.1.2.2 pgoyette size_t snamelen = qinfo->qname_len;
832 1.1.1.1.2.2 pgoyette for(i=0; i<rep->an_numrrsets; i++) {
833 1.1.1.1.2.2 pgoyette uint16_t t = ntohs(rep->rrsets[i]->rk.type);
834 1.1.1.1.2.2 pgoyette if(t == LDNS_RR_TYPE_DNAME)
835 1.1.1.1.2.2 pgoyette continue; /* skip dnames; note TTL 0 not cached */
836 1.1.1.1.2.2 pgoyette /* verify that owner matches current sname */
837 1.1.1.1.2.2 pgoyette if(query_dname_compare(sname, rep->rrsets[i]->rk.dname) != 0){
838 1.1.1.1.2.2 pgoyette /* cname chain broken */
839 1.1.1.1.2.2 pgoyette return 0;
840 1.1.1.1.2.2 pgoyette }
841 1.1.1.1.2.2 pgoyette /* if this is a cname; move on */
842 1.1.1.1.2.2 pgoyette if(t == LDNS_RR_TYPE_CNAME) {
843 1.1.1.1.2.2 pgoyette get_cname_target(rep->rrsets[i], &sname, &snamelen);
844 1.1.1.1.2.2 pgoyette }
845 1.1.1.1.2.2 pgoyette }
846 1.1.1.1.2.2 pgoyette return 1;
847 1.1.1.1.2.2 pgoyette }
848 1.1.1.1.2.2 pgoyette
849 1.1.1.1.2.2 pgoyette int
850 1.1.1.1.2.2 pgoyette reply_all_rrsets_secure(struct reply_info* rep)
851 1.1.1.1.2.2 pgoyette {
852 1.1.1.1.2.2 pgoyette size_t i;
853 1.1.1.1.2.2 pgoyette for(i=0; i<rep->rrset_count; i++) {
854 1.1.1.1.2.2 pgoyette if( ((struct packed_rrset_data*)rep->rrsets[i]->entry.data)
855 1.1.1.1.2.2 pgoyette ->security != sec_status_secure )
856 1.1.1.1.2.2 pgoyette return 0;
857 1.1.1.1.2.2 pgoyette }
858 1.1.1.1.2.2 pgoyette return 1;
859 1.1.1.1.2.2 pgoyette }
860 1.1.1.1.2.2 pgoyette
861 1.1.1.1.2.2 pgoyette int edns_opt_append(struct edns_data* edns, struct regional* region,
862 1.1.1.1.2.2 pgoyette uint16_t code, size_t len, uint8_t* data)
863 1.1.1.1.2.2 pgoyette {
864 1.1.1.1.2.2 pgoyette struct edns_option** prevp;
865 1.1.1.1.2.2 pgoyette struct edns_option* opt;
866 1.1.1.1.2.2 pgoyette
867 1.1.1.1.2.2 pgoyette /* allocate new element */
868 1.1.1.1.2.2 pgoyette opt = (struct edns_option*)regional_alloc(region, sizeof(*opt));
869 1.1.1.1.2.2 pgoyette if(!opt)
870 1.1.1.1.2.2 pgoyette return 0;
871 1.1.1.1.2.2 pgoyette opt->next = NULL;
872 1.1.1.1.2.2 pgoyette opt->opt_code = code;
873 1.1.1.1.2.2 pgoyette opt->opt_len = len;
874 1.1.1.1.2.2 pgoyette opt->opt_data = regional_alloc_init(region, data, len);
875 1.1.1.1.2.2 pgoyette if(!opt->opt_data)
876 1.1.1.1.2.2 pgoyette return 0;
877 1.1.1.1.2.2 pgoyette
878 1.1.1.1.2.2 pgoyette /* append at end of list */
879 1.1.1.1.2.2 pgoyette prevp = &edns->opt_list;
880 1.1.1.1.2.2 pgoyette while(*prevp != NULL)
881 1.1.1.1.2.2 pgoyette prevp = &((*prevp)->next);
882 1.1.1.1.2.2 pgoyette *prevp = opt;
883 1.1.1.1.2.2 pgoyette return 1;
884 1.1.1.1.2.2 pgoyette }
885 1.1.1.1.2.2 pgoyette
886 1.1.1.1.2.2 pgoyette int edns_opt_inplace_reply(struct edns_data* edns, struct regional* region)
887 1.1.1.1.2.2 pgoyette {
888 1.1.1.1.2.2 pgoyette (void)region;
889 1.1.1.1.2.2 pgoyette /* remove all edns options from the reply, because only the
890 1.1.1.1.2.2 pgoyette * options that we understand should be in the reply
891 1.1.1.1.2.2 pgoyette * (sec 6.1.2 RFC 6891) */
892 1.1.1.1.2.2 pgoyette edns->opt_list = NULL;
893 1.1.1.1.2.2 pgoyette return 1;
894 1.1.1.1.2.2 pgoyette }
895 1.1.1.1.2.2 pgoyette
896 1.1.1.1.2.2 pgoyette struct edns_option* edns_opt_copy_region(struct edns_option* list,
897 1.1.1.1.2.2 pgoyette struct regional* region)
898 1.1.1.1.2.2 pgoyette {
899 1.1.1.1.2.2 pgoyette struct edns_option* result = NULL, *cur = NULL, *s;
900 1.1.1.1.2.2 pgoyette while(list) {
901 1.1.1.1.2.2 pgoyette /* copy edns option structure */
902 1.1.1.1.2.2 pgoyette s = regional_alloc_init(region, list, sizeof(*list));
903 1.1.1.1.2.2 pgoyette if(!s) return NULL;
904 1.1.1.1.2.2 pgoyette s->next = NULL;
905 1.1.1.1.2.2 pgoyette
906 1.1.1.1.2.2 pgoyette /* copy option data */
907 1.1.1.1.2.2 pgoyette if(s->opt_data) {
908 1.1.1.1.2.2 pgoyette s->opt_data = regional_alloc_init(region, s->opt_data,
909 1.1.1.1.2.2 pgoyette s->opt_len);
910 1.1.1.1.2.2 pgoyette if(!s->opt_data)
911 1.1.1.1.2.2 pgoyette return NULL;
912 1.1.1.1.2.2 pgoyette }
913 1.1.1.1.2.2 pgoyette
914 1.1.1.1.2.2 pgoyette /* link into list */
915 1.1.1.1.2.2 pgoyette if(cur)
916 1.1.1.1.2.2 pgoyette cur->next = s;
917 1.1.1.1.2.2 pgoyette else result = s;
918 1.1.1.1.2.2 pgoyette cur = s;
919 1.1.1.1.2.2 pgoyette
920 1.1.1.1.2.2 pgoyette /* examine next element */
921 1.1.1.1.2.2 pgoyette list = list->next;
922 1.1.1.1.2.2 pgoyette }
923 1.1.1.1.2.2 pgoyette return result;
924 1.1.1.1.2.2 pgoyette }
925 1.1.1.1.2.2 pgoyette
926 1.1.1.1.2.2 pgoyette int edns_opt_compare(struct edns_option* p, struct edns_option* q)
927 1.1.1.1.2.2 pgoyette {
928 1.1.1.1.2.2 pgoyette if(!p && !q) return 0;
929 1.1.1.1.2.2 pgoyette if(!p) return -1;
930 1.1.1.1.2.2 pgoyette if(!q) return 1;
931 1.1.1.1.2.2 pgoyette log_assert(p && q);
932 1.1.1.1.2.2 pgoyette if(p->opt_code != q->opt_code)
933 1.1.1.1.2.2 pgoyette return (int)q->opt_code - (int)p->opt_code;
934 1.1.1.1.2.2 pgoyette if(p->opt_len != q->opt_len)
935 1.1.1.1.2.2 pgoyette return (int)q->opt_len - (int)p->opt_len;
936 1.1.1.1.2.2 pgoyette if(p->opt_len != 0)
937 1.1.1.1.2.2 pgoyette return memcmp(p->opt_data, q->opt_data, p->opt_len);
938 1.1.1.1.2.2 pgoyette return 0;
939 1.1.1.1.2.2 pgoyette }
940 1.1.1.1.2.2 pgoyette
941 1.1.1.1.2.2 pgoyette int edns_opt_list_compare(struct edns_option* p, struct edns_option* q)
942 1.1.1.1.2.2 pgoyette {
943 1.1.1.1.2.2 pgoyette int r;
944 1.1.1.1.2.2 pgoyette while(p && q) {
945 1.1.1.1.2.2 pgoyette r = edns_opt_compare(p, q);
946 1.1.1.1.2.2 pgoyette if(r != 0)
947 1.1.1.1.2.2 pgoyette return r;
948 1.1.1.1.2.2 pgoyette p = p->next;
949 1.1.1.1.2.2 pgoyette q = q->next;
950 1.1.1.1.2.2 pgoyette }
951 1.1.1.1.2.2 pgoyette if(p || q) {
952 1.1.1.1.2.2 pgoyette /* uneven length lists */
953 1.1.1.1.2.2 pgoyette if(p) return 1;
954 1.1.1.1.2.2 pgoyette if(q) return -1;
955 1.1.1.1.2.2 pgoyette }
956 1.1.1.1.2.2 pgoyette return 0;
957 1.1.1.1.2.2 pgoyette }
958 1.1.1.1.2.2 pgoyette
959 1.1.1.1.2.2 pgoyette void edns_opt_list_free(struct edns_option* list)
960 1.1.1.1.2.2 pgoyette {
961 1.1.1.1.2.2 pgoyette struct edns_option* n;
962 1.1.1.1.2.2 pgoyette while(list) {
963 1.1.1.1.2.2 pgoyette free(list->opt_data);
964 1.1.1.1.2.2 pgoyette n = list->next;
965 1.1.1.1.2.2 pgoyette free(list);
966 1.1.1.1.2.2 pgoyette list = n;
967 1.1.1.1.2.2 pgoyette }
968 1.1.1.1.2.2 pgoyette }
969 1.1.1.1.2.2 pgoyette
970 1.1.1.1.2.2 pgoyette struct edns_option* edns_opt_copy_alloc(struct edns_option* list)
971 1.1.1.1.2.2 pgoyette {
972 1.1.1.1.2.2 pgoyette struct edns_option* result = NULL, *cur = NULL, *s;
973 1.1.1.1.2.2 pgoyette while(list) {
974 1.1.1.1.2.2 pgoyette /* copy edns option structure */
975 1.1.1.1.2.2 pgoyette s = memdup(list, sizeof(*list));
976 1.1.1.1.2.2 pgoyette if(!s) {
977 1.1.1.1.2.2 pgoyette edns_opt_list_free(result);
978 1.1.1.1.2.2 pgoyette return NULL;
979 1.1.1.1.2.2 pgoyette }
980 1.1.1.1.2.2 pgoyette s->next = NULL;
981 1.1.1.1.2.2 pgoyette
982 1.1.1.1.2.2 pgoyette /* copy option data */
983 1.1.1.1.2.2 pgoyette if(s->opt_data) {
984 1.1.1.1.2.2 pgoyette s->opt_data = memdup(s->opt_data, s->opt_len);
985 1.1.1.1.2.2 pgoyette if(!s->opt_data) {
986 1.1.1.1.2.2 pgoyette edns_opt_list_free(result);
987 1.1.1.1.2.2 pgoyette return NULL;
988 1.1.1.1.2.2 pgoyette }
989 1.1.1.1.2.2 pgoyette }
990 1.1.1.1.2.2 pgoyette
991 1.1.1.1.2.2 pgoyette /* link into list */
992 1.1.1.1.2.2 pgoyette if(cur)
993 1.1.1.1.2.2 pgoyette cur->next = s;
994 1.1.1.1.2.2 pgoyette else result = s;
995 1.1.1.1.2.2 pgoyette cur = s;
996 1.1.1.1.2.2 pgoyette
997 1.1.1.1.2.2 pgoyette /* examine next element */
998 1.1.1.1.2.2 pgoyette list = list->next;
999 1.1.1.1.2.2 pgoyette }
1000 1.1.1.1.2.2 pgoyette return result;
1001 1.1.1.1.2.2 pgoyette }
1002 1.1.1.1.2.2 pgoyette
1003 1.1.1.1.2.2 pgoyette struct edns_option* edns_opt_find(struct edns_option* list, uint16_t code)
1004 1.1.1.1.2.2 pgoyette {
1005 1.1.1.1.2.2 pgoyette struct edns_option* p;
1006 1.1.1.1.2.2 pgoyette for(p=list; p; p=p->next) {
1007 1.1.1.1.2.2 pgoyette if(p->opt_code == code)
1008 1.1.1.1.2.2 pgoyette return p;
1009 1.1.1.1.2.2 pgoyette }
1010 1.1.1.1.2.2 pgoyette return NULL;
1011 1.1.1.1.2.2 pgoyette }
1012