2017-07-03 06:33:17 +02:00
|
|
|
|
/****************************************************************************
|
|
|
|
|
* drivers/wireless/ieee802154/mrf24j40/mrf24j40.c
|
|
|
|
|
*
|
|
|
|
|
* Copyright (C) 2015-2016 Sebastien Lorquet. All rights reserved.
|
|
|
|
|
* Copyright (C) 2017 Verge Inc. All rights reserved.
|
|
|
|
|
* Author: Sebastien Lorquet <sebastien@lorquet.fr>
|
|
|
|
|
* Author: Anthony Merlino <anthony@vergeaero.com>
|
|
|
|
|
*
|
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
|
* are met:
|
|
|
|
|
*
|
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
|
* notice, this list of conditions and the following disclaimer in
|
|
|
|
|
* the documentation and/or other materials provided with the
|
|
|
|
|
* distribution.
|
|
|
|
|
* 3. Neither the name NuttX nor the names of its contributors may be
|
|
|
|
|
* used to endorse or promote products derived from this software
|
|
|
|
|
* without specific prior written permission.
|
|
|
|
|
*
|
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
|
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
|
|
|
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
|
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
|
|
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
|
|
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
|
|
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
* Included Files
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include <nuttx/config.h>
|
2017-07-12 05:41:49 +02:00
|
|
|
|
|
2017-07-03 06:33:17 +02:00
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <debug.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <semaphore.h>
|
|
|
|
|
|
2017-07-12 05:41:49 +02:00
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
|
2017-07-03 06:33:17 +02:00
|
|
|
|
#include <nuttx/kmalloc.h>
|
|
|
|
|
#include <nuttx/wqueue.h>
|
|
|
|
|
#include <nuttx/semaphore.h>
|
|
|
|
|
|
|
|
|
|
#include <nuttx/mm/iob.h>
|
|
|
|
|
|
|
|
|
|
#include <nuttx/wireless/ieee802154/mrf24j40.h>
|
|
|
|
|
#include <nuttx/wireless/ieee802154/ieee802154_radio.h>
|
|
|
|
|
#include <nuttx/wireless/ieee802154/ieee802154_mac.h>
|
|
|
|
|
|
|
|
|
|
#include "mrf24j40.h"
|
2017-07-10 09:10:20 +02:00
|
|
|
|
#include "mrf24j40_reg.h"
|
|
|
|
|
#include "mrf24j40_radif.h"
|
2017-07-12 03:12:50 +02:00
|
|
|
|
#include "mrf24j40_getset.h"
|
2017-07-10 09:10:20 +02:00
|
|
|
|
#include "mrf24j40_regops.h"
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
* Pre-processor Definitions
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2017-07-12 03:12:50 +02:00
|
|
|
|
/****************************************************************************
|
|
|
|
|
* Private Function Prototypes
|
|
|
|
|
****************************************************************************/
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-12 03:12:50 +02:00
|
|
|
|
#if 0
|
|
|
|
|
static int mrf24j40_energydetect(FAR struct mrf24j40_radio_s *dev,
|
|
|
|
|
FAR uint8_t *energy);
|
|
|
|
|
#endif
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-12 03:12:50 +02:00
|
|
|
|
/****************************************************************************
|
|
|
|
|
* Private Functions
|
|
|
|
|
****************************************************************************/
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
2017-07-12 03:12:50 +02:00
|
|
|
|
* Name: mrf24j40_energydetect
|
|
|
|
|
*
|
|
|
|
|
* Description:
|
|
|
|
|
* Measure the RSSI level for the current channel.
|
|
|
|
|
*
|
2017-07-03 06:33:17 +02:00
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2017-07-12 03:12:50 +02:00
|
|
|
|
#if 0
|
|
|
|
|
static int mrf24j40_energydetect(FAR struct mrf24j40_radio_s *dev,
|
|
|
|
|
FAR uint8_t *energy)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
2017-07-10 09:10:20 +02:00
|
|
|
|
uint8_t reg;
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2019-02-27 15:41:08 +01:00
|
|
|
|
/* Manually enable the LNA */
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-12 03:12:50 +02:00
|
|
|
|
mrf24j40_setpamode(dev, MRF24J40_PA_ED);
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-12 03:12:50 +02:00
|
|
|
|
/* Set RSSI average duration to 8 symbols */
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-12 03:12:50 +02:00
|
|
|
|
reg = mrf24j40_getreg(dev->spi, MRF24J40_TXBCON1);
|
|
|
|
|
reg |= 0x30;
|
|
|
|
|
mrf24j40_setreg(dev->spi, MRF24J40_TXBCON1, reg);
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-12 03:12:50 +02:00
|
|
|
|
/* 1. Set RSSIMODE1 0x3E<7> – Initiate RSSI calculation. */
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-12 03:12:50 +02:00
|
|
|
|
mrf24j40_setreg(dev->spi, MRF24J40_BBREG6, 0x80);
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-12 03:12:50 +02:00
|
|
|
|
/* 2. Wait until RSSIRDY 0x3E<0> is set to ‘1’ – RSSI calculation is
|
|
|
|
|
* complete.
|
|
|
|
|
*/
|
|
|
|
|
|
2019-02-27 15:41:08 +01:00
|
|
|
|
while (!(mrf24j40_getreg(dev->spi, MRF24J40_BBREG6) & 0x01));
|
2017-07-12 03:12:50 +02:00
|
|
|
|
|
|
|
|
|
/* 3. Read RSSI 0x210<7:0> – The RSSI register contains the averaged RSSI
|
|
|
|
|
* received power level for 8 symbol periods.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
*energy = mrf24j40_getreg(dev->spi, MRF24J40_RSSI);
|
|
|
|
|
mrf24j40_setreg(dev->spi, MRF24J40_BBREG6, 0x40);
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-12 03:12:50 +02:00
|
|
|
|
/* Back to automatic control */
|
|
|
|
|
|
|
|
|
|
mrf24j40_setpamode(dev, MRF24J40_PA_AUTO);
|
|
|
|
|
|
|
|
|
|
return OK;
|
2017-07-03 06:33:17 +02:00
|
|
|
|
}
|
2017-07-12 03:12:50 +02:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
* Public Functions
|
|
|
|
|
****************************************************************************/
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
2017-07-10 09:10:20 +02:00
|
|
|
|
* Function: mrf24j40_dopoll_csma
|
2017-07-03 06:33:17 +02:00
|
|
|
|
*
|
|
|
|
|
* Description:
|
2017-07-10 09:10:20 +02:00
|
|
|
|
* This function is called in order to preform an out-of-sequence TX poll.
|
|
|
|
|
* This is done:
|
|
|
|
|
*
|
|
|
|
|
* 1. After completion of a transmission (mrf24j40_txdone_csma),
|
|
|
|
|
* 2. When new TX data is available (mrf24j40_txnotify), and
|
|
|
|
|
* 3. After a TX timeout to restart the sending process
|
|
|
|
|
* (mrf24j40_txtimeout_csma).
|
2017-07-03 06:33:17 +02:00
|
|
|
|
*
|
2018-03-13 16:52:27 +01:00
|
|
|
|
* Input Parameters:
|
2017-07-03 06:33:17 +02:00
|
|
|
|
* radio - Reference to the radio driver state structure
|
|
|
|
|
*
|
|
|
|
|
* Returned Value:
|
|
|
|
|
* None
|
|
|
|
|
*
|
|
|
|
|
* Assumptions:
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
void mrf24j40_dopoll_csma(FAR void *arg)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
2017-07-10 09:10:20 +02:00
|
|
|
|
FAR struct mrf24j40_radio_s *dev = (FAR struct mrf24j40_radio_s *)arg;
|
|
|
|
|
int len = 0;
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
/* Get exclusive access to the driver */
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-10-04 23:22:27 +02:00
|
|
|
|
while (nxsem_wait(&dev->exclsem) < 0)
|
|
|
|
|
{
|
|
|
|
|
}
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
/* If this a CSMA transaction and we have room in the CSMA fifo */
|
|
|
|
|
|
|
|
|
|
if (!dev->csma_busy)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
2017-07-10 09:10:20 +02:00
|
|
|
|
wlinfo("Polling for frame\n");
|
|
|
|
|
len = dev->radiocb->poll(dev->radiocb, false, &dev->csma_desc);
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
if (len > 0)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
2017-07-10 09:10:20 +02:00
|
|
|
|
wlinfo("Frame received. Frame length: %d\n", len);
|
|
|
|
|
|
|
|
|
|
/* Now the txdesc is in use */
|
|
|
|
|
|
|
|
|
|
dev->csma_busy = 1;
|
|
|
|
|
|
|
|
|
|
/* Setup the transaction on the device in the CSMA FIFO */
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
mrf24j40_norm_setup(dev, dev->csma_desc->frame, true);
|
|
|
|
|
mrf24j40_norm_trigger(dev);
|
2017-07-03 06:33:17 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-03 23:35:24 +02:00
|
|
|
|
nxsem_post(&dev->exclsem);
|
2017-07-03 06:33:17 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
2017-07-10 09:10:20 +02:00
|
|
|
|
* Function: mrf24j40_dopoll_gts
|
2017-07-03 06:33:17 +02:00
|
|
|
|
*
|
|
|
|
|
* Description:
|
2017-07-10 09:10:20 +02:00
|
|
|
|
* This function is called in order to preform an out-of-sequence TX poll.
|
|
|
|
|
* This is done:
|
|
|
|
|
*
|
|
|
|
|
* 1. After completion of a transmission (mrf24j40_txdone_gts),
|
|
|
|
|
* 2. When new TX data is available (mrf24j40_txnotify), and
|
|
|
|
|
* 3. After a TX timeout to restart the sending process
|
|
|
|
|
* (mrf24j40_txtimeout_gts).
|
2017-07-03 06:33:17 +02:00
|
|
|
|
*
|
2018-03-13 16:52:27 +01:00
|
|
|
|
* Input Parameters:
|
2017-07-10 09:10:20 +02:00
|
|
|
|
* arg - Reference to the radio driver state structure
|
2017-07-03 06:33:17 +02:00
|
|
|
|
*
|
|
|
|
|
* Returned Value:
|
|
|
|
|
* None
|
|
|
|
|
*
|
|
|
|
|
* Assumptions:
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
void mrf24j40_dopoll_gts(FAR void *arg)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
2017-07-10 09:10:20 +02:00
|
|
|
|
FAR struct mrf24j40_radio_s *dev = (FAR struct mrf24j40_radio_s *)arg;
|
|
|
|
|
int gts = 0;
|
|
|
|
|
int len = 0;
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
/* Get exclusive access to the driver */
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-10-04 23:22:27 +02:00
|
|
|
|
while (nxsem_wait(&dev->exclsem) < 0)
|
|
|
|
|
{
|
|
|
|
|
}
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
for (gts = 0; gts < MRF24J40_GTS_SLOTS; gts++)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
2017-07-10 09:10:20 +02:00
|
|
|
|
if (!dev->gts_busy[gts])
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
2017-07-10 09:10:20 +02:00
|
|
|
|
len = dev->radiocb->poll(dev->radiocb, true, &dev->gts_desc[gts]);
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
if (len > 0)
|
|
|
|
|
{
|
|
|
|
|
/* Now the txdesc is in use */
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2019-02-27 15:41:08 +01:00
|
|
|
|
dev->gts_busy[gts] = 1;
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
/* Setup the transaction on the device in the open GTS FIFO */
|
|
|
|
|
|
|
|
|
|
mrf24j40_gts_setup(dev, gts, dev->gts_desc[gts]->frame);
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-07-03 06:33:17 +02:00
|
|
|
|
}
|
|
|
|
|
|
2017-10-03 23:35:24 +02:00
|
|
|
|
nxsem_post(&dev->exclsem);
|
2017-07-03 06:33:17 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
* Name: mrf24j40_norm_setup
|
|
|
|
|
*
|
|
|
|
|
* Description:
|
|
|
|
|
* Setup a transaction in the normal TX FIFO
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
void mrf24j40_norm_setup(FAR struct mrf24j40_radio_s *dev,
|
|
|
|
|
FAR struct iob_s *frame, bool csma)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
|
|
|
|
uint8_t reg;
|
|
|
|
|
|
|
|
|
|
/* Enable tx int */
|
|
|
|
|
|
|
|
|
|
reg = mrf24j40_getreg(dev->spi, MRF24J40_INTCON);
|
|
|
|
|
reg &= ~MRF24J40_INTCON_TXNIE;
|
|
|
|
|
mrf24j40_setreg(dev->spi, MRF24J40_INTCON, reg);
|
|
|
|
|
|
|
|
|
|
/* Enable/Disable CSMA mode */
|
|
|
|
|
|
|
|
|
|
reg = mrf24j40_getreg(dev->spi, MRF24J40_TXMCR);
|
|
|
|
|
|
|
|
|
|
if (csma)
|
|
|
|
|
{
|
|
|
|
|
reg &= ~MRF24J40_TXMCR_NOCSMA;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
reg |= MRF24J40_TXMCR_NOCSMA;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mrf24j40_setreg(dev->spi, MRF24J40_TXMCR, reg);
|
|
|
|
|
|
|
|
|
|
/* Setup the FIFO */
|
|
|
|
|
|
2019-02-27 15:41:08 +01:00
|
|
|
|
mrf24j40_setup_fifo(dev, frame->io_data, frame->io_len,
|
|
|
|
|
MRF24J40_TXNORM_FIFO);
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
|
|
|
|
/* If the frame control field contains an acknowledgment request, set the
|
|
|
|
|
* TXNACKREQ bit. See IEEE 802.15.4/2003 7.2.1.1 page 112 for info.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
reg = mrf24j40_getreg(dev->spi, MRF24J40_TXNCON);
|
|
|
|
|
|
|
|
|
|
if (frame->io_data[0] & IEEE802154_FRAMECTRL_ACKREQ)
|
|
|
|
|
{
|
|
|
|
|
reg |= MRF24J40_TXNCON_TXNACKREQ;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
reg &= ~MRF24J40_TXNCON_TXNACKREQ;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mrf24j40_setreg(dev->spi, MRF24J40_TXNCON, reg);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
* Name: mrf24j40_norm_trigger
|
|
|
|
|
*
|
|
|
|
|
* Description:
|
|
|
|
|
* Trigger the normal TX FIFO
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
void mrf24j40_norm_trigger(FAR struct mrf24j40_radio_s *dev)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
|
|
|
|
uint8_t reg;
|
|
|
|
|
|
|
|
|
|
reg = mrf24j40_getreg(dev->spi, MRF24J40_TXNCON);
|
|
|
|
|
reg |= MRF24J40_TXNCON_TXNTRIG;
|
|
|
|
|
mrf24j40_setreg(dev->spi, MRF24J40_TXNCON, reg);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
* Name: mrf24j40_beacon_trigger
|
|
|
|
|
*
|
|
|
|
|
* Description:
|
|
|
|
|
* Trigger the beacon TX FIFO
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
void mrf24j40_beacon_trigger(FAR struct mrf24j40_radio_s *dev)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
|
|
|
|
uint8_t reg;
|
|
|
|
|
|
|
|
|
|
reg = mrf24j40_getreg(dev->spi, MRF24J40_TXBCON0);
|
|
|
|
|
|
|
|
|
|
reg |= MRF24J40_TXBCON0_TXBTRIG;
|
|
|
|
|
|
|
|
|
|
mrf24j40_setreg(dev->spi, MRF24J40_TXBCON0, reg);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
* Name: mrf24j40_gts_setup
|
|
|
|
|
*
|
|
|
|
|
* Description:
|
|
|
|
|
* Setup a GTS transaction in one of the GTS FIFOs
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2017-07-10 09:10:20 +02:00
|
|
|
|
void mrf24j40_gts_setup(FAR struct mrf24j40_radio_s *dev, uint8_t fifo,
|
|
|
|
|
FAR struct iob_s *frame)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
* Name: mrf24j40_setup_fifo
|
|
|
|
|
*
|
|
|
|
|
* Description:
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2019-02-27 15:41:08 +01:00
|
|
|
|
void mrf24j40_setup_fifo(FAR struct mrf24j40_radio_s *dev,
|
|
|
|
|
FAR const uint8_t *buf, uint8_t length,
|
|
|
|
|
uint32_t fifo_addr)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
int hlen = 3; /* Include frame control and seq number */
|
|
|
|
|
int i;
|
|
|
|
|
uint16_t frame_ctrl;
|
|
|
|
|
|
|
|
|
|
/* Analyze frame control to compute header length */
|
|
|
|
|
|
|
|
|
|
frame_ctrl = buf[0];
|
|
|
|
|
frame_ctrl |= (buf[1] << 8);
|
|
|
|
|
|
2019-02-27 15:41:08 +01:00
|
|
|
|
if ((frame_ctrl & IEEE802154_FRAMECTRL_DADDR) ==
|
|
|
|
|
IEEE802154_ADDRMODE_SHORT)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
|
|
|
|
hlen += 2 + 2; /* Destination PAN + shortaddr */
|
|
|
|
|
}
|
2019-02-27 15:41:08 +01:00
|
|
|
|
else if ((frame_ctrl & IEEE802154_FRAMECTRL_DADDR) ==
|
|
|
|
|
IEEE802154_ADDRMODE_EXTENDED)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
|
|
|
|
hlen += 2 + 8; /* Destination PAN + extaddr */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!(frame_ctrl & IEEE802154_FRAMECTRL_PANIDCOMP))
|
|
|
|
|
{
|
|
|
|
|
hlen += 2; /* No PAN compression, source PAN is different from dest PAN */
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-27 15:41:08 +01:00
|
|
|
|
if ((frame_ctrl & IEEE802154_FRAMECTRL_SADDR) ==
|
|
|
|
|
IEEE802154_ADDRMODE_SHORT)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
|
|
|
|
hlen += 2; /* Source saddr */
|
|
|
|
|
}
|
2019-02-27 15:41:08 +01:00
|
|
|
|
else if ((frame_ctrl & IEEE802154_FRAMECTRL_SADDR) ==
|
|
|
|
|
IEEE802154_ADDRMODE_EXTENDED)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
|
|
|
|
hlen += 8; /* Ext saddr */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Header len, 0, TODO for security modes */
|
|
|
|
|
|
|
|
|
|
mrf24j40_setreg(dev->spi, fifo_addr++, hlen);
|
|
|
|
|
|
|
|
|
|
/* Frame length */
|
|
|
|
|
|
|
|
|
|
mrf24j40_setreg(dev->spi, fifo_addr++, length);
|
|
|
|
|
|
|
|
|
|
/* Frame data */
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < length; i++)
|
|
|
|
|
{
|
|
|
|
|
mrf24j40_setreg(dev->spi, fifo_addr++, buf[i]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
* Public Functions
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
* Name: mrf24j40_init
|
|
|
|
|
*
|
|
|
|
|
* Description:
|
|
|
|
|
* Return an mrf24j40 device for use by other drivers.
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2017-11-03 02:17:33 +01:00
|
|
|
|
FAR struct ieee802154_radio_s *
|
|
|
|
|
mrf24j40_init(FAR struct spi_dev_s *spi,
|
|
|
|
|
FAR const struct mrf24j40_lower_s *lower)
|
2017-07-03 06:33:17 +02:00
|
|
|
|
{
|
|
|
|
|
FAR struct mrf24j40_radio_s *dev;
|
|
|
|
|
|
|
|
|
|
dev = kmm_zalloc(sizeof(struct mrf24j40_radio_s));
|
|
|
|
|
if (dev == NULL)
|
|
|
|
|
{
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Attach irq */
|
|
|
|
|
|
|
|
|
|
if (lower->attach(lower, mrf24j40_interrupt, dev) != OK)
|
|
|
|
|
{
|
|
|
|
|
#if 0
|
|
|
|
|
free(dev);
|
|
|
|
|
#endif
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Allow exclusive access to the privmac struct */
|
|
|
|
|
|
2017-10-03 20:51:15 +02:00
|
|
|
|
nxsem_init(&dev->exclsem, 0, 1);
|
2017-07-03 06:33:17 +02:00
|
|
|
|
|
|
|
|
|
dev->radio.bind = mrf24j40_bind;
|
|
|
|
|
dev->radio.reset = mrf24j40_reset;
|
|
|
|
|
dev->radio.getattr = mrf24j40_getattr;
|
|
|
|
|
dev->radio.setattr = mrf24j40_setattr;
|
|
|
|
|
dev->radio.txnotify = mrf24j40_txnotify;
|
|
|
|
|
dev->radio.txdelayed = mrf24j40_txdelayed;
|
|
|
|
|
dev->radio.rxenable = mrf24j40_rxenable;
|
2019-08-10 19:03:24 +02:00
|
|
|
|
dev->radio.energydetect = mrf24j40_energydetect;
|
2017-07-03 06:33:17 +02:00
|
|
|
|
dev->radio.beaconstart = mrf24j40_beaconstart;
|
|
|
|
|
dev->radio.beaconupdate = mrf24j40_beaconupdate;
|
|
|
|
|
dev->radio.beaconstop = mrf24j40_beaconstop;
|
|
|
|
|
dev->radio.sfupdate = mrf24j40_sfupdate;
|
|
|
|
|
|
|
|
|
|
dev->lower = lower;
|
|
|
|
|
dev->spi = spi;
|
|
|
|
|
|
|
|
|
|
mrf24j40_reset(&dev->radio);
|
|
|
|
|
|
|
|
|
|
dev->lower->enable(dev->lower, true);
|
|
|
|
|
return &dev->radio;
|
|
|
|
|
}
|
|
|
|
|
|