-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathLinkGPIO.hpp
118 lines (102 loc) · 3.58 KB
/
LinkGPIO.hpp
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
#ifndef LINK_GPIO_H
#define LINK_GPIO_H
// --------------------------------------------------------------------------
// A General Purpose Input-Output handler for the Link Port.
// --------------------------------------------------------------------------
// Usage:
// - 1) Include this header in your main.cpp file and add:
// LinkGPIO* linkGPIO = new LinkGPIO();
// - 2) Initialize the library with:
// linkGPIO->reset();
// - 3) Write pins by using:
// linkGPIO->setMode(LinkGPIO::Pin::SD, LinkGPIO::Direction::OUTPUT);
// linkGPIO->writePin(LinkGPIO::Pin::SD, true);
// - 4) Read pins by using:
// linkGPIO->setMode(LinkGPIO::Pin::SC, LinkGPIO::Direction::INPUT);
// bool isHigh = linkGPIO->readPin(LinkGPIO::Pin::SC);
// - 5) Subscribe to SI falling:
// linkGPIO->setSIInterrupts(true);
// // (when SI changes from high to low, an IRQ will be generated)
// --------------------------------------------------------------------------
// considerations:
// - always set the SI terminal to an input!
// - call reset() when you finish doing GPIO stuff!
// --------------------------------------------------------------------------
#ifndef LINK_DEVELOPMENT
#pragma GCC system_header
#endif
#include "_link_common.hpp"
/**
* @brief A General Purpose Input-Output handler for the Link Port.
*/
class LinkGPIO {
private:
using u32 = unsigned int;
using u16 = unsigned short;
using u8 = unsigned char;
static constexpr int RCNT_GENERAL_PURPOSE = (1 << 15);
static constexpr int SIOCNT_GENERAL_PURPOSE = 0;
static constexpr int BIT_SI_INTERRUPT = 8;
static constexpr u8 DATA_BITS[] = {2, 3, 1, 0};
static constexpr u8 DIRECTION_BITS[] = {6, 7, 5, 4};
public:
enum Pin { SI, SO, SD, SC };
enum Direction { INPUT, OUTPUT };
/**
* @brief Resets communication mode to General Purpose (same as
* `Link::reset()`).
* \warning Required to initialize the library!
*/
void reset() {
Link::_REG_RCNT = RCNT_GENERAL_PURPOSE;
Link::_REG_SIOCNT = SIOCNT_GENERAL_PURPOSE;
}
/**
* @brief Configures a `pin` to use a `direction` (input or output).
* @param pin One of the enum values from `LinkGPIO::Pin`.
* @param direction One of the enum values from `LinkGPIO::Direction`.
*/
void setMode(Pin pin, Direction direction) {
setBit(Link::_REG_RCNT, DIRECTION_BITS[pin],
direction == Direction::OUTPUT);
}
/**
* @brief Returns the direction set at `pin`.
*/
[[nodiscard]] Direction getMode(Pin pin) {
return Direction(getBit(Link::_REG_RCNT, DIRECTION_BITS[pin]));
}
/**
* @brief Returns whether a `pin` is *HIGH* or not (when set as an input).
* @param pin One of the enum values from `LinkGPIO::Pin`.
*/
[[nodiscard]] bool readPin(Pin pin) {
return (Link::_REG_RCNT & (1 << DATA_BITS[pin])) != 0;
}
/**
* @brief Sets a `pin` to be high or not (when set as an output).
* @param pin One of the enum values from `LinkGPIO::Pin`.
* @param isHigh `true` = HIGH, `false` = LOW.
*/
void writePin(Pin pin, bool isHigh) {
setBit(Link::_REG_RCNT, DATA_BITS[pin], isHigh);
}
/**
* @brief If it `isEnabled`, an IRQ will be generated when `SI` changes from
* HIGH to LOW.
* @param isEnabled Enable SI-falling interrupts.
*/
void setSIInterrupts(bool isEnabled) {
setBit(Link::_REG_RCNT, BIT_SI_INTERRUPT, isEnabled);
}
private:
int getBit(u16 reg, int bit) { return (reg >> bit) & 1; }
void setBit(volatile u16& reg, int bit, bool data) {
if (data)
reg |= 1 << bit;
else
reg &= ~(1 << bit);
}
};
extern LinkGPIO* linkGPIO;
#endif // LINK_GPIO_H