164 lines
4.8 KiB
C++
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[]);
|