blob: 4f4705a6c0be07e8fdfdd0d8384f70f5c1b81e7f [file] [log] [blame]
/*
*
* Copyright (c) 2020 Project CHIP Authors
* Copyright (c) 2018-2019 Google LLC
* All rights reserved.
*
* 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.
*/
/**
* @file
* This file implements a process to effect a functional test for
* the InetLayer Internet Protocol stack abstraction interfaces
* for handling IP (v4 or v6) multicast on either bare IP (i.e.,
* "raw") or UDP endpoints.
*
*/
#ifndef __STDC_LIMIT_MACROS
#define __STDC_LIMIT_MACROS
#endif
#include <signal.h>
#include <stdint.h>
#include <string.h>
#include <unistd.h>
#include <nlbyteorder.hpp>
#include <CHIPVersion.h>
#include <inet/IPAddress.h>
#include <inet/InetArgParser.h>
#include <lib/support/CHIPArgParser.hpp>
#include "TestInetCommon.h"
#include "TestInetCommonOptions.h"
#include "TestInetLayerCommon.hpp"
#include "TestSetupFaultInjection.h"
#include "TestSetupSignalling.h"
using namespace chip;
using namespace chip::ArgParser;
using namespace chip::Inet;
using namespace chip::System;
/* Preprocessor Macros */
#define kToolName "TestInetLayerMulticast"
#define kToolOptNoLoopback 'L'
#define kToolOptGroup 'g'
#define kToolOptExpectedGroupRxPackets (kToolOptBase + 0)
#define kToolOptExpectedGroupTxPackets (kToolOptBase + 1)
/* Type Definitions */
enum OptFlags
{
kOptFlagNoLoopback = 0x00010000
};
struct GroupAddress
{
uint32_t mGroup;
TransferStats mStats;
IPAddress mMulticastAddress;
};
template <size_t tCapacity>
struct GroupAddresses
{
size_t mSize;
const size_t mCapacity = tCapacity;
GroupAddress mAddresses[tCapacity];
};
template <size_t tCapacity>
struct TestState
{
GroupAddresses<tCapacity> & mGroupAddresses;
TestStatus mStatus;
};
/* Function Declarations */
static void HandleSignal(int aSignal);
static bool HandleOption(const char * aProgram, OptionSet * aOptions, int aIdentifier, const char * aName, const char * aValue);
static bool HandleNonOptionArgs(const char * aProgram, int argc, char * argv[]);
static bool ParseGroupOpt(const char * aProgram, const char * aValue, bool aIPv6, uint32_t & aOutLastGroupIndex);
static bool ParseAndUpdateExpectedGroupPackets(const char * aProgram, const char * aValue, uint32_t aGroup,
const char * aDescription, uint32_t & aOutExpected);
static void StartTest();
static void CleanupTest();
/* Global Variables */
// clang-format off
static const uint32_t kOptFlagsDefault = (kOptFlagUseIPv6 | kOptFlagUseRawIP);
static RawEndPoint * sRawIPEndPoint = nullptr;
static UDPEndPoint * sUDPIPEndPoint = nullptr;
static GroupAddresses<4> sGroupAddresses;
static TestState<4> sTestState =
{
sGroupAddresses,
{ false, false }
};
static uint32_t sLastGroupIndex = 0;
static OptionDef sToolOptionDefs[] =
{
{ "interface", kArgumentRequired, kToolOptInterface },
{ "group", kArgumentRequired, kToolOptGroup },
{ "group-expected-rx-packets", kArgumentRequired, kToolOptExpectedGroupRxPackets },
{ "group-expected-tx-packets", kArgumentRequired, kToolOptExpectedGroupTxPackets },
{ "interval", kArgumentRequired, kToolOptInterval },
#if INET_CONFIG_ENABLE_IPV4
{ "ipv4", kNoArgument, kToolOptIPv4Only },
#endif // INET_CONFIG_ENABLE_IPV4
{ "ipv6", kNoArgument, kToolOptIPv6Only },
{ "listen", kNoArgument, kToolOptListen },
{ "no-loopback", kNoArgument, kToolOptNoLoopback },
{ "raw", kNoArgument, kToolOptRawIP },
{ "send-size", kArgumentRequired, kToolOptSendSize },
{ "udp", kNoArgument, kToolOptUDPIP },
{ }
};
static const char * sToolOptionHelp =
" -I, --interface <interface>\n"
" The network interface to bind to and from which to send and receive all packets.\n"
"\n"
" -L, --no-loopback\n"
" Suppress the loopback of multicast packets.\n"
"\n"
" -g, --group <group>\n"
" Multicast group number to join.\n"
"\n"
" --group-expected-rx-packets <packets>\n"
" Expect to receive this number of packets for the previously-specified multicast group.\n"
"\n"
" --group-expected-tx-packets <packets>\n"
" Expect to send this number of packets for the previously-specified multicast group.\n"
"\n"
" -i, --interval <interval>\n"
" Wait interval milliseconds between sending each packet (default: 1000 ms).\n"
"\n"
" -l, --listen\n"
" Act as a server (i.e., listen) for packets rather than send them.\n"
"\n"
#if INET_CONFIG_ENABLE_IPV4
" -4, --ipv4\n"
" Use IPv4 only.\n"
"\n"
#endif // INET_CONFIG_ENABLE_IPV4
" -6, --ipv6\n"
" Use IPv6 only (default).\n"
"\n"
" -s, --send-size <size>\n"
" Send size bytes of user data (default: 59 bytes)\n"
"\n"
" -r, --raw\n"
" Use raw IP (default).\n"
"\n"
" -u, --udp\n"
" Use UDP over IP.\n"
"\n";
static OptionSet sToolOptions =
{
HandleOption,
sToolOptionDefs,
"GENERAL OPTIONS",
sToolOptionHelp
};
static HelpOptions sHelpOptions(
kToolName,
"Usage: " kToolName " [ <options> ] [ -g <group> [ ... ] -I <interface> ]\n",
CHIP_VERSION_STRING "\n" CHIP_TOOL_COPYRIGHT
);
static OptionSet * sToolOptionSets[] =
{
&sToolOptions,
&gNetworkOptions,
&gFaultInjectionOptions,
&sHelpOptions,
nullptr
};
// clang-format on
static void CheckSucceededOrFailed(const GroupAddress & aAddress, bool & aOutSucceeded, bool & aOutFailed)
{
const TransferStats & lStats = aAddress.mStats;
#if DEBUG
printf("Group %u: %u/%u sent, %u/%u received\n", aAddress.mGroup, lStats.mTransmit.mActual, lStats.mTransmit.mExpected,
lStats.mReceive.mActual, lStats.mReceive.mExpected);
#endif
if (((lStats.mTransmit.mExpected > 0) && (lStats.mTransmit.mActual > lStats.mTransmit.mExpected)) ||
((lStats.mReceive.mExpected > 0) && (lStats.mReceive.mActual > lStats.mReceive.mExpected)))
{
aOutFailed = true;
}
else if (((lStats.mTransmit.mExpected > 0) && (lStats.mTransmit.mActual < lStats.mTransmit.mExpected)) ||
((lStats.mReceive.mExpected > 0) && (lStats.mReceive.mActual < lStats.mReceive.mExpected)))
{
aOutSucceeded = false;
}
}
template <size_t tCapacity>
static void CheckSucceededOrFailed(TestState<tCapacity> & aTestState, bool & aOutSucceeded, bool & aOutFailed)
{
for (size_t i = 0; i < aTestState.mGroupAddresses.mSize; i++)
{
const GroupAddress & lGroup = aTestState.mGroupAddresses.mAddresses[i];
CheckSucceededOrFailed(lGroup, aOutSucceeded, aOutFailed);
}
if (aOutSucceeded || aOutFailed)
{
if (aOutSucceeded)
aTestState.mStatus.mSucceeded = true;
if (aOutFailed)
SetStatusFailed(aTestState.mStatus);
}
}
static void HandleSignal(int aSignal)
{
switch (aSignal)
{
case SIGUSR1:
SetStatusFailed(sTestState.mStatus);
break;
}
}
namespace TestInetLayerMulticast {
int main(int argc, char * argv[])
{
bool lSuccessful = true;
CHIP_ERROR lStatus;
InitTestInetCommon();
SetupFaultInjectionContext(argc, argv);
SetSignalHandler(HandleSignal);
if (argc == 1)
{
sHelpOptions.PrintBriefUsage(stderr);
lSuccessful = false;
goto exit;
}
if (!ParseArgsFromEnvVar(kToolName, TOOL_OPTIONS_ENV_VAR_NAME, sToolOptionSets, nullptr, true) ||
!ParseArgs(kToolName, argc, argv, sToolOptionSets, HandleNonOptionArgs))
{
lSuccessful = false;
goto exit;
}
InitSystemLayer();
InitNetwork();
// At this point, we should have valid network interfaces,
// including LwIP TUN/TAP shim interfaces. Validate the
// -I/--interface argument, if present.
if (gInterfaceName != nullptr)
{
lStatus = InterfaceNameToId(gInterfaceName, gInterfaceId);
if (lStatus != CHIP_NO_ERROR)
{
PrintArgError("%s: unknown network interface %s\n", kToolName, gInterfaceName);
lSuccessful = false;
goto shutdown;
}
}
// If any multicast groups have been specified, ensure that a
// network interface identifier has been specified and is valid.
if ((sGroupAddresses.mSize > 0) && !IsInterfaceIdPresent(gInterfaceId))
{
PrintArgError("%s: a network interface is required when specifying one or more multicast groups\n", kToolName);
lSuccessful = false;
goto shutdown;
}
StartTest();
while (Common::IsTesting(sTestState.mStatus))
{
bool lSucceeded = true;
bool lFailed = false;
constexpr uint32_t kSleepTimeMilliseconds = 10;
ServiceNetwork(kSleepTimeMilliseconds);
CheckSucceededOrFailed(sTestState, lSucceeded, lFailed);
#if DEBUG
printf("%s %s number of expected packets\n", ((lSucceeded) ? "successfully" : ((lFailed) ? "failed to" : "has not yet")),
((lSucceeded)
? (Common::IsReceiver() ? "received" : "sent")
: ((lFailed) ? (Common::IsReceiver() ? "receive" : "send") : Common::IsReceiver() ? "received" : "sent")));
#endif
}
CleanupTest();
shutdown:
ShutdownNetwork();
ShutdownSystemLayer();
lSuccessful = Common::WasSuccessful(sTestState.mStatus);
exit:
return (lSuccessful ? EXIT_SUCCESS : EXIT_FAILURE);
}
} // namespace TestInetLayerMulticast
static bool HandleOption(const char * aProgram, OptionSet * aOptions, int aIdentifier, const char * aName, const char * aValue)
{
bool retval = true;
switch (aIdentifier)
{
case kToolOptInterval:
if (!ParseInt(aValue, gSendIntervalMs))
{
PrintArgError("%s: invalid value specified for send interval: %s\n", aProgram, aValue);
retval = false;
}
break;
case kToolOptListen:
gOptFlags |= kOptFlagListen;
break;
case kToolOptNoLoopback:
gOptFlags |= kOptFlagNoLoopback;
break;
case kToolOptGroup:
if (!ParseGroupOpt(aProgram, aValue, gOptFlags & kOptFlagUseIPv6, sLastGroupIndex))
{
retval = false;
}
break;
case kToolOptExpectedGroupRxPackets: {
GroupAddress & lGroupAddress = sGroupAddresses.mAddresses[sLastGroupIndex];
if (!ParseAndUpdateExpectedGroupPackets(aProgram, aValue, lGroupAddress.mGroup, "received",
lGroupAddress.mStats.mReceive.mExpected))
{
retval = false;
}
}
break;
case kToolOptExpectedGroupTxPackets: {
GroupAddress & lGroupAddress = sGroupAddresses.mAddresses[sLastGroupIndex];
if (!ParseAndUpdateExpectedGroupPackets(aProgram, aValue, lGroupAddress.mGroup, "sent",
lGroupAddress.mStats.mTransmit.mExpected))
{
retval = false;
}
}
break;
#if INET_CONFIG_ENABLE_IPV4
case kToolOptIPv4Only:
if (gOptFlags & kOptFlagUseIPv6)
{
PrintArgError("%s: the use of --ipv4 is exclusive with --ipv6. Please select only one of the two options.\n", aProgram);
retval = false;
}
gOptFlags |= kOptFlagUseIPv4;
break;
#endif // INET_CONFIG_ENABLE_IPV4
case kToolOptIPv6Only:
if (gOptFlags & kOptFlagUseIPv4)
{
PrintArgError("%s: the use of --ipv6 is exclusive with --ipv4. Please select only one of the two options.\n", aProgram);
retval = false;
}
gOptFlags |= kOptFlagUseIPv6;
break;
case kToolOptInterface:
// NOTE: When using LwIP on a hosted OS, the interface will
// not actually be available until AFTER InitNetwork,
// consequently, we cannot do any meaningful validation
// here. Simply save the value off and we will validate it
// later.
gInterfaceName = aValue;
break;
case kToolOptRawIP:
if (gOptFlags & kOptFlagUseUDPIP)
{
PrintArgError("%s: the use of --raw is exclusive with --udp. Please select only one of the two options.\n", aProgram);
retval = false;
}
gOptFlags |= kOptFlagUseRawIP;
break;
case kToolOptSendSize:
if (!ParseInt(aValue, gSendSize))
{
PrintArgError("%s: invalid value specified for send size: %s\n", aProgram, aValue);
return false;
}
break;
case kToolOptUDPIP:
if (gOptFlags & kOptFlagUseRawIP)
{
PrintArgError("%s: the use of --udp is exclusive with --raw. Please select only one of the two options.\n", aProgram);
retval = false;
}
gOptFlags |= kOptFlagUseUDPIP;
break;
default:
PrintArgError("%s: INTERNAL ERROR: Unhandled option: %s\n", aProgram, aName);
retval = false;
break;
}
return (retval);
}
bool HandleNonOptionArgs(const char * aProgram, int argc, char * argv[])
{
if ((gOptFlags & (kOptFlagListen | kOptFlagNoLoopback)) == (kOptFlagListen | kOptFlagNoLoopback))
{
PrintArgError("%s: the listen option is exclusive with the loopback suppression option. Please select one or the other.\n",
aProgram);
return false;
}
// If there were any additional, non-parsed arguments, it's an error.
if (argc > 0)
{
PrintArgError("%s: unexpected argument: %s\n", aProgram, argv[0]);
return false;
}
// If no IP version or transport flags were specified, use the defaults.
if (!(gOptFlags & (kOptFlagUseIPv4 | kOptFlagUseIPv6 | kOptFlagUseRawIP | kOptFlagUseUDPIP)))
{
gOptFlags |= kOptFlagsDefault;
}
return true;
}
// Create an IPv4 administratively-scoped multicast address
static IPAddress MakeIPv4Multicast(uint32_t aGroupIdentifier)
{
IPAddress lAddress;
lAddress.Addr[0] = 0;
lAddress.Addr[1] = 0;
lAddress.Addr[2] = nlByteOrderSwap32HostToBig(0xFFFF);
lAddress.Addr[3] = nlByteOrderSwap32HostToBig((239 << 24) | (aGroupIdentifier & 0xFFFFFF));
return (lAddress);
}
// Create an IPv6 site-scoped multicast address
static IPAddress MakeIPv6Multicast(uint32_t aGroupIdentifier)
{
const uint8_t lFlags = kIPv6MulticastFlag_Transient;
return (IPAddress::MakeIPv6Multicast(lFlags, kIPv6MulticastScope_Site, aGroupIdentifier));
}
static void SetGroup(GroupAddress & aGroupAddress, uint32_t aGroupIdentifier, uint32_t aExpectedRx, uint32_t aExpectedTx)
{
aGroupAddress.mGroup = aGroupIdentifier;
aGroupAddress.mStats.mReceive.mExpected = aExpectedRx;
aGroupAddress.mStats.mReceive.mActual = 0;
aGroupAddress.mStats.mTransmit.mExpected = aExpectedTx;
aGroupAddress.mStats.mTransmit.mActual = 0;
}
static bool ParseGroupOpt(const char * aProgram, const char * aValue, bool aIPv6, uint32_t & aOutLastGroupIndex)
{
uint32_t lGroupIdentifier;
if (sGroupAddresses.mSize == sGroupAddresses.mCapacity)
{
PrintArgError("%s: the maximum number of allowed groups (%zu) have been specified\n", aProgram, sGroupAddresses.mCapacity);
return false;
}
if (!ParseInt(aValue, lGroupIdentifier))
{
PrintArgError("%s: unrecognized group %s\n", aProgram, aValue);
return false;
}
aOutLastGroupIndex = sGroupAddresses.mSize++;
SetGroup(sGroupAddresses.mAddresses[aOutLastGroupIndex], lGroupIdentifier, lGroupIdentifier, lGroupIdentifier);
return true;
}
static bool ParseAndUpdateExpectedGroupPackets(const char * aProgram, const char * aValue, uint32_t aGroup,
const char * aDescription, uint32_t & aOutExpected)
{
uint32_t lExpectedGroupPackets;
if (!ParseInt(aValue, lExpectedGroupPackets))
{
PrintArgError("%s: invalid value specified for expected group %u %s packets: %s\n", aProgram, aGroup, aDescription, aValue);
return false;
}
aOutExpected = lExpectedGroupPackets;
return true;
}
static GroupAddress * FindGroupAddress(const IPAddress & aSourceAddress)
{
GroupAddress * lResult = nullptr;
for (size_t i = 0; i < sGroupAddresses.mSize; i++)
{
GroupAddress & lGroupAddress = sGroupAddresses.mAddresses[i];
if (lGroupAddress.mMulticastAddress == aSourceAddress)
{
lResult = &lGroupAddress;
break;
}
}
return (lResult);
}
static void PrintReceivedStats(const GroupAddress & aGroupAddress)
{
printf("%u/%u received for multicast group %u\n", aGroupAddress.mStats.mReceive.mActual,
aGroupAddress.mStats.mReceive.mExpected, aGroupAddress.mGroup);
}
static bool HandleDataReceived(const PacketBufferHandle & aBuffer, GroupAddress & aGroupAddress, bool aCheckBuffer)
{
constexpr bool lStatsByPacket = true;
if (!Common::HandleDataReceived(aBuffer, aGroupAddress.mStats, lStatsByPacket, aCheckBuffer))
{
return false;
}
PrintReceivedStats(aGroupAddress);
return true;
}
static bool HandleDataReceived(const PacketBufferHandle & aBuffer, const IPPacketInfo & aPacketInfo, bool aCheckBuffer)
{
GroupAddress * const lGroupAddress = FindGroupAddress(aPacketInfo.DestAddress);
if (lGroupAddress != nullptr)
{
return HandleDataReceived(aBuffer, *lGroupAddress, aCheckBuffer);
}
return true;
}
// Raw Endpoint Callbacks
static void HandleRawMessageReceived(IPEndPointBasis * aEndPoint, PacketBufferHandle && aBuffer, const IPPacketInfo * aPacketInfo)
{
const bool lCheckBuffer = true;
const bool lStatsByPacket = true;
IPAddressType lAddressType;
bool lStatus = true;
GroupAddress * lGroupAddress;
VerifyOrExit(aEndPoint != nullptr, lStatus = false);
VerifyOrExit(!aBuffer.IsNull(), lStatus = false);
VerifyOrExit(aPacketInfo != nullptr, lStatus = false);
Common::HandleRawMessageReceived(aEndPoint, aBuffer, aPacketInfo);
lGroupAddress = FindGroupAddress(aPacketInfo->DestAddress);
if (lGroupAddress != nullptr)
{
lAddressType = aPacketInfo->DestAddress.Type();
if (lAddressType == kIPAddressType_IPv4)
{
const uint16_t kIPv4HeaderSize = 20;
aBuffer->ConsumeHead(kIPv4HeaderSize);
lStatus = Common::HandleICMPv4DataReceived(std::move(aBuffer), lGroupAddress->mStats, lStatsByPacket, lCheckBuffer);
}
else if (lAddressType == kIPAddressType_IPv6)
{
lStatus = Common::HandleICMPv6DataReceived(std::move(aBuffer), lGroupAddress->mStats, lStatsByPacket, lCheckBuffer);
}
else
{
lStatus = false;
}
if (lStatus)
{
PrintReceivedStats(*lGroupAddress);
}
}
exit:
if (!lStatus)
{
SetStatusFailed(sTestState.mStatus);
}
}
static void HandleRawReceiveError(IPEndPointBasis * aEndPoint, CHIP_ERROR aError, const IPPacketInfo * aPacketInfo)
{
Common::HandleRawReceiveError(aEndPoint, aError, aPacketInfo);
SetStatusFailed(sTestState.mStatus);
}
// UDP Endpoint Callbacks
static void HandleUDPMessageReceived(IPEndPointBasis * aEndPoint, PacketBufferHandle && aBuffer, const IPPacketInfo * aPacketInfo)
{
const bool lCheckBuffer = true;
bool lStatus;
VerifyOrExit(aEndPoint != nullptr, lStatus = false);
VerifyOrExit(!aBuffer.IsNull(), lStatus = false);
VerifyOrExit(aPacketInfo != nullptr, lStatus = false);
Common::HandleUDPMessageReceived(aEndPoint, aBuffer, aPacketInfo);
lStatus = HandleDataReceived(std::move(aBuffer), *aPacketInfo, lCheckBuffer);
exit:
if (!lStatus)
{
SetStatusFailed(sTestState.mStatus);
}
}
static void HandleUDPReceiveError(IPEndPointBasis * aEndPoint, CHIP_ERROR aError, const IPPacketInfo * aPacketInfo)
{
Common::HandleUDPReceiveError(aEndPoint, aError, aPacketInfo);
SetStatusFailed(sTestState.mStatus);
}
static bool IsTransportReadyForSend()
{
bool lStatus = false;
if ((gOptFlags & (kOptFlagUseRawIP)) == (kOptFlagUseRawIP))
{
lStatus = (sRawIPEndPoint != nullptr);
}
else if ((gOptFlags & kOptFlagUseUDPIP) == kOptFlagUseUDPIP)
{
lStatus = (sUDPIPEndPoint != nullptr);
}
return (lStatus);
}
static CHIP_ERROR PrepareTransportForSend()
{
CHIP_ERROR lStatus = CHIP_NO_ERROR;
return (lStatus);
}
static CHIP_ERROR DriveSendForDestination(const IPAddress & aAddress, uint16_t aSize)
{
PacketBufferHandle lBuffer;
if ((gOptFlags & (kOptFlagUseRawIP)) == (kOptFlagUseRawIP))
{
// For ICMP (v4 or v6), we'll send n aSize or smaller
// datagrams (with overhead for the ICMP header), each
// patterned from zero to aSize - 1, following the ICMP
// header.
if ((gOptFlags & kOptFlagUseIPv6) == (kOptFlagUseIPv6))
{
lBuffer = Common::MakeICMPv6DataBuffer(aSize);
VerifyOrReturnError(!lBuffer.IsNull(), CHIP_ERROR_NO_MEMORY);
}
#if INET_CONFIG_ENABLE_IPV4
else if ((gOptFlags & kOptFlagUseIPv4) == (kOptFlagUseIPv4))
{
lBuffer = Common::MakeICMPv4DataBuffer(aSize);
VerifyOrReturnError(!lBuffer.IsNull(), CHIP_ERROR_NO_MEMORY);
}
#endif // INET_CONFIG_ENABLE_IPV4
return sRawIPEndPoint->SendTo(aAddress, std::move(lBuffer));
}
if ((gOptFlags & kOptFlagUseUDPIP) == kOptFlagUseUDPIP)
{
const uint8_t lFirstValue = 0;
// For UDP, we'll send n aSize or smaller datagrams, each
// patterned from zero to aSize - 1.
lBuffer = Common::MakeDataBuffer(aSize, lFirstValue);
VerifyOrReturnError(!lBuffer.IsNull(), CHIP_ERROR_NO_MEMORY);
return sUDPIPEndPoint->SendTo(aAddress, kUDPPort, std::move(lBuffer));
}
return CHIP_NO_ERROR;
}
static CHIP_ERROR DriveSendForGroup(GroupAddress & aGroupAddress)
{
if (aGroupAddress.mStats.mTransmit.mActual < aGroupAddress.mStats.mTransmit.mExpected)
{
ReturnErrorOnFailure(DriveSendForDestination(aGroupAddress.mMulticastAddress, gSendSize));
aGroupAddress.mStats.mTransmit.mActual++;
printf("%u/%u transmitted for multicast group %u\n", aGroupAddress.mStats.mTransmit.mActual,
aGroupAddress.mStats.mTransmit.mExpected, aGroupAddress.mGroup);
}
return CHIP_NO_ERROR;
}
template <size_t tCapacity>
static CHIP_ERROR DriveSendForGroups(GroupAddresses<tCapacity> & aGroupAddresses)
{
// Iterate over each multicast group for which this node is a
// member and send a packet.
for (size_t i = 0; i < aGroupAddresses.mSize; i++)
{
ReturnErrorOnFailure(DriveSendForGroup(aGroupAddresses.mAddresses[i]));
}
return CHIP_NO_ERROR;
}
void DriveSend()
{
CHIP_ERROR lStatus = CHIP_NO_ERROR;
if (!Common::IsSender())
goto exit;
if (!gSendIntervalExpired)
goto exit;
if (!IsTransportReadyForSend())
{
lStatus = PrepareTransportForSend();
SuccessOrExit(lStatus);
}
else
{
gSendIntervalExpired = false;
gSystemLayer.StartTimer(gSendIntervalMs, Common::HandleSendTimerComplete, nullptr);
lStatus = DriveSendForGroups(sGroupAddresses);
SuccessOrExit(lStatus);
}
exit:
if (lStatus != CHIP_NO_ERROR)
{
SetStatusFailed(sTestState.mStatus);
}
}
static void StartTest()
{
IPAddressType lIPAddressType = kIPAddressType_IPv6;
IPProtocol lIPProtocol = kIPProtocol_ICMPv6;
IPVersion lIPVersion = kIPVersion_6;
IPAddress lAddress = IPAddress::Any;
IPEndPointBasis * lEndPoint = nullptr;
const bool lUseLoopback = ((gOptFlags & kOptFlagNoLoopback) == 0);
CHIP_ERROR lStatus;
#if INET_CONFIG_ENABLE_IPV4
if (gOptFlags & kOptFlagUseIPv4)
{
lIPAddressType = kIPAddressType_IPv4;
lIPProtocol = kIPProtocol_ICMPv4;
lIPVersion = kIPVersion_4;
}
#endif // INET_CONFIG_ENABLE_IPV4
// clang-format off
printf("Using %sIP%s, device interface: %s (w/%c LwIP)\n",
((gOptFlags & kOptFlagUseRawIP) ? "" : "UDP/"),
((gOptFlags & kOptFlagUseIPv4) ? "v4" : "v6"),
((gInterfaceName) ? gInterfaceName : "<none>"),
(CHIP_SYSTEM_CONFIG_USE_LWIP ? '\0' : 'o'));
// clang-format ob
// Allocate the endpoints for sending or receiving.
if (gOptFlags & kOptFlagUseRawIP)
{
lStatus = gInet.NewRawEndPoint(lIPVersion, lIPProtocol, &sRawIPEndPoint);
INET_FAIL_ERROR(lStatus, "InetLayer::NewRawEndPoint failed");
lStatus = sRawIPEndPoint->Bind(lIPAddressType, lAddress);
INET_FAIL_ERROR(lStatus, "RawEndPoint::Bind failed");
if (gOptFlags & kOptFlagUseIPv6)
{
lStatus = sRawIPEndPoint->SetICMPFilter(kICMPv6_FilterTypes, gICMPv6Types);
INET_FAIL_ERROR(lStatus, "RawEndPoint::SetICMPFilter (IPv6) failed");
}
if (IsInterfaceIdPresent(gInterfaceId))
{
lStatus = sRawIPEndPoint->BindInterface(lIPAddressType, gInterfaceId);
INET_FAIL_ERROR(lStatus, "RawEndPoint::BindInterface failed");
}
lStatus = sRawIPEndPoint->Listen(HandleRawMessageReceived, HandleRawReceiveError);
INET_FAIL_ERROR(lStatus, "RawEndPoint::Listen failed");
lEndPoint = sRawIPEndPoint;
}
else if (gOptFlags & kOptFlagUseUDPIP)
{
lStatus = gInet.NewUDPEndPoint(&sUDPIPEndPoint);
INET_FAIL_ERROR(lStatus, "InetLayer::NewUDPEndPoint failed");
lStatus = sUDPIPEndPoint->Bind(lIPAddressType, lAddress, kUDPPort);
INET_FAIL_ERROR(lStatus, "UDPEndPoint::Bind failed");
if (IsInterfaceIdPresent(gInterfaceId))
{
lStatus = sUDPIPEndPoint->BindInterface(lIPAddressType, gInterfaceId);
INET_FAIL_ERROR(lStatus, "UDPEndPoint::BindInterface failed");
}
lStatus = sUDPIPEndPoint->Listen(HandleUDPMessageReceived, HandleUDPReceiveError);
INET_FAIL_ERROR(lStatus, "UDPEndPoint::Listen failed");
lEndPoint = sUDPIPEndPoint;
}
// If loopback suppression has been requested, attempt to disable
// it; otherwise, attempt to enable it.
lStatus = lEndPoint->SetMulticastLoopback(lIPVersion, lUseLoopback);
INET_FAIL_ERROR(lStatus, "SetMulticastLoopback failed");
// Configure and join the multicast groups
for (size_t i = 0; i < sGroupAddresses.mSize; i++)
{
char lAddressBuffer[INET6_ADDRSTRLEN];
GroupAddress & lGroupAddress = sGroupAddresses.mAddresses[i];
IPAddress & lMulticastAddress = lGroupAddress.mMulticastAddress;
if ((lEndPoint != nullptr) && IsInterfaceIdPresent(gInterfaceId))
{
if (gOptFlags & kOptFlagUseIPv4)
{
lMulticastAddress = MakeIPv4Multicast(lGroupAddress.mGroup);
}
else
{
lMulticastAddress = MakeIPv6Multicast(lGroupAddress.mGroup);
}
lMulticastAddress.ToString(lAddressBuffer, sizeof(lAddressBuffer));
printf("Will join multicast group %s\n", lAddressBuffer);
lStatus = lEndPoint->JoinMulticastGroup(gInterfaceId, lMulticastAddress);
INET_FAIL_ERROR(lStatus, "Could not join multicast group");
}
}
if (Common::IsReceiver())
printf("Listening...\n");
else
DriveSend();
}
static void CleanupTest()
{
IPEndPointBasis * lEndPoint = nullptr;
CHIP_ERROR lStatus;
gSendIntervalExpired = false;
gSystemLayer.CancelTimer(Common::HandleSendTimerComplete, nullptr);
// Leave the multicast groups
if (gOptFlags & kOptFlagUseRawIP)
{
lEndPoint = sRawIPEndPoint;
}
else if (gOptFlags & kOptFlagUseUDPIP)
{
lEndPoint = sUDPIPEndPoint;
}
for (size_t i = 0; i < sGroupAddresses.mSize; i++)
{
char lAddressBuffer[INET6_ADDRSTRLEN];
GroupAddress & lGroupAddress = sGroupAddresses.mAddresses[i];
IPAddress & lMulticastAddress = lGroupAddress.mMulticastAddress;
if ((lEndPoint != nullptr) && IsInterfaceIdPresent(gInterfaceId))
{
lMulticastAddress.ToString(lAddressBuffer, sizeof(lAddressBuffer));
printf("Will leave multicast group %s\n", lAddressBuffer);
lStatus = lEndPoint->LeaveMulticastGroup(gInterfaceId, lMulticastAddress);
INET_FAIL_ERROR(lStatus, "Could not leave multicast group");
}
}
// Release the resources associated with the allocated end points.
if (sRawIPEndPoint != nullptr)
{
sRawIPEndPoint->Free();
}
if (sUDPIPEndPoint != nullptr)
{
sUDPIPEndPoint->Free();
}
}