mirror of
https://github.com/espressif/openthread.git
synced 2026-06-05 21:14:49 +00:00
6e7ed5cb55
This commit refactors the code to eliminate the use of `using` for `BigEndian` or `LittleEndian` functions like `HostSwap` and `ReadUint`. As these functions are frequently used in header files, using the direct namespace enhances code safety by mitigating potential conflicts arising from the order of included headers in the absence of explicit `using` declarations. Generally, avoiding the `using` keyword in headers is considered a recommended practice. Additionally, this commit removes the `Encoding` namespace to shorten the full function names.
141 lines
5.7 KiB
C++
141 lines
5.7 KiB
C++
/*
|
|
* Copyright (c) 2022, 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.
|
|
*/
|
|
|
|
#include "common/encoding.hpp"
|
|
#include "net/ip6_headers.hpp"
|
|
|
|
#include "test_util.hpp"
|
|
|
|
namespace ot {
|
|
namespace Ip6 {
|
|
|
|
void VerifyVersionTcFlow(const Header &aHeader, uint8_t aDscp, Ecn aEcn, uint32_t aFlow)
|
|
{
|
|
uint8_t expectedTc = static_cast<uint8_t>((aDscp << 2) + aEcn);
|
|
uint32_t expectedVerTcFlow = 0x60000000 + (static_cast<uint32_t>(expectedTc) << 20) + aFlow;
|
|
|
|
printf("%08x {dscp:%d, ecn:%d, flow:%d}\n", aHeader.GetVerionTrafficClassFlow(), aHeader.GetDscp(),
|
|
aHeader.GetEcn(), aHeader.GetFlow());
|
|
|
|
VerifyOrQuit(aHeader.IsVersion6());
|
|
VerifyOrQuit(aHeader.GetDscp() == aDscp);
|
|
VerifyOrQuit(aHeader.GetEcn() == aEcn);
|
|
VerifyOrQuit(aHeader.GetFlow() == aFlow);
|
|
VerifyOrQuit(aHeader.GetTrafficClass() == expectedTc);
|
|
VerifyOrQuit(aHeader.GetVerionTrafficClassFlow() == expectedVerTcFlow);
|
|
}
|
|
|
|
void TestIp6Header(void)
|
|
{
|
|
static constexpr uint16_t kPayloadLength = 650;
|
|
static constexpr uint8_t kHopLimit = 0xd1;
|
|
|
|
const uint32_t kFlows[] = {0x0, 0x1, 0xfff, 0xffff, 0xff000, 0xfffff};
|
|
const uint8_t kDscps[] = {0x0, 0x1, 0x3, 0xf, 0x30, 0x2f, 0x3f};
|
|
const Ecn kEcns[] = {kEcnNotCapable, kEcnCapable0, kEcnCapable1, kEcnMarked};
|
|
|
|
Header header;
|
|
Address source;
|
|
Address destination;
|
|
const uint8_t *headerBytes = reinterpret_cast<const uint8_t *>(&header);
|
|
|
|
SuccessOrQuit(source.FromString("0102:0304:0506:0708:090a:0b0c:0d0e:0f12"), "Address::FromString() failed");
|
|
SuccessOrQuit(destination.FromString("1122:3344:5566::7788:99aa:bbcc:ddee:ff23"), "Address::FromString() failed");
|
|
|
|
header.InitVersionTrafficClassFlow();
|
|
VerifyVersionTcFlow(header, kDscpCs0, kEcnNotCapable, 0);
|
|
|
|
header.Clear();
|
|
header.InitVersionTrafficClassFlow();
|
|
VerifyOrQuit(header.IsValid());
|
|
VerifyOrQuit(header.GetPayloadLength() == 0);
|
|
VerifyOrQuit(header.GetNextHeader() == 0);
|
|
VerifyOrQuit(header.GetHopLimit() == 0);
|
|
VerifyOrQuit(header.GetSource().IsUnspecified());
|
|
VerifyOrQuit(header.GetDestination().IsUnspecified());
|
|
|
|
header.SetPayloadLength(kPayloadLength);
|
|
header.SetNextHeader(kProtoUdp);
|
|
header.SetHopLimit(kHopLimit);
|
|
header.SetSource(source);
|
|
header.SetDestination(destination);
|
|
|
|
VerifyOrQuit(header.IsValid());
|
|
VerifyVersionTcFlow(header, kDscpCs0, kEcnNotCapable, 0);
|
|
VerifyOrQuit(header.GetPayloadLength() == kPayloadLength);
|
|
VerifyOrQuit(header.GetNextHeader() == kProtoUdp);
|
|
VerifyOrQuit(header.GetHopLimit() == kHopLimit);
|
|
VerifyOrQuit(header.GetSource() == source);
|
|
VerifyOrQuit(header.GetDestination() == destination);
|
|
|
|
// Verify the offsets to different fields.
|
|
|
|
VerifyOrQuit(BigEndian::ReadUint16(headerBytes + Header::kPayloadLengthFieldOffset) == kPayloadLength,
|
|
"kPayloadLengthFieldOffset is incorrect");
|
|
VerifyOrQuit(headerBytes[Header::kNextHeaderFieldOffset] == kProtoUdp, "kNextHeaderFieldOffset is incorrect");
|
|
VerifyOrQuit(headerBytes[Header::kHopLimitFieldOffset] == kHopLimit, "kHopLimitFieldOffset is incorrect");
|
|
VerifyOrQuit(memcmp(&headerBytes[Header::kSourceFieldOffset], &source, sizeof(source)) == 0,
|
|
"kSourceFieldOffset is incorrect");
|
|
VerifyOrQuit(memcmp(&headerBytes[Header::kDestinationFieldOffset], &destination, sizeof(destination)) == 0,
|
|
"kSourceFieldOffset is incorrect");
|
|
|
|
for (uint32_t flow : kFlows)
|
|
{
|
|
for (uint8_t dscp : kDscps)
|
|
{
|
|
for (Ecn ecn : kEcns)
|
|
{
|
|
printf("Expecting {dscp:%-2d, ecn:%d, flow:%-7d} => ", dscp, ecn, flow);
|
|
header.SetEcn(ecn);
|
|
header.SetDscp(dscp);
|
|
header.SetFlow(flow);
|
|
VerifyVersionTcFlow(header, dscp, ecn, flow);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Verify out of range values.
|
|
header.InitVersionTrafficClassFlow();
|
|
|
|
header.SetFlow(0xff000001);
|
|
VerifyVersionTcFlow(header, 0, kEcnNotCapable, 1);
|
|
|
|
header.SetDscp(0xef);
|
|
VerifyVersionTcFlow(header, 0x2f, kEcnNotCapable, 1);
|
|
}
|
|
|
|
} // namespace Ip6
|
|
} // namespace ot
|
|
|
|
int main(void)
|
|
{
|
|
ot::Ip6::TestIp6Header();
|
|
printf("All tests passed\n");
|
|
return 0;
|
|
}
|