-
Notifications
You must be signed in to change notification settings - Fork 36
/
Copy pathRF24.h
189 lines (167 loc) · 5.13 KB
/
RF24.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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
/*
* MYSBootloader 1.3.0-rc.1
* OTA RF24 bootloader for MySensors: https://www.mysensors.org
* Based on MySensors library 2.2
* Developed and maintained by tekka 2018
*/
#ifndef RF24_H
#define RF24_H
#include "RF24_registers.h"
extern nodeConfig_t _eepromNodeConfig;
#define RF24_SETUP ( ( (RF24_DATARATE & 0x02 ) << 4) | ((RF24_DATARATE & 0x01 ) << 3) | (RF24_PA_LEVEL << 1) ) + 1
typedef enum {
RF24_PA_MIN = 0,
RF24_PA_LOW,
RF24_PA_HIGH,
RF24_PA_MAX
} rf24_pa_t;
typedef enum {
RF24_1MBPS = 0,
RF24_2MBPS,
RF24_250KBPS
} rf24_datarate_t;
#define RF24_CONFIG _BV(CRCO) | _BV(EN_CRC)
#define _write_register(reg) ( (reg) | W_REGISTER)
#define _read_register(reg) ( (reg) | R_REGISTER)
uint8_t base_addr[RF24_ADDR_WIDTH] = { RF24_BASE_RADIO_ID };
static uint8_t SPIBytes(const uint8_t addr, uint8_t* buf, uint8_t len, const bool aReadMode) {
CSN_LOW();
_delay_us(10);
uint8_t status = SPItransfer( addr );
while ( len-- ) {
if (aReadMode) {
status = SPItransfer(NOP);
if(buf!=NULL) *buf++ = status;
} else status = SPItransfer(*buf++);
}
CSN_HIGH();
_delay_us(10);
return status;
}
static inline uint8_t burstWriteAddress(const uint8_t addr, uint8_t* buf, uint8_t len) {
return SPIBytes(addr,buf,len,false);
}
static inline uint8_t writeAddress(const uint8_t addr, uint8_t buf) {
return SPIBytes(addr,&buf,1,false);
}
#define _readAddress(addr) SPIBytes(addr,NULL,1,true)
#define _burstWriteRegister(reg,buf,len) burstWriteAddress(_write_register(reg), buf, len)
#define _burstReadRegister(reg,buf,len) SPIBytes(_read_register(reg), buf, len, true)
#define _writeRegister(reg,val) writeAddress(_write_register(reg), val)
#define _readRegister(reg) _readAddress(_read_register(reg))
#define _getStatus() SPIBytes(NOP, NULL, 0, false)
#define _dataAvailable() !(_readRegister(FIFO_STATUS) & ( _BV(RX_EMPTY) ))
#define _resetInterrupts() _writeRegister(STATUS, _BV(RX_DR) | _BV(TX_DS) | _BV(MAX_RT) )
#define _flushTX() burstWriteAddress(FLUSH_TX, NULL, 0)
#define _flushRX() burstWriteAddress(FLUSH_RX, NULL, 0)
static void Flush_RXTX_CLI(void) {
// flush RX and TX buffer
_flushRX();
_flushTX();
// clear interrupts
_resetInterrupts();
}
static bool writeBuf(uint8_t* buf, uint8_t len ) {
uint8_t status;
// write payload to FIFO, broadcasts do not require ACK
_burstWriteRegister(W_TX_PAYLOAD, buf, len) ;
// CE pulse to start transmission
CE_HIGH(); // 4 us until CSN_LOW
_delay_us(4);
do {
// get status byte
status = _getStatus();
} while( !( status & ( _BV(TX_DS) | _BV(MAX_RT) )) );
CE_LOW();
Flush_RXTX_CLI();
/*
//_writeRegister(STATUS, _BV(TX_DS) | _BV(MAX_RT) );
// max retries exceeded, flush tx buffer
if(status & _BV(MAX_RT)) {
// flush TX FIFO
Flush_RXTX_CLI(); // if Flush_RXTX_CLI() used, 4 bytes saved compared to _flushTX();
}
*/
return (status & _BV(TX_DS) );
}
/*
static uint8_t getDynamicPayloadSize(void) {
uint8_t result = _readAddress(R_RX_PL_WID);
// discard payloads with length > 32
if(result > 32) {
// flush RX FIFO
//_flushRX();
Flush_RXTX_CLI(); // saves 12 bytes
result = 0;
}
return result;
}
*/
static uint8_t readMessage(uint8_t* buf) {
uint8_t len = _readAddress(R_RX_PL_WID);
if(len > 32) {
len = 0;
}
_burstReadRegister(R_RX_PAYLOAD, buf, len);
Flush_RXTX_CLI();
return len;
/*
const uint8_t len = getDynamicPayloadSize();
// read payload
_burstReadRegister(R_RX_PAYLOAD, buf, len);
// reset IRQ and FIFOs since we drive the communication and do not expect several incoming packets simultaneously
Flush_RXTX_CLI();
//_writeRegister(STATUS, _BV(RX_DR) );
//#warning maybe clear IRQ only
return len;
*/
}
static void setPipeAddress(const uint8_t pipe) {
_burstWriteRegister(pipe, base_addr, RF24_ADDR_WIDTH);
}
static bool writeMessage(const uint8_t recipient, uint8_t* buf, const uint8_t len) {
//stop listening
CE_LOW();
// switch to TX mode
_writeRegister(CONFIG, RF24_CONFIG | _BV(PWR_UP) ) ;
// flush FIFO and interrupts
//Flush_RXTX_CLI();
// set pipe 0 RX/TX address
base_addr[0] = recipient;
setPipeAddress(RX_ADDR_P0);
setPipeAddress(TX_ADDR);
// write payload
const bool result = writeBuf(buf, len);
// start listening
_writeRegister(CONFIG, RF24_CONFIG | _BV(PWR_UP) | _BV(PRIM_RX) ) ;
// set pipe0 RX address
_writeRegister(RX_ADDR_P0, _eepromNodeConfig.nodeId);
// flush FIFO and interrupts
//Flush_RXTX_CLI();
// go!
CE_HIGH(); // 4 us until CSN_LOW. No delay needed here.
return result;
}
static bool initRadio(void) {
CE_LOW();
CSN_HIGH();
// set address width
_writeRegister(SETUP_AW, RF24_ADDR_WIDTH - 2);
// set channel
_writeRegister(RF_CH, RF24_CHANNEL);
// set data rate and pa level, +1 for Si24R1
_writeRegister(RF_SETUP, RF24_SETUP);
// unlock features (certain nRF24 clones and non-P)
writeAddress(ACTIVATE, 0x73); // adds 6 bytes
// enable dynamic payload length
_writeRegister(FEATURE, _BV(EN_DPL));
// enable pipe 0, no BC RX pipe needed
_writeRegister(EN_RXADDR, _BV(ERX_P0));
// enable autoACK on pipe 0
_writeRegister(EN_AA, _BV(ENAA_P0));
// Enable dynamic payload length on pipe 0
_writeRegister(DYNPD, _BV(DPL_P0));
// sanity check
return _readRegister(RF_SETUP)==RF24_SETUP;
}
#endif // RF24_H