blob: 69fda04ebba49c57bf0a1a17b0c703e0d3ddbaf4 [file] [log] [blame]
/*
*
* Copyright (c) 2020 Project CHIP Authors
* Copyright (c) 2016-2017 Nest Labs, Inc.
*
* 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 is a unit test suite for <tt>chip::System::Object</tt>, *
* the part of the CHIP System Layer that implements objects and
* their static allocation pools.
*
*/
#ifndef __STDC_LIMIT_MACROS
#define __STDC_LIMIT_MACROS
#endif
// Install a sleep in the high water mark function, to force
// collisions between the threads that call it.
// clang-format off
#define SYSTEM_OBJECT_HWM_TEST_HOOK() do { usleep(1000); } while(0)
// clang-format on
#include <system/SystemLayer.h>
#include <support/CodeUtils.h>
#include <support/ErrorStr.h>
#include <support/UnitTestRegistration.h>
#include <nlunit-test.h>
#if CHIP_SYSTEM_CONFIG_USE_LWIP
#include <lwip/init.h>
#include <lwip/tcpip.h>
#include <lwip/sys.h>
#endif // CHIP_SYSTEM_CONFIG_USE_LWIP
#if CHIP_SYSTEM_CONFIG_POSIX_LOCKING
#include <pthread.h>
#endif // CHIP_SYSTEM_CONFIG_POSIX_LOCKING
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
// Test context
using namespace chip::System;
namespace chip {
namespace System {
static int Initialize(void * aContext);
static int Finalize(void * aContext);
class TestObject : public Object
{
public:
CHIP_ERROR Init();
static void CheckRetention(nlTestSuite * inSuite, void * aContext);
static void CheckConcurrency(nlTestSuite * inSuite, void * aContext);
static void CheckHighWatermark(nlTestSuite * inSuite, void * aContext);
static void CheckHighWatermarkConcurrency(nlTestSuite * inSuite, void * aContext);
private:
enum
{
kPoolSize = 122 // a multiple of kNumThreads, less than CHIP_SYS_STATS_COUNT_MAX
};
static ObjectPool<TestObject, kPoolSize> sPool;
#if CHIP_SYSTEM_CONFIG_POSIX_LOCKING
unsigned int mDelay;
static constexpr int kNumThreads = 16;
static constexpr int kLoopIterations = 100000;
static constexpr int kMaxDelayIterations = 3;
void Delay(volatile unsigned int & aAccumulator);
static void * CheckConcurrencyThread(void * aContext);
static void * CheckHighWatermarkThread(void * aContext);
static void MultithreadedTest(nlTestSuite * inSuite, void * aContext, void * (*aStartRoutine)(void *) );
#endif // CHIP_SYSTEM_CONFIG_POSIX_LOCKING
// Not defined
TestObject(const TestObject &) = delete;
TestObject & operator=(const TestObject &) = delete;
};
ObjectPool<TestObject, TestObject::kPoolSize> TestObject::sPool;
CHIP_ERROR TestObject::Init()
{
#if CHIP_SYSTEM_CONFIG_POSIX_LOCKING
this->mDelay = kMaxDelayIterations > 0 ? 1 : 0;
if (kMaxDelayIterations > 1)
{
this->mDelay += static_cast<unsigned int>(rand() % kMaxDelayIterations);
}
#endif // CHIP_SYSTEM_CONFIG_POSIX_LOCKING
return CHIP_NO_ERROR;
}
namespace {
struct TestContext
{
nlTestSuite * mTestSuite;
void * mLayerContext;
volatile unsigned int mAccumulator;
};
struct TestContext sContext;
} // namespace
// Test Object retention
void TestObject::CheckRetention(nlTestSuite * inSuite, void * aContext)
{
TestContext & lContext = *static_cast<TestContext *>(aContext);
Layer lLayer;
unsigned int i, j;
lLayer.Init(lContext.mLayerContext);
memset(&sPool, 0, sizeof(sPool));
for (i = 0; i < kPoolSize; ++i)
{
TestObject * lCreated = sPool.TryCreate(lLayer);
NL_TEST_ASSERT(lContext.mTestSuite, lCreated != nullptr);
if (lCreated == nullptr)
continue;
NL_TEST_ASSERT(lContext.mTestSuite, lCreated->IsRetained(lLayer));
NL_TEST_ASSERT(lContext.mTestSuite, &(lCreated->SystemLayer()) == &lLayer);
lCreated->Init();
for (j = 0; j < kPoolSize; ++j)
{
TestObject * lGotten = sPool.Get(lLayer, j);
if (j > i)
{
NL_TEST_ASSERT(lContext.mTestSuite, lGotten == nullptr);
}
else
{
NL_TEST_ASSERT(lContext.mTestSuite, lGotten != nullptr);
lGotten->Retain();
}
}
}
for (i = 0; i < kPoolSize; ++i)
{
TestObject * lGotten = sPool.Get(lLayer, i);
NL_TEST_ASSERT(lContext.mTestSuite, lGotten != nullptr);
for (j = kPoolSize; j > i; --j)
{
NL_TEST_ASSERT(lContext.mTestSuite, lGotten->IsRetained(lLayer));
lGotten->Release();
}
NL_TEST_ASSERT(lContext.mTestSuite, lGotten->IsRetained(lLayer));
lGotten->Release();
NL_TEST_ASSERT(lContext.mTestSuite, !lGotten->IsRetained(lLayer));
}
for (i = 0; i < kPoolSize; ++i)
{
TestObject * lGotten = sPool.Get(lLayer, i);
NL_TEST_ASSERT(lContext.mTestSuite, lGotten == nullptr);
}
lLayer.Shutdown();
}
// Test Object concurrency
#if CHIP_SYSTEM_CONFIG_POSIX_LOCKING
void TestObject::Delay(volatile unsigned int & aAccumulator)
{
unsigned int lSum = 0;
if (kMaxDelayIterations > 0)
{
for (unsigned int z = 0; z < this->mDelay; ++z)
{
lSum += static_cast<unsigned int>(rand());
}
lSum = lSum / this->mDelay;
}
aAccumulator = lSum;
}
void * TestObject::CheckConcurrencyThread(void * aContext)
{
const unsigned int kNumObjects = kPoolSize / kNumThreads;
TestObject * lObject = nullptr;
TestContext & lContext = *static_cast<TestContext *>(aContext);
Layer lLayer;
unsigned int i;
lLayer.Init(lContext.mLayerContext);
// Take this thread's share of objects
for (i = 0; i < kNumObjects; ++i)
{
while (lObject == nullptr)
{
lObject = sPool.TryCreate(lLayer);
}
NL_TEST_ASSERT(lContext.mTestSuite, lObject->IsRetained(lLayer));
NL_TEST_ASSERT(lContext.mTestSuite, &(lObject->SystemLayer()) == &lLayer);
lObject->Init();
lObject->Delay(lContext.mAccumulator);
}
// Free the last object of the pool, if it belongs to
// this thread.
lObject = sPool.Get(lLayer, kPoolSize - 1);
if (lObject != nullptr)
{
lObject->Release();
NL_TEST_ASSERT(lContext.mTestSuite, !lObject->IsRetained(lLayer));
}
// For each iteration, take one more object, and free one starting from the end
// of the pool
for (i = 0; i < kLoopIterations; ++i)
{
unsigned int j;
lObject = nullptr;
while (lObject == nullptr)
{
lObject = sPool.TryCreate(lLayer);
}
NL_TEST_ASSERT(lContext.mTestSuite, lObject->IsRetained(lLayer));
NL_TEST_ASSERT(lContext.mTestSuite, &(lObject->SystemLayer()) == &lLayer);
lObject->Init();
lObject->Delay(lContext.mAccumulator);
j = kPoolSize;
lObject = nullptr;
while (j-- > 0)
{
lObject = sPool.Get(lLayer, j);
if (lObject == nullptr)
continue;
lObject->Release();
NL_TEST_ASSERT(lContext.mTestSuite, !lObject->IsRetained(lLayer));
break;
}
NL_TEST_ASSERT(lContext.mTestSuite, lObject != nullptr);
}
// Cleanup
for (i = 0; i < kPoolSize; ++i)
{
lObject = sPool.Get(lLayer, i);
if (lObject == nullptr)
continue;
lObject->Release();
NL_TEST_ASSERT(lContext.mTestSuite, !lObject->IsRetained(lLayer));
}
lLayer.Shutdown();
return aContext;
}
void * TestObject::CheckHighWatermarkThread(void * aContext)
{
TestContext & lContext = *static_cast<TestContext *>(aContext);
int i;
chip::System::Stats::count_t lNumInUse;
chip::System::Stats::count_t lHighWatermark;
i = (rand() % CHIP_SYS_STATS_COUNT_MAX);
sPool.UpdateHighWatermark(static_cast<unsigned int>(i));
sPool.GetStatistics(lNumInUse, lHighWatermark);
NL_TEST_ASSERT(lContext.mTestSuite, lHighWatermark >= i);
if (lHighWatermark < i)
{
printf("hwm: %d, i: %d\n", lHighWatermark, i);
}
return aContext;
}
void TestObject::MultithreadedTest(nlTestSuite * inSuite, void * aContext, void * (*aStartRoutine)(void *) )
{
TestContext & lContext = *static_cast<TestContext *>(aContext);
pthread_t lThread[kNumThreads];
memset(&sPool, 0, sizeof(sPool));
for (unsigned int i = 0; i < kNumThreads; ++i)
{
int lError = pthread_create(&lThread[i], nullptr, aStartRoutine, &lContext);
NL_TEST_ASSERT(lContext.mTestSuite, lError == 0);
}
for (pthread_t thread : lThread)
{
int lError = pthread_join(thread, nullptr);
NL_TEST_ASSERT(lContext.mTestSuite, lError == 0);
}
}
#endif // CHIP_SYSTEM_CONFIG_POSIX_LOCKING
void TestObject::CheckConcurrency(nlTestSuite * inSuite, void * aContext)
{
#if CHIP_SYSTEM_CONFIG_POSIX_LOCKING
MultithreadedTest(inSuite, aContext, CheckConcurrencyThread);
#endif // CHIP_SYSTEM_CONFIG_POSIX_LOCKING
}
void TestObject::CheckHighWatermarkConcurrency(nlTestSuite * inSuite, void * aContext)
{
#if CHIP_SYSTEM_CONFIG_POSIX_LOCKING
for (unsigned int i = 0; i < 1000; i++)
{
MultithreadedTest(inSuite, aContext, CheckHighWatermarkThread);
}
#endif // CHIP_SYSTEM_CONFIG_POSIX_LOCKING
}
void TestObject::CheckHighWatermark(nlTestSuite * inSuite, void * aContext)
{
memset(&sPool, 0, sizeof(sPool));
const int kNumObjects = kPoolSize;
TestObject * lObject = nullptr;
TestContext & lContext = *static_cast<TestContext *>(aContext);
Layer lLayer;
chip::System::Stats::count_t lNumInUse;
chip::System::Stats::count_t lHighWatermark;
lLayer.Init(lContext.mLayerContext);
// Take all objects one at a time and check the watermark
// increases monotonically
for (int i = 0; i < kNumObjects; ++i)
{
lObject = sPool.TryCreate(lLayer);
NL_TEST_ASSERT(lContext.mTestSuite, lObject->IsRetained(lLayer));
NL_TEST_ASSERT(lContext.mTestSuite, &(lObject->SystemLayer()) == &lLayer);
sPool.GetStatistics(lNumInUse, lHighWatermark);
NL_TEST_ASSERT(lContext.mTestSuite, lNumInUse == (i + 1));
NL_TEST_ASSERT(lContext.mTestSuite, lHighWatermark == lNumInUse);
lObject->Init();
}
// Fail an allocation and check that both stats don't change
lObject = sPool.TryCreate(lLayer);
NL_TEST_ASSERT(lContext.mTestSuite, lObject == nullptr);
sPool.GetStatistics(lNumInUse, lHighWatermark);
NL_TEST_ASSERT(lContext.mTestSuite, lNumInUse == kNumObjects);
NL_TEST_ASSERT(lContext.mTestSuite, lHighWatermark == kNumObjects);
// Free all objects one at a time and check that the watermark does not
// change.
for (int i = 0; i < kNumObjects; ++i)
{
lObject = sPool.Get(lLayer, static_cast<size_t>(i));
NL_TEST_ASSERT(lContext.mTestSuite, lObject != nullptr);
lObject->Release();
NL_TEST_ASSERT(lContext.mTestSuite, !lObject->IsRetained(lLayer));
sPool.GetStatistics(lNumInUse, lHighWatermark);
NL_TEST_ASSERT(lContext.mTestSuite, lNumInUse == (kNumObjects - i - 1));
NL_TEST_ASSERT(lContext.mTestSuite, lHighWatermark == kNumObjects);
}
// Take all objects one at a time again and check the watermark
// does not move
for (int i = 0; i < kNumObjects; ++i)
{
lObject = sPool.TryCreate(lLayer);
NL_TEST_ASSERT(lContext.mTestSuite, lObject->IsRetained(lLayer));
NL_TEST_ASSERT(lContext.mTestSuite, &(lObject->SystemLayer()) == &lLayer);
sPool.GetStatistics(lNumInUse, lHighWatermark);
NL_TEST_ASSERT(lContext.mTestSuite, lNumInUse == (i + 1));
NL_TEST_ASSERT(lContext.mTestSuite, lHighWatermark == kNumObjects);
lObject->Init();
}
// Cleanup
for (size_t i = 0; i < kPoolSize; ++i)
{
lObject = sPool.Get(lLayer, i);
if (lObject == nullptr)
continue;
lObject->Release();
NL_TEST_ASSERT(lContext.mTestSuite, !lObject->IsRetained(lLayer));
}
lLayer.Shutdown();
}
// Test Suite
/**
* Test Suite. It lists all the test functions.
*/
// clang-format off
static const nlTest sTests[] =
{
NL_TEST_DEF("Retention", TestObject::CheckRetention),
NL_TEST_DEF("Concurrency", TestObject::CheckConcurrency),
NL_TEST_DEF("HighWatermark", TestObject::CheckHighWatermark),
NL_TEST_DEF("HighWatermarkConcurrency", TestObject::CheckHighWatermarkConcurrency),
NL_TEST_SENTINEL()
};
static nlTestSuite sTestSuite =
{
"chip-system-object",
&sTests[0],
Initialize,
Finalize
};
// clang-format on
/**
* Initialize the test suite.
*/
static int Initialize(void * aContext)
{
TestContext & lContext = *reinterpret_cast<TestContext *>(aContext);
void * lLayerContext = nullptr;
#if CHIP_SYSTEM_CONFIG_USE_LWIP && LWIP_VERSION_MAJOR <= 2 && LWIP_VERSION_MINOR < 1
static sys_mbox_t * sLwIPEventQueue = NULL;
if (sLwIPEventQueue == NULL)
{
sys_mbox_new(sLwIPEventQueue, 100);
}
lLayerContext = &sLwIPEventQueue;
#endif // CHIP_SYSTEM_CONFIG_USE_LWIP
lContext.mTestSuite = &sTestSuite;
lContext.mLayerContext = lLayerContext;
lContext.mAccumulator = 0;
return SUCCESS;
}
/**
* Finalize the test suite.
*/
static int Finalize(void * aContext)
{
TestContext & lContext = *reinterpret_cast<TestContext *>(aContext);
lContext.mTestSuite = nullptr;
return SUCCESS;
}
} // namespace System
} // namespace chip
int TestSystemObject(void)
{
// Initialize standard pseudo-random number generator
srand(0);
// Run test suit againt one lContext.
nlTestRunner(&sTestSuite, &chip::System::sContext);
return nlTestRunnerStats(&sTestSuite);
}
CHIP_REGISTER_TEST_SUITE(TestSystemObject)