/* * Copyright (c) 2017, 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 #include "core/utils/heap.hpp" #include #include "common/debug.hpp" #include "crypto/aes_ccm.hpp" #include "test_platform.h" #include "test_util.h" namespace ot { #if !OPENTHREAD_CONFIG_HEAP_EXTERNAL_ENABLE /** * Verifies single variable allocating and freeing. * */ void TestAllocateSingle(void) { ot::Utils::Heap heap; const size_t totalSize = heap.GetFreeSize(); { void *p = heap.CAlloc(1, 0); VerifyOrQuit(p == nullptr && totalSize == heap.GetFreeSize(), "TestAllocateSingle allocate 1 x 0 byte failed!"); heap.Free(p); p = heap.CAlloc(0, 1); VerifyOrQuit(p == nullptr && totalSize == heap.GetFreeSize(), "TestAllocateSingle allocate 0 x 1 byte failed!"); heap.Free(p); } for (size_t size = 1; size <= heap.GetCapacity(); ++size) { printf("%s allocating %zu bytes...\n", __func__, size); void *p = heap.CAlloc(1, size); VerifyOrQuit(p != nullptr && !heap.IsClean() && heap.GetFreeSize() + size <= totalSize, "allocating failed!"); memset(p, 0xff, size); heap.Free(p); VerifyOrQuit(heap.IsClean() && heap.GetFreeSize() == totalSize, "freeing failed!\n"); } } /** * Verifies randomly allocating and freeing variables. * * @param[in] aSizeLimit The maximum allocation size. * @param[in] aSeed The seed for generating random sizes. * */ void TestAllocateRandomly(size_t aSizeLimit, unsigned int aSeed) { struct Node { Node *mNext; size_t mSize; }; ot::Utils::Heap heap; Node head; size_t nnodes = 0; srand(aSeed); const size_t totalSize = heap.GetFreeSize(); Node *last = &head; do { size_t size = sizeof(Node) + static_cast(rand()) % aSizeLimit; printf("TestAllocateRandomly allocating %zu bytes...\n", size); last->mNext = static_cast(heap.CAlloc(1, size)); // No more memory for allocation. if (last->mNext == nullptr) { break; } VerifyOrQuit(last->mNext->mNext == nullptr, "TestAllocateRandomly memory not initialized to zero!"); last = last->mNext; last->mSize = size; ++nnodes; // 50% probability to randomly free a node. size_t freeIndex = static_cast(rand()) % (nnodes * 2); if (freeIndex > nnodes) { freeIndex /= 2; Node *prev = &head; while (freeIndex--) { prev = prev->mNext; } Node *curr = prev->mNext; printf("TestAllocateRandomly freeing %zu bytes...\n", curr->mSize); prev->mNext = curr->mNext; heap.Free(curr); if (last == curr) { last = prev; } --nnodes; } } while (true); last = head.mNext; while (last) { Node *next = last->mNext; printf("TestAllocateRandomly freeing %zu bytes...\n", last->mSize); heap.Free(last); last = next; } VerifyOrQuit(heap.IsClean() && heap.GetFreeSize() == totalSize, "TestAllocateRandomly heap not clean after freeing all!"); } /** * Verifies allocating and free multiple variables. */ void TestAllocateMultiple(void) { for (unsigned int seed = 0; seed < 10; ++seed) { size_t sizeLimit = (1 << seed); printf("TestAllocateRandomly(%zu, %u)...\n", sizeLimit, seed); TestAllocateRandomly(sizeLimit, seed); } } void RunTimerTests(void) { TestAllocateSingle(); TestAllocateMultiple(); } #endif // !OPENTHREAD_CONFIG_HEAP_EXTERNAL_ENABLE } // namespace ot int main(void) { #if !OPENTHREAD_CONFIG_HEAP_EXTERNAL_ENABLE ot::RunTimerTests(); printf("All tests passed\n"); #endif // !OPENTHREAD_CONFIG_HEAP_EXTERNAL_ENABLE return 0; }