OLD | NEW |
---|---|
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "net/dns/dns_response.h" | 5 #include "net/dns/dns_response.h" |
6 | 6 |
7 #include "base/string_util.h" | 7 #include "base/string_util.h" |
8 #include "base/sys_byteorder.h" | 8 #include "base/sys_byteorder.h" |
9 #include "net/base/address_list.h" | 9 #include "net/base/address_list.h" |
10 #include "net/base/big_endian.h" | 10 #include "net/base/big_endian.h" |
(...skipping 157 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
168 return false; | 168 return false; |
169 } | 169 } |
170 | 170 |
171 // Construct the parser. | 171 // Construct the parser. |
172 parser_ = DnsRecordParser(io_buffer_->data(), | 172 parser_ = DnsRecordParser(io_buffer_->data(), |
173 nbytes, | 173 nbytes, |
174 hdr_size + question.size()); | 174 hdr_size + question.size()); |
175 return true; | 175 return true; |
176 } | 176 } |
177 | 177 |
178 bool DnsResponse::InitParse(int nbytes) { | |
179 if (nbytes >= io_buffer_->size()) | |
180 return false; | |
181 | |
182 // Construct the parser. | |
183 size_t response_offset = GetResponseOffset(); | |
184 if (response_offset == 0) | |
185 return false; | |
186 | |
187 parser_ = DnsRecordParser( | |
188 io_buffer_->data(), nbytes, response_offset); | |
189 return true; | |
190 } | |
191 | |
178 bool DnsResponse::IsValid() const { | 192 bool DnsResponse::IsValid() const { |
179 return parser_.IsValid(); | 193 return parser_.IsValid(); |
180 } | 194 } |
181 | 195 |
182 uint16 DnsResponse::flags() const { | 196 uint16 DnsResponse::flags() const { |
183 DCHECK(parser_.IsValid()); | 197 DCHECK(parser_.IsValid()); |
184 return base::NetToHost16(header()->flags) & ~(dns_protocol::kRcodeMask); | 198 return base::NetToHost16(header()->flags) & ~(dns_protocol::kRcodeMask); |
185 } | 199 } |
186 | 200 |
187 uint8 DnsResponse::rcode() const { | 201 uint8 DnsResponse::rcode() const { |
(...skipping 22 matching lines...) Expand all Loading... | |
210 const size_t type_offset = parser_.GetOffset() - 2 * sizeof(uint16); | 224 const size_t type_offset = parser_.GetOffset() - 2 * sizeof(uint16); |
211 uint16 type; | 225 uint16 type; |
212 ReadBigEndian<uint16>(io_buffer_->data() + type_offset, &type); | 226 ReadBigEndian<uint16>(io_buffer_->data() + type_offset, &type); |
213 return type; | 227 return type; |
214 } | 228 } |
215 | 229 |
216 std::string DnsResponse::GetDottedName() const { | 230 std::string DnsResponse::GetDottedName() const { |
217 return DNSDomainToString(qname()); | 231 return DNSDomainToString(qname()); |
218 } | 232 } |
219 | 233 |
234 size_t DnsResponse::GetResponseOffset() const { | |
szym
2013/04/17 22:11:12
Use DnsRecordParser instead. ReadName with |out| s
Noam Samuel
2013/04/18 19:03:17
Done.
| |
235 uint16 qdcount = base::NetToHost16(header()->qdcount); | |
236 const char* data = io_buffer_->data(); | |
237 size_t data_size = io_buffer_->size(); | |
238 size_t pos = sizeof(dns_protocol::Header); | |
239 | |
240 for (uint16 i = 0; i < qdcount; i++) { | |
241 while (pos < data_size && data[pos] != 0) { | |
242 unsigned txt_len = static_cast<unsigned>(data[pos]); | |
243 // Traversal and checks adapted from DNSDomainToString. | |
244 | |
245 // Handling pointers, which have both high bits set and an extra | |
246 // byte trailing them to indicate a location in the buffer. | |
247 if (txt_len >= 192) { | |
248 pos += 2; | |
249 continue; | |
szym
2013/04/17 22:11:12
This is an error. At this point you should be done
Noam Samuel
2013/04/18 19:03:17
Made irrelevant.
On 2013/04/17 22:11:12, szym wro
| |
250 } | |
251 | |
252 if (txt_len > 63) { | |
253 return 0; | |
254 } | |
255 | |
256 pos += txt_len + 1; | |
257 } | |
258 | |
259 pos += 5; // 1 for 0-length text, 2 for qtype, 2 for qclass | |
260 } | |
261 | |
262 if (pos > data_size) return 0; | |
263 | |
264 return pos; | |
265 } | |
266 | |
220 DnsRecordParser DnsResponse::Parser() const { | 267 DnsRecordParser DnsResponse::Parser() const { |
221 DCHECK(parser_.IsValid()); | 268 DCHECK(parser_.IsValid()); |
222 // Return a copy of the parser. | 269 // Return a copy of the parser. |
223 return parser_; | 270 return parser_; |
224 } | 271 } |
225 | 272 |
226 const dns_protocol::Header* DnsResponse::header() const { | 273 const dns_protocol::Header* DnsResponse::header() const { |
227 return reinterpret_cast<const dns_protocol::Header*>(io_buffer_->data()); | 274 return reinterpret_cast<const dns_protocol::Header*>(io_buffer_->data()); |
228 } | 275 } |
229 | 276 |
(...skipping 56 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
286 | 333 |
287 // getcanonname in eglibc returns the first owner name of an A or AAAA RR. | 334 // getcanonname in eglibc returns the first owner name of an A or AAAA RR. |
288 // If the response passed all the checks so far, then |expected_name| is it. | 335 // If the response passed all the checks so far, then |expected_name| is it. |
289 *addr_list = AddressList::CreateFromIPAddressList(ip_addresses, | 336 *addr_list = AddressList::CreateFromIPAddressList(ip_addresses, |
290 expected_name); | 337 expected_name); |
291 *ttl = base::TimeDelta::FromSeconds(ttl_sec); | 338 *ttl = base::TimeDelta::FromSeconds(ttl_sec); |
292 return DNS_PARSE_OK; | 339 return DNS_PARSE_OK; |
293 } | 340 } |
294 | 341 |
295 } // namespace net | 342 } // namespace net |
OLD | NEW |