前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >LoRaWAN实战 中国470频段的代码实现

LoRaWAN实战 中国470频段的代码实现

作者头像
twowinter
发布2020-04-17 14:42:34
1.1K0
发布2020-04-17 14:42:34
举报
文章被收录于专栏:twowinter

前言

LoRaWAN协议中文版_配套文件 地区参数(物理层)中已经为中国规划了470频段,因此国内开发者对此需求很强烈。

在最新(2017-02-27)的V4.3.1版本协议栈上已经新增了中国470频段。这篇文章从源码角度解析下其实现方式。

目前国内的LoRaWAN基站产品都和标准有一些不同,比如CLAA等,所以搞清楚整个代码实现还是很有必要的。只要熟悉了整个流程,对接任何一个基站都不是难事。

我正在陆续对协议的各个章节进行翻译,具体其他章节的译文,以及译文之外的代码解析,可点此查看帖子LoRa学习笔记_汇总

本文作者twowinter,转载请注明作者:http://blog.csdn.net/iotisan/

源码解析

1.前导码格式的源码实现

同步字的处理在SX1276的驱动中:

代码语言:javascript
复制
void SX1276SetPublicNetwork( bool enable )
{
	SX1276SetModem( MODEM_LORA );
	if( enable == true )
	{
		// Change LoRa modem SyncWord
		SX1276Write( REG_LR_SYNCWORD, LORA_MAC_PUBLIC_SYNCWORD );
	}
	else
	{
		// Change LoRa modem SyncWord
		SX1276Write( REG_LR_SYNCWORD, LORA_MAC_PRIVATE_SYNCWORD );
	}
}

而前导码长度则是在每次SetTxConfig和SetRxConfig时配置进去。

2.信道频率的源码实现

先说上行信道的处理。

第一步,初始化时把所有信道6*16=96个上行信道都使能了。

代码语言:javascript
复制
	LoRaMacParamsDefaults.ChannelsMask[0] = 0xFFFF;
	LoRaMacParamsDefaults.ChannelsMask[1] = 0xFFFF;
	LoRaMacParamsDefaults.ChannelsMask[2] = 0xFFFF;
	LoRaMacParamsDefaults.ChannelsMask[3] = 0xFFFF;
	LoRaMacParamsDefaults.ChannelsMask[4] = 0xFFFF;
	LoRaMacParamsDefaults.ChannelsMask[5] = 0xFFFF;

第二步,紧接着把96个信道的频点赋值一遍。

代码语言:javascript
复制
	for( uint8_t i = 0; i < LORA_MAX_NB_CHANNELS; i++ )
	{
		Channels[i].Frequency = 470.3e6 + i * 200e3;
		Channels[i].DrRange.Value = ( DR_5 << 4 ) | DR_0;
		Channels[i].Band = 0;
	}

第三步,发送时在SetNextChannel中选择合适的频点,默认是96个信道中随机选择。

代码语言:javascript
复制
Channel = enabledChannels[randr( 0, nbEnabledChannels - 1 )];

这上面是上行信道处理三部曲,下行信道处理则轻松多了。主要是配合接收窗口处理,由这个宏定义了下行的起始频点。具体可以看下面第7点。

代码语言:javascript
复制
#define LORAMAC_FIRST_RX1_CHANNEL           ( (uint32_t) 500.3e6 )

3.数据速率和节点发射功率编码

速率编码如下:

代码语言:javascript
复制
const uint8_t Datarates[]  = { 12, 11, 10,  9,  8,  7 };

发射功率编码如下:

代码语言:javascript
复制
const int8_t TxPowers[]    = { 17, 16, 14, 12, 10, 7, 5, 2 };

速率范围如下:

代码语言:javascript
复制
/*!
 * Minimal datarate that can be used by the node
 */
#define LORAMAC_TX_MIN_DATARATE                     DR_0

/*!
 * Maximal datarate that can be used by the node
 */
#define LORAMAC_TX_MAX_DATARATE                     DR_5

/*!
 * Minimal datarate that can be used by the node
 */
#define LORAMAC_RX_MIN_DATARATE                     DR_0

/*!
 * Maximal datarate that can be used by the node
 */
#define LORAMAC_RX_MAX_DATARATE                     DR_5

/*!
 * Default datarate used by the node
 */
#define LORAMAC_DEFAULT_DATARATE                    DR_0

发射功率范围如下:

代码语言:javascript
复制
/*!
 * Minimal Tx output power that can be used by the node
 */
#define LORAMAC_MIN_TX_POWER                        TX_POWER_2_DBM

/*!
 * Maximal Tx output power that can be used by the node
 */
#define LORAMAC_MAX_TX_POWER                        TX_POWER_17_DBM

/*!
 * Default Tx output power used by the node
 */
#define LORAMAC_DEFAULT_TX_POWER                    TX_POWER_14_DBM

4.CFList

中国没有。具体见OnRadioRxDone中的FRAME_TYPE_JOIN_ACCEPT分支。

5.LinkAdrReq命令

对于 ChMaskCntl 的处理都在 ProcessMacCommands() 的 SRV_MAC_LINK_ADR_REQ 分支中。

小彩蛋一个:你发现没,注释里写着Channel mask KO。不知是djaeckle (loramac-node的作者之一)调皮,还是语言习惯如此。

代码语言:javascript
复制
if( chMaskCntl == 6 )
{
	// Enable all 125 kHz channels
	for( uint8_t i = 0, k = 0; i < LORA_MAX_NB_CHANNELS; i += 16, k++ )
	{
		for( uint8_t j = 0; j < 16; j++ )
		{
			if( Channels[i + j].Frequency != 0 )
			{
				channelsMask[k] |= 1 << j;
			}
		}
	}
}
else if( chMaskCntl == 7 )
{
	status &= 0xFE; // Channel mask KO
}
else
{
	for( uint8_t i = 0; i < 16; i++ )
	{
		if( ( ( chMask & ( 1 << i ) ) != 0 ) &&
			( Channels[chMaskCntl * 16 + i].Frequency == 0 ) )
		{// Trying to enable an undefined channel
			status &= 0xFE; // Channel mask KO
		}
	}
	channelsMask[chMaskCntl] = chMask;
}

6.最大载荷长度

代码语言:javascript
复制
/*!
 * Maximum payload with respect to the datarate index. Cannot operate with repeater.
 */
const uint8_t MaxPayloadOfDatarate[] = { 51, 51, 51, 115, 222, 222 };

/*!
 * Maximum payload with respect to the datarate index. Can operate with repeater.
 */
const uint8_t MaxPayloadOfDatarateRepeater[] = { 51, 51, 51, 115, 222, 222 };

这在RxWindowSetup()进行处理,调用了最终的驱动函数。

代码语言:javascript
复制
if( RepeaterSupport == true )
{
	Radio.SetMaxPayloadLength( modem, MaxPayloadOfDatarateRepeater[datarate] + LORA_MAC_FRMPAYLOAD_OVERHEAD );
}
else
{
	Radio.SetMaxPayloadLength( modem, MaxPayloadOfDatarate[datarate] + LORA_MAC_FRMPAYLOAD_OVERHEAD );
}

7.接收窗口处理。

RX1的处理在OnRxWindow1TimerEvent()中,满足协议要求。

代码语言:javascript
复制
RxWindowSetup( LORAMAC_FIRST_RX1_CHANNEL + ( Channel % 48 ) * LORAMAC_STEPWIDTH_RX1_CHANNEL, datarate, bandwidth, symbTimeout, false );

RX2的默认参数见如下宏:

代码语言:javascript
复制
#define RX_WND_2_CHANNEL                                  { 505300000, DR_0 }

RX2的处理在OnRxWindow2TimerEvent()中:

代码语言:javascript
复制
if( RxWindowSetup( LoRaMacParams.Rx2Channel.Frequency, LoRaMacParams.Rx2Channel.Datarate, bandwidth, symbTimeout, rxContinuousMode ) == true )
{
	RxSlot = 1;
}

速率偏移处理如下:

代码语言:javascript
复制
datarate = LoRaMacParams.ChannelsDatarate - LoRaMacParams.Rx1DrOffset;
if( datarate < 0 )
{
	datarate = DR_0;
}

8.默认设置

目前基本各地区的参数都一样,因此协议栈也是直接共用如下参数:

代码语言:javascript
复制
/*!
 * Class A&B receive delay 1 in ms
 */
#define RECEIVE_DELAY1                              1000

/*!
 * Class A&B receive delay 2 in ms
 */
#define RECEIVE_DELAY2                              2000

/*!
 * Join accept receive delay 1 in ms
 */
#define JOIN_ACCEPT_DELAY1                          5000

/*!
 * Join accept receive delay 2 in ms
 */
#define JOIN_ACCEPT_DELAY2                          6000

/*!
 * Class A&B maximum receive window delay in ms
 */
#define MAX_RX_WINDOW                               3000

/*!
 * Maximum allowed gap for the FCNT field
 */
#define MAX_FCNT_GAP                                16384

/*!
 * ADR acknowledgement counter limit
 */
#define ADR_ACK_LIMIT                               64

/*!
 * Number of ADR acknowledgement requests before returning to default datarate
 */
#define ADR_ACK_DELAY                               32

/*!
 * Number of seconds after the start of the second reception window without
 * receiving an acknowledge.
 * AckTimeout = \ref ACK_TIMEOUT + Random( -\ref ACK_TIMEOUT_RND, \ref ACK_TIMEOUT_RND )
 */
#define ACK_TIMEOUT                                 2000

/*!
 * Random number of seconds after the start of the second reception window without
 * receiving an acknowledge
 * AckTimeout = \ref ACK_TIMEOUT + Random( -\ref ACK_TIMEOUT_RND, \ref ACK_TIMEOUT_RND )
 */
#define ACK_TIMEOUT_RND                             1000

End


本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 前言
  • 源码解析
    • 1.前导码格式的源码实现
      • 2.信道频率的源码实现
        • 3.数据速率和节点发射功率编码
          • 4.CFList
            • 5.LinkAdrReq命令
              • 6.最大载荷长度
                • 7.接收窗口处理。
                  • 8.默认设置
                  • End
                  领券
                  问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档