blob: 3341576bf4bf60cc4d0f8fe67a27d1d2bbfe84b8 [file] [log] [blame]
/*
*
* Copyright (c) 2020 Project CHIP Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#pragma once
#include <system/SystemPacketBuffer.h>
#include <lib/dnssd/minimal_mdns/Parser.h>
#include <lib/dnssd/minimal_mdns/core/DnsHeader.h>
#include <lib/dnssd/minimal_mdns/records/ResourceRecord.h>
namespace mdns {
namespace Minimal {
/// Writes a MDNS reply into a given packet buffer.
class ResponseBuilder
{
public:
ResponseBuilder() : mHeader(nullptr), mEndianOutput(nullptr, 0), mWriter(&mEndianOutput) {}
ResponseBuilder(chip::System::PacketBufferHandle && packet) :
mHeader(nullptr), mEndianOutput(nullptr, 0), mWriter(&mEndianOutput)
{
Reset(std::move(packet));
}
ResponseBuilder & Reset(chip::System::PacketBufferHandle && packet)
{
mPacket = std::move(packet);
mHeader = HeaderRef(mPacket->Start());
if (mPacket->AvailableDataLength() >= HeaderRef::kSizeBytes)
{
mPacket->SetDataLength(HeaderRef::kSizeBytes);
mHeader.Clear();
mBuildOk = true;
}
else
{
mBuildOk = false;
}
mHeader.SetFlags(mHeader.GetFlags().SetResponse().SetAuthoritative());
mEndianOutput =
chip::Encoding::BigEndian::BufferWriter(mPacket->Start(), mPacket->DataLength() + mPacket->AvailableDataLength());
mEndianOutput.Skip(mPacket->DataLength());
mWriter.Reset();
return *this;
}
CHECK_RETURN_VALUE
chip::System::PacketBufferHandle ReleasePacket()
{
mHeader = HeaderRef(nullptr);
mBuildOk = false;
return std::move(mPacket);
}
bool HasResponseRecords() const
{
return (mHeader.GetAnswerCount() != 0) || (mHeader.GetAuthorityCount() != 0) || (mHeader.GetAdditionalCount() != 0);
}
HeaderRef & Header() { return mHeader; }
/// Attempts to add a record to the currentsystem packet buffer.
/// On success, the packet buffer data length is updated.
/// On failure, the packet buffer data length is NOT updated and header is unchanged.
ResponseBuilder & AddRecord(ResourceType type, const ResourceRecord & record)
{
if (!mBuildOk)
{
return *this;
}
if (!record.Append(mHeader, type, mWriter))
{
mBuildOk = false;
}
else
{
VerifyOrDie(mEndianOutput.Fit()); // should be guaranteed because record Append succeeded
mPacket->SetDataLength(static_cast<uint16_t>(mEndianOutput.Needed()));
}
return *this;
}
ResponseBuilder & AddQuery(const QueryData & query)
{
if (!mBuildOk)
{
return *this;
}
if (!query.Append(mHeader, mWriter))
{
mBuildOk = false;
}
else
{
mPacket->SetDataLength(static_cast<uint16_t>(mEndianOutput.Needed()));
}
return *this;
}
bool Ok() const { return mBuildOk; }
bool HasPacketBuffer() const { return !mPacket.IsNull(); }
private:
chip::System::PacketBufferHandle mPacket;
HeaderRef mHeader;
chip::Encoding::BigEndian::BufferWriter mEndianOutput;
RecordWriter mWriter;
bool mBuildOk = false;
};
} // namespace Minimal
} // namespace mdns