目录

nRF24L01模块

迁移到STC8H

只需要调整SPI发送部分, 为适应nRF24L01的发送方式, 增加了多字节发送方法(节省了字节间拉高拉低CS的GPIO操作)

uint8_t SPI_TxRx(uint8_t dat)
{
SPDAT = dat;
while (!SPI_RxTxFinished());
SPI_ClearInterrupts();
return SPDAT;
} void SPI_TxRxBytes(uint8_t *pBuf, uint8_t len)
{
while(len--)
{
*pBuf++ = SPI_TxRx(*pBuf);
}
}

对应nRF24L01的SPI通信部分. 这里对发送作了一些优化

  1. 将命令和后续数据合并为字节数组一并发出, 节约发送开销
  2. 发送和接收使用同一段内存地址, 节约内存开销
  3. 每次交互后, 地址的第一个字节都是当前nRF24L01的状态数据, 在某些场景可以避免二次调用
void NRF24L01_WriteReg(uint8_t reg, uint8_t value)
{
NRF_CSN = 0;
xbuf[0] = reg;
xbuf[1] = value;
SPI_TxRxBytes(xbuf, 2);
NRF_CSN = 1;
} uint8_t NRF24L01_ReadReg(uint8_t reg)
{
NRF_CSN = 0;
xbuf[0] = reg;
xbuf[1] = NRF24_CMD_NOP;
SPI_TxRxBytes(xbuf, 2);
NRF_CSN = 1;
return xbuf[1];
} void NRF24L01_ReadToBuf(uint8_t reg, uint8_t len)
{
NRF_CSN = 0;
memset(xbuf, NRF24_CMD_NOP, NRF24_PLOAD_WIDTH + 1);
xbuf[0] = reg;
SPI_TxRxBytes(xbuf, len + 1);
NRF_CSN = 1;
} void NRF24L01_WriteFromBuf(uint8_t reg, const uint8_t *pBuf, uint8_t len)
{
NRF_CSN = 0;
xbuf[0] = reg;
memcpy(xbuf_data, pBuf, len);
SPI_TxRxBytes(xbuf, len + 1);
NRF_CSN = 1;
}

nRF24L01模块演示用例

接线方式

因为使用了20pin的 STC8H1K08, 所以可选的SPI引脚只有P3开头的这组, 连线方式如下

20pin的STC8H3K32S2的接线和这个一样. 如果使用的是32pin或者更多pin的型号, 可以选择其它组的SPI

/**
* Example code of SPI driving NRF24L01 module
*
* Pin connection:
* P35(SS, Ignored) => CSN
* P34(MOSI) => MOSI
* P33(MISO) => MISO
* P32(SPCLK) => CLK
* P36(INT2) => IRQ
* P37(IO) => CE
*
* test-board: Minimum System; test-MCU: STC8H1K08,STC8H3K32S2
*/

配置

接收和发送方的地址配置在 nrf24l01.c, 对于发送方和接收方, 需要将这两个地址互换.

const uint8_t TX_ADDRESS[NRF24_ADDR_WIDTH] = {0x32,0x4E,0x6F,0x64,0x22};
const uint8_t RX_ADDRESS[NRF24_ADDR_WIDTH] = {0x32,0x4E,0x6F,0x64,0x65};

pin脚的配置在 nrf24l01.h, 如果有变化, 这里要相应地调整

#define NRF_CSN  P35
#define NRF_MOSI P34
#define NRF_MISO P33
#define NRF_SCK P32
#define NRF_IRQ P36
#define NRF_CE P37

初始化方法

如果只发送, 则只需要进行SPI和GPIO初始化

const NRF24_SCEN CURRENT_SCEN = NRF24_SCEN_HALF_DUPLEX;
extern uint8_t __IDATA xbuf[NRF24_PLOAD_WIDTH + 1]; void SPI_Init(void)
{
// SPI预分频
SPI_SetClockPrescaler(SPI_ClockPreScaler_16);
// 时钟在空闲时保持低电平
SPI_SetClockPolarity(HAL_State_OFF);
// 由拉低SS脚触发数据传输
SPI_SetClockPhase(SPI_ClockPhase_LeadingEdge);
// 数据顺序MSB
SPI_SetDataOrder(SPI_DataOrder_MSB);
// 设定SPI的输出脚
SPI_SetPort(SPI_AlterPort_P35_P34_P33_P32);
// 忽略SS脚, 使用 MSTR 控制主从模式
SPI_IgnoreSlaveSelect(HAL_State_ON);
// 主模式
SPI_SetMasterMode(HAL_State_ON);
// 开启SPI
SPI_SetEnabled(HAL_State_ON);
} void GPIO_Init(void)
{
// 在配置SPI之前配置GPIO
// MISO(P33) MOSI(P34)
GPIO_P3_SetMode(GPIO_Pin_4, GPIO_Mode_InOut_QBD);
// SCLK(P32) CSN(P35) CE(P37)
GPIO_P3_SetMode(GPIO_Pin_2|GPIO_Pin_5|GPIO_Pin_7, GPIO_Mode_Output_PP);
// IRQ(P36)
GPIO_P3_SetMode(GPIO_Pin_6, GPIO_Mode_Input_HIP);
}

如果需要接收, 则还需要初始化中断和中断处理方法

void INT_Init()
{
EXTI_Int2_SetIntState(HAL_State_ON);
EXTI_Global_SetIntState(HAL_State_ON);
} INTERRUPT(Int2_Routine, EXTI_VectInt2)
{
NRF24L01_HandelIrqFlag();
}

接收模式

main()
{
...
NRF24L01_Init(NRF24_MODE_RX);
INT_Init();
while (1);

发送模式

发送部分使用了FIFO队列的快速写入模式

main()
{
...
NRF24L01_Init(NRF24_MODE_TX);
UART1_TxString("NRF24L01 Initialized\r\n");
while (1)
{
if (NRF24L01_WriteFast(tmp) == 0)
{
NRF24L01_ResetTX();
err++;
}
else
{
succ++;
}
if (err >= 255 || succ >= 255)
{
UART1_TxHex(err);
UART1_TxHex(succ);
UART1_TxChar('.');
err = 0;
succ = 0;
}
SYS_Delay(50);
}

半双工模式

在半双工模式下, 使用的还是普通的发送方法. 空闲时处于接收状态, 只有当发送数据时切换到发送状态, 发送后自动回到接收状态.

main()
{
...
NRF24L01_Init(NRF24_MODE_RX);
INT_Init();
while (1)
{
NRF24L01_Tx(tmp);
SYS_Delay(1000);
}

实际测试性能

使用 STC8H1K08 fastwrite发送, STC8H3K32S2 使用中断接收

  • 以下都是不带内容输出的测试结果
  • 发送间隔1ms时发送速率最高, 能达到720~748个package每秒, 每个package是32字节, 差不多23K字节每秒
  • 发送间隔为0ms时速度下降明显, 只有1ms间隔时的80%

完整代码

STC8H开发(五): SPI驱动nRF24L01无线模块的更多相关文章

  1. STC8H开发(十五): GPIO驱动Ci24R1无线模块

    目录 STC8H开发(一): 在Keil5中配置和使用FwLib_STC8封装库(图文详解) STC8H开发(二): 在Linux VSCode中配置和使用FwLib_STC8封装库(图文详解) ST ...

  2. STC8H开发(六): SPI驱动ADXL345三轴加速度检测模块

    目录 STC8H开发(一): 在Keil5中配置和使用FwLib_STC8封装库(图文详解) STC8H开发(二): 在Linux VSCode中配置和使用FwLib_STC8封装库(图文详解) ST ...

  3. STC8H开发(十): SPI驱动Nokia5110 LCD(PCD8544)

    目录 STC8H开发(一): 在Keil5中配置和使用FwLib_STC8封装库(图文详解) STC8H开发(二): 在Linux VSCode中配置和使用FwLib_STC8封装库(图文详解) ST ...

  4. STC8H开发(十六): GPIO驱动XL2400无线模块

    目录 STC8H开发(一): 在Keil5中配置和使用FwLib_STC8封装库(图文详解) STC8H开发(二): 在Linux VSCode中配置和使用FwLib_STC8封装库(图文详解) ST ...

  5. STC8H开发(七): I2C驱动MPU6050三轴加速度+三轴角速度检测模块

    目录 STC8H开发(一): 在Keil5中配置和使用FwLib_STC8封装库(图文详解) STC8H开发(二): 在Linux VSCode中配置和使用FwLib_STC8封装库(图文详解) ST ...

  6. STC8H开发(十三): I2C驱动DS3231高精度实时时钟芯片

    目录 STC8H开发(一): 在Keil5中配置和使用FwLib_STC8封装库(图文详解) STC8H开发(二): 在Linux VSCode中配置和使用FwLib_STC8封装库(图文详解) ST ...

  7. nRF2401A/nRF24L01/nRF24L01+无线模块最常见问题汇集(转)

    俗话说:每个人一生下来什么都会的,都是通过自己努力和探索出来的,NRF系列芯片,刚开始都好奇心加兴趣才来捣鼓它的,刚开始做硬件和软件,没有收发数据弄得整个人头都快炸开了,所以在此和大家分享一下前辈的经 ...

  8. nRF24L01无线模块笔记

    nRF24L01模块 官网链接: https://www.nordicsemi.com/Products/nRF24-series 常见的无线收发模块, 工作在2.4GHz频段, 适合近距离遥控和数据 ...

  9. 树莓派与Arduino Leonardo使用NRF24L01无线模块通信之基于RF24库 (五) 树莓派单子节点发送数据

    本项目中各个节点和树莓派的通信不区分信道,因此如果由树莓派发送给特定节点的数据会被所有节点接收到,因此子节点可以判别该数据是否发给自己的,需要在数据的第二个字节中加入目标节点的编号(第一个字节为源节点 ...

随机推荐

  1. MyBatis学习(三)MyBatis基于动态代理方式的增删改查

    1.前言 上一期讲到MyBatis-Statement版本的增删改查.可以发现.这种代码写下来冗余的地方特别多.写一套没啥.如果涉及到多表多查询的时候就容易出现问题.故.官方推荐了一种方法.即MyBa ...

  2. SpringBoot整合kafka的简单应用及配置说明

    引入依赖 <!-- https://mvnrepository.com/artifact/org.springframework.kafka/spring-kafka --> <de ...

  3. Centos(Linux)安装openoffice教程

    一.从官网下载openoffice软件 下载地址:http://www.openoffice.org/zh-cn/download/ 选择(RPM)类型进行下载,选择对应的版本,这里默认选择是最新的版 ...

  4. UE4之第一个飞机游戏

    开始之前 UE4官网 初识ue4教程(1~9节): https://www.bilibili.com/video/BV164411Y732?p=1 第一个飞机游戏: http://www.sikied ...

  5. 再谈多线程模型之生产者消费者(单一生产者和多消费者 )(c++11实现)

    0.关于 为缩短篇幅,本系列记录如下: 再谈多线程模型之生产者消费者(基础概念)(c++11实现) 再谈多线程模型之生产者消费者(单一生产者和单一消费者)(c++11实现) 再谈多线程模型之生产者消费 ...

  6. 【LeetCode】476. 数字的补数 Number Complement

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 公众号:负雪明烛 本文关键词:Leetcode, 力扣,476, 补数,二进制,Pyth ...

  7. 【剑指Offer】07. 重建二叉树 解题报告(Java & Python & C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 个人微信公众号:负雪明烛 目录 题目描述 解题方法 基本方法:线性查找根节点的位置 方法优 ...

  8. RMQ(ST(Sparse Table))(转载)

    1. 概述 RMQ(Range Minimum/Maximum Query),即区间最值查询,是指这样一个问题:对于长度为n的数列A,回答若干询问RMQ(A,i,j)(i,j<=n),返回数列A ...

  9. MCU变量加载过程

    前言 在开发mcu代码的时候经常会有些疑惑,变量是怎么在编译之后进入单片机的ram区的呢,特别是在使用keil开发的时候.后来在接触gcc编译器和自研的mcu后,终于明白了这个问题.实际上变量编译后被 ...

  10. 『动善时』JMeter基础 — 59、进行JMeter分布式测试遇到的坑

    目录 1.控制机端 (1)执行机没有关闭防火墙 (2)内存溢出 2.执行机端 (1)启动jmeter-server服务情况一 (2)启动jmeter-server服务情况二 (3)启动jmeter-s ...