完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
扫一扫,分享给好友
一,模块介绍
二,接口电路, 模块连接注意点: (1) VCC 脚接电压范围为 1.9V~3.6V 之间,不能在这个区间之外,超 过 3.6V 将会烧毁模块。推荐电压 3.3V 左右。 (2) 除电源 VCC 和接地端,其余脚都可以直接和普通的 5V 单片机 IO 口 直接相连,无需电平转换。当然对 3V 左右的单片机更加适用了。 (3) 硬件上面没有 SPI 的单片机也可以控制本模块,用普通单片机 IO 口模拟 SPI 不需要单片机真正的串口介入,只需要普通的单片机 IO 口 就可以了,当然用串口也可以了。 然后我接的就是CE连的A4,CSN连的C4,SCK连的A5,MOSI连的A7,MISO连的A6,IRQ是中断脚,可以接外部中断,不过我没有接。这里注意一下哈,中断脚要想低电平触发的三种情况:1,发送端发送完了数据并接收到接收端的ACK应答;2,接收端接受到数据;3,达到最大重发次数(外部中断的时候需要考虑的) 三,工作模式 工作模式由 PWR_UP register 、PRIM_RX register 和 CE 决定,有如下表格中那么多的工作模式 (英文版) (中文版) 其中收发模式有 Enhanced ShockBurstTM 收发模式、ShockBurstTM 收 发模式和直接收发模式三种,收发模式由器件配置字决定,但是主要用到的是 Enhanced ShockBurstTM 收发模式,只有该模式支持自动ANK,和自动重发。 下面也都是介绍的这个接收模式。 就是上面这张图,想象一下,这一个圈就是一个带有8个爪子的无线模块,RX是接收功能的(设为接收模式)模块,TX是发送功能的(设为发送模式)模块,这样就是被设为接收模式的端子有6个接受通道,而被,设为发送模式的端子有1个发送通道,它们的传输是这样子的: 发送端子的TX_add寄存器存放的是接收端子6个通道中的其中一个通道的地址,然后传送完数据之后,接收端在确认收到数据后记录发送端的地址,并以此地址(接收端某个通道的地址)为目标地址发送应答信号这个是自动的,不是程序设置的,应属于硬件实现) 发送端的2401会用自己接收通道0来接受接收端的2401发送来的相应信号。但是接收端的2401发送应答信号的时候,也发送的地址就是接收端的某个接受通道的地址。所以发送端的接收通道0的地址要设置成这个地址。注意就只能是发送通道的通道0作为应答信号 通道0有40位自身地址,通道1—5都为8位自身地址和32位公用地址。 例子: TX5:TX_ADDR=0xB3B4B5B605 TX5:RX_ADDR_P0=0xB3B4B5B605 RX:RX_ADDR_P5=0xB3B4B5B605 介绍完模式之后就来看 该模式的发送的流程以及初始化该发送模式 该模式的接收流程以及初始化该接收模式 知道了这个无线模块的接收和发送操作流程还有接收和发送的模式配置,然后就要关注它的通信方式spi了:发送spi读写指令,并对寄存器读写,(读出寄存器里面的值或者像寄存器里面写入数据)主要学习的是库函数的版本,与之有关的寄存器也要了解一下哈~ 四,相关寄存器介绍 这里是重点,找机会补上。 五, spi通信有关的库函数调用 stm32是有硬件spi的,这里连的是a5,6,7三个脚,(虽然一般上认为spi通信是4个脚,但是有一个是片选脚,在spi封装的库函数模块里面,没有加进去) 然后原子例程里是由现成的spi封装好的函数模块的,可以直接加进去用来着。 无线模块有关的库函数调用 NRF24L01初始化函数,因为是原子的代码,MINI板的测试代码, 这里NRF24L01也是使用的SPI1,和W25Q64以及SD卡等共用一个SPI接口,所以在使用的时候,他们分时复用SPI1。所以需要把SD卡和W25Q64的片选信号置高,以防止这两个器件对NRF24L01的通信造成干扰。所以也对没有用到的脚进行了拉高处理,就是代码中的A2,A3脚。所以自己实际在用的过程中可以不去处理这两个脚。emmmm大佬们都是可以自己写的,我只会拉人家的来用~(卑微) void NRF24L01_Init(void){GPIO_InitTypeDef GPIO_InitStructure;SPI_InitTypeDef SPI_InitStructure;RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOC, ENABLE );GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP ; //推挽输出GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP ; //推挽输出GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOC, &GPIO_InitStructure);GPIO_SetBits(GPIOC,GPIO_Pin_4);GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU ; //上拉输入GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);GPIO_SetBits(GPIOA,GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4);SPI1_Init(); //初始化 SPISPI_Cmd(SPI1, DISABLE); //SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;//设置 SPI 单向或者双向的数据模式:SPI 设置为双线双向全双工SPI_InitStructure.SPI_Mode = SPI_Mode_Master; //设置为主 SPISPI_InitStructure.SPI_DataSize = SPI_DataSize_8b; // 8 位帧结构SPI_InitStructure.SPI_CPOL = SPI_CPOL_Low; //时钟悬空低电平SPI_InitStructure.SPI_CPHA = SPI_CPHA_1Edge; //数据捕获于第一个时钟沿SPI_InitStructure.SPI_NSS = SPI_NSS_Soft; //内部 NSS 信号有 SSI 位控制SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;//定义波特率预分频的值:波特率预分频值为 256SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB; //数据传输从 MSB 位开始SPI_InitStructure.SPI_CRCPolynomial = 7; //CRC 值计算的多项式SPI_Init(SPI1, &SPI_InitStructure); //初始化外设 SPIx 寄存器NRF24L01_CE=0; //使能 24L01NRF24L01_CSN=1; //SPI 片选取消}//检测 24L01 是否存在//返回值:0,成功;1,失败1234567891011121314151617181920212223242526272829303132333435363738然后是模式设置,就是上面第三点所说的,如下是接收模式 void NRF24L01_RX_Mode(void){NRF24L01_CE=0;NRF24L01_Write_Buf(WRITE_REG+RX_ADDR_P0,(u8*)RX_ADDRESS,RX_ADR_WIDTH);//写 RX 节点地址NRF24L01_Write_Reg(WRITE_REG+EN_AA,0x01); //使能通道 0 的自动应答NRF24L01_Write_Reg(WRITE_REG+EN_RXADDR,0x01);//使能通道 0 的接收地址NRF24L01_Write_Reg(WRITE_REG+RF_CH,40); //设置 RF 通信频率NRF24L01_Write_Reg(WRITE_REG+RX_PW_P0,RX_PLOAD_WIDTH);//选择通道 0 的有效数据宽度NRF24L01_Write_Reg(WRITE_REG+RF_SETUP,0x0f);//设置 TX 发射参数,0db 增益,2Mbps,低噪声增益开启NRF24L01_Write_Reg(WRITE_REG+CONFIG, 0x0f);//配置基本工作模式的参数;PWR_UP,EN_CRC,16BIT_CRC,接收模式NRF24L01_CE = 1; //CE 为高,进入接收模式}12345678910111213141516还有发送模式 void NRF24L01_TX_Mode(void){NRF24L01_CE=0;NRF24L01_Write_Buf(WRITE_REG+TX_ADDR,(u8*)TX_ADDRESS,TX_ADR_WIDTH);//写 TX 节点地址NRF24L01_Write_Buf(WRITE_REG+RX_ADDR_P0,(u8*)RX_ADDRESS,RX_ADR_WIDTH);//设置 TX 节点地址,主要为了使能 ACKNRF24L01_Write_Reg(WRITE_REG+EN_AA,0x01); //使能通道 0 的自动应答NRF24L01_Write_Reg(WRITE_REG+EN_RXADDR,0x01); //使能通道 0 的接收地址NRF24L01_Write_Reg(WRITE_REG+SETUP_RETR,0x1a);//设置自动重发间隔时间:500us + 86us;最大自动重发次数:10 次NRF24L01_Write_Reg(WRITE_REG+RF_CH,40); //设置 RF 通道为 40NRF24L01_Write_Reg(WRITE_REG+RF_SETUP,0x0f);//设置 TX 发射参数,0db 增益,2Mbps,低噪声增益开启NRF24L01_Write_Reg(WRITE_REG+CONFIG,0x0e);//配置基本工作模式的参数;PWR_UP,EN_CRC,16BIT_CRC,接收模式,开启所有中断NRF24L01_CE=1;//CE 为高,10us 后启动发送}123456789101112131415161718模式确定完了就要写具体的收发流程,如果是接受模式,那接收流程如下 u8 NRF24L01_RxPacket(u8 *rxbuf){u8 sta;SPI2_SetSpeed(SPI_BaudRatePrescaler_8); //spi 速度为 9Mhzsta=NRF24L01_Read_Reg(STATUS); //读取状态寄存器的值NRF24L01_Write_Reg(WRITE_REG+STATUS,sta); //清除 TX_DS 或 MAX_RT 中断标志if(sta&RX_OK)//接收到数据{NRF24L01_Read_Buf(RD_RX_PLOAD,rxbuf,RX_PLOAD_WIDTH);//读取数据NRF24L01_Write_Reg(FLUSH_RX,0xff);//清除 RX FIFO 寄存器return 0;}return 1;//没收到任何数据}12345678910111213141516若为发送模式,发送流程为 u8 NRF24L01_TxPacket(u8 *txbuf){u8 sta;SPI2_SetSpeed(SPI_BaudRatePrescaler_8);//spi 速度为 9MhzNRF24L01_CE=0;NRF24L01_Write_Buf(WR_TX_PLOAD,txbuf,TX_PLOAD_WIDTH); //写数据到 TX BUFNRF24L01_CE=1; //启动发送while(NRF24L01_IRQ!=0); //等待发送完成sta=NRF24L01_Read_Reg(STATUS); //读取状态寄存器的值NRF24L01_Write_Reg(WRITE_REG+STATUS,sta); //清除 TX_DS 或 MAX_RT 中断标志if(sta&MAX_TX)//达到最大重发次数{NRF24L01_Write_Reg(FLUSH_TX,0xff);//清除 TX FIFO 寄存器return MAX_TX;}if(sta&TX_OK) return TX_OK;//发送完成return 0xff;//其他原因发送失败}123456789101112131415161718接收和发送的操作是基于用spi的通信方式对无线模块寄存器的读和写的操作,所以,spi读写寄存器的指令如下 u8 NRF24L01_Write_Reg(u8 reg,u8 value){u8 status;NRF24L01_CSN=0; //使能 SPI 传输status =SPI2_ReadWriteByte(reg);//发送寄存器号SPI2_ReadWriteByte(value); //写入寄存器的值NRF24L01_CSN=1; //禁止 SPI 传输return(status); //返回状态值}u8 NRF24L01_Read_Reg(u8 reg){u8 reg_val;NRF24L01_CSN = 0; //使能 SPI 传输SPI2_ReadWriteByte(reg); //发送寄存器号reg_val=SPI2_ReadWriteByte(0XFF);//读取寄存器内容NRF24L01_CSN = 1; //禁止 SPI 传输return(reg_val); //返回状态值}12345678910111213141516171819指令发送了,就是说好啦,寄存器,我要开始对你进行读写数据了,然后就是对寄存器里面的数据进行读写 读寄存器的值 u8 NRF24L01_Read_Buf(u8 reg,u8 *pBuf,u8 len){u8 status,u8_ctr;NRF24L01_CSN = 0; //使能 SPI 传输status=SPI2_ReadWriteByte(reg);//发送寄存器值(位置),并读取状态值for(u8_ctr=0;u8_ctr u8 NRF24L01_Write_Buf(u8 reg, u8 *pBuf, u8 len){u8 status,u8_ctr;NRF24L01_CSN = 0; //使能 SPI 传输status = SPI2_ReadWriteByte(reg);//发送寄存器值(位置),并读取状态值for(u8_ctr=0; u8_ctr u8 NRF24L01_Check(void){u8 buf[5]={0XA5,0XA5,0XA5,0XA5,0XA5}; u8 i;SPI2_SetSpeed(SPI_BaudRatePrescaler_8); //spi 速度为 9MhzNRF24L01_Write_Buf(WRITE_REG+TX_ADDR,buf,5);//写入 5 个字节的地址.NRF24L01_Read_Buf(TX_ADDR,buf,5); //读出写入的地址for(i=0;i<5;i++)if(buf!=0XA5)break;if(i!=5)return 1;//检测 24L01 错误return 0; //检测到 24L01} |
|
|
|
只有小组成员才能发言,加入小组>>
调试STM32H750的FMC总线读写PSRAM遇到的问题求解?
1884 浏览 1 评论
X-NUCLEO-IHM08M1板文档中输出电流为15Arms,15Arms是怎么得出来的呢?
1663 浏览 1 评论
1149 浏览 2 评论
STM32F030F4 HSI时钟温度测试过不去是怎么回事?
763 浏览 2 评论
ST25R3916能否对ISO15693的标签芯片进行分区域写密码?
1720 浏览 2 评论
1964浏览 9评论
STM32仿真器是选择ST-LINK还是选择J-LINK?各有什么优势啊?
790浏览 4评论
STM32F0_TIM2输出pwm2后OLED变暗或者系统重启是怎么回事?
614浏览 3评论
631浏览 3评论
stm32cubemx生成mdk-arm v4项目文件无法打开是什么原因导致的?
593浏览 3评论
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2025-1-13 17:29 , Processed in 0.640160 second(s), Total 46, Slave 40 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号