这些小活动你都参加了吗?快来围观一下吧!>>
电子产品世界 » 论坛首页 » 嵌入式开发 » STM32 » 利用STM32F103C8T6与NRF24L01无线通讯(从机部分)

共1条 1/1 1 跳转至

利用STM32F103C8T6与NRF24L01无线通讯(从机部分)

助工
2024-09-19 21:34:59     打赏

一:参照之前在论坛发布的帖子,此片文件作为无线通讯模块的接收端,完成USB与RS485、RS232、TTL之间的电平转换,从而将数据USB模块发送的数据成功的发送到模块当中。

下面和大家分享一下该从机的模块的硬件设计、软件开发及开发中遇到的问题

硬件设计部分:    

2.1 自动收发485电路

1726749225070.jpg

由于RS485是半双工通讯方式,而这里在开发的时候由于为了兼容其他通讯电平,这里将制作了一个485自动切换发送、接收状态的电路,为了电路简单这里使用3V3供电的MAX3485(也可以使用SP3485替换),而485的后端,并未使用TVS管保护电路、匹配电阻电路,只是在A相、B相增加上下拉电阻。

这里简单分析一下,电路知识:当有数据发送时,485发送数据使能引脚,由于Q2nMOS导通,而拉高变为高电平,当没有数据发送时,进入接收状态。

这个电路已经我长时间的测试,很稳定的,大家在设计自动收发电路时,可以参考一下。

2.2 485、232、TTL电平转换电路

1726750195098.jpg

2.2 CD4052介绍:

CD4052是一个差分4通道数字控制模拟开关,有A、B两个二进制控制输入端和INH输入,。幅值为4.5~20V的数字信号可控制峰峰值至20V的模拟信号。

例如,若V DD=+5V,VSS=0,VEE=-13.5V,则0~5V的数字信号可控制-13.5~4.5V的模拟信号,这些开关电路在整个VDD-VSS和VDD-VEE电源范围内具有极低的静态功耗,

与控制信号的逻辑状态无关,当INH输入端=“1”时,所有通道截止。二位二进制输入信号选通4对通道中的一通道,可连接该输入至输出。

内部引脚导通的真值表

1726750783649.jpg

应用时可以通过单片机对A/B的控制来选择输入哪一路,例如:需要从4路输入中选择第二路输入,假设使用的是Y组,那么单片机只需要分别给A和B送1和0即可选中该路,然后进行相应的处理。

在使用这个芯片时候我们需要注意一下,芯片6脚为使能引脚,当其为低电平时,主芯片才可以正常输出,这里i为了方便我就把其直接接入GND,使其一直处于工作状态,或者接入单片机的控制引脚,使信哦内部切换收到单片机的控制。

2.3 状态指示

1726751193404.png

主要代码:

void comm_mode(uint8_t mod)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	switch(mod)
	{
		case 1:
				 //USART2_TX   PA.2
			GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2; //PA.2
			GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
			GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;	//复用推挽输出
			GPIO_Init(GPIOA, &GPIO_InitStructure);  
			LED2 = 1;
			LED3 = 1;
			LED4 = 0;
			LED5 = 0;
			CD4052_A1 = 0;
			CD4052_B1 = 0;
			CD4052_A2 = 1;
			CD4052_B2 = 1;
			break;
		case 2:
				 //USART2_TX   PA.2
			GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2; //PA.2
			GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
			GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;	//复用推挽输出
			LED2 = 1;
			LED3 = 0;
			LED4 = 1;
			LED5 = 0;
			CD4052_A1 = 1;
			CD4052_B1 = 0;
			CD4052_A2 = 1;
			CD4052_B2 = 1;
			break;
		case 3:
				 //USART2_TX   PA.2
			GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2; //PA.2
			GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
			GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;	//复用推挽输出
			LED2 = 1;
			LED3 = 0;
			LED4 = 0;
			LED5 = 1;
			CD4052_A1 = 0;
			CD4052_B1 = 1;
			CD4052_A2 = 1;
			CD4052_B2 = 1;
			break;
		case 4:
				 //USART2_TX   PA.2
			GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2; //PA.2
			GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
			GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;	//复用推挽输出
			GPIO_Init(GPIOA, &GPIO_InitStructure);  
			LED2 = 0;
			LED3 = 1;
			LED4 = 1;
			LED5 = 0;
			CD4052_A1 = 0;
			CD4052_B1 = 0;
			CD4052_A2 = 0;
			CD4052_B2 = 0;
			break;
		case 5:
				 //USART2_TX   PA.2
			GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2; //PA.2
			GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
			GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;	//复用推挽输出
			GPIO_Init(GPIOA, &GPIO_InitStructure);  
			LED2 = 0;
			LED3 = 1;
			LED4 = 0;
			LED5 = 1;
			CD4052_A1 = 0;
			CD4052_B1 = 0;
			CD4052_A2 = 1;
			CD4052_B2 = 0;
			break;
		case 6:
				 //USART2_TX   PA.2
			GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2; //PA.2
			GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
			GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;	//复用推挽输出
			GPIO_Init(GPIOA, &GPIO_InitStructure);  
			LED2 = 0;
			LED3 = 0;
			LED4 = 1;
			LED5 = 1;
			CD4052_A1 = 1;
			CD4052_B1 = 0;
			CD4052_A2 = 1;
			CD4052_B2 = 0;
			break;
		}
}


//外部中断1服务程序 
void EXTI1_IRQHandler(void)
{
	
	uint8_t buffer[32] = {0};
	QelemType data = {0};
	uint8_t i;
	
	EXTI_ClearITPendingBit(EXTI_Line1); //清除LINE1上的中断标志位  
	
	if(NRF24L01_RxPacket(buffer) == 0)
	{
		for(i = 0; i < buffer[0]; i ++)
		{
			data.buffer = buffer[i + 1];
			enquene(&Qusart1Send,&data);
		}
	}
	
}
#include "24L01.h"
#include "delay.h"
#include "spi.h"
#include "QUEUE.h"
#include "usb_prop.h"
#include "usart.h"	 
#include "string.h"

//Mini STM32开发板
//NRF24L01 驱动函数 

// SPI总线速度设置 
#define SPI_SPEED_2   0
#define SPI_SPEED_4 	1
#define SPI_SPEED_8   2
#define SPI_SPEED_16  3
#define SPI_SPEED_256 4
 
	 

const unsigned char TX_ADDRESS[TX_ADR_WIDTH]={0x34,0x43,0x10,0x10,0x01}; //1发送地址
const unsigned char RX_ADDRESS[RX_ADR_WIDTH]={0x34,0x43,0x10,0x10,0x01}; //P0接收地址
const unsigned char TX1_ADDRESS[TX_ADR_WIDTH]={0x01,0xc2,0xc2,0xc2,0xc2}; //1发送地址
const unsigned char RX1_ADDRESS[RX_ADR_WIDTH]={0x01,0xc2,0xc2,0xc2,0xc2}; //P1接收地址
							    
//初始化24L01的IO口
void NRF24L01_Init(void)
	{
	GPIO_InitTypeDef GPIO_InitStructure;
		
	//RCC_APB2PeriphClockCmd(	RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB, ENABLE );	
	
		
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;          //初始化CE
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP ;   //推挽输出
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
		
	GPIO_SetBits(GPIOA,GPIO_Pin_4);	 


	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;           //初始化CS
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP ;   //推挽输出
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;
	GPIO_Init(GPIOB, &GPIO_InitStructure);
	
	Set_NRF24L01_CSN; 
	
	
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;          //初始化IRQ
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU  ;   //上拉输入
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;
	GPIO_Init(GPIOB, &GPIO_InitStructure);
	
	
	SPI1_Init();    //初始化SPI1
	
	Clr_NRF24L01_CE; 	//使能24L01
	
	Set_NRF24L01_CSN;	//SPI片选取消	
	
	Set_NRF24L01_IRQ;


	}
	
	
	
	//SPI 速度设置函数
//SpeedSet:
//SPI_SPEED_2   2分频   (SPI 36M@sys 72M)
//SPI_SPEED_8   8分频   (SPI 9M@sys 72M)
//SPI_SPEED_16  16分频  (SPI 4.5M@sys 72M)
//SPI_SPEED_256 256分频 (SPI 281.25K@sys 72M)
void SPIx_SetSpeed(u8 SpeedSet)
	{
	SPI1->CR1&=0XFFC7;//Fsck=Fcpu/256
	if(SpeedSet==SPI_SPEED_2)//二分频
		{
		SPI1->CR1|=0<<3;//Fsck=Fpclk/2=36Mhz	
		}
	else if(SpeedSet==SPI_SPEED_4)//八分频 
		{
		SPI1->CR1|=1<<3;//Fsck=Fpclk/8=9Mhz	
		}
	else if(SpeedSet==SPI_SPEED_8)//八分频 
		{
		SPI1->CR1|=2<<3;//Fsck=Fpclk/8=9Mhz	
		}
	else if(SpeedSet==SPI_SPEED_16)//十六分频
		{
		SPI1->CR1|=3<<3;//Fsck=Fpclk/16=4.5Mhz
		}
	else			 	 //256分频
		{
		SPI1->CR1|=7<<3; //Fsck=Fpclk/256=281.25Khz 低速模式
		}
	//SPI1->CR1|=1<<6; //SPI设备使能	
	/* Enable SPI1  */
	SPI_Cmd(SPI1, ENABLE); //使能SPI外设  
	}
	
	
	
	//SPIx 读写一个字节
//TxData:要写入的字节
//返回值:读取到的字节
unsigned char SPIx_ReadWriteByte(unsigned char TxData)
{		
	unsigned int retry=0;				 
	/* Loop while DR register in not emplty */
	
	while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET) //检查指定的SPI标志位设置与否:发送缓存空标志位
		{
			retry++;
			if(retry>2000)return 0;
		}			  
	/* Send byte through the SPI1 peripheral */
	SPI_I2S_SendData(SPI1, TxData); //通过外设SPIx发送一个数据
		
	retry=0;
		
	/* Wait to receive a byte */
	while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET) //检查指定的SPI标志位设置与否:接受缓存非空标志位
		{
			retry++;
			if(retry>2000)return 0;
		}	  						    
	/* Return the byte read from the SPI bus */
		
	return SPI_I2S_ReceiveData(SPI1); //返回通过SPIx最近接收的数据
		
}
	
	
	
	
	
	

//检测24L01是否存在
//返回值:0,成功;1,失败	
unsigned char NRF24L01_Check(void)
{
	unsigned char buf[5]={0XA5,0XA5,0XA5,0XA5,0XA5};
	
	unsigned char i;
	
	SPIx_SetSpeed(SPI_SPEED_4); //spi速度为9Mhz(24L01的最大SPI时钟为10Mhz)   
	
	NRF24L01_Write_Buf(NRF24L01_WRITE_REG+TX_ADDR,buf,5);//写入5个字节的地址.	
	
	NRF24L01_Read_Buf(TX_ADDR,buf,5); //读出写入的地址  
	
	for(i=0;i<5;i++)
	{
		
	  if(buf[i]!=0XA5)break;	 
		
	}
	if(i!=5)return 1;//检测24L01错误	
	
	return 0;		 //检测到24L01
	
}	
 	 
//SPI写寄存器
//reg:指定寄存器地址
//value:写入的值
unsigned char NRF24L01_Write_Reg(unsigned char reg,unsigned char value)
	{
		unsigned char status;	
		
		Clr_NRF24L01_CSN;                 //使能SPI传输
		
		status =SPIx_ReadWriteByte(reg);//发送寄存器号 
		
		SPIx_ReadWriteByte(value);      //写入寄存器的值
		
		Set_NRF24L01_CSN;                 //禁止SPI传输	 
		
		return(status);       			//返回状态值
	}

//读取SPI寄存器值
//reg:要读的寄存器
unsigned char NRF24L01_Read_Reg(unsigned char reg)
	{
		unsigned char reg_val;	
    
		Clr_NRF24L01_CSN;          //使能SPI传输		
		
		SPIx_ReadWriteByte(reg);   //发送寄存器号
		
		reg_val=SPIx_ReadWriteByte(0XFF);//读取寄存器内容
		
		Set_NRF24L01_CSN;          //禁止SPI传输		    
		
		return(reg_val);           //返回状态值
	}	

//在指定位置读出指定长度的数据
//reg:寄存器(位置)
//*pBuf:数据指针
//len:数据长度
//返回值,此次读到的状态寄存器值 
unsigned char NRF24L01_Read_Buf(unsigned char reg,unsigned char *pBuf,unsigned char len)
	{
		unsigned char status,NUM;	  
		
		Clr_NRF24L01_CSN;           //使能SPI传输
		
		status=SPIx_ReadWriteByte(reg);//发送寄存器值(位置),并读取状态值  
		
		for( NUM=0;NUM<len;NUM++)pBuf[NUM]=SPIx_ReadWriteByte(0XFF);//读出数据
		
		Set_NRF24L01_CSN;       //关闭SPI传输
		
		return status;        //返回读到的状态值
		
	}

//在指定位置写指定长度的数据
//reg:寄存器(位置)
//*pBuf:数据指针
//len:数据长度
//返回值,此次读到的状态寄存器值
unsigned char NRF24L01_Write_Buf(unsigned char reg, unsigned char *pBuf, unsigned char len)
	{
		
		unsigned char status,NUM;	 
			
		Clr_NRF24L01_CSN;          //使能SPI传输
			
		status = SPIx_ReadWriteByte(reg);//发送寄存器值(位置),并读取状态值
			
		for(NUM=0; NUM<len; NUM++)SPIx_ReadWriteByte(*pBuf++); //写入数据	
			
		Set_NRF24L01_CSN;       //关闭SPI传输
			
		return status;          //返回读到的状态值
		
	}
				
//数据写入自动应答中
void NRF24L01_Write_ACKPacket(uint8_t pipe, uint8_t *tx_pload)
{
  NRF24L01_Write_Buf(WR_ACK_PLOAD | pipe,tx_pload,TX_PLOAD_WIDTH);
//	NRF24L01_Write_Reg(NRF24L01_FLUSH_RX,0xff);//清除RX FIFO寄存器 
}

//读数据在自动应答中
void NRF24L01_Read_ACKPacket(uint8_t *tx_pload)
{
  NRF24L01_Read_Buf(NRF24L01_RD_RX_PLOAD,tx_pload,RX_PLOAD_WIDTH);//读取数据
	NRF24L01_Write_Reg(NRF24L01_FLUSH_RX,0xff);//清除RX FIFO寄存器 
}
	
//启动NRF24L01发送一次数据
//txbuf:待发送数据首地址
//返回值:发送完成状况
unsigned char NRF24L01_TxPacket(unsigned char *txbuf)
	{
	unsigned char sta;
		
	//SPIx_SetSpeed(SPI_SPEED_8);//spi速度为9Mhz(24L01的最大SPI时钟为10Mhz)   
		
	Clr_NRF24L01_CE;
		
	NRF24L01_Write_Buf(NRF24L01_WR_TX_PLOAD,txbuf,TX_PLOAD_WIDTH);//写数据到TX BUF  32个字节
		
	Set_NRF24L01_CE;//启动发送	
		
	while(NRF24L01_IRQ!=0);//等待发送完成
		
	sta=NRF24L01_Read_Reg(STATUS);  //读取状态寄存器的值	  
		
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+STATUS,sta); //清除TX_DS或MAX_RT中断标志
		
	if(sta&MAX_TX)//达到最大重发次数
		{
			
		NRF24L01_Write_Reg(NRF24L01_FLUSH_TX,0xff);//清除TX FIFO寄存器 
			
		  return MAX_TX; 
			
		}
	if(sta&TX_OK)//发送完成
		{
			NRF24L01_Read_ACKPacket(txbuf);
		  return TX_OK;
			
		}
	return 0xff;//其他原因发送失败
	}

//启动NRF24L01接收一次数据
//txbuf:待接收数据首地址
//返回值:0,接收完成;其他,错误代码
unsigned char NRF24L01_RxPacket(unsigned char *rxbuf)
	{
	unsigned char sta;	
		QelemType data = {0};
		uint8_t buffer[32]={0};
		uint16_t i = 0;
		LINE_CODING line = {0};
		static char flag = 0;
	//SPIx_SetSpeed(SPI_SPEED_8); //spi速度为9Mhz(24L01的最大SPI时钟为10Mhz)  
		
	sta=NRF24L01_Read_Reg(STATUS);  //读取状态寄存器的值    	 
		
	//NRF24L01_Write_Reg(NRF24L01_WRITE_REG+STATUS,sta|0x40); //清除RX_OK中断标志
		
	if(sta&RX_OK)//接收到数据
		{
			NRF24L01_Write_Reg(NRF24L01_WRITE_REG+STATUS,sta|0x40); //清除RX_OK中断标志
		  NRF24L01_Read_Buf(NRF24L01_RD_RX_PLOAD,rxbuf,RX_PLOAD_WIDTH);//读取数据
			
		  NRF24L01_Write_Reg(NRF24L01_FLUSH_RX,0xff);//清除RX FIFO寄存器 
			if(rxbuf[0] == 33)//串口配置指令
			{
				memcpy((void *)&line,(void *)(rxbuf+1),sizeof(line));
				uart2_config_change(line);
				NRF24L01_Write_ACKPacket(0,rxbuf);
				PBout(5) = 0;
				TIM_Cmd(TIM3,ENABLE); //使能TIM3
				flag = 1;
				return 2;//串口配置指令
			}
			if(flag == 1)
			{
				if(length(&Qusart1Rec) > 0)
				{
					for(i = 1; i < 32; i ++)
					{
						if(dequene(&Qusart1Rec,&data) == -1 )
							break;
						buffer[i] = data.buffer;
					}
					buffer[0] = i - 1;	
					PBout(5) = 0;
					TIM_Cmd(TIM3,ENABLE); //使能TIM3
				}
			}
			else
			{
				buffer[0] = 33;//上电后自动要求配置串口
			}
			NRF24L01_Write_ACKPacket(0,buffer);
		  return 0; 
		}	
		
	return 1;//没收到任何数据
		
	}	
					    
//该函数初始化NRF24L01到RX模式
//设置RX地址,写RX数据宽度,选择RF频道,波特率和LNA HCURR
//当CE变高后,即进入RX模式,并可以接收数据了		   
void RX_Mode(void)
{
	Clr_NRF24L01_CE;	  
	NRF24L01_Write_Buf(NRF24L01_WRITE_REG+RX_ADDR_P0,(unsigned char*)RX1_ADDRESS,RX_ADR_WIDTH);//写RX节点地址

	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+EN_AA,0x01);    //使能通道0的自动应答    
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+EN_RXADDR,0x01);//使能通道0的接收地址  	 
	
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+RF_CH,110);	     //设置RF通信频率		  
	
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+RX_PW_P0,RX_PLOAD_WIDTH);//选择通道0的有效数据宽度 	    
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+RF_SETUP,0x0f);//设置TX发射参数,0db增益,2Mbps,低噪声增益开启  

	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+FEATURE, 0x02);
	
	if(NRF24L01_Read_Reg(FEATURE) == 0x00 && (NRF24L01_Read_Reg(DYNPD) == 0x00))
  {
		Clr_NRF24L01_CSN;                 //使能SPI传输
		
		SPIx_ReadWriteByte(LOCK_UNLOCK);//发送寄存器号 
		
		SPIx_ReadWriteByte(0x73);      //写入寄存器的值
		
		Set_NRF24L01_CSN;                 //禁止SPI传输	 
    NRF24L01_Write_Reg(NRF24L01_WRITE_REG+FEATURE, 0x02);  // Enables payload in ack
  }
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+FEATURE, (NRF24L01_Read_Reg(FEATURE) | 0x04));
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+DYNPD, ALL_PIPES & ~0xC0); 

	
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+CONFIG, 0x0f);//配置基本工作模式的参数;PWR_UP,EN_CRC,16BIT_CRC,接收模式 
	Set_NRF24L01_CE; //CE为高,进入接收模式 
}
							 
//该函数初始化NRF24L01到TX模式
//设置TX地址,写TX数据宽度,设置RX自动应答的地址,填充TX发送数据,选择RF频道,波特率和LNA HCURR
//PWR_UP,CRC使能
//当CE变高后,即进入RX模式,并可以接收数据了		   
//CE为高大于10us,则启动发送.	 

void TX_Mode(void)
{														 
	Clr_NRF24L01_CE;	  
	
	NRF24L01_Write_Buf(NRF24L01_WRITE_REG+TX_ADDR,(unsigned char*)TX1_ADDRESS,TX_ADR_WIDTH);//写TX节点地址 
	
	NRF24L01_Write_Buf(NRF24L01_WRITE_REG+RX_ADDR_P0,(unsigned char*)RX1_ADDRESS,RX_ADR_WIDTH); //设置RX节点地址,主要为了使能ACK	  
	 
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+EN_AA,0x01);     //使能通道0的自动应答    
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+EN_RXADDR,0x01); //使能通道0的接收地址  
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+SETUP_RETR,0x1a);//设置自动重发间隔时间:500us + 86us;最大自动重发次数:10次
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+RF_CH,40);       //设置RF通道为40
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+RF_SETUP,0x0f);  //设置TX发射参数,0db增益,2Mbps,低噪声增益开启   
	
	
	
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+RX_PW_P0,RX_PLOAD_WIDTH);//选择通道0的有效数据宽度 	   
	
	
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+FEATURE, 0x02);
	
	if(NRF24L01_Read_Reg(FEATURE) == 0x00 && (NRF24L01_Read_Reg(DYNPD) == 0x00))
  {
		Clr_NRF24L01_CSN;                 //使能SPI传输
		
		SPIx_ReadWriteByte(LOCK_UNLOCK);//发送寄存器号 
		
		SPIx_ReadWriteByte(0x73);      //写入寄存器的值
		
		Set_NRF24L01_CSN;                 //禁止SPI传输	 
    NRF24L01_Write_Reg(NRF24L01_WRITE_REG+FEATURE, 0x02);  // Enables payload in ack
  }
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+FEATURE, (NRF24L01_Read_Reg(FEATURE) | 0x04));
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+DYNPD, ALL_PIPES & ~0xC0); 
	
	NRF24L01_Write_Reg(NRF24L01_WRITE_REG+CONFIG,0x0e);    //配置基本工作模式的参数;PWR_UP,EN_CRC,16BIT_CRC,发送模式,开启所有中断

	Set_NRF24L01_CE;//CE为高,10us后启动发送
	
	delay_us(10); //CE要拉高一段时间才进入发送模式
		
}		

		

实物图片:

项目开发中的心得:在开始设计项目时,仅仅时为了自己调试代码方便,后来慢慢的发现,调试代码时候,在调试两个主控之间SPI通讯还是有些难度的,而且时利用SPI方式进行无线传输,开始调试时候,对无线模块还不是很熟悉,遇到的硬件设计不合理、软件代码问题,真的是很头疼不过还是建议大家调试代码时候,遇到问题需要模块化处理,一个模块功能调试完毕后,进行一段时间的老化测试,再去调试其他的功能,对于不太熟悉的底层驱动配置,大家可以使用STM32CUBE软件去生成,开始使用的时候可能会有些不习惯,但是图形化配置还是很方便的。








关键词: STM32F103C8T6     NRF24L01              

共1条 1/1 1 跳转至

回复

匿名不能发帖!请先 [ 登陆 注册 ]