OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 29|回复: 0

《STM32H7R7开发指南 V1.1 》第四十七章 无线通信实验

[复制链接]

1334

主题

1350

帖子

2

精华

超级版主

Rank: 8Rank: 8

积分
5699
金钱
5699
注册时间
2019-5-8
在线时间
1536 小时
发表于 3 小时前 | 显示全部楼层 |阅读模式
第四十七章 无线通信实验

1)实验平台:正点原子STM32H7R7开发板

2)章节摘自【正点原子】STM32H7R7开发指南 V1.1

3)购买链接: https://detail.tmall.com/item.htm?id=820823382459

4)全套实验源码+手册+视频下载地址:http://www.openedv.com/docs/boards/stm32/zdyz_stm32h7rx.html

5)正点原子官方B站:https://space.bilibili.com/394620890

6)正点原子STM32开发板技术交流群:756580169


2.jpg

3.png

本章,我们将介绍如何使用2.4G无线模块NRF24L01实现无线通信。将使用两块STM32开发板,一块用于发送,一块用于接收,从而实现无线数据传输,并把数据显示在LCD上。本章分为如下几个小节:
47.1 NRF24L01无线模块介绍
47.2 硬件设计
47.3 程序设计
47.4 下载验证


47.1 NRF24L01无线模块介绍

47.1.1 NRF24L01简介
NRF24L01无线模块,采用的芯片是NRF24L01+。该芯片是由NORDIC公司生产,并且集成NORDIC自家的Enhance ShortBurst协议,主要特点如下:
1)2.4G全球开放的ISM频段,免许可证使用
2)最高工作速率2Mbps,高效的GFSK调制,抗干扰能力强
3)126个可选的频道,满足多点通信和调频通信的需要
4)6个数据通道可支持点对多点的通信地址控制
5)低工作电压(1.9~3.6V)
6)硬件CRC和自动处理字头
7)可设置自动应答,确保数据可靠传输
由于高速信号是由芯片内部的射频协议处理后进行无线高速通信,对MCU的时钟频率要求不高,只需要对NRF24L01某些寄存器进行配置即可。芯片与外部MCU是通过SPI通信接口进行数据通信,并且最大的SPI速度可达10MHz。
这个芯片是NRF24L01的升级版。相比NRF24L01,升级版支持250k,1M,2M三种传输速率;支持更多种功率配置,根据不同应用有效节省功耗;稳定性及可靠性更高。
该模块的外形和引脚图如图47.1.1.1所示:


第四十七章 无线通信实验666.png
图47.1.1.1 NRF24L01无线模块外形和引脚图

模块VCC脚的电压范围为1.9~3.6V,建议不要超过3.6V,否则可能烧坏模块,一般用3.3V电压比较合适。除了VCC和GND脚,其他引脚都可以和5V单片机的IO口直连,正是因为其兼容5V单片机的IO,所以使用上具有很大优势。
具体引脚介绍如表47.1.1.1所示。


1.png
表47.1.1.1 引脚介绍表

引脚部分主要分为电源相关的VCC和GND,SPI通信接口相关的CSN/SCK/MOSI/MISO,模式选择相关的CE,中断相关的IRQ。CE引脚会与CONFIG寄存器共同控制NRF24L01进入某个工作模式。IRQ引脚会在寄存器的配置下生效,当收到数据、成功发送数据或达到最大重发次数时,IRQ引脚会变为低电平。
SPI通信接口将会在47.1.4小节进行讲解。
NRF24L01的Enhance ShockBurstTM模式具体表现在自动应答和重发机制,发送端要求接收端在接收到数据后要有应答信号,便于发送端检测有无数据丢失,一旦有数据丢失,则通过重发功能将丢失的数据恢复,这个过程无需MCU。Enhance ShockBurstTM模式可以通过EN_AA寄存器进行配置。
接下来看一下Enhanced ShockBurstTM模式下NRF24L01通信图,如图47.1.1.2所示:


第四十七章 无线通信实验1320.png
图47.1.1.2 NRF24L01通信图

这里我们抽离PTX6和PRX出来,分析一下通信过程。
PTX6作为发送端,它就需要设置发送地址,可以看到TX_ADDR为0x7878787878,PRX作为接收端,它使能接收通道0并设置接收通道0接收地址0x7878787878。通信时,发送端发送数据接收端接收到数据并记录TX地址接收端以TX地址为目的地址发送应答信号发送端会以通道0接收应答信号。
NRF24L01规定:发送端中的数据通道0是用来接收接收端发送的应答信号,所以数据通道0的接收地址要与发送地址要相同才能确保收到正确的应答信号,这里十分重要,必须要在相关寄存器中配置正确。


47.1.2 NRF24L01工作模式介绍
NRF24L01作为无线通信模块,功耗问题十分重要,有数据发送与空闲状态下能耗肯定是需要调整,所以设计者给芯片设计了多种工作模块,如表47.1.2.1所示:

2.png
表47.1.2.1 NRF24L01工作模式表

NRF24L01工作模式是由CE引脚和CONFIG寄存器的PWR_UP位和PRIM_RX位共同控制。CE引脚在前面也说到是模式控制线,而PWR_UP位是上电位,PRIM_RX位可以理解为配置身份位(TX or RX)。可以看到发送模式有两种,待机模式也有两种,功耗上各不相同,没有标红的发送模式和待机模式I是官方推荐使用,更加节能,但是本实验用到的模式就是上表中标红色部分,因为标红的的模式使用起来更加方便。单看发送模式,使用官方推荐的发送模式,你要发送三级TX_FIFO数据需要产生三个边沿信号(CE从高电平变为低电平)。而我们使用的发送模式,从CE引脚的操作上看,只需要拉高,就可以把所有TX_FIFO里的数据发送完成。
NRF24L01的发送和接收都有三级FIFO,每一级FIFO就有32个字节。发送和接收都是对FIFO进行操作,并且最大操作的数据量就是一级FIFO即32字节。发送时,只需要把数据存进TX_FIFO并按照发送模式下的操作(参考NRF24L01工作模式表中的发送模式)即可让NRF24L01启动发射,这个发射过程就包括:无线系统上电,启动内部16MHz时钟,无线发送数据打包,高速发送数据。接收时,也是通过读取RX_FIFO里的内容。


47.1.3 NRF24L01寄存器
在这里简单介绍一下本实验用到的NRF24L01比较重要的寄存器。
配置寄存器(CONFIG)
寄存器地址0x01,复位值为0x80,用来配置NRF24L01工作状态以及中断相关,描述如表47.1.3.1所示:


3.png
表47.1.3.1 配置寄存器图

需要配置成发送模式,可以把该寄存器赋值为0x0E,如果配置成接收模式,可以把该寄存器赋值为0x0F。无论是发送模式还是接收模式,都使能16位CRC以及使能接收中断、发送中断和最大重发次数中断,这里发送端和接收端配置需要一致。
自动应答功能寄存器(EN_AA)
寄存器地址0x01,复位值为0x3F,用来设置通道0~5的自动应答功能,描述如表47.1.3.2所示:


4.png
表47.1.3.2 自动应答功能寄存器图

本实验,接收端是以数据通道0作为接收通道,并且前面也提及Enhanced ShockBurstTM模式的自动应答流程,接收端接收到数据后,需要回复应答信号,通过该寄存器ENAA_P0置1即可实现。另外,使能自动应答也相当于配置成Enhanced模式,所以发送端也需要进行自动应答允许。
接收地址允许寄存器(EN_RXADDR)
寄存器地址0x02,复位值为0x03,用于使能接收通道0~5,描述如图47.1.3.3所示:


5.png
图47.1.3.3 接收地址允许寄存器图

前面也说到接收端使用的是通道0进行接收数据,所以ERX_P0需要置1处理。同样的,发送端也需要使能数据通道0来接收应答信号。
地址宽度设置寄存器(SETUP_AW)
寄存器地址0x03,复位值为0x03,对接收/发送地址宽度设置位,描述如图47.1.3.4所示:


6.png
图47.1.3.4 地址宽度设置寄存器图

本实验中,无论是发送地址还是接收地址都是使用5字节,也就是默认设置便是使用5字节宽度的地址。
自动重发配置寄存器(SETUP_RETR)
寄存器地址0x04,复位值为0x00,对发送端的自动重发数值和延时进行设置,描述如图47.1.3.5所示:


7.png
图47.1.3.5 自动重发配置寄存器图

本实验中,直接对该寄存器写入0x1A,即自动重发间隔时间为586us,最大自动重发次数为10次。在使能了MAX_RT中断时,连续重发10次还是发送失败的时候,IRQ中断引脚就会拉低。
射频频率设置寄存器(RF_CH)
寄存器地址0x05,复位值0x05,对NRF24L01的频段进行设置,描述如图47.1.3.6所示:


8.png
图47.1.3.6 射频频率设置寄存器图

频率计算公式:2400 + RF_CH(MHz)
本实验中,直接对该寄存器写入40即射频频率为2440MHz。通信双方该寄存器必须配置一样才能通信成功。
发射参数设置寄存器(RF_SETUP)
寄存器地址0x06,复位值为0x0E,对NRF24L01的发射功率、无线速率进行设置,描述如图47.1.3.7所示:


9.png
图47.1.3.7 发射参数设置寄存器图

本实验中,直接对该寄存器写入0x0F即射频输出功率为0dBm增益,传输速率为2MHz。发送端和接收端该寄存器的配置需一样。功率越小耗电越少,同等条件下,传输距离越小,这里我们设置射频部分功耗为最大,当然大家可以根据实际应用而选择对应的功率配置。
状态寄存器(STATUS)
地址0x07,复位值为0x0E,反映NRF24L01当前工作状态,描述如图47.1.3.8所示:


10.png
图47.1.3.8 状态寄存器图

该寄存器作为查询作用,作为发送端,发送完数据后,可以查询一下TX_DS位状态便知是否成功发送数据,发送数据异常时,也可以通过查询MAX_RT位状态获知是否达到最大重发次数。作为接收端,就可以通过查询RX_OK位状态获知是否接收到数据。我们查询相关位后都需要将该位置1清除中断。
此外,我们还用到设置接收通道0地址寄存器RX_ADDR_P0(0x0A)和发送地址设置寄存器TX_ADDR(0x10)以及接收通道0有效数据看度设置寄存器RX_PW_P0(0x11),由于这三个寄存器比较简单,所以这里就不列出来了。


47.1.4 SPI接口简介
SPI 是英语Serial Peripheral interface的缩写,顾名思义就是串行外围设备接口。是Motorola首先在其MC68HCXX系列处理器上定义的。SPI接口主要应用在 EEPROM,FLASH,实时时钟,AD转换器,还有数字信号处理器和数字信号解码器之间。
SPI,是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚,同时为PCB的布局上节省空间,提供方便,正是出于这种简单易用的特性,现在越来越多的芯片集成了这种通信协议,STM32H7R7也有SPI接口,其框图如图47.1.4.1所示:


第四十七章 无线通信实验6088.png
图47.1.4.1 STM32H7R7的SPI框图

图中我们挑出了4处和我们本章例程相关的位置进行重点介绍,其中①处是SPI的对外信号,一般由4根信号线组成(MCK是I2S用的,不属于SPI信号线):
MISO(Master In / Slave Out)主设备数据输入,从设备数据输出。
MOSI(Master Out / Slave In)主设备数据输出,从设备数据输入。
SCK(CK)时钟信号,由主设备产生。
SS(WS)从设备片选信号(也称CS),由主设备产生。
图中②处是SPI的时钟生成器,将来自spi_ker_ck的时钟进行分频输出,最终产生SCK信号,并驱动RX/TX移位寄存器接收/发送数据。SPI1~6的spi_ker_ck时通过RCC_CCIPR3寄存器进行设置,例程中,我们使用到SPI2,并设置其时钟源来自pll1_q_ck,为300Mhz。
图中③处重点由:TX移位寄存器、RX移位寄存器等组成,分别用于发送数据和接收数据,TxFIFO用于发送数据,RxFIFO用于接收数据,我们可以设置FIFO大小为1,这样写入TXDR的数据就可以直接输出到TX移位寄存器,在SCK时钟的驱动下,输出到MOSI脚。同理,当FIFO大小为1时,RX移位寄存器接收到的数据,也会直接输出给RXDR寄存器。
图中④处是SPI的一堆控制寄存器,用于设置SPI的各种工作情况(或模式),该区域由spi_pclk时钟驱动。一般我们需要合理的设置这些寄存器,SPI才可以正常使用。   
SPI总线有四种工作模式,如表47.1.4.1所示:


11.png
表47.1.4.1 SPI四种工作模式

其中CPOL和CPHA分别控制SCK的时钟极性和相位时钟,可以通过相关寄存器设置。 不同时钟/相位下的总线数据传输时序如图47.1.4.2所示:

第四十七章 无线通信实验7011.png
图47.1.4.2 不同时钟相位下的总线传输时序(CPHA=0/1)

SPI在驱动不同的器件时,必须注意其支持的工作模式,需要设置合适的SPI工作模式,才可以正常通信。
对于STM32H7R7来说,SPI的MSB和LSB是可以配置的,通过SPI_CFG2的LSBFIRST位进行控制,当该位为1时,表示LSB在前;当该位为0时,表示MSB在前;
STM32H7R7的SPI功能很强大,SPI时钟最高可以到200Mhz,支持DMA,可以配置为SPI协议或者I2S协议(支持全双工I2S)。
本章,我们将使用STM32H7R7的SPI来驱动NRF24L01无线模块。这里对SPI我们只简单介绍一下SPI的使用,STM32H7R7的SPI详细介绍请参考《STM32H7Rx参考手册_V6(英文版)》第2557页,第55章。
SPI主模式配置步骤
1)配置相关引脚的复用功能,使能SPI2时钟。
我们要用SPI2,第一步就要使能SPI2的时钟,SPI2的时钟通过APB1LENR的第14位来设置。其次要设置SPI2的相关引脚为复用(AF5)输出,这样才会连接到SPI2上。这里我们使用的是PD3、PC2、PC3这3个(SCK.、MISO、MOSI,NSS使用软件管理方式),所以设置这三个为复用IO,复用功能为AF5。
2)设置SPI2的波特率和数据长度。
这一步通过SPI2_CFG1来设置,SPI2的波特率通过SPI2_CFG1寄存器的MBR[2:0]来设置,可以设置为spi_ker_ck的2~256分频。而数据格式通过SPI2_CFG1寄存器的DSIZE[4:0]位设置,数据长度等于DSIZE[4:0]+1,比如,我们要设置为8位数据格式,则设置DSIZE[4:0]=7即可。
3)设置SPI2工作模式。
这一步通过SPI2_CFG2来设置,我们设置SPI2为主机模式,全双工,然后通过CPOL和CPHA位来设置SCK时钟极性及采样方式。并设置SPI2的SS(即CS)控制方式为软件控制。
4)使能SPI2。
通过SPI2_CR1的bit0来设置,以启动SPI2,在启动之后,我们就可以开始SPI通讯了。


47.2 硬件设计

1. 例程功能
开机的时候先检测NRF24L01模块是否存在,在检测到NRF24L01模块之后,根据KEY0和KEY1的设置来决定模块的工作模式。在设定好工作模式之后,就会不停的发送/接收数据,同时在LCD上面显示相关信息。LED0闪烁用于提示程序正在运行。

2. 硬件资源
1)LED灯  
       LED0 – PD14
2 ) 独立按键  
       KEY0 – PE9
       KEY1 – PE8
3)2.4G无线模块  NRF24L01模块
4)正点原子2.8/3.5/4.3/7/10寸TFTLCD模块(包括MCU屏和RGB屏,都支持)
5)串口1(PB14/PB15连接在板载USB转串口芯片CH340上面)
6)SPI2(连接在PD3/PC2/PC3上)

3. 原理图
NRF24L01模块与STM32的连接关系,如下图所示:


第四十七章 无线通信实验8292.png
图47.2.1 NRF24L01模块接口与STM32连接原理图

这里NRF24L01使用的是SPI2,连接在PD3/PC2/PC3上。注意:NRF_IRQ和GBC_KEY共用了PF2,所以,他们不能同时使用,需要分时复用。
由于无线通信实验是双向的,所以至少要有两个模块同时能工作,这里我们使用2套开发板来向大家演示。


47.3 程序设计
NRF24L01配置步骤
1)SPI参数初始化(工作模式、数据时钟极性、时钟相位等)。
HAL库通过调用SPI初始化函数HAL_SPI_Init完成对SPI参数初始化,详见例程源码。
注意:该函数会调用:HAL_SPI_MspInit函数来完成对SPI底层的初始化,包括:SPI及GPIO时钟使能、GPIO模式设置等。
2)使能SPI时钟和配置相关引脚的复用功能以及NRF24L01的其他相关管脚。
本实验用到SPI2,使用PD3、PC2和PC3作为SPI_SCK、SPI_MISO和SPI_MOSI,以及NRF24L01的CE、CSN和IRQ分别对应PM14,PM13和PF2,因此需要先使能SPI2、GPIOD和GPIOC时钟。参考代码如下:

  1. __HAL_RCC_GPIOx_CLK_ENABLE();            /* 使能GPIOx时钟,x=A……K */
  2. __HAL_RCC_SPI2_CLK_ENABLE ();            /* 使能SPI时钟 */
复制代码
GPIO模式设置通过调用HAL_GPIO_Init函数实现,详见本例程源码。
3)使能SPI
通过__HAL_SPI_ENABLE函数使能SPI,便可进行数据传输。
4)SPI传输数据
通过HAL_SPI_Transmit函数进行发送数据。
通过HAL_SPI_Receive函数进行接收数据。
也可以通过HAL_SPI_TransmitReceive函数进行发送与接收操作。
5)编写NRF24L01的读写函数
基于SPI的读写函数的基础上,编写NRF24L01的读写函数。
6)编写NRF24L01接收模式与发送模式函数
通过查看寄存器,编写配置NRF24L01接收和发送模式的函数。


47.3.1 程序解析

1. SPI驱动代码
这里我们只讲解核心代码,详细的源码请大家参考光盘本实验对应源码。SPI驱动源码包括两个文件:spi.c和spi.h。
首先介绍的是spi.h文件,具体定义如下:

  1. /* SPI定义 */
  2. #define SPI2_SCK_GPIO_PORT              GPIOD
  3. #define SPI2_SCK_GPIO_PIN               GPIO_PIN_3
  4. #define SPI2_SCK_GPIO_AF                GPIO_AF5_SPI2
  5. #define SPI2_SCK_GPIO_CLK_ENABLE()      do{ __HAL_RCC_GPIOD_CLK_ENABLE(); }while(0)   /* PD口时钟使能 */

  6. #define SPI2_MISO_GPIO_PORT             GPIOC
  7. #define SPI2_MISO_GPIO_PIN              GPIO_PIN_2
  8. #define SPI2_MISO_GPIO_AF               GPIO_AF5_SPI2
  9. #define SPI2_MISO_GPIO_CLK_ENABLE()     do{ __HAL_RCC_GPIOC_CLK_ENABLE(); }while(0)   /* PC口时钟使能 */

  10. #define SPI2_MOSI_GPIO_PORT             GPIOC
  11. #define SPI2_MOSI_GPIO_PIN              GPIO_PIN_3
  12. #define SPI2_MOSI_GPIO_AF               GPIO_AF5_SPI2
  13. #define SPI2_MOSI_GPIO_CLK_ENABLE()     do{ __HAL_RCC_GPIOC_CLK_ENABLE(); }while(0)   /* PC口时钟使能 */

  14. /* SPI2相关定义 */
  15. #define SPI2_SPI                        SPI2
  16. #define SPI2_SPI_CLK_ENABLE()           do{ __HAL_RCC_SPI2_CLK_ENABLE(); }while(0)    /* SPI2时钟使能 */
复制代码
上面的宏定义是SPI2的接口(PD3\PC2\PC3)的相关宏定义。
下面介绍spi.c文件,首先是SPI初始化相关函数,具体如下:

  1. /**
  2. * @brief       SPI初始化代码
  3. *   [url=home.php?mod=space&uid=60778]@note[/url]      主机模式,8位数据,禁止硬件片选
  4. * [url=home.php?mod=space&uid=271674]@param[/url]       无
  5. * @retval      无
  6. */
  7. void spi2_init(void)
  8. {
  9.     g_spi_handle.Instance = SPI2;       /* SP2 */
  10. g_spi_handle.Init.Mode = SPI_MODE_MASTER;   
  11. /* 设置SPI工作模式,设置为主模式 */
  12. g_spi_handle.Init.Direction = SPI_DIRECTION_2LINES;                     
  13. /* 设置SPI单向或者双向的数据模式:SPI设置为双线模式 */
  14. g_spi_handle.Init.DataSize = SPI_DATASIZE_8BIT;                        
  15. /* 设置SPI的数据大小:SPI发送接收8位帧结构 */
  16. g_spi_handle.Init.CLKPolarity = SPI_POLARITY_HIGH;                     
  17. /* 串行同步时钟的空闲状态为高电平 */
  18. g_spi_handle.Init.CLKPhase = SPI_PHASE_2EDGE;                           
  19. /* 串行同步时钟的第二个跳变沿(上升或下降)数据被采样 */
  20. g_spi_handle.Init.NSS = SPI_NSS_SOFT;                                   
  21. /* NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制 */
  22. g_spi_handle.Init.NSSPMode = SPI_NSS_PULSE_DISABLE;                     
  23. /* NSS信号脉冲失能 */
  24. g_spi_handle.Init.MasterKeepIOState = SPI_MASTER_KEEP_IO_STATE_ENABLE;  
  25. /* SPI主模式IO状态保持使能 */
  26. g_spi_handle.Init.BaudRatePrescaler = SPI_BAUDRATEPRESCALER_256;        
  27. /* 定义波特率预分频的值:波特率预分频值为256 */
  28. g_spi_handle.Init.FirstBit = SPI_FIRSTBIT_MSB;                          
  29. /* 指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始 */
  30.     g_spi_handle.Init.TIMode = SPI_TIMODE_DISABLE;   /* 关闭TI模式 */
  31. g_spi_handle.Init.CRCCalculation = SPI_CRCCALCULATION_DISABLE;         
  32. /* 关闭硬件CRC校验 */
  33.     g_spi_handle.Init.CRCPolynomial = 7;    /* CRC值计算的多项式 */
  34.     HAL_SPI_Init(&g_spi_handle);
  35.    
  36.     __HAL_SPI_ENABLE(&g_spi_handle);        /* 使能SPI2 */
  37.     spi2_read_write_byte(0Xff);             /* 启动传输 */
  38. }

  39. /**
  40. * @brief       SPI底层驱动,时钟使能,引脚配置
  41. * @param       hspi:SPI句柄
  42. * @note        此函数会被HAL_SPI_Init()调用
  43. * @retval      无
  44. */
  45. void HAL_SPI_MspInit(SPI_HandleTypeDef *hspi)
  46. {
  47.     GPIO_InitTypeDef gpio_init_struct = {0};
  48.     RCC_PeriphCLKInitTypeDef rcc_periph_clk_init = {0};

  49.     SPI2_SPI_CLK_ENABLE();                            /* SPI2时钟使能 */
  50.     SPI2_SCK_GPIO_CLK_ENABLE();                       /* SPI2_SCK脚时钟使能 */
  51.     SPI2_MISO_GPIO_CLK_ENABLE();                      /* SPI2_MISO脚时钟使能 */
  52.     SPI2_MOSI_GPIO_CLK_ENABLE();                      /* SPI2_MOSI脚时钟使能 */

  53.     /* 设置SPI2的时钟源 */
  54. rcc_periph_clk_init.PeriphClockSelection = RCC_PERIPHCLK_SPI23;     
  55. /* 设置SPI2时钟源 */
  56. rcc_periph_clk_init.Spi23ClockSelection = RCC_SPI23CLKSOURCE_PLL1Q;
  57. /* SPI2时钟源使用PLL1Q */
  58.     HAL_RCCEx_PeriphCLKConfig(&rcc_periph_clk_init);

  59.     gpio_init_struct.Pin = SPI2_SCK_GPIO_PIN;
  60.     gpio_init_struct.Mode = GPIO_MODE_AF_PP;               /* 复用推挽输出 */
  61.     gpio_init_struct.Pull = GPIO_PULLUP;                   /* 上拉 */
  62.     gpio_init_struct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;    /* 高速 */
  63.     gpio_init_struct.Alternate = SPI2_SCK_GPIO_AF;         /* 复用 */
  64.     HAL_GPIO_Init(SPI2_SCK_GPIO_PORT, &gpio_init_struct);  /* 初始化SCK引脚 */

  65.     gpio_init_struct.Pin = SPI2_MISO_GPIO_PIN;
  66.     gpio_init_struct.Alternate = SPI2_MISO_GPIO_AF;        /* 复用 */
  67.     HAL_GPIO_Init(SPI2_MISO_GPIO_PORT, &gpio_init_struct); /* 初始化MISO引脚 */
  68.    
  69.     gpio_init_struct.Pin = SPI2_MOSI_GPIO_PIN;
  70.     gpio_init_struct.Alternate = SPI2_MOSI_GPIO_AF;        /* 复用 */
  71.     HAL_GPIO_Init(SPI2_MOSI_GPIO_PORT, &gpio_init_struct); /* 初始化MOSI引脚 */
  72. }
复制代码
上面介绍的是SPI2初始化函数(spi2_init)和SPI初始化回调函数HAL_SPI_MspInit。
最后介绍的是SPI2速度设置函数和SPI2读写一个字节数据函数,具体如下:

  1. /**
  2. * @brief       SPI2速度设置函数
  3. * @note        SPI2时钟选择来自pll1_q_ck, 为250Mhz
  4. *              SPI速度 = spi_ker_ck / 2^(speed + 1)
  5. * @param       speed: SPI2时钟分频系数,
  6. SPI_BAUDRATEPRESCALER_2~SPI_BAUDRATEPRESCALER_256
  7. * @retval      无
  8. */
  9. void spi2_set_speed(uint32_t speed)
  10. {
  11.     assert_param(IS_SPI_BAUDRATE_PRESCALER(speed));          /* 判断有效性 */
  12.     __HAL_SPI_DISABLE(&g_spi_handle);                        /* 关闭SPI */
  13.     g_spi_handle.Instance->CFG1 &= ~(0X7<<28);  /* 位30-28清零,用来设置波特率 */
  14.     g_spi_handle.Instance->CFG1 |= speed;                   /* 设置SPI速度 */
  15.     __HAL_SPI_ENABLE(&g_spi_handle);                        /* 使能SPI */
  16. }

  17. /**
  18. * @brief       SPI2读写一个字节数据
  19. * @param       txdata: 要发送的数据(1字节)
  20. * @retval      接收到的数据(1字节)
  21. */
  22. uint8_t spi2_read_write_byte(uint8_t txdata)
  23. {
  24.     uint8_t rxdata;
  25.     HAL_SPI_TransmitReceive(&g_spi_handle, &txdata, &rxdata, 1, 1000);
  26.     return rxdata;                                       /* 返回收到的数据 */
  27. }
复制代码
其中,spi2_set_speed函数用于设置SPI2的传输速度也就是波特率,SPI2的传输速度是通过SPI2->CR1寄存器的位5-3来设置。形参speed是SPI2时钟分频系数,取值范围是:SPI_BAUDRATEPRESCALER_2~SPI_BAUDRATEPRESCALER_256。
而spi2_read_write_byte函数主要是通过调用HAL库中SPI发送接收函数HAL_SPI_TransmitReceive来实现数据的发送和接收。

2. NRF24L01驱动代码
这里我们只讲解核心代码,详细的源码请大家参考光盘本实验对应源码。NRF24L01驱动源码包括两个文件:nrf24l01.c和nrf24l01.h。
我们要在spi.c文件中封装好的函数的基础上进行调用,实现nrf24l01的发送与接收。下面先看一下nrf24l01.h文件中定义的信息,其代码如下:  

  1. /* NRF24L01 操作引脚 定义(不包含SPI_SCK/MISO/MISO等三根线) */
  2. #define NRF24L01_CE_GPIO_PORT                           GPIOM
  3. #define NRF24L01_CE_GPIO_PIN                            GPIO_PIN_14
  4. #define NRF24L01_CE_GPIO_CLK_ENABLE()      
  5. do{ __HAL_RCC_GPIOM_CLK_ENABLE(); }while(0)   /* PM口时钟使能 */

  6. #define NRF24L01_CSN_GPIO_PORT                            GPIOM
  7. #define NRF24L01_CSN_GPIO_PIN                      GPIO_PIN_13
  8. #define NRF24L01_CSN_GPIO_CLK_ENABLE()     
  9. do{ __HAL_RCC_GPIOM_CLK_ENABLE(); }while(0)   /* PM口时钟使能 */

  10. #define NRF24L01_IRQ_GPIO_PORT                     GPIOF
  11. #define NRF24L01_IRQ_GPIO_PIN                      GPIO_PIN_2
  12. #define NRF24L01_IRQ_GPIO_CLK_ENABLE()     
  13. do{ __HAL_RCC_GPIOF_CLK_ENABLE(); }while(0)   /* PF口时钟使能 */
  14. /* 24L01操作线 */
  15. #define NRF24L01_CE(x)    do{ x ? \
  16. HAL_GPIO_WritePin(NRF24L01_CE_GPIO_PORT,NRF24L01_CE_GPIO_PIN,GPIO_PIN_SET):\
  17. HAL_GPIO_WritePin(NRF24L01_CE_GPIO_PORT,NRF24L01_CE_GPIO_PIN,GPIO_PIN_RESET);\
  18.                                  }while(0)            /* 24L01模式选择信号 */

  19. #define NRF24L01_CSN(x)   do{ x ? \
  20. HAL_GPIO_WritePin(NRF24L01_CSN_GPIO_PORT,NRF24L01_CSN_GPIO_PIN,GPIO_PIN_SET):\
  21. HAL_GPIO_WritePin(NRF24L01_CSN_GPIO_PORT,
  22. NRF24L01_CSN_GPIO_PIN,GPIO_PIN_RESET);\
  23.                                  }while(0)            /* 24L01片选信号 */

  24. #define NRF24L01_IRQ      HAL_GPIO_ReadPin(NRF24L01_IRQ_GPIO_PORT,
  25. NRF24L01_IRQ_GPIO_PIN)      /* IRQ主机数据输入 */
复制代码
以上除了有NRF24L01的引脚定义及引脚操作函数,此外还有一些NRF24L01寄存器操作命令以及其寄存器地址,由于篇幅太大,所以这里就不列出来了,大家可以去看一下工程文件。
下面看一下NRF24L01的初始化函数,其定义如下:

  1. /**
  2. * @brief       初始化24L01的IO口
  3. * @note        将SPI2模式改成SCK空闲低电平,及SPI 模式0
  4. * @param       无
  5. * @retval      无
  6. */
  7. void nrf24l01_init(void)
  8. {
  9.     GPIO_InitTypeDef gpio_init_struct;

  10.     NRF24L01_CE_GPIO_CLK_ENABLE();                   /* CE脚时钟使能 */
  11.     NRF24L01_CSN_GPIO_CLK_ENABLE();                  /* CSN脚时钟使能 */
  12.     NRF24L01_IRQ_GPIO_CLK_ENABLE();                  /* IRQ脚时钟使能 */
  13.     HAL_PWREx_EnableUSBVoltageDetector();            /* 使用GPIOM需使能该时钟 */
  14.    
  15.     gpio_init_struct.Pin = NRF24L01_CE_GPIO_PIN;
  16.     gpio_init_struct.Mode = GPIO_MODE_OUTPUT_PP;             /* 推挽输出 */
  17.     gpio_init_struct.Pull = GPIO_PULLUP;                     /* 上拉 */
  18.     gpio_init_struct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;      /* 高速 */
  19.     HAL_GPIO_Init(NRF24L01_CE_GPIO_PORT, &gpio_init_struct); /* 初始化CE引脚 */

  20.     gpio_init_struct.Pin = NRF24L01_CSN_GPIO_PIN;
  21.     HAL_GPIO_Init(NRF24L01_CSN_GPIO_PORT, &gpio_init_struct);/* 初始化CSN引脚 */

  22.     gpio_init_struct.Pin = NRF24L01_IRQ_GPIO_PIN;
  23.     gpio_init_struct.Mode = GPIO_MODE_INPUT;                 /* 输入 */
  24.     gpio_init_struct.Pull = GPIO_PULLUP;                     /* 上拉 */
  25.     gpio_init_struct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;      /* 高速 */
  26.     HAL_GPIO_Init(NRF24L01_IRQ_GPIO_PORT, &gpio_init_struct);/* 初始化CE引脚 */

  27.     spi2_init();                                /* 初始化SPI2 */
  28.     nrf24l01_spi_init();                        /* 针对NRF的特点修改SPI的设置 */
  29.     NRF24L01_CE(0);                             /* 使能24L01 */
  30.     NRF24L01_CSN(1);                            /* SPI片选取消 */
  31. }
复制代码
在初始化函数中,我们主要对该模块用到的管脚进行配置以及从初始化工作以及需要调用spi.c文件中的spi2_init函数对SPI2的引脚进行初始化。现在让我们看一下 NRF24L01的工作时序图见下图47.3.2.1所示。

第四十七章 无线通信实验17975.png
图47.3.2.1 NRF24L01的工作时序图

大家可以比对一下前面章节的SPI工作时序图,符合工作模式1的时序,即在奇数边沿上升沿进行数据的采集。所以我们调用nrf24l01_init函数针对NRF的特点修改SPI的设置。该函数就是将SPI的工作模式配置成串行同步时钟空闲状态为低电平,在奇数边沿数据被采集,也就是前面SPI实验章节中SPI的工作模式0。看看工作时序图的某些标号意义,Cn代表指令位,Sn代表状态寄存器位,Dn代表数据位。
下面介绍一下NRF24L01的读写函数,其代码如下:

  1. /**
  2. * @brief       NRF24L01写寄存器
  3. * @param       reg   : 寄存器地址
  4. * @param       value : 写入寄存器的值
  5. * @retval      状态寄存器值
  6. */
  7. static uint8_t nrf24l01_write_reg(uint8_t reg, uint8_t value)
  8. {
  9.     uint8_t status;
  10.     NRF24L01_CSN(0);                        /* 使能SPI传输 */
  11.     status = spi2_read_write_byte(reg);     /* 发送寄存器号 */
  12.     spi2_read_write_byte(value);            /* 写入寄存器的值 */
  13.     NRF24L01_CSN(1);                        /* 禁止SPI传输 */
  14.     return status;                          /* 返回状态值 */
  15. }

  16. /**
  17. * @brief       NRF24L01读寄存器
  18. * @param       reg   : 寄存器地址
  19. * @retval      读取到的寄存器值;
  20. */
  21. static uint8_t nrf24l01_read_reg(uint8_t reg)
  22. {
  23.     uint8_t reg_val;
  24.     NRF24L01_CSN(0);                        /* 使能SPI传输 */
  25.     spi2_read_write_byte(reg);              /* 发送寄存器号 */
  26.     reg_val = spi2_read_write_byte(0XFF);   /* 读取寄存器内容 */
  27.     NRF24L01_CSN(1);                        /* 禁止SPI传输 */
  28.     return reg_val;                         /* 返回状态值 */
  29. }
复制代码
以上是NRF24L01的写寄存器函数和读寄存器函数,以及扩展的函数:配置NRF24L01为接收模式函数和配置NRF24L01为发送模式函数。
先讲一下NRF24L01读写寄存器函数实现的具体过程:
先拉低片选线→发送寄存器号→发送数据/接收数据→拉高片选线。
这里提及一下SPI的相关知识:SPI是通过移位寄存器进行数据传输,所以发一字节数据就会收到一个字节数据。那么发数据就可以直接发送数据,接收数据只需要发送0xFF,寄存器会返回要读取的数据。
配置NRF24L01为接收模式函数和配置NRF24L01为发送模式函数是通过寄存器的方式配置,这里就不做展开了。
下面看一下这两种模式的初始化过程:
Rx模式初始化过程:
1)写Rx节点的地址
2)使能通道x自动应答
3)使能通道x接收地址
4)设置通信频率
5)选择通道x的有效数据宽度
6)配置发射参数(发射功率、无线速率)
7)配置NRF24L01的基本参数以及切换工作模式
其代码如下:

  1. /**
  2. * @brief       NRF24L01进入接收模式
  3. * @note        设置RX地址,写RX数据宽度,选择RF频道,波特率
  4. *              当CE变高后,即进入RX模式,并可以接收数据了
  5. * @param       无
  6. * @retval      无
  7. */
  8. void nrf24l01_rx_mode(void)
  9. {
  10.     NRF24L01_CE(0);
  11. nrf24l01_write_buf(NRF_WRITE_REG + RX_ADDR_P0, (uint8_t *)RX_ADDRESS,
  12. RX_ADR_WIDTH);    /* 写RX节点地址 */

  13.     nrf24l01_write_reg(NRF_WRITE_REG + EN_AA, 0x01);    /* 使能通道0的自动应答 */
  14.     nrf24l01_write_reg(NRF_WRITE_REG + EN_RXADDR, 0x01);/* 使能通道0的接收地址 */
  15.     nrf24l01_write_reg(NRF_WRITE_REG + RF_CH, 40);      /* 设置RF通信频率 */
  16. nrf24l01_write_reg(NRF_WRITE_REG + RX_PW_P0, RX_PLOAD_WIDTH);   
  17. /* 选择通道0的有效数据宽度 */
  18. nrf24l01_write_reg(NRF_WRITE_REG + RF_SETUP, 0x0f);            
  19. /* 设置TX发射参数,0db增益,2Mbps */
  20. nrf24l01_write_reg(NRF_WRITE_REG + CONFIG, 0x0f);               
  21. /* 配置基本工作模式的参数;PWR_UP,EN_CRC,16BIT_CRC,接收模式 */
  22.     NRF24L01_CE(1);   /* CE为高,进入接收模式 */
  23. }
复制代码
Tx模式初始化过程:
1)写Tx节点的地址
2)写Rx节点的地址,主要为了使能硬件的自动应答
3)使能通道x的自动应答
4)使能通道x接收地址
5)配置自动重发次数
6)配置通信频率
7)选择通道x的有效数据宽度
8)配置发射参数(发射功率、无线速率)
9)配置NRF24L01的基本参数以及切换工作模式
其代码如下:

  1. /**
  2. * @brief       NRF24L01进入发送模式
  3. *   @note      设置TX地址,写TX数据宽度,设置RX自动应答的地址,填充TX发送数据,
  4. *              选择RF频道,波特率和PWR_UP,CRC使能
  5. *              当CE变高后,即进入TX模式,并可以发送数据了, CE为高大于10us,则启动发送.
  6. * @param       无
  7. * @retval      无
  8. */
  9. void nrf24l01_tx_mode(void)
  10. {
  11.     NRF24L01_CE(0);
  12. nrf24l01_write_buf(NRF_WRITE_REG + TX_ADDR, (uint8_t *)TX_ADDRESS,
  13. TX_ADR_WIDTH);       /* 写TX节点地址 */
  14. nrf24l01_write_buf(NRF_WRITE_REG + RX_ADDR_P0, (uint8_t *)RX_ADDRESS,
  15. RX_ADR_WIDTH);       /* 设置RX节点地址,主要为了使能ACK */

  16.     nrf24l01_write_reg(NRF_WRITE_REG + EN_AA, 0x01);   /* 使能通道0的自动应答 */
  17.     nrf24l01_write_reg(NRF_WRITE_REG + EN_RXADDR, 0x01);/* 使能通道0的接收地址 */
  18. nrf24l01_write_reg(NRF_WRITE_REG + SETUP_RETR, 0x1a);   
  19. /* 设置自动重发间隔时间:500us + 86us;最大自动重发次数:10次 */
  20.     nrf24l01_write_reg(NRF_WRITE_REG + RF_CH, 40);      /* 设置RF通道为40 */
  21. nrf24l01_write_reg(NRF_WRITE_REG + RF_SETUP, 0x0f);     
  22. /* 设置TX发射参数,0db增益,2Mbps */
  23. nrf24l01_write_reg(NRF_WRITE_REG + CONFIG, 0x0e);      
  24. /* 配置基本工作模式的参数;PWR_UP,EN_CRC,16BIT_CRC,接收模式,开启所有中断 */
  25.     NRF24L01_CE(1);     /* CE为高,10us后启动发送 */
  26. }
复制代码
以上就是两种模式的配置,看过完整代码的,会发现TX_ADDR和RX_ADDR两个地址是一样的,跟前面说法一致,我们必须保持地址的匹配才能通信成功。以上代码中的发送函数都有一个特点,并不是单纯发送寄存器地址,而是操作指令+寄存器地址,这一点需要记得。NRF24L01的操作指令也有好几个,它是配合寄存器完成特定的操作,其定义如下:
  1. /* NRF24L01命令定义 */
  2. #define NRF_READ_REG           0x00           /* 读配置寄存器,低5位为寄存器地址 */
  3. #define NRF_WRITE_REG          0x20          /* 写配置寄存器,低5位为寄存器地址 */
  4. #define RD_RX_PLOAD           0x61          /* 读RX有效数据,1~32字节 */
  5. #define WR_TX_PLOAD            0xA0          /* 写TX有效数据,1~32字节 */
  6. #define FLUSH_TX              0xE1          /* 清除TX FIFO寄存器.发射模式下用 */
  7. #define FLUSH_RX             0xE2    /* 清除RX FIFO寄存器.接收模式下用 */
  8. #define REUSE_TX_PL           0xE3    /* 重新使用上一包数据,CE为高,数据包被不断发送. */
  9. #define NOP                    0xFF    /* 空操作,可以用来读状态寄存器 */
复制代码
经过上面的发送或者接收模式初始化步骤后,NRF24L01就可以准备启动发送数据或者等待接收数据了。
下面来看一下启动NRF24L01发送一次数据的函数,其定义如下:

  1. /**
  2. * @brief       启动NRF24L01发送一次数据(数据长度 = TX_PLOAD_WIDTH)
  3. * @param       ptxbuf : 待发送数据首地址
  4. * @retval      发送完成状态
  5. *   @arg       0    : 发送成功
  6. *   @arg       1    : 达到最大发送次数,失败
  7. *   @arg       0XFF : 其他错误
  8. */
  9. uint8_t nrf24l01_tx_packet(uint8_t *ptxbuf)
  10. {
  11.     uint8_t sta;
  12.     uint8_t rval = 0XFF;
  13.    
  14.     NRF24L01_CE(0);
  15. nrf24l01_write_buf(WR_TX_PLOAD, ptxbuf, TX_PLOAD_WIDTH);   
  16. /* 写数据到TX BUF  TX_PLOAD_WIDTH个字节 */
  17.     NRF24L01_CE(1);                                      /* 启动发送 */

  18.     while (NRF24L01_IRQ != 0);                           /* 等待发送完成 */

  19.     sta = nrf24l01_read_reg(STATUS);                     /* 读取状态寄存器的值 */
  20. nrf24l01_write_reg(NRF_WRITE_REG + STATUS, sta);     
  21. /* 清除TX_DS或MAX_RT中断标志 */

  22.     if (sta & MAX_TX)                                    /* 达到最大重发次数 */
  23.     {
  24.         nrf24l01_write_reg(FLUSH_TX, 0xff);              /* 清除TX FIFO寄存器 */
  25.         rval = 1;
  26.     }

  27.     if (sta & TX_OK)                                     /* 发送完成 */
  28.     {
  29.         rval = 0;                                        /* 标记发送成功 */
  30.     }

  31.     return rval;                                         /* 返回结果 */
  32. }
复制代码
在这里启动发送数据函数中,具体实现很简单,拉低片选信号→向发送数据寄存器写入数据→拉高片选信号。这里说明一下,在发送完寄存器号后都会返回一个status值,返回的这个值就是前面介绍的STATUS寄存器的内容。在这个基础上就可以知道数据是否发送完成以及现在的状态。
然后介绍一下NRF24L01接收一次数据函数,其定义如下:

  1. /**
  2. * @brief       启动NRF24L01接收一次数据(数据长度 = RX_PLOAD_WIDTH)
  3. * @param       prxbuf : 接收数据缓冲区首地址
  4. * @retval      接收完成状态
  5. *   @arg       0 : 接收成功
  6. *   @arg       1 : 失败
  7. */
  8. uint8_t nrf24l01_rx_packet(uint8_t *prxbuf)
  9. {
  10.     uint8_t sta;
  11.     uint8_t rval = 1;
  12.    
  13.     sta = nrf24l01_read_reg(STATUS);                 /* 读取状态寄存器的值 */
  14.     nrf24l01_write_reg(NRF_WRITE_REG + STATUS, sta); /* 清除RX_OK中断标志 */

  15.     if (sta & RX_OK)                                 /* 接收到数据 */
  16.     {
  17.         nrf24l01_read_buf(RD_RX_PLOAD, prxbuf, RX_PLOAD_WIDTH); /* 读取数据 */
  18.         nrf24l01_write_reg(FLUSH_RX, 0xff);          /* 清除RX FIFO寄存器 */
  19.         rval = 0;                                    /* 标记接收完成 */
  20.     }

  21.     return rval;                                     /* 返回结果 */
  22. }
复制代码
在启动接收的过程中,首先需要判断当前NRF24L01的状态,往后才是真正的读取数据,清除接收寄存器的缓冲,完成数据的接收。这里需要注意的是我们通过NRF24L01_RX_PLOAD_WIDTH和NRF24L01_TX_PLOAD_WIDTH决定了接收和发送的数据宽度,这也决定每次发送和接收的有效字节数。NRF24L01每次最多传输32个字节,再多的字节传输则需要多次传输。通信双方的发送和接收数据宽度必须一致才能正常通信。

3. main.c代码
在main.c里编写如下代码:

  1. int main(void)
  2. {
  3.     uint8_t key, mode;
  4.     uint16_t t = 0;
  5.     uint8_t tmp_buf[33];
  6.    
  7.     sys_mpu_config();                   /* 配置MPU */
  8.     sys_cache_enable();                 /* 使能Cache */
  9.     HAL_Init();                         /* 初始化HAL库 */
  10.     sys_stm32_clock_init(300, 6, 2);    /* 配置时钟,600MHz */
  11.     delay_init(600);                    /* 初始化延时 */
  12.     usart_init(115200);                 /* 初始化串口 */
  13.     led_init();                         /* 初始化LED */
  14.     key_init();                         /* 初始化按键 */
  15.     hyperram_init();                    /* 初始化HyperRAM */
  16.     lcd_init();                         /* 初始化LCD */
  17.     nrf24l01_init();                    /* NRF24L01初始化 */
  18.    
  19.     lcd_show_string(30, 50, 200, 16, 16, "STM32", RED);
  20.     lcd_show_string(30, 70, 200, 16, 16, "NRF24L01 TEST", RED);
  21.     lcd_show_string(30, 90, 200, 16, 16, "ATOM@ALIENTEK", RED);

  22.     while (nrf24l01_check())            /* 检查NRF24L01是否在线 */
  23.     {
  24.         lcd_show_string(30, 130, 200, 16, 16, "NRF24L01 Error", RED);
  25.         delay_ms(200);
  26.         lcd_fill(30, 130, 239, 130 + 16, WHITE);
  27.         delay_ms(200);
  28.     }

  29.     lcd_show_string(30, 130, 200, 16, 16, "NRF24L01 OK", RED);
  30.    
  31.     while (1)                   /* 提醒用户选择模式 */
  32.     {
  33.         key = key_scan(0);

  34.         if (key == KEY0_PRES)
  35.         {
  36.             mode = 0;           /* 接收模式 */
  37.             break;
  38.         }
  39.         else if (key == KEY1_PRES)
  40.         {
  41.             mode = 1;           /* 发送模式 */
  42.             break;
  43.         }

  44.         t++;

  45.         if (t == 100)           /* 显示提示信息 */
  46.         {
  47.             lcd_show_string(10, 150, 230, 16, 16, "KEY0:RX_Mode  KEY1:TX_Mode",
  48.   RED);
  49.         }
  50.         
  51.         if (t == 200)           /* 关闭提示信息 */
  52.         {
  53.             lcd_fill(10, 150, 230, 150 + 16, WHITE);
  54.             t = 0;
  55.         }

  56.         delay_ms(5);
  57.     }

  58.     lcd_fill(10, 150, 240, 166, WHITE); /* 清空上面的显示 */

  59.     if (mode == 0)                      /* RX模式 */
  60.     {
  61.         lcd_show_string(30, 150, 200, 16, 16, "NRF24L01 RX_Mode", BLUE);
  62.         lcd_show_string(30, 170, 200, 16, 16, "Received DATA:", BLUE);
  63.         nrf24l01_rx_mode();             /* 进入RX模式 */

  64.         while (1)
  65.         {
  66.             if (nrf24l01_rx_packet(tmp_buf) == 0) /* 一旦接收到信息,则显示出来. */
  67.             {
  68.                 tmp_buf[32] = 0;        /* 加入字符串结束符 */
  69.                 lcd_show_string(0, 190, lcddev.width - 1, 32, 16,
  70.         (char*)tmp_buf, BLUE);
  71.             }
  72.             else delay_us(100);

  73.             t++;

  74.             if (t == 10000)             /* 大约1s钟改变一次状态 */
  75.             {
  76.                 t = 0;
  77.                 LED0_TOGGLE();
  78.             }
  79.         }
  80.     }
  81.     else    /* TX模式 */
  82.     {
  83.         lcd_show_string(30, 150, 200, 16, 16, "NRF24L01 TX_Mode", BLUE);
  84.         nrf24l01_tx_mode();             /* 进入TX模式 */
  85.         mode = ' ';                     /* 从空格键开始发送 */

  86.         while (1)
  87.         {
  88.             if (nrf24l01_tx_packet(tmp_buf) == 0)   /* 发送成功 */
  89.             {
  90.                 lcd_show_string(30, 170, 239, 32, 16, "Sended DATA:", BLUE);
  91.                 lcd_show_string(0, 190, lcddev.width - 1, 32, 16,
  92. (char*)tmp_buf, BLUE);
  93.                 key = mode;

  94.                 for (t = 0; t < 32; t++)
  95.                 {
  96.                     key++;

  97.                     if (key > ('~'))key = ' ';

  98.                     tmp_buf[t] = key;
  99.                 }

  100.                 mode++;

  101.                 if (mode > '~')mode = ' ';

  102.                 tmp_buf[32] = 0;         /* 加入结束符 */
  103.             }
  104.             else
  105.             {
  106.                 lcd_fill(0, 170, lcddev.width, 170 + 16 * 3, WHITE);   
  107. /* 清空显示 */
  108.                 lcd_show_string(30, 170, lcddev.width - 1, 32, 16,
  109. "Send Failed ", BLUE);
  110.             }

  111.             LED0_TOGGLE();
  112.             delay_ms(200);
  113.         };
  114.     }
  115. }
复制代码
程序运行时先通过nrf24l01_cheak函数检测NRF24L01是否存在,如果存在,则让用户选择发送模式还是接收模式,在确定模式之后,设置NRF24L01的工作模式,然后执行相对应的数据发送/接收处理。

47.4 下载验证
将程序下载到开发板后,可以看到LCD显示的内容如图47.4.1所示:

第四十七章 无线通信实验28538.png
图47.4.1  选择工作模式界面

通过KEY0和KEY1来选择NRF24L01模块所要进入的工作模式,我们两个开发板一个选择发送模式,一个选择接收模式就可以了。设置好的通信界面如图47.4.2和图47.4.3所示:

第四十七章 无线通信实验28652.png
图47.4.2 开发板A通信界面(发送)

第四十七章 无线通信实验28675.png
图47.4.3 开发板B通信界面(接收)

图47.4.2来自于开发板A,工作在发送模式。图47.4.3来自于开发板B,工作在接收模式,A发送,B接收。发送和接收图片的数据不一样,是因为我们拍照的时间不一样导致的。大家看到收发数据是一致,那就说明实验成功。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /1 下一条

正点原子公众号

如发现本坛存在违规或侵权内容, 请点击这里发送邮件举报 (或致电020-38271790)。请提供侵权说明和联系方式。我们将及时审核依法处理,感谢配合。

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2026-5-22 13:33

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表