diff --git a/src/app/CASESessionManager.h b/src/app/CASESessionManager.h index bf771945348875..d2f096fa4d804f 100644 --- a/src/app/CASESessionManager.h +++ b/src/app/CASESessionManager.h @@ -103,7 +103,7 @@ class CASESessionManager : public SessionReleaseDelegate, public Dnssd::Resolver OperationalDeviceProxy * FindSession(SessionHandle session); void ReleaseSession(OperationalDeviceProxy * device); - BitMapObjectPool mActiveSessions; + BitMapObjectPool mActiveSessions; CASESessionManagerConfig mConfig; }; diff --git a/src/controller/CHIPDeviceControllerFactory.cpp b/src/controller/CHIPDeviceControllerFactory.cpp index df423e1b4620f3..95bb7f69a1de97 100644 --- a/src/controller/CHIPDeviceControllerFactory.cpp +++ b/src/controller/CHIPDeviceControllerFactory.cpp @@ -223,6 +223,15 @@ CHIP_ERROR DeviceControllerSystemState::Shutdown() // Shut down the interaction model app::InteractionModelEngine::GetInstance()->Shutdown(); + // Shut down the TransportMgr. This holds Inet::UDPEndPoints so it must be shut down + // before PlatformMgr().Shutdown() shuts down Inet. + if (mTransportMgr != nullptr) + { + mTransportMgr->Close(); + chip::Platform::Delete(mTransportMgr); + mTransportMgr = nullptr; + } + #if CONFIG_DEVICE_LAYER // // We can safely call PlatformMgr().Shutdown(), which like DeviceController::Shutdown(), @@ -247,11 +256,6 @@ CHIP_ERROR DeviceControllerSystemState::Shutdown() mSystemLayer = nullptr; mInetLayer = nullptr; - if (mTransportMgr != nullptr) - { - chip::Platform::Delete(mTransportMgr); - mTransportMgr = nullptr; - } if (mMessageCounterManager != nullptr) { diff --git a/src/lib/support/Pool.h b/src/lib/support/Pool.h index ec9540756af73f..b8f051731c0b1c 100644 --- a/src/lib/support/Pool.h +++ b/src/lib/support/Pool.h @@ -23,6 +23,7 @@ #pragma once #include +#include #include #include @@ -159,6 +160,16 @@ struct HeapObjectList : HeapObjectListNode } // namespace internal +/** + * Action taken if objects remain allocated when a pool is destroyed. + */ +enum class OnObjectPoolDestruction +{ + AutoRelease, ///< Release any objects still allocated. + Die, ///< Abort if any objects remain allocated. + UnsafeDoNotUse, ///< Do nothing; keep historical behaviour until leaks are fixed. +}; + /** * @class ObjectPool * @@ -191,14 +202,24 @@ struct HeapObjectList : HeapObjectListNode * @tparam T type of element to be allocated. * @tparam N a positive integer max number of elements the pool provides. */ -template +template class BitMapObjectPool : public internal::StaticAllocatorBitmap, public internal::PoolCommon { public: BitMapObjectPool() : StaticAllocatorBitmap(mData.mMemory, mUsage, N, sizeof(T)) {} ~BitMapObjectPool() { - // ReleaseAll(); + switch (Action) + { + case OnObjectPoolDestruction::AutoRelease: + ReleaseAll(); + break; + case OnObjectPoolDestruction::Die: + VerifyOrDie(Allocated() == 0); + break; + case OnObjectPoolDestruction::UnsafeDoNotUse: + break; + } } template @@ -264,15 +285,24 @@ class BitMapObjectPool : public internal::StaticAllocatorBitmap, public internal * * @tparam T type to be allocated. */ -template +template class HeapObjectPool : public internal::Statistics, public internal::PoolCommon { public: HeapObjectPool() {} ~HeapObjectPool() { - // TODO(#11880): Release all active objects (or verify that none are active) when destroying the pool. - // ReleaseAll(); + switch (Action) + { + case OnObjectPoolDestruction::AutoRelease: + ReleaseAll(); + break; + case OnObjectPoolDestruction::Die: + VerifyOrDie(Allocated() == 0); + break; + case OnObjectPoolDestruction::UnsafeDoNotUse: + break; + } } template @@ -338,11 +368,11 @@ class HeapObjectPool : public internal::Statistics, public internal::PoolCommon< #endif // CHIP_SYSTEM_CONFIG_POOL_USE_HEAP #if CHIP_SYSTEM_CONFIG_POOL_USE_HEAP -template -using ObjectPool = HeapObjectPool; +template +using ObjectPool = HeapObjectPool; #else // CHIP_SYSTEM_CONFIG_POOL_USE_HEAP -template -using ObjectPool = BitMapObjectPool; +template +using ObjectPool = BitMapObjectPool; #endif // CHIP_SYSTEM_CONFIG_POOL_USE_HEAP enum class ObjectPoolMem @@ -353,17 +383,17 @@ enum class ObjectPoolMem #endif // CHIP_SYSTEM_CONFIG_POOL_USE_HEAP }; -template +template class MemTypeObjectPool; -template -class MemTypeObjectPool : public BitMapObjectPool +template +class MemTypeObjectPool : public BitMapObjectPool { }; #if CHIP_SYSTEM_CONFIG_POOL_USE_HEAP -template -class MemTypeObjectPool : public HeapObjectPool +template +class MemTypeObjectPool : public HeapObjectPool { }; #endif // CHIP_SYSTEM_CONFIG_POOL_USE_HEAP diff --git a/src/lib/support/PoolWrapper.h b/src/lib/support/PoolWrapper.h index 9bd3699ae769fc..4fefabc59ea87a 100644 --- a/src/lib/support/PoolWrapper.h +++ b/src/lib/support/PoolWrapper.h @@ -50,11 +50,11 @@ class PoolInterface virtual bool ForEachActiveObjectInner(void * context, Lambda lambda) = 0; }; -template +template class PoolProxy; -template -class PoolProxy> : public PoolInterface +template +class PoolProxy> : public PoolInterface { public: static_assert(std::is_base_of::value, "Interface type is not derived from Pool type"); @@ -81,7 +81,7 @@ class PoolProxy> : public PoolInter return Impl().ForEachActiveObject([&](T * target) { return lambda(context, static_cast(target)); }); } - virtual BitMapObjectPool & Impl() = 0; + virtual BitMapObjectPool & Impl() = 0; }; /* @@ -95,18 +95,18 @@ class PoolProxy> : public PoolInter * PoolInterface, the PoolImpl can be converted to the interface type * and passed around */ -template -class PoolImpl : public PoolProxy... +template +class PoolImpl : public PoolProxy... { public: PoolImpl() {} virtual ~PoolImpl() override {} protected: - virtual BitMapObjectPool & Impl() override { return mImpl; } + virtual BitMapObjectPool & Impl() override { return mImpl; } private: - BitMapObjectPool mImpl; + BitMapObjectPool mImpl; }; } // namespace chip diff --git a/src/lib/support/tests/TestPool.cpp b/src/lib/support/tests/TestPool.cpp index 2245a9322baf9a..aaca635ba9f71f 100644 --- a/src/lib/support/tests/TestPool.cpp +++ b/src/lib/support/tests/TestPool.cpp @@ -136,6 +136,8 @@ void TestCreateReleaseObjectStatic(nlTestSuite * inSuite, void * inContext) NL_TEST_ASSERT(inSuite, GetNumObjectsInUse(pool) == kSize); NL_TEST_ASSERT(inSuite, pool.Allocated() == kSize); NL_TEST_ASSERT(inSuite, pool.Exhausted()); + + pool.ReleaseAll(); } #if CHIP_SYSTEM_CONFIG_POOL_USE_HEAP @@ -325,6 +327,8 @@ void TestForEachActiveObject(nlTestSuite * inSuite, void * inContext) } NL_TEST_ASSERT(inSuite, count >= kSize / 2); NL_TEST_ASSERT(inSuite, count <= kSize); + + pool.ReleaseAll(); } void TestForEachActiveObjectStatic(nlTestSuite * inSuite, void * inContext) diff --git a/src/transport/SecureSessionTable.h b/src/transport/SecureSessionTable.h index 62a2429e172a5f..e117d7539520bf 100644 --- a/src/transport/SecureSessionTable.h +++ b/src/transport/SecureSessionTable.h @@ -124,7 +124,7 @@ class SecureSessionTable private: Time::TimeSource mTimeSource; - BitMapObjectPool mEntries; + BitMapObjectPool mEntries; }; } // namespace Transport diff --git a/src/transport/SessionManager.h b/src/transport/SessionManager.h index 3cd472ff49f6be..50c66f96510eca 100644 --- a/src/transport/SessionManager.h +++ b/src/transport/SessionManager.h @@ -286,16 +286,19 @@ class DLL_EXPORT SessionManager : public TransportMgrDelegate State mState; // < Initialization state of the object SessionMessageDelegate * mCB = nullptr; - BitMapObjectPool, CHIP_CONFIG_MAX_SESSION_CREATION_DELEGATES> + BitMapObjectPool, CHIP_CONFIG_MAX_SESSION_CREATION_DELEGATES, + OnObjectPoolDestruction::UnsafeDoNotUse> mSessionCreationDelegates; // TODO: This is a temporary solution to release sessions, in the near future, SessionReleaseDelegate will be // directly associated with the every SessionHandle. Then the callback function is called on over the handle // delegate directly, in order to prevent dangling handles. - BitMapObjectPool, CHIP_CONFIG_MAX_SESSION_RELEASE_DELEGATES> + BitMapObjectPool, CHIP_CONFIG_MAX_SESSION_RELEASE_DELEGATES, + OnObjectPoolDestruction::UnsafeDoNotUse> mSessionReleaseDelegates; - BitMapObjectPool, CHIP_CONFIG_MAX_SESSION_RECOVERY_DELEGATES> + BitMapObjectPool, CHIP_CONFIG_MAX_SESSION_RECOVERY_DELEGATES, + OnObjectPoolDestruction::UnsafeDoNotUse> mSessionRecoveryDelegates; TransportMgrBase * mTransportMgr = nullptr; diff --git a/src/transport/UnauthenticatedSessionTable.h b/src/transport/UnauthenticatedSessionTable.h index c265d43fe249e7..179c967820c0a4 100644 --- a/src/transport/UnauthenticatedSessionTable.h +++ b/src/transport/UnauthenticatedSessionTable.h @@ -223,7 +223,7 @@ class UnauthenticatedSessionTable } Time::TimeSource mTimeSource; - BitMapObjectPool mEntries; + BitMapObjectPool mEntries; }; } // namespace Transport diff --git a/src/transport/raw/TCP.h b/src/transport/raw/TCP.h index 16cc64c5b6cc88..2b7d2a1d2e5733 100644 --- a/src/transport/raw/TCP.h +++ b/src/transport/raw/TCP.h @@ -280,7 +280,7 @@ class TCP : public TCPBase private: friend class TCPTest; TCPBase::ActiveConnectionState mConnectionsBuffer[kActiveConnectionsSize]; - PoolImpl mPendingPackets; + PoolImpl mPendingPackets; }; } // namespace Transport