第十七章 宝马1768——NRF24L01 2.4G无线通信
开发环境:集成开发环境μVision4 IDE版本4.60.0.0 主机系统:Microsoft Windows XP 开发平台:旺宝NXP LPC1768开发板
17.1 2.4G无线模块 17.2 硬件描述 17.3 程序说明 17.4 实验现象 本章我们主要介绍的是宝马开发板的预留接口2.4G无线模块,学习本章可以了解到2.4G通信原理,及使用两板进行通信。我们到目前为止,使用SPI接口的器件有字库(W25Q16)、SD卡、2.4G、还有我们后面会接触到的XPT2046触摸芯片。都是使用SPI接口与MCU通信,看来掌握SPI的操作是很重要的。 1、2.4G无线模块:nRF24L01是一款新型单片射频收发器件无线模块,工作全球免申请2.4 GHz~2.5 GHz ISM频段,最大发射功率0dBm(1mW)。内置频率合成器、功率放大器、晶体振荡器、调制器等功能模块,并融合了增强型ShockBurst技术,其中输出功率和通信频道可通过程序进行配置。nRF24L01功耗低,在以-6dBm的功率发射时,工作电流也只有9 mA;接收时,工作电流只有12.3 mA,多种低功率工作模式(掉电模式和空闲模式)使节能设计更方便。 主要特性: 1)支持六路通道的数据接收 2)低工作电压:1.9-3.6V低电压工作 3)高速率:2Mbps(软件设置1Mbps或2Mbps) 4)多点频:125频点,满足多点通信和调频通信需要(2400+RF_CH) 5)nRF24L01集成了所有与RF协议相关的高速信号处理部分 6)SPI接口可以方便和单片机进行通信 如上图,nRF24L01与单片机的接口,具体功能如下表描述 工作模式: 1) nRF24L01有四种工作模式: 2) 收发模式 3) 配置模式 4) 空闲模式 5) 关机模式 数据发射:发射数据时,首先将nRF24L01配置为发射模式:接着把接收节点地址TX_ADDR和有效数据TX_PLD按照时序由SPI口写入nRF24L01缓存区,TX_PLD必须在CSN为低时连续写入,而TX_ADDR在发射时写入一次即可,然后CE置为高电平并保持至少10μs,延迟130μs后发射数据;若自动应答开启,那么nRF24L01在发射数据后立即进入接收模式,接收应答信号(自动应答接收地址应该与接收节点地址TX_ADDR一致)。如果收到应答,则认为此次通信成功,TX_DS置高,同时TX_PLD从TX FIFO中清除;若未收到应答,则自动重新发射该数据(自动重发已开启),若重发次数(ARC)达到上限,MAX_RT置高,TX FIFO中数据保留以便再次重发;MAX_RT或TX_DS置高时,使IRQ变低,产生中断,通知MCU。最后发射成功时,若CE为低则nRF24L01进入空闲模式1;若发送堆栈中有数据且CE为高,则进入下一次发射;若发送堆栈中无数据且CE为高,则进入空闲模式2。 数据接收:接收数据时,首先将nRF24L01配置为接收模式,接着延迟130μs进入接收状态等待数据的到来。当接收方检测到有效的地址和CRC时,就将数据包存储在RX FIFO中,同时中断标志位RX_DR置高,IRQ变低,产生中断,通知MCU去取数据。若此时自动应答开启,接收方则同时进入发射状态回传应答信号。最后接收成功时,若CE变低,则nRF24L01进入空闲模式1。 NRF24L01的读写时序: LPC1768通过SSP0接口对nRF24L01进行配置,具体指令及寄存器请参考手册,其指令格式如下 命令字:由高位到低位(每字节) 数据字节:低字节到高字节,每一字节高位在前 任何一条新指令均由CSN 的由高到低的转换开始,下图为SPI操作及时序。 在写寄存器之前一定要进入待机模式或掉电模式。 读操作 写操作 2、硬件描述: 了解了NRF24L01的基本工作模式和读写时序,我们再来看看它是如何与1768连接的。 3、程序说明: 本次实验使用两块宝马开发板(带nRF24L01无线模块)进行通信,板A为主动发射端,板B为被动接收端。 板A:初始化完成后,进入主循环通过无线模块发送数据 板B:初始化完成后进入循环查询NRF24L01是否有数据接收到,当接收到板A发送过来的数据后,将数据通过串口打印出来。 如下图是LPC1768与nRF24L01的接口描述,其中P1.20、P1.23和P1.24分别为SSP0的SCK、MISO和MOSI引脚,P1.21接CSN,P2.0接CE,P2.1接可屏蔽中断引脚。 LPC1768对nRF24L01的配置及读写代码如下: uchar SPI1_NRF_SendByte(unsigned char byte) { LPC_SSP0->DR = byte; while (LPC_SSP0->SR & 0x10); //等待发送完成 return (LPC_SSP0->DR); //返回接收到的值 } /*写数据到寄存器,并返回寄存器读回的数据*/ uchar SPI_RW_Reg(uchar reg, uchar value) { unsigned int Data = 0; Select_NRF(); //选择NRF24L01片选 Data=SPI1_NRF_SendByte(reg); //指定NRF24L01寄存器 SPI_NRF_SendByte(value); //写入数据 NotSelect_NRF(); //禁止NRF24L01片选 return(Data); //返回NRF24L01 写寄存器的状态信息 } /*读取寄存器中的数据*/ unsigned char SPI_Read(unsigned char reg) { unsigned char Data; Select_NRF(); //选择NRF24L01片选 SPI1_NRF_SendByte(reg); //指定NRF24L01寄存器 Data=SPI1_NRF_SendByte(0); //读出数据 NotSelect_NRF(); //禁止NRF24L01片选 return (Data); } /*读取批量数据并存在pBuf缓冲区*/ uchar SPI_Read_Buf(uchar reg, uchar *pBuf, uchar bytes) { unsigned char status,i; Select_NRF(); //选择NRF24L01片选 status=SPI_NRF_SendByte(reg); //读出指定NRF24L01寄存器的状态信息 for(i=0; i<bytes; i++) //读出指定长度的数据 { pBuf=SPI1_NRF_SendByte(0); } NotSelect_NRF(); //禁止NRF24L01片选 return(status); //返回指定NRF24L01寄存器的状态信息 } /*写缓冲*/ uchar SPI_Write_Buf(uchar reg, uchar *pBuf, uchar bytes) { unsigned char status,byte_ctr; Select_NRF(); //选择NRF24L01片选 status=SPI1_NRF_SendByte(reg); //指定NRF24L01寄存器 for(byte_ctr=0; byte_ctr<bytes; byte_ctr++) //写入指定长度的数据 { SPI1_NRF_SendByte(*pBuf++); } NotSelect_NRF(); //禁止NRF24L01片选 return(status); //返回NRF24L01 写寄存器的状态信息 } /*ssp0初始化*/ void ssp0_init(void) { LPC_SC->PCONP |= (1 << 21); //Enable power to SSPI0 block LPC_GPIO1->FIODIR |= (1<<21); //P1.21初始化SSEL LPC_GPIO1->FIOPIN |= (1<<21); //设置为高电平 LPC_PINCON->PINSEL3 &= ~(0<<10); //P1.21做普通IO LPC_GPIO3->FIODIR |= (1<<26); //P3.26输出 LPC_GPIO3->FIOPIN &= ~(1<<26); //P3.26输出低电平 关闭SD卡 LPC_PINCON->PINSEL3 &= ~(3<<8); LPC_PINCON->PINSEL3 |= (3<<8); // P1.20设置为SCK0 LPC_PINCON->PINSEL3 &= ~((3<<14) | (3<<16)); LPC_PINCON->PINSEL3 |= ((3<<14) | (3<<16)); // P1.23设置为MISO0, P1.24 设置为MOSI0 LPC_SC->PCLKSEL1 &= ~(3<<10); // PCLKSP0 = CCLK/4 ( 25MHz) LPC_SC->PCLKSEL1 |= (1<<10); // PCLKSP0 = CCLK (100MHz) LPC_SSP0->CPSR = 250; //72MHz / 250 = 400kBit //maximum of 18MHz is possible LPC_SSP0->CR0 = 0x0007; // 8Bit, CPOL=0, CPHA=0 LPC_SSP0->CR1 = 0x0002; // SSP0 enable, master } /*初始化RF24L01内部数据*/ void config_24L01() { RF24L01_Gpio_Init(); //IO初始化 NotSelect_NRF(); //禁止SPI2 NRF24L01+的片选。 ssp0_init(); //ssp0初始化 } /*24L01初始化*/ void Init_24L01(void) { uchar Channel=2; //工作通道频率=2400+2=2.402G config_24L01(); SPI_RW_Reg(WRITE_REGL + CONFIG,0x79); //屏蔽所有中断、接收模式 SPI_RW_Reg(WRITE_REGL + EN_AA,0x00); //禁止自动应答 SPI_RW_Reg(WRITE_REGL + EN_RXADDR,0x01); //接收数据通道0允许 SPI_RW_Reg(WRITE_REGL + SETUP_AW,0x03);//接收发射地址宽度5字节 SPI_RW_Reg(WRITE_REGL + SETUP_RETR,0x00); //禁止自动重发 SPI_RW_Reg(WRITE_REGL + RF_CH,Channel); //设置工作通道频率 SPI_RW_Reg(WRITE_REGL + RF_SETUP,0x07); //传输速率1Mbps、0dBm SPI_RW_Reg(WRITE_REGL + STATUS,0x70); //清空中断标志 SPI_Write_Buf(WRITE_REGL+RX_ADDR_P0,TX_ADDRESS, RX_ADR_WIDTH); //数据通道0接收地址 SPI_Write_Buf(WRITE_REGL + TX_ADDR, TX_ADDRESS, TX_ADR_WIDTH); //发送地址 SPI_RW_Reg(WRITE_REGL + RX_PW_P0,0x04); //接收数据通道0有效数据宽度4字节 CSN(0); SPI_NRF_SendByte(0xE2); //清除RX FIFO寄存器 CSN(1); CSN(0); SPI_NRF_SendByte(0xE1); //清除TX FIFO寄存器 CSN(1); } /*发送数据*/ void send(uchar *Buffdata,uchar Length) { CE(0); SPI_RW_Reg(WRITE_REGL + CONFIG,0x1e);//CRC16位校验 发送模式 SPI_RW_Reg(FLUSH_TX,0); //清除TX FIFO寄存器 用于发射模式下 SPI_Write_Buf(WR_TX_PLOAD, Buffdata ,Length); //写TX有效数据Buffdata Length字节 CE(1); Delay1ms(5); CE(0); SPI_RW_Reg(WRITE_REGL + STATUS,0x70); //清除中断标志 } /*设置接收模式*/ void Set_RF_RX_Mode(uchar DataLength) { SPI_RW_Reg(WRITE_REGL + RX_PW_P0,DataLength); //设置接收数据通道0有效数据宽度DataLength SPI_RW_Reg(WRITE_REGL + CONFIG,0x5f);//CRC 16位 接收模式 SPI_RW_Reg(WRITE_REGL + STATUS,0x70);//清除中断标志 SPI_RW_Reg(FLUSH_RX,0); //清除RX FIFO寄存器 用于接收模式下 CE(1); } 其中具体的寄存器定义及其设置可以参考源程序中24L01头文件的定义和NRF24L01手册。本历程包含了发送和接收,通过宏定义来设置 当作为接收板时修改宏定义如下: 当作为发送板时修改宏定义如下: 当定义为接收板时条件编译选择运行红色框框内的程序,当定义为发送板时,条件编译选择运行蓝色框框内的程序。从下面的程序中可以看出当作为接收板时开始串口打印数据“NRF24L01 接收板”后初始化NRF 模块,循环查询是否有数据接收到。当作为发送板时从串口打印数据“NRF24L01发送板”后初始化NRF模块,每隔100ms通过NRF24L01发送四个字符“ab0d”
4、实验现象: 将开发板USB转串口(usb1)与电脑相连,插上nRF24L01,下载程序后,重新上电,打开串口调试助手(波特率9600),板A从串口打印出来“NRF24L01发送板”,板B从串口打印出来“NRF24L01接收板”,随后打印NRF24L01接收到的数据。
|