463 lines
12 KiB
C
463 lines
12 KiB
C
/****************************************************************************
|
|
* drivers/wireless/ieee802154/mrf24j40/mrf24j40.c
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed with
|
|
* this work for additional information regarding copyright ownership. The
|
|
* ASF licenses this file to you under the Apache License, Version 2.0 (the
|
|
* "License"); you may not use this file except in compliance with the
|
|
* License. You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
* License for the specific language governing permissions and limitations
|
|
* under the License.
|
|
*
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Included Files
|
|
****************************************************************************/
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
#include <assert.h>
|
|
#include <debug.h>
|
|
#include <stdio.h>
|
|
#include <stdint.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <nuttx/kmalloc.h>
|
|
#include <nuttx/wqueue.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"
|
|
#include "mrf24j40_reg.h"
|
|
#include "mrf24j40_radif.h"
|
|
#include "mrf24j40_getset.h"
|
|
#include "mrf24j40_regops.h"
|
|
|
|
/****************************************************************************
|
|
* Pre-processor Definitions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Private Function Prototypes
|
|
****************************************************************************/
|
|
|
|
#if 0
|
|
static int mrf24j40_energydetect(FAR struct mrf24j40_radio_s *dev,
|
|
FAR uint8_t *energy);
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Private Functions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Name: mrf24j40_energydetect
|
|
*
|
|
* Description:
|
|
* Measure the RSSI level for the current channel.
|
|
*
|
|
****************************************************************************/
|
|
|
|
#if 0
|
|
static int mrf24j40_energydetect(FAR struct mrf24j40_radio_s *dev,
|
|
FAR uint8_t *energy)
|
|
{
|
|
uint8_t reg;
|
|
|
|
/* Manually enable the LNA */
|
|
|
|
mrf24j40_setpamode(dev, MRF24J40_PA_ED);
|
|
|
|
/* Set RSSI average duration to 8 symbols */
|
|
|
|
reg = mrf24j40_getreg(dev->spi, MRF24J40_TXBCON1);
|
|
reg |= 0x30;
|
|
mrf24j40_setreg(dev->spi, MRF24J40_TXBCON1, reg);
|
|
|
|
/* 1. Set RSSIMODE1 0x3E<7> - Initiate RSSI calculation. */
|
|
|
|
mrf24j40_setreg(dev->spi, MRF24J40_BBREG6, 0x80);
|
|
|
|
/* 2. Wait until RSSIRDY 0x3E<0> is set to '1' - RSSI calculation is
|
|
* complete.
|
|
*/
|
|
|
|
while (!(mrf24j40_getreg(dev->spi, MRF24J40_BBREG6) & 0x01));
|
|
|
|
/* 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);
|
|
|
|
/* Back to automatic control */
|
|
|
|
mrf24j40_setpamode(dev, MRF24J40_PA_AUTO);
|
|
|
|
return OK;
|
|
}
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Public Functions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Function: mrf24j40_dopoll_csma
|
|
*
|
|
* Description:
|
|
* This function is called in order to perform 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).
|
|
*
|
|
* Input Parameters:
|
|
* radio - Reference to the radio driver state structure
|
|
*
|
|
* Returned Value:
|
|
* None
|
|
*
|
|
* Assumptions:
|
|
*
|
|
****************************************************************************/
|
|
|
|
void mrf24j40_dopoll_csma(FAR void *arg)
|
|
{
|
|
FAR struct mrf24j40_radio_s *dev = (FAR struct mrf24j40_radio_s *)arg;
|
|
int len = 0;
|
|
|
|
/* Get exclusive access to the driver */
|
|
|
|
while (nxmutex_lock(&dev->lock) < 0)
|
|
{
|
|
}
|
|
|
|
/* If this a CSMA transaction and we have room in the CSMA fifo */
|
|
|
|
if (!dev->csma_busy)
|
|
{
|
|
wlinfo("Polling for frame\n");
|
|
len = dev->radiocb->poll(dev->radiocb, false, &dev->csma_desc);
|
|
|
|
if (len > 0)
|
|
{
|
|
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 */
|
|
|
|
mrf24j40_norm_setup(dev, dev->csma_desc->frame, true);
|
|
mrf24j40_norm_trigger(dev);
|
|
}
|
|
}
|
|
|
|
nxmutex_unlock(&dev->lock);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Function: mrf24j40_dopoll_gts
|
|
*
|
|
* Description:
|
|
* This function is called in order to perform 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).
|
|
*
|
|
* Input Parameters:
|
|
* arg - Reference to the radio driver state structure
|
|
*
|
|
* Returned Value:
|
|
* None
|
|
*
|
|
* Assumptions:
|
|
*
|
|
****************************************************************************/
|
|
|
|
void mrf24j40_dopoll_gts(FAR void *arg)
|
|
{
|
|
FAR struct mrf24j40_radio_s *dev = (FAR struct mrf24j40_radio_s *)arg;
|
|
int gts = 0;
|
|
int len = 0;
|
|
|
|
/* Get exclusive access to the driver */
|
|
|
|
while (nxmutex_lock(&dev->lock) < 0)
|
|
{
|
|
}
|
|
|
|
for (gts = 0; gts < MRF24J40_GTS_SLOTS; gts++)
|
|
{
|
|
if (!dev->gts_busy[gts])
|
|
{
|
|
len = dev->radiocb->poll(dev->radiocb, true, &dev->gts_desc[gts]);
|
|
|
|
if (len > 0)
|
|
{
|
|
/* Now the txdesc is in use */
|
|
|
|
dev->gts_busy[gts] = 1;
|
|
|
|
/* Setup the transaction on the device in the open GTS FIFO */
|
|
|
|
mrf24j40_gts_setup(dev, gts, dev->gts_desc[gts]->frame);
|
|
}
|
|
}
|
|
}
|
|
|
|
nxmutex_unlock(&dev->lock);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: mrf24j40_norm_setup
|
|
*
|
|
* Description:
|
|
* Setup a transaction in the normal TX FIFO
|
|
*
|
|
****************************************************************************/
|
|
|
|
void mrf24j40_norm_setup(FAR struct mrf24j40_radio_s *dev,
|
|
FAR struct iob_s *frame, bool csma)
|
|
{
|
|
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 */
|
|
|
|
mrf24j40_setup_fifo(dev, frame->io_data, frame->io_len,
|
|
MRF24J40_TXNORM_FIFO);
|
|
|
|
/* 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
|
|
*
|
|
****************************************************************************/
|
|
|
|
void mrf24j40_norm_trigger(FAR struct mrf24j40_radio_s *dev)
|
|
{
|
|
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
|
|
*
|
|
****************************************************************************/
|
|
|
|
void mrf24j40_beacon_trigger(FAR struct mrf24j40_radio_s *dev)
|
|
{
|
|
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
|
|
*
|
|
****************************************************************************/
|
|
|
|
void mrf24j40_gts_setup(FAR struct mrf24j40_radio_s *dev, uint8_t fifo,
|
|
FAR struct iob_s *frame)
|
|
{
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: mrf24j40_setup_fifo
|
|
*
|
|
* Description:
|
|
*
|
|
****************************************************************************/
|
|
|
|
void mrf24j40_setup_fifo(FAR struct mrf24j40_radio_s *dev,
|
|
FAR const uint8_t *buf, uint8_t length,
|
|
uint32_t fifo_addr)
|
|
{
|
|
uint16_t frame_ctrl;
|
|
uint16_t addrmode;
|
|
int hlen = 3; /* Include frame control and seq number */
|
|
int i;
|
|
|
|
/* Analyze frame control to compute header length */
|
|
|
|
frame_ctrl = buf[0];
|
|
frame_ctrl |= (buf[1] << 8);
|
|
|
|
addrmode = (frame_ctrl & IEEE802154_FRAMECTRL_DADDR) >>
|
|
IEEE802154_FRAMECTRL_SHIFT_DADDR;
|
|
|
|
if (addrmode == IEEE802154_ADDRMODE_SHORT)
|
|
{
|
|
hlen += 2 + 2; /* Destination PAN + shortaddr */
|
|
}
|
|
else if (addrmode == IEEE802154_ADDRMODE_EXTENDED)
|
|
{
|
|
hlen += 2 + 8; /* Destination PAN + extaddr */
|
|
}
|
|
|
|
if (!(frame_ctrl & IEEE802154_FRAMECTRL_PANIDCOMP))
|
|
{
|
|
hlen += 2; /* No PAN compression, source PAN is different from dest PAN */
|
|
}
|
|
|
|
addrmode = (frame_ctrl & IEEE802154_FRAMECTRL_SADDR) >>
|
|
IEEE802154_FRAMECTRL_SHIFT_SADDR;
|
|
|
|
if (addrmode == IEEE802154_ADDRMODE_SHORT)
|
|
{
|
|
hlen += 2; /* Source saddr */
|
|
}
|
|
else if (addrmode == IEEE802154_ADDRMODE_EXTENDED)
|
|
{
|
|
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.
|
|
*
|
|
****************************************************************************/
|
|
|
|
FAR struct ieee802154_radio_s *
|
|
mrf24j40_init(FAR struct spi_dev_s *spi,
|
|
FAR const struct mrf24j40_lower_s *lower)
|
|
{
|
|
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)
|
|
{
|
|
kmm_free(dev);
|
|
return NULL;
|
|
}
|
|
|
|
/* Allow exclusive access to the privmac struct */
|
|
|
|
nxmutex_init(&dev->lock);
|
|
|
|
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;
|
|
dev->radio.energydetect = mrf24j40_energydetect;
|
|
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;
|
|
}
|