// Copyright (c) 2011 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "net/dns/dns_query.h" #include #include "net/base/dns_util.h" #include "net/base/io_buffer.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(buf[0]) << 8 | static_cast(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(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(); } DnsQuery::~DnsQuery() { } uint16 DnsQuery::id() const { return UnpackUint16BE(&io_buffer_->data()[0]); } uint16 DnsQuery::qtype() const { return UnpackUint16BE(&io_buffer_->data()[kHeaderSize + qname_size_]); } DnsQuery* DnsQuery::CloneWithNewId() const { return new DnsQuery(qname(), qtype(), rand_int_cb_); } size_t DnsQuery::question_size() const { return qname_size_ // QNAME + sizeof(uint16) // QTYPE + sizeof(uint16); // QCLASS } const char* DnsQuery::question_data() const { return &io_buffer_->data()[kHeaderSize]; } const std::string DnsQuery::qname() const { return std::string(question_data(), qname_size_); } void DnsQuery::RandomizeId() { PackUint16BE(&io_buffer_->data()[0], rand_int_cb_.Run( std::numeric_limits::min(), std::numeric_limits::max())); } } // namespace net