项目设计的前言:作为一个嵌入式的工程师,在编写代码时候,难免会遇到不同电平之间的转换。例如:RS485通讯方式,RS232,TTL信号,为了查看通讯之间数据的正确性,经常使用到不同的电平转换模块,还需要一个USB转串口工具,将数据上传到PC端,这样我们才能将查看控制器或者是负载返回的数据,为了避免调试代码起来方便,让自己的工作变得整洁,方便我们自己的工作,前端时间正好研究SPI通讯方式的NRF24L01这款无线芯片,利用业余时间做了一个无线通讯模块。在这里分享一下开发的经验。
一:原理图展示部分:

这里使用的是STM32F103的主控。实物效果仿真图如下所示:

三:软硬件设计的思路:
在本次项目中使用主控为STM32F103C8T6芯片,当时使用该块芯片是因为该款芯片具有USB功能,该芯片可以虚拟一个USB的串口进行数据交互,而且封装也比较小,在PCB布局方面比较简单。
无线模块采用的是一款SPI通讯的NRF24L01芯片,在实际的使用过程中比较稳定,可以满足数据之间的交互。
软件设计思路:利用串口1的引脚的接收发送引脚,使用消息队列的方式,可以将USB下发的的数据发送出去。
使用串口2完成与SPI之间的交互,
软件配置参数:使用STM32cube MX软件生成USB,SPI,串口底层的驱动部分,这样开发起来也比较容易;
cube软件配置图如下,主要是对该项目中使用到的引脚进行配置

部分代码分享如下:
if(bDeviceState == CONFIGURED) //判断 USB的连接状态被置位时
{
NRF24L01_Stop(); //停止NRF24L01的工作
if(GetEPRxStatus(ENDP4) == EP_RX_NAK) //判断一下USB下发的字节长度
{
if(CDC_MAXSIZE - queue_size(&Qusart2Send) >= 64 ) //由于无线模块本身的限制,每次只能传输64个字节长度,这里做分包处理
SetEPRxValid(ENDP4);
}
/*下述代码,是判断与主机模块的波特率,校验位,停止位不一致时候,主机需要重新下发串口的配置信息*/
if( linecoding_old.bitrate != linecoding.bitrate ||
linecoding_old.datatype != linecoding.datatype ||
linecoding_old.format != linecoding.format ||
linecoding_old.paritytype != linecoding.paritytype )
{
if( usart2_flag == 0 ) //下发配置信息命令
{
uart2_config_change(linecoding);
memcpy((void *)(&linecoding_old),(void *)&linecoding,sizeof(linecoding));
}
}
else if(queue_size(&Qusart2Send) > 0 && usart2_flag == 0 && GetEPRxStatus(ENDP4) == EP_RX_VALID) //该处代码为下发串口数据指令
{
QueueElementByte data_byte;
usart2_flag = 1;
if(dequene(&Qusart2Send,&data_byte) == 0 )
{
usart2_send(data_byte.buffer);
// USART_SendData(USART2,data_byte.buffer);
}
USART_ITConfig(USART2, USART_IT_TXE, ENABLE);//开启ENABLE/关闭DISABLE中断
}
if((queue_size(&Daplink_Rec) > 0)) //当下发用作无线DAP时候,需要将数据传输到从机模块的数据引脚
{
QueueElement64Bytes response;
dequene(&Daplink_Rec,&data);
if(data.buffer[0] == 0 && data.buffer[1] == 2)
{
quene_clean(&Daplink_Send);
}
response.len = DAP_ExecuteCommand(data.buffer,response.buffer);
enquene(&Daplink_Send,&response);
SetEPRxValid(ENDP6);
}
else
{
if(GetEPRxStatus(ENDP6) == EP_RX_NAK)
{
SetEPRxValid(ENDP6);
}
}
if(GetEPTxStatus(ENDP6) == EP_TX_NAK && GetEPTxStatus(ENDP4) == EP_TX_NAK && GetEPTxStatus(ENDP0) == EP_TX_STALL)
{
if (dequene(&Daplink_Send,&data) == 0)
{
UserToPMABufferCopy(data.buffer, ENDP6_TXADDR, data.len);
SetEPTxCount(ENDP6, data.len);
SetEPTxValid(ENDP6);
}
else
{
// if(usart2Rec_flag == 0 || queue_size(&Qusart2Rec) >= 64)
if(usart2Rec_flag == 0)
{
QueueElementByte data1;
if(queue_size(&Qusart2Rec) > 0)
{
for(i = 0; i < 64; i ++)
{
if(dequene(&Qusart2Rec,&data1) == -1 )
break;
buffer[i] = data1.buffer;
}
UserToPMABufferCopy(buffer, ENDP4_TXADDR, i);
SetEPTxCount(ENDP4, i);
SetEPTxValid(ENDP4);
}
}
}
}
}
else
{
NRF24L01_Start();
if (dequene(&Daplink_Rec,&data) == 0)
{
QueueElement64Bytes data1 = {0};
data1.len = DAP_ExecuteCommand(data.buffer, data1.buffer) & 0xFFFF;
enquene(&Daplink_Send,&data1);
}
if(queue_size(&Qusart2Send) > 0 && usart2_flag == 0)
{
QueueElementByte data_byte;
usart2_flag = 1;
if(dequene(&Qusart2Send,&data_byte) == 0 )
{
usart2_send(data_byte.buffer);
}
USART_ITConfig(USART2, USART_IT_TXE, ENABLE);//开启ENABLE/关闭DISABLE中断
}
}
}//外部中断1服务程序
void EXTI4_IRQHandler(void)
{
uint8_t buffer[32] = {0};
QueueElementByte data = {0};
uint8_t i;
LINE_CODING line = {0};
static char flag = 0;
if(NRF24L01_RxPacket(buffer) == 0)
{
if(nrf2dap_Data(buffer) == -1)
{
if(buffer[0] == 33)//串口配置指令
{
memcpy((void *)&line,(void *)(buffer+1),sizeof(line));
uart2_config_change(line);
NRF24L01_Write_ACKPacket(0,buffer,sizeof(line) + 1);
LED1 = 0;
usart_led = 1;
flag = 1;
return ;
}
else if(buffer[0] < 32 && buffer[0] != 0)
{
for(i = 0; i < buffer[0]; i ++) //从机模块将无线接收到的数据,通过串口2 发出去,并作出提示
{
data.buffer = buffer[i + 1];
enquene(&Qusart2Send,&data);
}
LED1 = 0;
usart_led = 1;
}
}
if(dap2nrf_Data() == -1)
{
if(flag == 1)
{
buffer[0] = 0;
if(queue_size(&Qusart2Rec) > 0)
{
for(i = 1; i < 32; i ++)
{
if(dequene(&Qusart2Rec,&data) == -1 )
break;
buffer[i] = data.buffer;
}
buffer[0] = i - 1;
LED1 = 0;
usart_led = 1;
}
NRF24L01_Write_ACKPacket(0,buffer,buffer[0] + 1);
}
else
{
buffer[0] = 33;//上电后自动要求配置串口
NRF24L01_Write_ACKPacket(0,buffer,1);
}
}
}
EXTI_ClearITPendingBit(EXTI_Line4); //清除LINE4上的中断标志位
}int dap2nrf_Data(void)
{
static uint8_t state = 0,send_len = 0,send_maxlen = 29;
static QueueElement64Bytes data={0};
uint8_t buffer[32];
if(state == 0)
{
if((queue_size(&Daplink_Send) > 0))
{
dequene(&Daplink_Send,&data);
state ++;
}
else
{
return -1;
}
}
if(state == 1)
{
if((int)data.len - send_len >= send_maxlen)
{
buffer[0] = 34;
buffer[1] = id;
buffer[2] = send_maxlen;
memcpy(buffer+3,data.buffer + send_len,buffer[2]);
NRF24L01_Write_ACKPacket(0,buffer,32);;
send_len += buffer[2];
}
else
{
buffer[0] = 34;
buffer[1] = id;
buffer[2] = (int)data.len - send_len;
memcpy(buffer+3,data.buffer + send_len,buffer[2]);
NRF24L01_Write_ACKPacket(0,buffer,buffer[2]+3);
send_len = 0;
id ++;
state = 0;
}
}
return 0;
}
int nrf2dap_Data(uint8_t *buf)
{
static uint8_t state = 0,id_last = 0,rec_len = 0,rec_maxlen = 29;
static QueueElement64Bytes data={0};
if(buf[0] != 34)
return -1;
if(buf[1] != id_last)//此条件成立即数据传输出错
{
id_last = buf[1];
quene_clean(&Daplink_Rec);
quene_clean(&Daplink_Send);
rec_len = 0;
id ++;
}
if(buf[2] == rec_maxlen)
{
memcpy(data.buffer + rec_len,buf+3,buf[2]);
rec_len += buf[2];
}
else
{
memcpy(data.buffer + rec_len,buf+3,buf[2]);
rec_len += buf[2];
data.len = rec_len;
enquene(&Daplink_Rec,&data);
rec_len = 0;
id_last ++;
}
return 0;
}在实际使用中发现,硬件功耗大概在130ma左右,在数据交互的时候功耗会增加至170ma左右,不过并不影响模块的使用。
使用该模块的最大好处就是可以让硬件工程师的桌面更加的整洁。
硬件设计的时候,设计注意事项:
使用PS7516用作升压芯片,将电池电压3.7V升压至5V,为外设可以提供5v的电压输出。
MP2155为降压型LDO,具有可关断功能,当不使用时候,可以使整个系统处于低功耗状态,从而延长电池寿命。
PW4054为锂电池充电芯片,可使用type-c为电池充电,充电电流由R26取样电阻设定。
同时PL5356A芯片为锂电池电压检测芯片,可以显示当前的电池电量,后期将电池电压分压至单片机的ADC内,从而精准的检测电池电压。
在软件开发时候,有以下的注意事项:
抗干扰能力:NRF24L01工作在2.4GHz频段,该频段易受到WiFi、蓝牙等设备的干扰。在布局布线时要尽量减少干扰源的影响,并合理设置通信参数以提高抗干扰能力。
功耗管理:在不需要通信时及时将NRF24L01置于待机模式或掉电模式以降低功耗。
距离与障碍物:无线通信距离受环境因素影响较大,障碍物会严重影响通信质量。在实际应用中要考虑通信距离和障碍物的影响并采取相应的措施。
NRF24L01的开发过程需要仔细规划硬件连接、合理配置软件参数、掌握调试技巧并关注实际应用中的注意事项。通过不断的实践和总结可以逐渐提高开发效率和应用效果。
实物效果如下所示:

测试效果如下所示:

原理图如下所示:

我要赚赏金
