一:参照之前在论坛发布的帖子,此片文件作为无线通讯模块的接收端,完成USB与RS485、RS232、TTL之间的电平转换,从而将数据USB模块发送的数据成功的发送到模块当中。
下面和大家分享一下该从机的模块的硬件设计、软件开发及开发中遇到的问题
硬件设计部分:
2.1 自动收发485电路
由于RS485是半双工通讯方式,而这里在开发的时候由于为了兼容其他通讯电平,这里将制作了一个485自动切换发送、接收状态的电路,为了电路简单这里使用3V3供电的MAX3485(也可以使用SP3485替换),而485的后端,并未使用TVS管保护电路、匹配电阻电路,只是在A相、B相增加上下拉电阻。
这里简单分析一下,电路知识:当有数据发送时,485发送数据使能引脚,由于Q2nMOS导通,而拉高变为高电平,当没有数据发送时,进入接收状态。
这个电路已经我长时间的测试,很稳定的,大家在设计自动收发电路时,可以参考一下。
2.2 485、232、TTL电平转换电路
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对通道中的一通道,可连接该输入至输出。
内部引脚导通的真值表
应用时可以通过单片机对A/B的控制来选择输入哪一路,例如:需要从4路输入中选择第二路输入,假设使用的是Y组,那么单片机只需要分别给A和B送1和0即可选中该路,然后进行相应的处理。
在使用这个芯片时候我们需要注意一下,芯片6脚为使能引脚,当其为低电平时,主芯片才可以正常输出,这里i为了方便我就把其直接接入GND,使其一直处于工作状态,或者接入单片机的控制引脚,使信哦内部切换收到单片机的控制。
2.3 状态指示
主要代码:
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软件去生成,开始使用的时候可能会有些不习惯,但是图形化配置还是很方便的。