-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcc1101.go
439 lines (364 loc) · 8.79 KB
/
cc1101.go
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
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
package shinywaffle
import (
"context"
"flag"
"fmt"
"log"
"sync"
"time"
"github.com/hatstand/shinywaffle/config"
"github.com/kidoman/embd"
_ "github.com/kidoman/embd/host/rpi"
)
var gdo0pin = flag.Int("gdo0", 25, "GPIO pin connected to CC1101 GDO0 (BCM numbering)")
var gdo2pin = flag.Int("gdo2", 24, "GPIO pin connected to CC1101 GDO2 (BCM numbering)")
const (
// Read/write flags.
WRITE_SINGLE_BYTE = 0x00
WRITE_BURST = 0x40
READ_SINGLE_BYTE = 0x80
READ_BURST = 0xc0
BYTES_IN_RXFIFO = 0x7f
RXFIFO = 0x3f
TXFIFO = 0x3f
OVERFLOW = 0x80
// Bitmask for reading state out of chip status byte.
STATE = 0x70
CRC_OK = 0x80
RSSI = 0
LQI = 1
RSSI_OFFSET = 74
// Strobes
SRES = 0x30 // Reset
SRX = 0x34 // Set receive mode
STX = 0x35 // Set transmit mode
SIDLE = 0x36
SFRX = 0x3a // Flush RX FIFO buffer
SFTX = 0x3b // Flush TX FIFO buffer
SNOP = 0x3d
// Status Registers
PARTNUM = 0xf0
VERSION = 0xf1
RXBYTES = 0x3b
// Config Registers
IOCFG2 = 0x00
IOCFG1 = 0x01
IOCFG0 = 0x02
FIFOTHR = 0x03
SYNC1 = 0x04
SYNC0 = 0x05
PKTLEN = 0x06
PKTCTRL1 = 0x07
PKTCTRL0 = 0x08
ADDR = 0x09
CHANNR = 0x0a
FSCTRL1 = 0x0b
FSCTRL0 = 0x0c
FREQ2 = 0x0d
FREQ1 = 0x0e
FREQ0 = 0x0f
MDMCFG4 = 0x10
MDMCFG3 = 0x11
MDMCFG2 = 0x12
MDMCFG1 = 0x13
MDMCFG0 = 0x14
DEVIATN = 0x15
MCSM2 = 0x16
MCSM1 = 0x17
MCSM0 = 0x18
FOCCFG = 0x19
BSCFG = 0x1a
AGCCTRL2 = 0x1b
AGCCTRL1 = 0x1c
AGCCTRL0 = 0x1d
WOREVT1 = 0x1e
WOREVT0 = 0x1f
WORCTRL = 0x20
FREND1 = 0x21
FREND0 = 0x22
FSCAL3 = 0x23
FSCAL2 = 0x24
FSCAL1 = 0x25
FSCAL0 = 0x26
RCCTRL1 = 0x27
RCCTRL0 = 0x28
FSTEST = 0x29
PTEST = 0x2a
AGCTEST = 0x2b
TEST2 = 0x2c
TEST1 = 0x2d
TEST0 = 0x2e
)
// Copied from TI datasheet.
func convertRSSI(rssi int) int {
if rssi >= 128 {
return (rssi-256)/2 - RSSI_OFFSET
} else {
return rssi/2 - RSSI_OFFSET
}
}
type CC1101 struct {
bus embd.SPIBus
// Configured to emit a rising edge on receiving packets that pass the CRC check.
// See IOCFG0.
gdo0 embd.DigitalPin
// Configured to emit a rising edge when the sync word has been transmitted
// and to emit a falling edge once a packet has been transmitted.
// See IOCFG2.
gdo2 embd.DigitalPin
lock sync.Mutex
}
func NewCC1101(packetCh chan<- []byte) CC1101 {
err := embd.InitSPI()
if err != nil {
panic(err)
}
err = embd.InitGPIO()
if err != nil {
panic(err)
}
gdo0, err := embd.NewDigitalPin(*gdo0pin)
if err != nil {
panic(err)
}
gdo0.SetDirection(embd.In)
gdo2, err := embd.NewDigitalPin(*gdo2pin)
if err != nil {
panic(err)
}
gdo2.SetDirection(embd.In)
bus := embd.NewSPIBus(embd.SPIMode0, 0, 50000, 8, 0)
cc1101 := CC1101{
bus: bus,
gdo0: gdo0,
gdo2: gdo2,
}
cc1101.Reset()
cc1101.SelfTest()
cc1101.Init()
if packetCh != nil {
cc1101.SetIdle()
cc1101.SetRx()
log.Print("Waiting for packets...")
gdo0.Watch(embd.EdgeRising, func(pin embd.DigitalPin) {
log.Println("Packet arrived")
defer cc1101.SetRx()
defer cc1101.SetIdle()
recv, err := cc1101.Receive()
if err != nil {
log.Println("Failed to receive: ", err)
} else {
packetCh <- recv
}
})
}
return cc1101
}
func (c *CC1101) Close() {
c.Strobe(SRES)
c.bus.Close()
c.gdo0.Close()
c.gdo2.Close()
embd.CloseGPIO()
embd.CloseSPI()
}
func (cc1101 *CC1101) Strobe(address byte) (byte, error) {
data := []byte{address, 0x00}
err := cc1101.bus.TransferAndReceiveData(data)
if err != nil {
return 0, err
}
return data[0], nil
}
func (cc1101 *CC1101) ReadSingleByte(address byte) (byte, error) {
data := []byte{address | READ_SINGLE_BYTE, 0x00}
err := cc1101.bus.TransferAndReceiveData(data)
if err != nil {
return 0x00, err
}
return data[1], nil
}
func (c *CC1101) ReadBurst(address byte, num byte) ([]byte, error) {
var buf []byte
for i := byte(0); i < num+1; i++ {
addr := (address + i*8) | READ_BURST
buf = append(buf, addr)
}
err := c.bus.TransferAndReceiveData(buf)
if err != nil {
return nil, err
}
return buf[1:], nil
}
func (cc1101 *CC1101) WriteSingleByte(address byte, in byte) error {
data := []byte{address | WRITE_SINGLE_BYTE, in}
return cc1101.bus.TransferAndReceiveData(data)
}
func (c *CC1101) WriteBurst(address byte, data []byte) error {
var buf []byte
buf = append(buf, address|WRITE_BURST)
buf = append(buf, data...)
err := c.bus.TransferAndReceiveData(buf)
if err != nil {
return err
}
return nil
}
func (cc1101 *CC1101) Reset() error {
_, err := cc1101.Strobe(SRES)
return err
}
func (c *CC1101) Init() error {
c.WriteSingleByte(FSCTRL1, config.FSCTRL1)
c.WriteSingleByte(FSCTRL0, config.FSCTRL0)
c.WriteSingleByte(FREQ2, config.FREQ2)
c.WriteSingleByte(FREQ1, config.FREQ1)
c.WriteSingleByte(FREQ0, config.FREQ0)
c.WriteSingleByte(MDMCFG4, config.MDMCFG4)
c.WriteSingleByte(MDMCFG3, config.MDMCFG3)
c.WriteSingleByte(MDMCFG2, config.MDMCFG2)
c.WriteSingleByte(MDMCFG1, config.MDMCFG1)
c.WriteSingleByte(MDMCFG0, config.MDMCFG0)
c.WriteSingleByte(CHANNR, config.CHANNR)
c.WriteSingleByte(DEVIATN, config.DEVIATN)
c.WriteSingleByte(FREND1, config.FREND1)
c.WriteSingleByte(FREND0, config.FREND0)
c.WriteSingleByte(MCSM0, config.MCSM0)
c.WriteSingleByte(FOCCFG, config.FOCCFG)
c.WriteSingleByte(BSCFG, config.BSCFG)
c.WriteSingleByte(AGCCTRL2, config.AGCCTRL2)
c.WriteSingleByte(AGCCTRL1, config.AGCCTRL1)
c.WriteSingleByte(AGCCTRL0, config.AGCCTRL0)
c.WriteSingleByte(FSCAL3, config.FSCAL3)
c.WriteSingleByte(FSCAL2, config.FSCAL2)
c.WriteSingleByte(FSCAL1, config.FSCAL1)
c.WriteSingleByte(FSCAL0, config.FSCAL0)
c.WriteSingleByte(FSTEST, config.FSTEST)
c.WriteSingleByte(TEST2, config.TEST2)
c.WriteSingleByte(TEST1, config.TEST1)
c.WriteSingleByte(TEST0, config.TEST0)
c.WriteSingleByte(IOCFG2, config.IOCFG2)
c.WriteSingleByte(IOCFG1, config.IOCFG1)
c.WriteSingleByte(IOCFG0, config.IOCFG0)
// Two status bytes appended to payload: RSSI LQI and CRC OK.
c.WriteSingleByte(PKTCTRL1, config.PKTCTRL1)
// No address check, data whitening off, CRC enable, variable length packets.
c.WriteSingleByte(PKTCTRL0, config.PKTCTRL0)
c.WriteSingleByte(ADDR, config.ADDR)
// Max packet length 61 bytes.
c.WriteSingleByte(PKTLEN, config.PKTLEN)
return nil
}
func (cc1101 *CC1101) SelfTest() error {
version, err := cc1101.ReadSingleByte(VERSION)
if err != nil {
return err
}
log.Printf("Version: 0x%x", version)
partnum, err := cc1101.ReadSingleByte(PARTNUM)
if err != nil {
return err
}
log.Printf("Partnum: 0x%x", partnum)
if version != 0x14 || partnum != 0x00 {
return fmt.Errorf("Self test failed.\nGot Version: 0x%x Partnum: 0x%x", version, partnum)
}
return nil
}
func (cc1101 *CC1101) SetSyncWord(word uint16) error {
err := cc1101.WriteSingleByte(SYNC1, byte(word>>8))
if err != nil {
return err
}
return cc1101.WriteSingleByte(SYNC0, byte(word&0xff))
}
func (c *CC1101) SetState(state byte) error {
_, err := c.Strobe(state)
// Worst case state change is ~1ms for IDLE -> RX with calibration.
time.Sleep(1)
return err
}
func (c *CC1101) SetRx() error {
return c.SetState(SRX)
}
func (c *CC1101) SetTx() error {
return c.SetState(STX)
}
func (c *CC1101) SetIdle() error {
return c.SetState(SIDLE)
}
func (c *CC1101) FlushRx() {
c.SetState(SIDLE)
c.Strobe(SFRX)
}
func (c *CC1101) Receive() ([]byte, error) {
c.lock.Lock()
defer c.lock.Unlock()
rxbytes, err := c.ReadSingleByte(RXBYTES)
if err != nil {
return nil, err
}
// Flush RX buffer.
defer c.FlushRx()
if rxbytes&OVERFLOW > 0 {
return nil, fmt.Errorf("FIFO Overflow")
}
if rxbytes&BYTES_IN_RXFIFO > 0 {
numBytes, err := c.ReadSingleByte(RXFIFO)
if err != nil {
return nil, err
}
var recv []byte
if numBytes > 0 {
recv, err = c.ReadBurst(RXFIFO, numBytes)
if err != nil {
return nil, err
}
}
_, err = c.ReadBurst(RXFIFO, 2)
if err != nil {
return nil, err
}
return recv, nil
} else {
return []byte{}, nil
}
}
func (c *CC1101) Send(packet []byte) error {
c.lock.Lock()
defer c.lock.Unlock()
if len(packet) > 60 {
return fmt.Errorf("Packet too long: %d", len(packet))
}
if err := c.WriteSingleByte(TXFIFO, byte(len(packet))); err != nil {
return err
}
if err := c.WriteBurst(TXFIFO, packet); err != nil {
return err
}
c.SetTx()
defer c.Strobe(SFTX)
defer c.Strobe(SIDLE)
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()
doneCh := make(chan interface{})
if err := c.gdo2.Watch(embd.EdgeFalling, func(dp embd.DigitalPin) {
doneCh <- nil
}); err != nil {
return fmt.Errorf("Error waiting for packet send: %w", err)
}
// Watch should always be unregistered.
defer func() {
if err := c.gdo2.StopWatching(); err != nil {
log.Printf("failed to stop watching GDO2: %v", err)
}
}()
for {
select {
case <-doneCh:
return nil
case <-ctx.Done():
return fmt.Errorf("Timed out waiting for packet send")
}
}
}