speeduino/test/test_utils.h

164 lines
4.8 KiB
C++

#pragma once
#include "table3d.h"
#include <stdint.h>
#include <unity.h>
#include <Arduino.h>
#include <unity.h>
#include "table2d.h"
#include "table3d.h"
template<size_t MAX_LEN, size_t N>
constexpr void STR_LEN_CHECK(char const (&)[N])
{
static_assert(N < MAX_LEN, "String overflow!");
}
#if !defined(_countof)
#define _countof(x) (sizeof(x) / sizeof (x[0]))
#endif
// Unity macro to reduce memory usage (RAM, .bss)
//
// Unity supplied RUN_TEST captures the function name
// using #func directly in the call to UnityDefaultTestRun.
// This is a raw string that is placed in the data segment,
// which consumes RAM.
//
// So instead, place the function name in flash memory and
// load it at run time.
#define RUN_TEST_P(func) \
{ \
char funcName[128]; \
constexpr size_t bufferLen = _countof(funcName); \
STR_LEN_CHECK<bufferLen>(#func); \
strcpy_P(funcName, PSTR(#func)); \
UnityDefaultTestRun(func, funcName, __LINE__); \
}
// ============================ SET_UNITY_FILENAME ============================
static inline uint8_t ufname_set(const char *newFName)
{
Unity.TestFile = newFName;
return 1;
}
static inline void ufname_szrestore(char** __s)
{
Unity.TestFile = *__s;
__asm__ volatile ("" ::: "memory");
}
#define UNITY_FILENAME_RESTORE char* _ufname_saved \
__attribute__((__cleanup__(ufname_szrestore))) = (char*)Unity.TestFile
#define SET_UNITY_FILENAME() \
for ( UNITY_FILENAME_RESTORE, _ufname_done = ufname_set(__FILE__); \
_ufname_done; _ufname_done = 0 )
// ============================ end SET_UNITY_FILENAME ============================
// Store test data in flash, if feasible.
#if defined(PROGMEM)
#define TEST_DATA_P static constexpr PROGMEM
#else
#define TEST_DATA_P static constexpr
#endif
template <typename table3d_t>
static inline void fill_table_values(table3d_t &table, table3d_value_t value) {
// for (uint8_t i=0; i<table.values.row_size*table.values.num_rows; ++i) {
// table.values.values[i] = value;
// }
table_value_iterator itZ = table.values.begin();
while (!itZ.at_end())
{
table_row_iterator itRow = *itZ;
while (!itRow.at_end())
{
*itRow = value;
++itRow;
}
++itZ;
}
invalidate_cache(&table.get_value_cache);
}
static inline void populate_table_axis_P(table_axis_iterator it,
const table3d_axis_t *pXValues) { // PROGMEM if available
while (!it.at_end())
{
#if defined(PROGMEM)
*it = (table3d_axis_t)pgm_read_word(pXValues);
#else
*it = *pXValues;
#endif
++pXValues;
++it;
}
}
// Populate a 3d table (from PROGMEM if available)
// You wuld typically declare the 3 source arrays usin TEST_DATA_P
template <typename table3d_t>
static inline void populate_table_P(table3d_t &table,
const table3d_axis_t *pXValues, // PROGMEM if available
const table3d_axis_t *pYValues, // PROGMEM if available
const table3d_value_t *pZValues) // PROGMEM if available
{
populate_table_axis_P(table.axisX.begin(), pXValues);
populate_table_axis_P(table.axisY.begin(), pYValues);
{
table_value_iterator itZ = table.values.begin();
while (!itZ.at_end())
{
table_row_iterator itRow = *itZ;
while (!itRow.at_end())
{
#if defined(PROGMEM)
*itRow = pgm_read_byte(pZValues);
#else
*itRow = *pZValues;
#endif
++pZValues;
++itRow;
}
++itZ;
}
}
}
// Populate a 2d table with constant values
static inline void populate_2dtable(table2D *pTable, uint8_t value, uint8_t bin) {
for (uint8_t index=0; index<pTable->xSize; ++index) {
((uint8_t*)pTable->values)[index] = value;
((uint8_t*)pTable->axisX)[index] = bin;
}
pTable->cacheTime = UINT8_MAX;
}
template <typename TValue, typename TBin>
static inline void populate_2dtable(table2D *pTable, const TValue values[], const TBin bins[]) {
memcpy(pTable->axisX, bins, pTable->xSize * sizeof(TBin));
memcpy(pTable->values, values, pTable->xSize * sizeof(TValue));
pTable->cacheTime = UINT8_MAX;
}
// Populate a 2d table (from PROGMEM if available)
// You would typically declare the 2 source arrays using TEST_DATA_P
template <typename TValue, typename TBin>
static inline void populate_2dtable_P(table2D *pTable, const TValue values[], const TBin bins[]) {
#if defined(PROGMEM)
memcpy_P(pTable->axisX, bins, pTable->xSize * sizeof(TBin));
memcpy_P(pTable->values, values, pTable->xSize * sizeof(TValue));
pTable->cacheTime = UINT8_MAX;
#else
populate_2dtable(pTable, values, bins)
#endif
}
void populateTable(table3d16RpmLoad &table, const table3d_value_t values[], const table3d_axis_t xAxis[], const table3d_axis_t yAxis[]);