From 23720032c6273886b3523dc4d2002fd1a92d9f1d Mon Sep 17 00:00:00 2001 From: Boris Zbarsky Date: Thu, 26 May 2022 14:38:43 -0400 Subject: [PATCH] Fix packetbuffer handling in UDPEndPointImplOpenThread. (#18851) What the code was doing was: 1. Allocate a buffer of size msgLen. 2. Copy sizeof(IPPacketInfo) + msgLen bytes into it. This is a buffer overrun if the packetbuffer was not over-allocated. 3. memmove the contents of the buffer over by sizeof(IPPacketInfo), point Start() at the location the data was moved to. This would fail if the buffer was not even more over-allocated. 4. Read the memory from ((Start() - sizeof(IPPacketInfo) & ~3) (which might _under-run_ the buffer depending on how its allocation is aligned in memory), and which anyway conceptually represents garbage data that the packetbuffer code is well withiin its rights to have overwritten and treat that memory as an IPPacketInfo. 5. Advance Start() by sizeof(IPPacketInfo) to point to the actual data. What the new code does is: 1. Allocate a buffer with msgLen data space and enough reserved space that we can place an aligned IPPacketInfo in the reserved space. 2. Copy the incoming data into the data space. 3. Fill in an IPPacketInfo in the reserved space. 4. Later read the IPPacketInfo from the reserved space. --- src/inet/UDPEndPointImplOpenThread.cpp | 50 ++++++++++++++++---------- 1 file changed, 32 insertions(+), 18 deletions(-) diff --git a/src/inet/UDPEndPointImplOpenThread.cpp b/src/inet/UDPEndPointImplOpenThread.cpp index 1668cc99a3c369..02f11ba30bdf75 100644 --- a/src/inet/UDPEndPointImplOpenThread.cpp +++ b/src/inet/UDPEndPointImplOpenThread.cpp @@ -32,11 +32,21 @@ namespace Inet { otInstance * globalOtInstance; +namespace { +// We want to reserve space for an IPPacketInfo in our buffer, but it needs to +// be 4-byte aligned. We ensure the alignment by masking off the low bits of +// the pointer that we get by doing `Start() - sizeof(IPPacketInfo)`. That +// might move it backward by up to kPacketInfoAlignmentBytes, so we need to make +// sure we allocate enough reserved space that this will still be within our +// buffer. +constexpr uint16_t kPacketInfoAlignmentBytes = sizeof(uint32_t) - 1; +constexpr uint16_t kPacketInfoReservedSize = sizeof(IPPacketInfo) + kPacketInfoAlignmentBytes; +} // namespace + void UDPEndPointImplOT::handleUdpReceive(void * aContext, otMessage * aMessage, const otMessageInfo * aMessageInfo) { UDPEndPointImplOT * ep = static_cast(aContext); - IPPacketInfo pktInfo; - uint16_t msgLen = otMessageGetLength(aMessage); + uint16_t msgLen = otMessageGetLength(aMessage); System::PacketBufferHandle payload; #if CHIP_DETAIL_LOGGING static uint16_t msgReceivedCount = 0; @@ -50,12 +60,7 @@ void UDPEndPointImplOT::handleUdpReceive(void * aContext, otMessage * aMessage, return; } - pktInfo.SrcAddress = IPAddress::FromOtAddr(aMessageInfo->mPeerAddr); - pktInfo.DestAddress = IPAddress::FromOtAddr(aMessageInfo->mSockAddr); - pktInfo.SrcPort = aMessageInfo->mPeerPort; - pktInfo.DestPort = aMessageInfo->mSockPort; - - payload = System::PacketBufferHandle::New(msgLen, 0); + payload = System::PacketBufferHandle::New(msgLen, kPacketInfoReservedSize); if (payload.IsNull()) { @@ -63,25 +68,35 @@ void UDPEndPointImplOT::handleUdpReceive(void * aContext, otMessage * aMessage, return; } + IPPacketInfo * pktInfo = GetPacketInfo(payload); + if (pktInfo == nullptr) + { + ChipLogError(Inet, "Failed to pre-allocate reserved space for an IPPacketInfo for UDP Message reception."); + return; + } + + pktInfo->SrcAddress = IPAddress::FromOtAddr(aMessageInfo->mPeerAddr); + pktInfo->DestAddress = IPAddress::FromOtAddr(aMessageInfo->mSockAddr); + pktInfo->SrcPort = aMessageInfo->mPeerPort; + pktInfo->DestPort = aMessageInfo->mSockPort; + #if CHIP_DETAIL_LOGGING - pktInfo.SrcAddress.ToString(sourceStr, Inet::IPAddress::kMaxStringLength); - pktInfo.DestAddress.ToString(destStr, Inet::IPAddress::kMaxStringLength); + pktInfo->SrcAddress.ToString(sourceStr, Inet::IPAddress::kMaxStringLength); + pktInfo->DestAddress.ToString(destStr, Inet::IPAddress::kMaxStringLength); ChipLogDetail(Inet, "UDP Message Received packet nb : %d SrcAddr : %s[%d] DestAddr " ": %s[%d] Payload Length %d", - ++msgReceivedCount, sourceStr, pktInfo.SrcPort, destStr, pktInfo.DestPort, msgLen); + ++msgReceivedCount, sourceStr, pktInfo->SrcPort, destStr, pktInfo->DestPort, msgLen); #endif - memcpy(payload->Start(), &pktInfo, sizeof(IPPacketInfo)); - - if (otMessageRead(aMessage, 0, payload->Start() + sizeof(IPPacketInfo), msgLen) != msgLen) + if (otMessageRead(aMessage, 0, payload->Start(), msgLen) != msgLen) { ChipLogError(Inet, "Failed to copy OpenThread buffer into System Packet buffer"); return; } - payload->SetDataLength(static_cast(msgLen + sizeof(IPPacketInfo))); + payload->SetDataLength(static_cast(msgLen)); ep->Retain(); auto * buf = std::move(payload).UnsafeRelease(); @@ -160,7 +175,6 @@ void UDPEndPointImplOT::HandleDataReceived(System::PacketBufferHandle && msg) const IPPacketInfo pktInfoCopy = *pktInfo; // copy the address info so that the app can free the // PacketBuffer without affecting access to address info. - msg->ConsumeHead(sizeof(IPPacketInfo)); OnMessageReceived(this, std::move(msg), &pktInfoCopy); } else @@ -279,7 +293,7 @@ CHIP_ERROR UDPEndPointImplOT::IPv6JoinLeaveMulticastGroupImpl(InterfaceId aInter IPPacketInfo * UDPEndPointImplOT::GetPacketInfo(const System::PacketBufferHandle & aBuffer) { - if (!aBuffer->EnsureReservedSize(sizeof(IPPacketInfo))) + if (!aBuffer->EnsureReservedSize(kPacketInfoReservedSize)) { return nullptr; } @@ -288,7 +302,7 @@ IPPacketInfo * UDPEndPointImplOT::GetPacketInfo(const System::PacketBufferHandle uintptr_t lPacketInfoStart = lStart - sizeof(IPPacketInfo); // Align to a 4-byte boundary - return reinterpret_cast(lPacketInfoStart & ~(sizeof(uint32_t) - 1)); + return reinterpret_cast(lPacketInfoStart & ~kPacketInfoAlignmentBytes); } } // namespace Inet