d1d46335df
Signed-off-by: anjiahao <anjiahao@xiaomi.com> Signed-off-by: Xiang Xiao <xiaoxiang@xiaomi.com>
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;
|
||
}
|