Blame view

RIOT/drivers/include/mrf24j40.h 11.9 KB
a752c7ab   elopes   add first test an...
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
  /*
   * Copyright (C) 2017 Neo Nenaco <neo@nenaco.de>
   * Copyright (C) 2017 Koen Zandberg <koen@bergzand.net>
   *
   * This file is subject to the terms and conditions of the GNU Lesser General
   * Public License v2.1. See the file LICENSE in the top level directory for more
   * details.
   */
  
  /**
   * @defgroup    drivers_mrf24j40 MRF24J40 based drivers
   * @ingroup     drivers_netdev
   *
   * This module contains drivers for radio devices in Microchip MRF24J40 series.
   * The driver is aimed to work with all devices of this series.
   *
   * Default TX power is 0dBm.
   *
   * TX power mapping:
   * * 0 -> -36dB
   * * 1 -> -35dB
   * * 2 -> -34dB
   * * 3 -> -33dB
   * * 4 -> -32dB
   * * 5 -> -31dB
   * * 6 -> -30dB
   * * 7 -> -30dB
   * * 8 -> -26dB
   * * 9 -> -25dB
   * * 10 -> -24dB
   * * 11 -> -23dB
   * * 12 -> -22dB
   * * 13 -> -21dB
   * * 14 -> -20dB
   * * 15 -> -20dB
   * * 16 -> -16dB
   * * 17 -> -15dB
   * * 18 -> -14dB
   * * 19 -> -13dB
   * * 20 -> -12dB
   * * 21 -> -11dB
   * * 22 -> -10dB
   * * 23 -> -10dB
   * * 24 -> -6dB
   * * 25 -> -5dB
   * * 26 -> -4dB
   * * 27 -> -3dB
   * * 28 -> -2dB
   * * 29 -> -1dB
   * * 30 -> -0dB
   * * 31 -> -0dB
   *
   * @{
   *
   * @file
   * @brief       Interface definition for MRF24J40 based drivers
   *
   * @author      Neo Nenaco <neo@nenaco.de>
   * @author      Koen Zandberg <koen@bergzand.net>
   */
  
  #ifndef MRF24J40_H
  #define MRF24J40_H
  
  #include <stdint.h>
  
  #include "board.h"
  #include "periph/spi.h"
  #include "periph/gpio.h"
  #include "net/netdev.h"
  #include "net/netdev/ieee802154.h"
  #include "net/gnrc/nettype.h"
  
  #ifdef __cplusplus
  extern "C" {
  #endif
  
  /**
   * @name    Flags for pseudo device internal states
   * @{
   */
  #define MRF24J40_PSEUDO_STATE_IDLE      (0x01)      /**< Idle, ready to transmit or receive */
  #define MRF24J40_PSEUDO_STATE_SLEEP     (0x02)      /**< sleep mode, registers functional, but no RF */
  #define MRF24J40_PSEUDO_STATE_RESET     (0x04)      /**< Reset device, next state is idle */
  /** @} */
  
  /**
   * @name    Internal device option flags
   *
   * `0x00ff` is reserved for general IEEE 802.15.4 flags
   * (see @ref netdev_ieee802154_t)
   *
   * @{
   */
  #define MRF24J40_OPT_CSMA               (0x0100)    /**< CSMA active */
  #define MRF24J40_OPT_PROMISCUOUS        (0x0200)    /**< promiscuous mode
                                                       *   active */
  #define MRF24J40_OPT_PRELOADING         (0x0400)    /**< preloading enabled */
  #define MRF24J40_OPT_TELL_TX_START      (0x0800)    /**< notify MAC layer on TX
                                                       *   start */
  #define MRF24J40_OPT_TELL_TX_END        (0x1000)    /**< notify MAC layer on TX
                                                       *   finished */
  #define MRF24J40_OPT_TELL_RX_START      (0x2000)    /**< notify MAC layer on RX
                                                       *   start */
  #define MRF24J40_OPT_TELL_RX_END        (0x4000)    /**< notify MAC layer on RX
                                                       *   finished */
  #define MRF24J40_OPT_REQ_AUTO_ACK       (0x8000)    /**< notify MAC layer on RX
                                                       *   finished */
  /** @} */
  
  
  #define MRF24J40_TASK_TX_DONE           (0x01)      /**< TX operation is done */
  #define MRF24J40_TASK_TX_READY          (0x02)      /**< TX operation results ready for processing */
  #define MRF24J40_TASK_RX_READY          (0x04)      /**< RX processing needed */
  
  #define MRF24J40_MAX_FRAME_RETRIES      (3U)        /**< Number of frame retries (fixed) */
  
  /**
   * @brief   struct holding all params needed for device initialization
   */
  typedef struct mrf24j40_params {
      spi_t spi;              /**< SPI bus the device is connected to */
      spi_clk_t spi_clk;      /**< SPI speed to use */
      spi_cs_t cs_pin;        /**< GPIO pin connected to chip select */
      gpio_t int_pin;         /**< GPIO pin connected to the interrupt pin */
      gpio_t reset_pin;       /**< GPIO pin connected to the reset pin */
  } mrf24j40_params_t;
  
  /**
   * @brief   Device descriptor for MRF24J40 radio devices
   */
  typedef struct {
      netdev_ieee802154_t netdev;             /**< netdev parent struct */
      /*  device specific fields  */
      mrf24j40_params_t params;               /**< parameters for initialization */
      uint8_t state;                          /**< current state of the radio */
      uint8_t idle_state;                     /**< state to return to after sending */
      uint8_t tx_frame_len;                   /**< length of the current TX frame */
      uint8_t header_len;                     /**< length of the header */
      uint8_t pending;                        /**< Flags for pending tasks */
      uint8_t irq_flag;                       /**< Flags for IRQs */
      uint8_t tx_retries;                     /**< Number of retries needed for last transmission */
  } mrf24j40_t;
  
  /**
   * @brief   Setup an MRF24J40 based device state
   *
   * @param[out] dev          device descriptor
   * @param[in]  params       parameters for device initialization
   */
  void mrf24j40_setup(mrf24j40_t *dev, const mrf24j40_params_t *params);
  
  /**
   * @brief   Trigger a hardware reset and configure radio with default values
   *
   * @param[in] dev           device to reset
   */
  void mrf24j40_reset(mrf24j40_t *dev);
  
  /**
   * @brief   Trigger a clear channel assessment
   *
   * @param[in] dev           device to use
   *
   * @return                  true if channel is clear
   * @return                  false if channel is busy
   */
  bool mrf24j40_cca(mrf24j40_t *dev);
  
  /**
   * @brief   Get the short address of the given device
   *
   * @param[in] dev           device to read from
   *
   * @return                  the currently set (2-byte) short address
   */
  uint16_t mrf24j40_get_addr_short(mrf24j40_t *dev);
  
  /**
   * @brief   Set the short address of the given device
   *
   * @param[in] dev           device to write to
   * @param[in] addr          (2-byte) short address to set
   */
  void mrf24j40_set_addr_short(mrf24j40_t *dev, uint16_t addr);
  
  /**
   * @brief   Get the configured long address of the given device
   *
   * @param[in] dev           device to read from
   *
   * @return                  the currently set (8-byte) long address
   */
  uint64_t mrf24j40_get_addr_long(mrf24j40_t *dev);
  
  /**
   * @brief   Set the long address of the given device
   *
   * @param[in] dev           device to write to
   * @param[in] addr          (8-byte) long address to set
   */
  void mrf24j40_set_addr_long(mrf24j40_t *dev, uint64_t addr);
  
  /**
   * @brief   Get the configured channel number of the given device
   *
   * @param[in] dev           device to read from
   *
   * @return                  the currently set channel number
   */
  uint8_t mrf24j40_get_chan(mrf24j40_t *dev);
  
  /**
   * @brief   Set the channel number of the given device
   *
   * @param[in] dev           device to write to
   * @param[in] chan          channel number to set
   */
  void mrf24j40_set_chan(mrf24j40_t *dev, uint8_t chan);
  
  /**
   * @brief   Get the configured PAN ID of the given device
   *
   * @param[in] dev           device to read from
   *
   * @return                  the currently set PAN ID
   */
  uint16_t mrf24j40_get_pan(mrf24j40_t *dev);
  
  /**
   * @brief   Set the PAN ID of the given device
   *
   * @param[in] dev           device to write to
   * @param[in] pan           PAN ID to set
   */
  void mrf24j40_set_pan(mrf24j40_t *dev, uint16_t pan);
  
  /**
   * @brief   Get the configured transmission power of the given device [in dBm]
   *
   * @param[in] dev           device to read from
   *
   * @return                  configured transmission power in dBm
   */
  int16_t mrf24j40_get_txpower(mrf24j40_t *dev);
  
  /**
   * @brief   Set the transmission power of the given device [in dBm]
   *
   * If the device does not support the exact dBm value given, it will set a value
   * as close as possible to the given value. If the given value is larger or
   * lower then the maximal or minimal possible value, the min or max value is
   * set, respectively.
   *
   * @param[in] dev           device to write to
   * @param[in] txpower       transmission power in dBm
   */
  void mrf24j40_set_txpower(mrf24j40_t *dev, int16_t txpower);
  
  /**
   * @brief   Get the maximum number of channel access attempts per frame (CSMA)
   *
   * @param[in] dev           device to read from
   *
   * @return                  configured number of retries
   */
  uint8_t mrf24j40_get_csma_max_retries(mrf24j40_t *dev);
  
  /**
   * @brief   Set the maximum number of channel access attempts per frame (CSMA)
   *
   * This setting specifies the number of attempts to access the channel to
   * transmit a frame. If the channel is busy @p retries times, then frame
   * transmission fails.
   * Valid values: 0 to 5, -1 means CSMA disabled
   *
   * @param[in] dev           device to write to
   * @param[in] retries       the maximum number of retries
   */
  void mrf24j40_set_csma_max_retries(mrf24j40_t *dev, int8_t retries);
  
  /**
   * @brief   Set the min and max backoff exponent for CSMA/CA
   *
   * - Maximum BE: 0 - 8
   * - Minimum BE: 0 - [max]
   *
   * @param[in] dev           device to write to
   * @param[in] min           the minimum BE
   * @param[in] max           the maximum BE
   */
  void mrf24j40_set_csma_backoff_exp(mrf24j40_t *dev, uint8_t min, uint8_t max);
  
  /**
   * @brief   Get the CCA threshold value
   *
   * @param[in] dev           device to read value from
   *
   * @return                  the current CCA threshold value
   */
  int8_t mrf24j40_get_cca_threshold(mrf24j40_t *dev);
  
  /**
   * @brief   Set the CCA threshold value
   *
   * @param[in] dev           device to write to
   * @param[in] value         the new CCA threshold value
   */
  void mrf24j40_set_cca_threshold(mrf24j40_t *dev, int8_t value);
  
  /**
   * @brief   Enable or disable driver specific options
   *
   * @param[in] dev           device to set/clear option flag for
   * @param[in] option        option to enable/disable
   * @param[in] state         true for enable, false for disable
   */
  void mrf24j40_set_option(mrf24j40_t *dev, uint16_t option, bool state);
  
  /**
   * @brief   Set the state of the given device (trigger a state change)
   *
   * @param[in] dev           device to change state of
   * @param[in] state         the targeted new state
   */
  void mrf24j40_set_state(mrf24j40_t *dev, uint8_t state);
  
  /**
   * @brief   Put in sleep mode
   *
   * @param[in] dev       device to put to sleep
   */
  void mrf24j40_sleep(mrf24j40_t *dev);
  
  /**
   * @brief   Put in sleep mode if idle_state is sleep
   *
   * @param[in] dev       device to put to sleep
   */
  void mrf24j40_assert_sleep(mrf24j40_t *dev);
  
  /**
   * @brief   Wake up from sleep mode
   *
   * @param[in] dev       device to eventually wake up
   */
  void mrf24j40_assert_awake(mrf24j40_t *dev);
  
  /**
   * @brief   Reset the internal state machine to TRX_OFF mode.
   *
   * This will force a transition to TRX_OFF regardless of whether the transceiver
   * is currently busy sending or receiving. This function is used to get back to
   * a known state during driver initialization.
   *
   * @param[in] dev           device to operate on
   */
  void mrf24j40_reset_state_machine(mrf24j40_t *dev);
  
  /**
   * @brief   Software Reset.
   *
   * This will force the power management circuitry, the baseband circuitry and the MAC circuitry
   * to be reset
   *
   * @param[in] dev           device to operate on
   */
  void mrf24j40_software_reset(mrf24j40_t *dev);
  
  /**
   * @brief   Prepare for sending of data
   *
   * This function puts the given device into the TX state, so no receiving of
   * data is possible after it was called.
   *
   * @param[in] dev            device to prepare for sending
   */
  void mrf24j40_tx_prepare(mrf24j40_t *dev);
  
  /**
   * @brief   Load chunks of data into the transmit buffer of the given device
   *
   * @param[in] dev           device to write data to
   * @param[in] data          buffer containing the data to load
   * @param[in] len           number of bytes in @p buffer
   * @param[in] offset        offset used when writing data to internal buffer
   *
   * @return                  offset + number of bytes written
   */
  size_t mrf24j40_tx_load(mrf24j40_t *dev, uint8_t *data, size_t len,
                          size_t offset);
  
  /**
   * @brief   Trigger sending of data previously loaded into transmit buffer
   *
   * @param[in] dev           device to trigger
   */
  void mrf24j40_tx_exec(mrf24j40_t *dev);
  
  #ifdef __cplusplus
  }
  #endif
  
  #endif /* MRF24J40_H */
  /** @} */