-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathcommon.h
119 lines (98 loc) · 5.31 KB
/
common.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
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
#ifndef COMMON_H
#define COMMON_H
#include <limits>
#include <cstdint>
#include <cstddef>
/// Type to store data memory offset
using memptr_t = uint_fast32_t;
/// Type to store I/O memory offset
using ioptr_t = uint_fast16_t;
/// Type to store flash memory offset
using flashptr_t = uint_fast32_t;
/// Interrupt vector number
using ivnum_t = uint_fast8_t;
/** @brief Interrupt levels
*
* Bit index in PMIC.STATUS can be obtained decrementing the value by 1.
*/
enum IntLvl {
INTLVL_NONE = 0,
INTLVL_LO = 1,
INTLVL_MED = 2,
INTLVL_HI = 3,
INTLVL_NMI = 8,
};
/// Map bit size to an appropriated unsigned integer type
template <unsigned N> class nbits_to_utype;
template <> struct nbits_to_utype<8> { using type = uint8_t; };
template <> struct nbits_to_utype<16> { using type = uint16_t; };
template <> struct nbits_to_utype<24> { using type = uint32_t; };
template <> struct nbits_to_utype<32> { using type = uint32_t; };
/// Read a multibyte little-endian value from an 8-bit array-compatible type
template <unsigned N> typename nbits_to_utype<N>::type register_get(const uint8_t* p);
template <> inline uint8_t register_get<8>(const uint8_t* p) { return p[0]; }
template <> inline uint16_t register_get<16>(const uint8_t* p) { return p[0] + (p[1] << 8); }
template <> inline uint32_t register_get<24>(const uint8_t* p) { return p[0] + (p[1] << 8) + (p[2] << 16); }
template <> inline uint32_t register_get<32>(const uint8_t* p) { return p[0] + (p[1] << 8) + (p[2] << 16) + (p[3] << 24); }
/// Write a multibyte little-endian value to an 8-bit array-compatible type
template <unsigned N> void register_set(uint8_t* p, typename nbits_to_utype<N>::type v);
template <> inline void register_set<8>(uint8_t* p, uint8_t v) { p[0] = v; }
template <> inline void register_set<16>(uint8_t* p, uint16_t v) { p[0] = v & 0xff; p[1] = (v >> 8) & 0xff; }
template <> inline void register_set<24>(uint8_t* p, uint32_t v) { p[0] = v & 0xff; p[1] = (v >> 8) & 0xff; p[2] = (v >> 16) & 0xff; }
template <> inline void register_set<32>(uint8_t* p, uint32_t v) { p[0] = v & 0xff; p[1] = (v >> 8) & 0xff; p[2] = (v >> 16) & 0xff; p[3] = (v >> 24) & 0xff; }
/// Like register_get(), but for stack access (assume pre-increment)
template <unsigned N> typename nbits_to_utype<N>::type stack_get(const uint8_t* p);
template <> inline uint8_t stack_get<8>(const uint8_t* p) { return p[0]; }
template <> inline uint16_t stack_get<16>(const uint8_t* p) { return p[0] + (p[-1] << 8); }
template <> inline uint32_t stack_get<24>(const uint8_t* p) { return p[0] + (p[-1] << 8) + (p[-2] << 16); }
template <> inline uint32_t stack_get<32>(const uint8_t* p) { return p[0] + (p[-1] << 8) + (p[-2] << 16) + (p[-3] << 24); }
/// Like register_set(), but for stack access (assume post-decrement)
template <unsigned N> void stack_set(uint8_t* p, typename nbits_to_utype<N>::type v);
template <> inline void stack_set<8>(uint8_t* p, uint8_t v) { p[0] = v; }
template <> inline void stack_set<16>(uint8_t* p, uint16_t v) { p[0] = v & 0xff; p[-1] = (v >> 8) & 0xff; }
template <> inline void stack_set<24>(uint8_t* p, uint32_t v) { p[0] = v & 0xff; p[-1] = (v >> 8) & 0xff; p[-2] = (v >> 16) & 0xff; }
template <> inline void stack_set<32>(uint8_t* p, uint32_t v) { p[0] = v & 0xff; p[-1] = (v >> 8) & 0xff; p[-2] = (v >> 16) & 0xff; p[-3] = (v >> 24) & 0xff; }
/// Wrap memory into 8/16/32-bit register
template <unsigned N>
class Register
{
static_assert(N % 8 == 0, "invalid bit count");
uint8_t data_[N/8];
public:
using value_type = typename nbits_to_utype<N>::type;
Register& operator=(value_type v) { register_set<N>(data_, v); return *this; }
operator value_type() const { return register_get<N>(data_); }
Register& operator++() { register_set<N>(data_, register_get<N>(data_)+1); return *this; }
Register& operator--() { register_set<N>(data_, register_get<N>(data_)-1); return *this; }
};
static_assert(sizeof(Register<32>) == 4, "Register class not compatible with your compiler");
/// Portable bitfields
template <unsigned B, unsigned N=1, typename T=uint8_t>
struct BitField
{
static_assert(!std::numeric_limits<T>::is_signed, "T is not unsigned");
static_assert(std::numeric_limits<T>::digits >= (int)(B+N), "T is too short");
static constexpr T mask = (1 << N)-1;
T data;
BitField& operator=(T v) { data = (data & ~(mask << B)) | ((N == 1 ? !!v : (v & mask)) << B); return *this; }
operator T() const { return (data >> B) & mask; }
};
/// Convert unsigned value to signed, the portable way
template <typename S, typename U, unsigned nbits=std::numeric_limits<U>::digits>
constexpr S unsigned_to_signed(U v) {
static_assert(std::numeric_limits<S>::is_signed, "S is not signed");
static_assert(!std::numeric_limits<U>::is_signed, "U is not unsigned");
static_assert(std::numeric_limits<S>::digits >= (int)nbits-1, "S non-sign part is shorter than nbits");
static_assert(std::numeric_limits<U>::digits >= (int)nbits-1, "U is shorter than nbits");
return (v & (1 << (nbits-1))) ? (S)v - (1 << nbits) : v;
}
template <unsigned nbits> constexpr int16_t u8_to_s8(uint8_t v) {
return unsigned_to_signed<int8_t, uint8_t, nbits>(v);
};
template <unsigned nbits> constexpr int16_t u16_to_s16(uint16_t v) {
return unsigned_to_signed<int16_t, uint16_t, nbits>(v);
};
template <unsigned nbits=8> constexpr int16_t u8_to_s16(uint8_t v) {
return unsigned_to_signed<int16_t, uint8_t, nbits>(v);
};
#endif