Chromium Code Reviews| Index: net/dns/dns_query.cc |
| diff --git a/net/dns/dns_query.cc b/net/dns/dns_query.cc |
| index 788d653a304cab9a1297920cc6108cc4b206836c..6ddd152cab3620141e326b739dede340a07b8656 100644 |
| --- a/net/dns/dns_query.cc |
| +++ b/net/dns/dns_query.cc |
| @@ -6,91 +6,78 @@ |
| #include <limits> |
| +#include "net/base/big_endian.h" |
| #include "net/base/dns_util.h" |
| #include "net/base/io_buffer.h" |
| +#include "net/base/sys_byteorder.h" |
| +#include "net/dns/dns_protocol.h" |
| namespace net { |
| -namespace { |
| - |
| -void PackUint16BE(char buf[2], uint16 v) { |
| - buf[0] = v >> 8; |
| - buf[1] = v & 0xff; |
| -} |
| - |
| -uint16 UnpackUint16BE(char buf[2]) { |
| - return static_cast<uint8>(buf[0]) << 8 | static_cast<uint8>(buf[1]); |
| -} |
| - |
| -} // namespace |
| - |
| // DNS query consists of a 12-byte header followed by a question section. |
| // For details, see RFC 1035 section 4.1.1. This header template sets RD |
| // bit, which directs the name server to pursue query recursively, and sets |
| -// the QDCOUNT to 1, meaning the question section has a single entry. The |
| -// first two bytes of the header form a 16-bit random query ID to be copied |
| -// in the corresponding reply by the name server -- randomized during |
| -// DnsQuery construction. |
| -static const char kHeader[] = {0x00, 0x00, 0x01, 0x00, 0x00, 0x01, |
| - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; |
| -static const size_t kHeaderSize = arraysize(kHeader); |
| - |
| -DnsQuery::DnsQuery(const std::string& qname, |
| - uint16 qtype, |
| - const RandIntCallback& rand_int_cb) |
| - : qname_size_(qname.size()), |
| - rand_int_cb_(rand_int_cb) { |
| - DCHECK(DnsResponseBuffer(reinterpret_cast<const uint8*>(qname.c_str()), |
| - qname.size()).DNSName(NULL)); |
| - DCHECK(qtype == kDNS_A || qtype == kDNS_AAAA); |
| - |
| - io_buffer_ = new IOBufferWithSize(kHeaderSize + question_size()); |
| - |
| - int byte_offset = 0; |
| - char* buffer_head = io_buffer_->data(); |
| - memcpy(&buffer_head[byte_offset], kHeader, kHeaderSize); |
| - byte_offset += kHeaderSize; |
| - memcpy(&buffer_head[byte_offset], &qname[0], qname_size_); |
| - byte_offset += qname_size_; |
| - PackUint16BE(&buffer_head[byte_offset], qtype); |
| - byte_offset += sizeof(qtype); |
| - PackUint16BE(&buffer_head[byte_offset], kClassIN); |
| - RandomizeId(); |
| +// the QDCOUNT to 1, meaning the question section has a single entry. |
| +DnsQuery::DnsQuery(uint16 id, const base::StringPiece& qname, uint16 qtype) |
| + : qname_size_(qname.size()) { |
| + DCHECK(!DNSDomainToString(qname).empty()); |
| + // QNAME + QTYPE + QCLASS |
| + size_t question_size = qname_size_ + sizeof(uint16) + sizeof(uint16); |
| + io_buffer_ = new IOBufferWithSize(sizeof(dns_protocol::Header) + |
| + question_size); |
| + dns_protocol::Header* header = |
| + reinterpret_cast<dns_protocol::Header*>(io_buffer_->data()); |
| + memset(header, 0, sizeof(dns_protocol::Header)); |
| + header->id = htons(id); |
| + header->flags[0] = 0x1; // RD bit |
| + header->qdcount = htons(1); |
| + |
| + // Write question section after the header. |
| + BigEndianWriter writer(reinterpret_cast<char*>(header + 1), question_size); |
| + writer.WriteBytes(qname.data(), qname.size()); |
| + writer.WriteU16(qtype); |
| + writer.WriteU16(dns_protocol::kClassIN); |
| } |
| DnsQuery::~DnsQuery() { |
| } |
| -uint16 DnsQuery::id() const { |
| - return UnpackUint16BE(&io_buffer_->data()[0]); |
| +DnsQuery* DnsQuery::CloneWithNewId(uint16 id) const { |
| + return new DnsQuery(*this, id); |
| } |
| -uint16 DnsQuery::qtype() const { |
| - return UnpackUint16BE(&io_buffer_->data()[kHeaderSize + qname_size_]); |
| -} |
| - |
| -DnsQuery* DnsQuery::CloneWithNewId() const { |
| - return new DnsQuery(qname(), qtype(), rand_int_cb_); |
| +uint16 DnsQuery::id() const { |
| + const dns_protocol::Header* h = |
|
mmenke
2011/12/06 20:43:01
nit: Rename variable to header or something.
szym
2011/12/06 21:06:43
Done. (I thought I caught all of them.)
|
| + reinterpret_cast<const dns_protocol::Header*>(io_buffer_->data()); |
| + return ntohs(h->id); |
| } |
| -size_t DnsQuery::question_size() const { |
| - return qname_size_ // QNAME |
| - + sizeof(uint16) // QTYPE |
| - + sizeof(uint16); // QCLASS |
| +base::StringPiece DnsQuery::qname() const { |
| + return base::StringPiece(io_buffer_->data() + sizeof(dns_protocol::Header), |
| + qname_size_); |
| } |
| -const char* DnsQuery::question_data() const { |
| - return &io_buffer_->data()[kHeaderSize]; |
| +uint16 DnsQuery::qtype() const { |
| + uint16 type; |
| + ReadBigEndian<uint16>(io_buffer_->data() + |
| + sizeof(dns_protocol::Header) + |
| + qname_size_, &type); |
| + return type; |
| } |
| -const std::string DnsQuery::qname() const { |
| - return std::string(question_data(), qname_size_); |
| +base::StringPiece DnsQuery::question() const { |
| + return base::StringPiece(io_buffer_->data() + sizeof(dns_protocol::Header), |
| + qname_size_ + sizeof(uint16) + sizeof(uint16)); |
| } |
| -void DnsQuery::RandomizeId() { |
| - PackUint16BE(&io_buffer_->data()[0], rand_int_cb_.Run( |
| - std::numeric_limits<uint16>::min(), |
| - std::numeric_limits<uint16>::max())); |
| +DnsQuery::DnsQuery(const DnsQuery& orig, uint16 id) { |
| + qname_size_ = orig.qname_size_; |
| + io_buffer_ = new IOBufferWithSize(orig.io_buffer()->size()); |
| + memcpy(io_buffer_.get()->data(), orig.io_buffer()->data(), |
| + io_buffer_.get()->size()); |
| + dns_protocol::Header* header = |
| + reinterpret_cast<dns_protocol::Header*>(io_buffer_->data()); |
| + header->id = htons(id); |
| } |
| } // namespace net |