blob: 447c34701f6e80f0acec13e22f6cf73f2552157f [file] [log] [blame]
/*
* Copyright (c) 2016, The OpenThread Authors.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of the copyright holder nor the
* names of its contributors may be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
/**
* @file
* This file implements common methods for manipulating MLE TLVs.
*/
#include "tlvs.hpp"
#include "common/code_utils.hpp"
#include "common/debug.hpp"
#include "common/message.hpp"
using ot::Encoding::BigEndian::HostSwap16;
using ot::Encoding::BigEndian::HostSwap32;
namespace ot {
uint32_t Tlv::GetSize(void) const
{
return IsExtended() ? sizeof(ExtendedTlv) + static_cast<const ExtendedTlv *>(this)->GetLength()
: sizeof(Tlv) + GetLength();
}
uint8_t *Tlv::GetValue(void)
{
return reinterpret_cast<uint8_t *>(this) + (IsExtended() ? sizeof(ExtendedTlv) : sizeof(Tlv));
}
const uint8_t *Tlv::GetValue(void) const
{
return reinterpret_cast<const uint8_t *>(this) + (IsExtended() ? sizeof(ExtendedTlv) : sizeof(Tlv));
}
otError Tlv::AppendTo(Message &aMessage) const
{
uint32_t size = GetSize();
// OT_ASSERT(size <= UINT16_MAX);
return aMessage.Append(this, static_cast<uint16_t>(size));
}
otError Tlv::FindTlv(const Message &aMessage, uint8_t aType, uint16_t aMaxSize, Tlv &aTlv)
{
otError error;
uint16_t offset;
uint16_t size;
SuccessOrExit(error = Find(aMessage, aType, &offset, &size, nullptr));
if (aMaxSize > size)
{
aMaxSize = size;
}
aMessage.Read(offset, aMaxSize, &aTlv);
exit:
return error;
}
otError Tlv::FindTlvOffset(const Message &aMessage, uint8_t aType, uint16_t &aOffset)
{
return Find(aMessage, aType, &aOffset, nullptr, nullptr);
}
otError Tlv::FindTlvValueOffset(const Message &aMessage, uint8_t aType, uint16_t &aValueOffset, uint16_t &aLength)
{
otError error;
uint16_t offset;
uint16_t size;
bool isExtendedTlv;
SuccessOrExit(error = Find(aMessage, aType, &offset, &size, &isExtendedTlv));
if (!isExtendedTlv)
{
aValueOffset = offset + sizeof(Tlv);
aLength = size - sizeof(Tlv);
}
else
{
aValueOffset = offset + sizeof(ExtendedTlv);
aLength = size - sizeof(ExtendedTlv);
}
exit:
return error;
}
otError Tlv::Find(const Message &aMessage, uint8_t aType, uint16_t *aOffset, uint16_t *aSize, bool *aIsExtendedTlv)
{
otError error = OT_ERROR_NOT_FOUND;
uint16_t offset = aMessage.GetOffset();
uint16_t remainingLen = aMessage.GetLength();
Tlv tlv;
uint32_t size;
VerifyOrExit(offset <= remainingLen, OT_NOOP);
remainingLen -= offset;
while (true)
{
VerifyOrExit(sizeof(Tlv) <= remainingLen, OT_NOOP);
aMessage.Read(offset, sizeof(Tlv), &tlv);
if (tlv.mLength != kExtendedLength)
{
size = tlv.GetSize();
}
else
{
ExtendedTlv extTlv;
VerifyOrExit(sizeof(ExtendedTlv) <= remainingLen, OT_NOOP);
aMessage.Read(offset, sizeof(ExtendedTlv), &extTlv);
VerifyOrExit(extTlv.GetLength() <= (remainingLen - sizeof(ExtendedTlv)), OT_NOOP);
size = extTlv.GetSize();
}
VerifyOrExit(size <= remainingLen, OT_NOOP);
if (tlv.GetType() == aType)
{
if (aOffset != nullptr)
{
*aOffset = offset;
}
if (aSize != nullptr)
{
*aSize = static_cast<uint16_t>(size);
}
if (aIsExtendedTlv != nullptr)
{
*aIsExtendedTlv = (tlv.mLength == kExtendedLength);
}
error = OT_ERROR_NONE;
ExitNow();
}
offset += size;
remainingLen -= size;
}
exit:
return error;
}
otError Tlv::ReadUint8Tlv(const Message &aMessage, uint16_t aOffset, uint8_t &aValue)
{
return ReadTlv(aMessage, aOffset, &aValue, sizeof(uint8_t));
}
otError Tlv::ReadUint16Tlv(const Message &aMessage, uint16_t aOffset, uint16_t &aValue)
{
otError error;
SuccessOrExit(error = ReadTlv(aMessage, aOffset, &aValue, sizeof(uint16_t)));
aValue = HostSwap16(aValue);
exit:
return error;
}
otError Tlv::ReadUint32Tlv(const Message &aMessage, uint16_t aOffset, uint32_t &aValue)
{
otError error;
SuccessOrExit(error = ReadTlv(aMessage, aOffset, &aValue, sizeof(uint32_t)));
aValue = HostSwap32(aValue);
exit:
return error;
}
otError Tlv::ReadTlv(const Message &aMessage, uint16_t aOffset, void *aValue, uint8_t aLength)
{
otError error = OT_ERROR_NONE;
Tlv tlv;
VerifyOrExit(aMessage.Read(aOffset, sizeof(Tlv), &tlv) == sizeof(Tlv), error = OT_ERROR_PARSE);
VerifyOrExit(!tlv.IsExtended() && (tlv.GetLength() >= aLength), error = OT_ERROR_PARSE);
VerifyOrExit(tlv.GetSize() + aOffset <= aMessage.GetLength(), error = OT_ERROR_PARSE);
aMessage.Read(aOffset + sizeof(Tlv), aLength, aValue);
exit:
return error;
}
otError Tlv::FindUint8Tlv(const Message &aMessage, uint8_t aType, uint8_t &aValue)
{
otError error = OT_ERROR_NONE;
uint16_t offset;
SuccessOrExit(error = FindTlvOffset(aMessage, aType, offset));
error = ReadUint8Tlv(aMessage, offset, aValue);
exit:
return error;
}
otError Tlv::FindUint16Tlv(const Message &aMessage, uint8_t aType, uint16_t &aValue)
{
otError error = OT_ERROR_NONE;
uint16_t offset;
SuccessOrExit(error = FindTlvOffset(aMessage, aType, offset));
error = ReadUint16Tlv(aMessage, offset, aValue);
exit:
return error;
}
otError Tlv::FindUint32Tlv(const Message &aMessage, uint8_t aType, uint32_t &aValue)
{
otError error = OT_ERROR_NONE;
uint16_t offset;
SuccessOrExit(error = FindTlvOffset(aMessage, aType, offset));
error = ReadUint32Tlv(aMessage, offset, aValue);
exit:
return error;
}
otError Tlv::FindTlv(const Message &aMessage, uint8_t aType, void *aValue, uint8_t aLength)
{
otError error;
uint16_t offset;
uint16_t length;
SuccessOrExit(error = FindTlvValueOffset(aMessage, aType, offset, length));
VerifyOrExit(length >= aLength, error = OT_ERROR_PARSE);
aMessage.Read(offset, aLength, static_cast<uint8_t *>(aValue));
exit:
return error;
}
otError Tlv::AppendUint8Tlv(Message &aMessage, uint8_t aType, uint8_t aValue)
{
uint8_t value8 = aValue;
return AppendTlv(aMessage, aType, &value8, sizeof(uint8_t));
}
otError Tlv::AppendUint16Tlv(Message &aMessage, uint8_t aType, uint16_t aValue)
{
uint16_t value16 = HostSwap16(aValue);
return AppendTlv(aMessage, aType, &value16, sizeof(uint16_t));
}
otError Tlv::AppendUint32Tlv(Message &aMessage, uint8_t aType, uint32_t aValue)
{
uint32_t value32 = HostSwap32(aValue);
return AppendTlv(aMessage, aType, &value32, sizeof(uint32_t));
}
otError Tlv::AppendTlv(Message &aMessage, uint8_t aType, const void *aValue, uint8_t aLength)
{
otError error = OT_ERROR_NONE;
Tlv tlv;
OT_ASSERT(aLength <= Tlv::kBaseTlvMaxLength);
tlv.SetType(aType);
tlv.SetLength(aLength);
SuccessOrExit(error = aMessage.Append(&tlv, sizeof(tlv)));
VerifyOrExit(aLength > 0, OT_NOOP);
error = aMessage.Append(aValue, aLength);
exit:
return error;
}
} // namespace ot