-
Notifications
You must be signed in to change notification settings - Fork 190
/
Copy pathmozzi_pgmspace.h
70 lines (65 loc) · 3 KB
/
mozzi_pgmspace.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
/*
* mozzi_pgmspace.h
*
* This file is part of Mozzi.
*
* Copyright 2018-2024 Thomas Friedrichsmeier and the Mozzi Team
*
* Mozzi is licensed under the GNU Lesser General Public Licence (LGPL) Version 2.1 or later.
*
*/
#ifndef MOZZI_PGMSPACE_H
#define MOZZI_PGMSPACE_H
/* Cross-platform wrapper around avr/pgmspace.h, i.e. macros and functions to
* store data in and retrieve data from flash memory. */
#include "hardware_defines.h"
#if IS_ESP8266() || IS_ESP32() || IS_RP2040() || IS_RENESAS()
template<typename T> inline T FLASH_OR_RAM_READ(T* address) {
return (T) (*address);
}
#define CONSTTABLE_STORAGE(X) const X
#else
#include <avr/pgmspace.h>
// work around missing std::is_const
template<typename T> inline bool mozzi_is_const_pointer(T*) { return false; }
template<typename T> inline bool mozzi_is_const_pointer(const T*) { return true; }
/** @ingroup core
* Helper function to FLASH_OR_RAM_READ(). You do not want to call this, directly. */
template<typename T> inline T mozzi_pgm_read_wrapper(const T* address) {
static_assert(sizeof(T) == 1 || sizeof(T) == 2 || sizeof(T) == 4 || sizeof(T) == 8, "Data type not supported");
switch (sizeof(T)) {
case 1: return (T) pgm_read_byte_near(address);
case 2: return (T) pgm_read_word_near(address);
case 4: return (T) pgm_read_dword_near(address);
}
// case 8: AVR-libc does not provide a read function for this, so we combine two 32-bit reads. TODO: is this MSB/LSB safe?
return (T) (pgm_read_dword_near(address) | (uint64_t) pgm_read_dword_near(((byte*) address) + 4) << 32);
}
template<> inline float mozzi_pgm_read_wrapper(const float* address) {
return pgm_read_float_near(address);
}
template<> inline double mozzi_pgm_read_wrapper(const double* address) {
static_assert(sizeof(uint64_t) == sizeof(double) || sizeof(float) == sizeof(double), "Reading double from pgmspace memory not supported on this architecture");
if (sizeof(double) == sizeof(uint64_t)) {
union u { uint64_t i; double d; };
return u{mozzi_pgm_read_wrapper((uint64_t*) address)}.d;
}
return pgm_read_float_near(address);
}
/** @ingroup core
* Read a value from flash or RAM. The specified address is read from flash, if T is const, _and_ const
* tables are stored in flash on this platform (i.e. not on ESP8266). It is read from RAM, if T is not-const
* or tables are always stored in RAM on this platform. @see CONSTTABLE_STORAGE . */
template<typename T> inline T FLASH_OR_RAM_READ(T* address) {
if(mozzi_is_const_pointer(address)) {
return mozzi_pgm_read_wrapper(address);
}
return (T) *address;
}
/** @ingroup core
* Declare a variable such that it will be stored in flash memory, instead of RAM, on platforms where this
* is reasonably possible (i.e. not on ESP8266, where random location flash memory access is too slow).
* To read the variable in a cross-platform compatible way, use FLASH_OR_RAM_READ(). */
#define CONSTTABLE_STORAGE(X) const X __attribute__((section(".progmem.data")))
#endif
#endif