1*6777b538SAndroid Build Coastguard Worker // Copyright 2012 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker
5*6777b538SAndroid Build Coastguard Worker #include "net/dns/dns_query.h"
6*6777b538SAndroid Build Coastguard Worker
7*6777b538SAndroid Build Coastguard Worker #include <optional>
8*6777b538SAndroid Build Coastguard Worker #include <string_view>
9*6777b538SAndroid Build Coastguard Worker #include <utility>
10*6777b538SAndroid Build Coastguard Worker
11*6777b538SAndroid Build Coastguard Worker #include "base/big_endian.h"
12*6777b538SAndroid Build Coastguard Worker #include "base/containers/span.h"
13*6777b538SAndroid Build Coastguard Worker #include "base/containers/span_writer.h"
14*6777b538SAndroid Build Coastguard Worker #include "base/logging.h"
15*6777b538SAndroid Build Coastguard Worker #include "base/memory/ptr_util.h"
16*6777b538SAndroid Build Coastguard Worker #include "base/numerics/byte_conversions.h"
17*6777b538SAndroid Build Coastguard Worker #include "base/numerics/safe_conversions.h"
18*6777b538SAndroid Build Coastguard Worker #include "base/sys_byteorder.h"
19*6777b538SAndroid Build Coastguard Worker #include "net/base/io_buffer.h"
20*6777b538SAndroid Build Coastguard Worker #include "net/dns/dns_names_util.h"
21*6777b538SAndroid Build Coastguard Worker #include "net/dns/opt_record_rdata.h"
22*6777b538SAndroid Build Coastguard Worker #include "net/dns/public/dns_protocol.h"
23*6777b538SAndroid Build Coastguard Worker #include "net/dns/record_rdata.h"
24*6777b538SAndroid Build Coastguard Worker
25*6777b538SAndroid Build Coastguard Worker namespace net {
26*6777b538SAndroid Build Coastguard Worker
27*6777b538SAndroid Build Coastguard Worker namespace {
28*6777b538SAndroid Build Coastguard Worker
29*6777b538SAndroid Build Coastguard Worker const size_t kHeaderSize = sizeof(dns_protocol::Header);
30*6777b538SAndroid Build Coastguard Worker
31*6777b538SAndroid Build Coastguard Worker // Size of the fixed part of an OPT RR:
32*6777b538SAndroid Build Coastguard Worker // https://tools.ietf.org/html/rfc6891#section-6.1.2
33*6777b538SAndroid Build Coastguard Worker static const size_t kOptRRFixedSize = 11;
34*6777b538SAndroid Build Coastguard Worker
35*6777b538SAndroid Build Coastguard Worker // https://tools.ietf.org/html/rfc6891#section-6.2.5
36*6777b538SAndroid Build Coastguard Worker // TODO(robpercival): Determine a good value for this programmatically.
37*6777b538SAndroid Build Coastguard Worker const uint16_t kMaxUdpPayloadSize = 4096;
38*6777b538SAndroid Build Coastguard Worker
QuestionSize(size_t qname_size)39*6777b538SAndroid Build Coastguard Worker size_t QuestionSize(size_t qname_size) {
40*6777b538SAndroid Build Coastguard Worker // QNAME + QTYPE + QCLASS
41*6777b538SAndroid Build Coastguard Worker return qname_size + sizeof(uint16_t) + sizeof(uint16_t);
42*6777b538SAndroid Build Coastguard Worker }
43*6777b538SAndroid Build Coastguard Worker
44*6777b538SAndroid Build Coastguard Worker // Buffer size of Opt record for |rdata| (does not include Opt record or RData
45*6777b538SAndroid Build Coastguard Worker // added for padding).
OptRecordSize(const OptRecordRdata * rdata)46*6777b538SAndroid Build Coastguard Worker size_t OptRecordSize(const OptRecordRdata* rdata) {
47*6777b538SAndroid Build Coastguard Worker return rdata == nullptr ? 0 : kOptRRFixedSize + rdata->buf().size();
48*6777b538SAndroid Build Coastguard Worker }
49*6777b538SAndroid Build Coastguard Worker
50*6777b538SAndroid Build Coastguard Worker // Padding size includes Opt header for the padding. Does not include OptRecord
51*6777b538SAndroid Build Coastguard Worker // header (kOptRRFixedSize) even when added just for padding.
DeterminePaddingSize(size_t unpadded_size,DnsQuery::PaddingStrategy padding_strategy)52*6777b538SAndroid Build Coastguard Worker size_t DeterminePaddingSize(size_t unpadded_size,
53*6777b538SAndroid Build Coastguard Worker DnsQuery::PaddingStrategy padding_strategy) {
54*6777b538SAndroid Build Coastguard Worker switch (padding_strategy) {
55*6777b538SAndroid Build Coastguard Worker case DnsQuery::PaddingStrategy::NONE:
56*6777b538SAndroid Build Coastguard Worker return 0;
57*6777b538SAndroid Build Coastguard Worker case DnsQuery::PaddingStrategy::BLOCK_LENGTH_128:
58*6777b538SAndroid Build Coastguard Worker size_t padding_size = OptRecordRdata::Opt::kHeaderSize;
59*6777b538SAndroid Build Coastguard Worker size_t remainder = (padding_size + unpadded_size) % 128;
60*6777b538SAndroid Build Coastguard Worker padding_size += (128 - remainder) % 128;
61*6777b538SAndroid Build Coastguard Worker DCHECK_EQ((unpadded_size + padding_size) % 128, 0u);
62*6777b538SAndroid Build Coastguard Worker return padding_size;
63*6777b538SAndroid Build Coastguard Worker }
64*6777b538SAndroid Build Coastguard Worker }
65*6777b538SAndroid Build Coastguard Worker
AddPaddingIfNecessary(const OptRecordRdata * opt_rdata,DnsQuery::PaddingStrategy padding_strategy,size_t no_opt_buffer_size)66*6777b538SAndroid Build Coastguard Worker std::unique_ptr<OptRecordRdata> AddPaddingIfNecessary(
67*6777b538SAndroid Build Coastguard Worker const OptRecordRdata* opt_rdata,
68*6777b538SAndroid Build Coastguard Worker DnsQuery::PaddingStrategy padding_strategy,
69*6777b538SAndroid Build Coastguard Worker size_t no_opt_buffer_size) {
70*6777b538SAndroid Build Coastguard Worker // If no input OPT record rdata and no padding, no OPT record rdata needed.
71*6777b538SAndroid Build Coastguard Worker if (!opt_rdata && padding_strategy == DnsQuery::PaddingStrategy::NONE)
72*6777b538SAndroid Build Coastguard Worker return nullptr;
73*6777b538SAndroid Build Coastguard Worker
74*6777b538SAndroid Build Coastguard Worker std::unique_ptr<OptRecordRdata> merged_opt_rdata;
75*6777b538SAndroid Build Coastguard Worker if (opt_rdata) {
76*6777b538SAndroid Build Coastguard Worker merged_opt_rdata = OptRecordRdata::Create(
77*6777b538SAndroid Build Coastguard Worker std::string_view(opt_rdata->buf().data(), opt_rdata->buf().size()));
78*6777b538SAndroid Build Coastguard Worker } else {
79*6777b538SAndroid Build Coastguard Worker merged_opt_rdata = std::make_unique<OptRecordRdata>();
80*6777b538SAndroid Build Coastguard Worker }
81*6777b538SAndroid Build Coastguard Worker DCHECK(merged_opt_rdata);
82*6777b538SAndroid Build Coastguard Worker
83*6777b538SAndroid Build Coastguard Worker size_t unpadded_size =
84*6777b538SAndroid Build Coastguard Worker no_opt_buffer_size + OptRecordSize(merged_opt_rdata.get());
85*6777b538SAndroid Build Coastguard Worker size_t padding_size = DeterminePaddingSize(unpadded_size, padding_strategy);
86*6777b538SAndroid Build Coastguard Worker
87*6777b538SAndroid Build Coastguard Worker if (padding_size > 0) {
88*6777b538SAndroid Build Coastguard Worker // |opt_rdata| must not already contain padding if DnsQuery is to add
89*6777b538SAndroid Build Coastguard Worker // padding.
90*6777b538SAndroid Build Coastguard Worker DCHECK(!merged_opt_rdata->ContainsOptCode(dns_protocol::kEdnsPadding));
91*6777b538SAndroid Build Coastguard Worker // OPT header is the minimum amount of padding.
92*6777b538SAndroid Build Coastguard Worker DCHECK(padding_size >= OptRecordRdata::Opt::kHeaderSize);
93*6777b538SAndroid Build Coastguard Worker
94*6777b538SAndroid Build Coastguard Worker merged_opt_rdata->AddOpt(std::make_unique<OptRecordRdata::PaddingOpt>(
95*6777b538SAndroid Build Coastguard Worker padding_size - OptRecordRdata::Opt::kHeaderSize));
96*6777b538SAndroid Build Coastguard Worker }
97*6777b538SAndroid Build Coastguard Worker
98*6777b538SAndroid Build Coastguard Worker return merged_opt_rdata;
99*6777b538SAndroid Build Coastguard Worker }
100*6777b538SAndroid Build Coastguard Worker
101*6777b538SAndroid Build Coastguard Worker } // namespace
102*6777b538SAndroid Build Coastguard Worker
103*6777b538SAndroid Build Coastguard Worker // DNS query consists of a 12-byte header followed by a question section.
104*6777b538SAndroid Build Coastguard Worker // For details, see RFC 1035 section 4.1.1. This header template sets RD
105*6777b538SAndroid Build Coastguard Worker // bit, which directs the name server to pursue query recursively, and sets
106*6777b538SAndroid Build Coastguard Worker // the QDCOUNT to 1, meaning the question section has a single entry.
DnsQuery(uint16_t id,base::span<const uint8_t> qname,uint16_t qtype,const OptRecordRdata * opt_rdata,PaddingStrategy padding_strategy)107*6777b538SAndroid Build Coastguard Worker DnsQuery::DnsQuery(uint16_t id,
108*6777b538SAndroid Build Coastguard Worker base::span<const uint8_t> qname,
109*6777b538SAndroid Build Coastguard Worker uint16_t qtype,
110*6777b538SAndroid Build Coastguard Worker const OptRecordRdata* opt_rdata,
111*6777b538SAndroid Build Coastguard Worker PaddingStrategy padding_strategy)
112*6777b538SAndroid Build Coastguard Worker : qname_size_(qname.size()) {
113*6777b538SAndroid Build Coastguard Worker #if DCHECK_IS_ON()
114*6777b538SAndroid Build Coastguard Worker std::optional<std::string> dotted_name =
115*6777b538SAndroid Build Coastguard Worker dns_names_util::NetworkToDottedName(qname);
116*6777b538SAndroid Build Coastguard Worker DCHECK(dotted_name && !dotted_name.value().empty());
117*6777b538SAndroid Build Coastguard Worker #endif // DCHECK_IS_ON()
118*6777b538SAndroid Build Coastguard Worker
119*6777b538SAndroid Build Coastguard Worker size_t buffer_size = kHeaderSize + QuestionSize(qname_size_);
120*6777b538SAndroid Build Coastguard Worker std::unique_ptr<OptRecordRdata> merged_opt_rdata =
121*6777b538SAndroid Build Coastguard Worker AddPaddingIfNecessary(opt_rdata, padding_strategy, buffer_size);
122*6777b538SAndroid Build Coastguard Worker if (merged_opt_rdata)
123*6777b538SAndroid Build Coastguard Worker buffer_size += OptRecordSize(merged_opt_rdata.get());
124*6777b538SAndroid Build Coastguard Worker
125*6777b538SAndroid Build Coastguard Worker io_buffer_ = base::MakeRefCounted<IOBufferWithSize>(buffer_size);
126*6777b538SAndroid Build Coastguard Worker
127*6777b538SAndroid Build Coastguard Worker dns_protocol::Header* header = header_in_io_buffer();
128*6777b538SAndroid Build Coastguard Worker *header = {};
129*6777b538SAndroid Build Coastguard Worker header->id = base::HostToNet16(id);
130*6777b538SAndroid Build Coastguard Worker header->flags = base::HostToNet16(dns_protocol::kFlagRD);
131*6777b538SAndroid Build Coastguard Worker header->qdcount = base::HostToNet16(1);
132*6777b538SAndroid Build Coastguard Worker
133*6777b538SAndroid Build Coastguard Worker // Write question section after the header.
134*6777b538SAndroid Build Coastguard Worker auto writer = base::SpanWriter(
135*6777b538SAndroid Build Coastguard Worker base::as_writable_bytes(io_buffer_->span()).subspan(kHeaderSize));
136*6777b538SAndroid Build Coastguard Worker writer.Write(qname);
137*6777b538SAndroid Build Coastguard Worker writer.WriteU16BigEndian(qtype);
138*6777b538SAndroid Build Coastguard Worker writer.WriteU16BigEndian(dns_protocol::kClassIN);
139*6777b538SAndroid Build Coastguard Worker
140*6777b538SAndroid Build Coastguard Worker if (merged_opt_rdata) {
141*6777b538SAndroid Build Coastguard Worker DCHECK_NE(merged_opt_rdata->OptCount(), 0u);
142*6777b538SAndroid Build Coastguard Worker
143*6777b538SAndroid Build Coastguard Worker header->arcount = base::HostToNet16(1);
144*6777b538SAndroid Build Coastguard Worker // Write OPT pseudo-resource record.
145*6777b538SAndroid Build Coastguard Worker writer.WriteU8BigEndian(0); // empty domain name (root domain)
146*6777b538SAndroid Build Coastguard Worker writer.WriteU16BigEndian(OptRecordRdata::kType); // type
147*6777b538SAndroid Build Coastguard Worker writer.WriteU16BigEndian(kMaxUdpPayloadSize); // class
148*6777b538SAndroid Build Coastguard Worker // ttl (next 3 fields)
149*6777b538SAndroid Build Coastguard Worker writer.WriteU8BigEndian(0); // rcode does not apply to requests
150*6777b538SAndroid Build Coastguard Worker writer.WriteU8BigEndian(0); // version
151*6777b538SAndroid Build Coastguard Worker // TODO(robpercival): Set "DNSSEC OK" flag if/when DNSSEC is supported:
152*6777b538SAndroid Build Coastguard Worker // https://tools.ietf.org/html/rfc3225#section-3
153*6777b538SAndroid Build Coastguard Worker writer.WriteU16BigEndian(0); // flags
154*6777b538SAndroid Build Coastguard Worker
155*6777b538SAndroid Build Coastguard Worker // rdata
156*6777b538SAndroid Build Coastguard Worker writer.WriteU16BigEndian(merged_opt_rdata->buf().size()); // rdata length
157*6777b538SAndroid Build Coastguard Worker writer.Write(base::as_byte_span(merged_opt_rdata->buf()));
158*6777b538SAndroid Build Coastguard Worker }
159*6777b538SAndroid Build Coastguard Worker }
160*6777b538SAndroid Build Coastguard Worker
DnsQuery(scoped_refptr<IOBufferWithSize> buffer)161*6777b538SAndroid Build Coastguard Worker DnsQuery::DnsQuery(scoped_refptr<IOBufferWithSize> buffer)
162*6777b538SAndroid Build Coastguard Worker : io_buffer_(std::move(buffer)) {}
163*6777b538SAndroid Build Coastguard Worker
DnsQuery(const DnsQuery & query)164*6777b538SAndroid Build Coastguard Worker DnsQuery::DnsQuery(const DnsQuery& query) {
165*6777b538SAndroid Build Coastguard Worker CopyFrom(query);
166*6777b538SAndroid Build Coastguard Worker }
167*6777b538SAndroid Build Coastguard Worker
operator =(const DnsQuery & query)168*6777b538SAndroid Build Coastguard Worker DnsQuery& DnsQuery::operator=(const DnsQuery& query) {
169*6777b538SAndroid Build Coastguard Worker CopyFrom(query);
170*6777b538SAndroid Build Coastguard Worker return *this;
171*6777b538SAndroid Build Coastguard Worker }
172*6777b538SAndroid Build Coastguard Worker
173*6777b538SAndroid Build Coastguard Worker DnsQuery::~DnsQuery() = default;
174*6777b538SAndroid Build Coastguard Worker
CloneWithNewId(uint16_t id) const175*6777b538SAndroid Build Coastguard Worker std::unique_ptr<DnsQuery> DnsQuery::CloneWithNewId(uint16_t id) const {
176*6777b538SAndroid Build Coastguard Worker return base::WrapUnique(new DnsQuery(*this, id));
177*6777b538SAndroid Build Coastguard Worker }
178*6777b538SAndroid Build Coastguard Worker
Parse(size_t valid_bytes)179*6777b538SAndroid Build Coastguard Worker bool DnsQuery::Parse(size_t valid_bytes) {
180*6777b538SAndroid Build Coastguard Worker if (io_buffer_ == nullptr || io_buffer_->span().empty()) {
181*6777b538SAndroid Build Coastguard Worker return false;
182*6777b538SAndroid Build Coastguard Worker }
183*6777b538SAndroid Build Coastguard Worker auto reader =
184*6777b538SAndroid Build Coastguard Worker base::SpanReader(base::as_bytes(io_buffer_->span()).first(valid_bytes));
185*6777b538SAndroid Build Coastguard Worker dns_protocol::Header header;
186*6777b538SAndroid Build Coastguard Worker if (!ReadHeader(&reader, &header)) {
187*6777b538SAndroid Build Coastguard Worker return false;
188*6777b538SAndroid Build Coastguard Worker }
189*6777b538SAndroid Build Coastguard Worker if (header.flags & dns_protocol::kFlagResponse) {
190*6777b538SAndroid Build Coastguard Worker return false;
191*6777b538SAndroid Build Coastguard Worker }
192*6777b538SAndroid Build Coastguard Worker if (header.qdcount != 1) {
193*6777b538SAndroid Build Coastguard Worker VLOG(1) << "Not supporting parsing a DNS query with multiple (or zero) "
194*6777b538SAndroid Build Coastguard Worker "questions.";
195*6777b538SAndroid Build Coastguard Worker return false;
196*6777b538SAndroid Build Coastguard Worker }
197*6777b538SAndroid Build Coastguard Worker std::string qname;
198*6777b538SAndroid Build Coastguard Worker if (!ReadName(&reader, &qname)) {
199*6777b538SAndroid Build Coastguard Worker return false;
200*6777b538SAndroid Build Coastguard Worker }
201*6777b538SAndroid Build Coastguard Worker uint16_t qtype;
202*6777b538SAndroid Build Coastguard Worker uint16_t qclass;
203*6777b538SAndroid Build Coastguard Worker if (!reader.ReadU16BigEndian(qtype) || !reader.ReadU16BigEndian(qclass) ||
204*6777b538SAndroid Build Coastguard Worker qclass != dns_protocol::kClassIN) {
205*6777b538SAndroid Build Coastguard Worker return false;
206*6777b538SAndroid Build Coastguard Worker }
207*6777b538SAndroid Build Coastguard Worker // |io_buffer_| now contains the raw packet of a valid DNS query, we just
208*6777b538SAndroid Build Coastguard Worker // need to properly initialize |qname_size_|.
209*6777b538SAndroid Build Coastguard Worker qname_size_ = qname.size();
210*6777b538SAndroid Build Coastguard Worker return true;
211*6777b538SAndroid Build Coastguard Worker }
212*6777b538SAndroid Build Coastguard Worker
id() const213*6777b538SAndroid Build Coastguard Worker uint16_t DnsQuery::id() const {
214*6777b538SAndroid Build Coastguard Worker return base::NetToHost16(header_in_io_buffer()->id);
215*6777b538SAndroid Build Coastguard Worker }
216*6777b538SAndroid Build Coastguard Worker
qname() const217*6777b538SAndroid Build Coastguard Worker base::span<const uint8_t> DnsQuery::qname() const {
218*6777b538SAndroid Build Coastguard Worker return base::as_bytes(io_buffer_->span()).subspan(kHeaderSize, qname_size_);
219*6777b538SAndroid Build Coastguard Worker }
220*6777b538SAndroid Build Coastguard Worker
qtype() const221*6777b538SAndroid Build Coastguard Worker uint16_t DnsQuery::qtype() const {
222*6777b538SAndroid Build Coastguard Worker return base::U16FromBigEndian(base::as_bytes(io_buffer_->span())
223*6777b538SAndroid Build Coastguard Worker .subspan(kHeaderSize + qname_size_)
224*6777b538SAndroid Build Coastguard Worker .first<2u>());
225*6777b538SAndroid Build Coastguard Worker }
226*6777b538SAndroid Build Coastguard Worker
question() const227*6777b538SAndroid Build Coastguard Worker std::string_view DnsQuery::question() const {
228*6777b538SAndroid Build Coastguard Worker auto s = io_buffer_->span().subspan(kHeaderSize, QuestionSize(qname_size_));
229*6777b538SAndroid Build Coastguard Worker return std::string_view(s.begin(), s.end());
230*6777b538SAndroid Build Coastguard Worker }
231*6777b538SAndroid Build Coastguard Worker
question_size() const232*6777b538SAndroid Build Coastguard Worker size_t DnsQuery::question_size() const {
233*6777b538SAndroid Build Coastguard Worker return QuestionSize(qname_size_);
234*6777b538SAndroid Build Coastguard Worker }
235*6777b538SAndroid Build Coastguard Worker
set_flags(uint16_t flags)236*6777b538SAndroid Build Coastguard Worker void DnsQuery::set_flags(uint16_t flags) {
237*6777b538SAndroid Build Coastguard Worker header_in_io_buffer()->flags = flags;
238*6777b538SAndroid Build Coastguard Worker }
239*6777b538SAndroid Build Coastguard Worker
DnsQuery(const DnsQuery & orig,uint16_t id)240*6777b538SAndroid Build Coastguard Worker DnsQuery::DnsQuery(const DnsQuery& orig, uint16_t id) {
241*6777b538SAndroid Build Coastguard Worker CopyFrom(orig);
242*6777b538SAndroid Build Coastguard Worker header_in_io_buffer()->id = base::HostToNet16(id);
243*6777b538SAndroid Build Coastguard Worker }
244*6777b538SAndroid Build Coastguard Worker
CopyFrom(const DnsQuery & orig)245*6777b538SAndroid Build Coastguard Worker void DnsQuery::CopyFrom(const DnsQuery& orig) {
246*6777b538SAndroid Build Coastguard Worker qname_size_ = orig.qname_size_;
247*6777b538SAndroid Build Coastguard Worker io_buffer_ = base::MakeRefCounted<IOBufferWithSize>(orig.io_buffer()->size());
248*6777b538SAndroid Build Coastguard Worker io_buffer_->span().copy_from(orig.io_buffer()->span());
249*6777b538SAndroid Build Coastguard Worker }
250*6777b538SAndroid Build Coastguard Worker
ReadHeader(base::SpanReader<const uint8_t> * reader,dns_protocol::Header * header)251*6777b538SAndroid Build Coastguard Worker bool DnsQuery::ReadHeader(base::SpanReader<const uint8_t>* reader,
252*6777b538SAndroid Build Coastguard Worker dns_protocol::Header* header) {
253*6777b538SAndroid Build Coastguard Worker return (reader->ReadU16BigEndian(header->id) &&
254*6777b538SAndroid Build Coastguard Worker reader->ReadU16BigEndian(header->flags) &&
255*6777b538SAndroid Build Coastguard Worker reader->ReadU16BigEndian(header->qdcount) &&
256*6777b538SAndroid Build Coastguard Worker reader->ReadU16BigEndian(header->ancount) &&
257*6777b538SAndroid Build Coastguard Worker reader->ReadU16BigEndian(header->nscount) &&
258*6777b538SAndroid Build Coastguard Worker reader->ReadU16BigEndian(header->arcount));
259*6777b538SAndroid Build Coastguard Worker }
260*6777b538SAndroid Build Coastguard Worker
ReadName(base::SpanReader<const uint8_t> * reader,std::string * out)261*6777b538SAndroid Build Coastguard Worker bool DnsQuery::ReadName(base::SpanReader<const uint8_t>* reader,
262*6777b538SAndroid Build Coastguard Worker std::string* out) {
263*6777b538SAndroid Build Coastguard Worker DCHECK(out != nullptr);
264*6777b538SAndroid Build Coastguard Worker out->clear();
265*6777b538SAndroid Build Coastguard Worker out->reserve(dns_protocol::kMaxNameLength + 1);
266*6777b538SAndroid Build Coastguard Worker uint8_t label_length;
267*6777b538SAndroid Build Coastguard Worker if (!reader->ReadU8BigEndian(label_length)) {
268*6777b538SAndroid Build Coastguard Worker return false;
269*6777b538SAndroid Build Coastguard Worker }
270*6777b538SAndroid Build Coastguard Worker while (label_length) {
271*6777b538SAndroid Build Coastguard Worker if (out->size() + 1 + label_length > dns_protocol::kMaxNameLength) {
272*6777b538SAndroid Build Coastguard Worker return false;
273*6777b538SAndroid Build Coastguard Worker }
274*6777b538SAndroid Build Coastguard Worker
275*6777b538SAndroid Build Coastguard Worker out->push_back(static_cast<char>(label_length));
276*6777b538SAndroid Build Coastguard Worker
277*6777b538SAndroid Build Coastguard Worker std::optional<base::span<const uint8_t>> label = reader->Read(label_length);
278*6777b538SAndroid Build Coastguard Worker if (!label) {
279*6777b538SAndroid Build Coastguard Worker return false;
280*6777b538SAndroid Build Coastguard Worker }
281*6777b538SAndroid Build Coastguard Worker out->append(base::as_string_view(*label));
282*6777b538SAndroid Build Coastguard Worker
283*6777b538SAndroid Build Coastguard Worker if (!reader->ReadU8BigEndian(label_length)) {
284*6777b538SAndroid Build Coastguard Worker return false;
285*6777b538SAndroid Build Coastguard Worker }
286*6777b538SAndroid Build Coastguard Worker }
287*6777b538SAndroid Build Coastguard Worker DCHECK_LE(out->size(), static_cast<size_t>(dns_protocol::kMaxNameLength));
288*6777b538SAndroid Build Coastguard Worker out->append(1, '\0');
289*6777b538SAndroid Build Coastguard Worker return true;
290*6777b538SAndroid Build Coastguard Worker }
291*6777b538SAndroid Build Coastguard Worker
292*6777b538SAndroid Build Coastguard Worker } // namespace net
293