blob: 5c1b5ca5b89bfd79dc60f4aae2bdfa55afed9bde [file] [log] [blame]
Jason Jeremy Imana21be272020-10-21 17:53:45 +09001// Copyright 2020 The Chromium OS Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "patchpanel/dns/dns_query.h"
6
7#include <utility>
8
9#include "base/big_endian.h"
10#include "base/logging.h"
11#include "base/memory/ptr_util.h"
12#include "base/numerics/safe_conversions.h"
13#include "base/optional.h"
14#include "base/sys_byteorder.h"
15
16#include "patchpanel/dns/dns_protocol.h"
17#include "patchpanel/dns/dns_util.h"
18#include "patchpanel/dns/io_buffer.h"
19
Qijiang Fan713061e2021-03-08 15:45:12 +090020#include <base/check.h>
21
Jason Jeremy Imana21be272020-10-21 17:53:45 +090022namespace patchpanel {
23
24namespace {
25
26const size_t kHeaderSize = sizeof(dns_protocol::Header);
27
28size_t QuestionSize(size_t qname_size) {
29 // QNAME + QTYPE + QCLASS
30 return qname_size + sizeof(uint16_t) + sizeof(uint16_t);
31}
32
33} // namespace
34
35DnsQuery::DnsQuery(scoped_refptr<IOBufferWithSize> buffer)
36 : io_buffer_(std::move(buffer)) {}
37
38DnsQuery::~DnsQuery() = default;
39
40bool DnsQuery::Parse(size_t valid_bytes) {
41 if (io_buffer_ == nullptr || io_buffer_->data() == nullptr) {
42 return false;
43 }
44 CHECK(valid_bytes <= base::checked_cast<size_t>(io_buffer_->size()));
45 // We should only parse the query once if the query is constructed from a raw
46 // buffer. If we have constructed the query from data or the query is already
47 // parsed after constructed from a raw buffer, |header_| is not null.
48 DCHECK(header_ == nullptr);
49 base::BigEndianReader reader(io_buffer_->data(), valid_bytes);
50 dns_protocol::Header header;
51 if (!ReadHeader(&reader, &header)) {
52 return false;
53 }
54 if (header.flags & dns_protocol::kFlagResponse) {
55 return false;
56 }
57 if (header.qdcount > 1) {
58 LOG(ERROR) << "Not supporting parsing a DNS query with multiple questions.";
59 return false;
60 }
61 std::string qname;
62 if (!ReadName(&reader, &qname)) {
63 return false;
64 }
65 uint16_t qtype;
66 uint16_t qclass;
67 if (!reader.ReadU16(&qtype) || !reader.ReadU16(&qclass) ||
68 qclass != dns_protocol::kClassIN) {
69 return false;
70 }
71 // |io_buffer_| now contains the raw packet of a valid DNS query, we just
72 // need to properly initialize |qname_size_| and |header_|.
73 qname_size_ = qname.size();
74 header_ = reinterpret_cast<dns_protocol::Header*>(io_buffer_->data());
75 return true;
76}
77
78uint16_t DnsQuery::id() const {
79 return base::NetToHost16(header_->id);
80}
81
82base::StringPiece DnsQuery::qname() const {
83 return base::StringPiece(io_buffer_->data() + kHeaderSize, qname_size_);
84}
85
86uint16_t DnsQuery::qtype() const {
87 uint16_t type;
88 base::ReadBigEndian<uint16_t>(io_buffer_->data() + kHeaderSize + qname_size_,
89 &type);
90 return type;
91}
92
93base::StringPiece DnsQuery::question() const {
94 return base::StringPiece(io_buffer_->data() + kHeaderSize,
95 QuestionSize(qname_size_));
96}
97
98size_t DnsQuery::question_size() const {
99 return QuestionSize(qname_size_);
100}
101
102bool DnsQuery::ReadHeader(base::BigEndianReader* reader,
103 dns_protocol::Header* header) {
104 return (
105 reader->ReadU16(&header->id) && reader->ReadU16(&header->flags) &&
106 reader->ReadU16(&header->qdcount) && reader->ReadU16(&header->ancount) &&
107 reader->ReadU16(&header->nscount) && reader->ReadU16(&header->arcount));
108}
109
110bool DnsQuery::ReadName(base::BigEndianReader* reader, std::string* out) {
111 DCHECK(out != nullptr);
112 out->clear();
113 out->reserve(dns_protocol::kMaxNameLength);
114 uint8_t label_length;
115 if (!reader->ReadU8(&label_length)) {
116 return false;
117 }
118 out->append(reinterpret_cast<char*>(&label_length), 1);
119 while (label_length) {
120 base::StringPiece label;
121 if (!reader->ReadPiece(&label, label_length)) {
122 return false;
123 }
124 out->append(label.data(), label.size());
125 if (!reader->ReadU8(&label_length)) {
126 return false;
127 }
128 out->append(reinterpret_cast<char*>(&label_length), 1);
129 }
130 return true;
131}
132
133} // namespace patchpanel