
git-subtree-dir: deps/memkind/src git-subtree-split: bb9f19dd1b3ed6cc5e1b35919564ccf6f4b32f69
192 lines
6.2 KiB
C++
192 lines
6.2 KiB
C++
/*
|
|
* Copyright (C) 2015 - 2018 Intel Corporation.
|
|
* 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(s),
|
|
* this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice(s),
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
* and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) ``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(S) 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.
|
|
*/
|
|
|
|
/*
|
|
This file contain self-tests for Allocator Perf Tool.
|
|
*/
|
|
#pragma once
|
|
|
|
#include <unistd.h>
|
|
|
|
#include "Configuration.hpp"
|
|
#include "AllocatorFactory.hpp"
|
|
#include "AllocationSizes.hpp"
|
|
#include "VectorIterator.hpp"
|
|
#include "StandardAllocatorWithTimer.hpp"
|
|
#include "ScenarioWorkload.h"
|
|
#include "TaskFactory.hpp"
|
|
#include "Task.hpp"
|
|
|
|
//Basic test for allocators. Allocate 32 bytes by malloc, then deallocate it by free.
|
|
void test_allocator(Allocator &allocator, const char *allocator_name)
|
|
{
|
|
printf("\n============= Allocator (%s) test ============= \n",allocator_name);
|
|
|
|
memory_operation data = allocator.wrapped_malloc(32);
|
|
printf("malloc: %p \n",data.ptr);
|
|
|
|
assert(data.ptr!=NULL);
|
|
|
|
allocator.wrapped_free(data.ptr);
|
|
printf("free: %p \n",data.ptr);
|
|
|
|
printf("\n==================================================== \n");
|
|
}
|
|
|
|
//Test for the workload classes. This test count and validate the number or runs.
|
|
void test_workload(Workload &workload, const int N, const char *workload_name)
|
|
{
|
|
printf("\n============= Work load (%s) test ============= \n",workload_name);
|
|
|
|
assert(workload.run()); // Do "single" memory operation (allocate or deallocate).
|
|
|
|
int i;
|
|
for (i=1; workload.run(); i++); // Do the rest of operations.
|
|
assert(i == N);
|
|
|
|
printf("\n==================================================== \n");
|
|
}
|
|
|
|
//Simulate time consuming memory operation.
|
|
memory_operation time_consuming_memory_operation(unsigned int seconds)
|
|
{
|
|
size_t size;
|
|
|
|
START_TEST(0,0)
|
|
sleep(seconds);
|
|
END_TEST
|
|
}
|
|
|
|
//This test check if timer has measured time.
|
|
void test_timer()
|
|
{
|
|
float total_time = time_consuming_memory_operation(2).total_time;
|
|
|
|
bool test_res = (total_time >= 2.0) && (total_time < 2.2);
|
|
if(!test_res)
|
|
printf("test_timer(): unexpected timing %f", total_time);
|
|
assert(test_res);
|
|
}
|
|
|
|
//Test for iterators. Check the number of iterations, such as:
|
|
//N == number of iterations.
|
|
template<class T>
|
|
void test_iterator(Iterator<T> &it, const int N)
|
|
{
|
|
printf("\n================= Iteartor test ==================== \n");
|
|
|
|
assert(it.size() == N);
|
|
int i;
|
|
for (i=0; it.has_next(); i++) {
|
|
it.next();
|
|
}
|
|
printf("iterations=%d/%d\n",i,N);
|
|
assert(i == N);
|
|
|
|
printf("\n==================================================== \n");
|
|
}
|
|
|
|
//This test validate range of values generated by random generators.
|
|
template<class T, class C>
|
|
void test_iterator_values(VectorIterator<T> &it, const C from, const C to)
|
|
{
|
|
for (int i=0; it.has_next(); i++) {
|
|
T val = it.next();
|
|
|
|
if(val < from) std::cout << "ivalid value: actual=" << val << ", expected= >="
|
|
<< from << std::endl;
|
|
|
|
if(val > to) std::cout << "ivalid value: actual=" << val << ", expected= <=" <<
|
|
to << std::endl;
|
|
}
|
|
}
|
|
|
|
//Test time counting instrumentation in StandardAllocatorWithTimer.
|
|
//The instrumentation is made with START_TEST and END_TEST macros from WrapperMacros.h.
|
|
void test_allocator_with_timer(int N, int seed)
|
|
{
|
|
printf("\n============= Allocator with timer test ============= \n");
|
|
|
|
StandardAllocatorWithTimer allocator;
|
|
VectorIterator<size_t> allocation_sizes =
|
|
AllocationSizes::generate_random_sizes(N, 32, 2048, seed);
|
|
memory_operation data;
|
|
|
|
double elaspsed_time = 0;
|
|
for (int i=0; i<N; i++) {
|
|
data = allocator.wrapped_malloc(allocation_sizes.next());
|
|
elaspsed_time += data.total_time;
|
|
allocator.wrapped_free(data.ptr);
|
|
}
|
|
|
|
printf("%d allocations and frees done in time: %f \n", N, elaspsed_time);
|
|
|
|
printf("\n==================================================== \n");
|
|
}
|
|
|
|
//Test behavior of TypesConfiguration class, by enabling and disabling types.
|
|
void test_types_conf()
|
|
{
|
|
TypesConf types;
|
|
|
|
for (unsigned i=0; i<FunctionCalls::NUM_OF_FUNCTIONS; i++) {
|
|
types.enable_type(i);
|
|
assert(types.is_enabled(i));
|
|
types.disable_type(i);
|
|
assert(!types.is_enabled(i));
|
|
}
|
|
}
|
|
|
|
void execute_self_tests()
|
|
{
|
|
const int N = 10000;
|
|
const size_t size_from = 32, size_to = 2048;
|
|
const unsigned seed = 11;
|
|
|
|
test_types_conf();
|
|
|
|
{
|
|
VectorIterator<size_t> it = AllocationSizes::generate_random_sizes(N, size_from,
|
|
size_to,seed);
|
|
test_iterator_values(it, size_from, size_to);
|
|
}
|
|
|
|
{
|
|
TypesConf enable_func_calls;
|
|
enable_func_calls.enable_type(FunctionCalls::MALLOC);
|
|
|
|
VectorIterator<int> it = FunctionCalls::generate_random_allocator_func_calls(N,
|
|
seed, enable_func_calls);
|
|
test_iterator(it, N);
|
|
}
|
|
|
|
//Timer implementation __cplusplus < 201100L is based on CPU clocks counting and it will fail on this test.
|
|
#if __cplusplus > 201100L
|
|
test_timer();
|
|
#endif
|
|
|
|
printf("Test completed! (press ENTER to continue)\n");
|
|
}
|
|
|