STM32
直播中

陈伟

7年用户 1484经验值
私信 关注
[问答]

如何利用STM32F103串口通信获取GY-53红外测距模块数据?

如何利用STM32F103串口通信获取GY-53红外测距模块数据?

回帖(1)

王艳

2021-12-8 11:06:52
STM32F103的串口功能强大,主要用于不同模块的通信,在使用GY-53模块时, 我选择使用STN32F1的串口用于读GY-53发送的数据。

GY-53 是一款低成本数字红外测距传感器模块,有两种方式读取数据,即串口 UART(TTL 电平)+PWM(1 线)或者芯片 IIC 模式,串口的波特率有 9600bps 与115200bps,可配置,有连续,询问输出两种方式,可掉电保存设置。 文章底部有使用手册的文档
思路导图






STMF103串口配置


/***
  * @brief  
  * @note
  * @param
  * @retval
  */
void laser_uart1_Configuration(u32 bound){
  //GPIO端口设置
  GPIO_InitTypeDef GPIO_InitStructure;
        USART_InitTypeDef USART_InitStructure;
        NVIC_InitTypeDef NVIC_InitStructure;
         
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE);        //使能USART1,GPIOA时钟
  
        USART_DeInit(USART1);
        //USART1_TX   GPIOA.9
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;        //复用推挽输出
  GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.9
   
  //USART1_RX          GPIOA.10初始化
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;//PA10
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入
  GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.10  


  //USART1 NVIC 配置
  NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//抢占优先级3
        NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;                //子优先级3
        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                        //IRQ通道使能
        NVIC_Init(&NVIC_InitStructure);        //根据指定的参数初始化VIC寄存器
  
   //USART 初始化设置


        USART_InitStructure.USART_BaudRate = bound;//串口波特率
        USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式
        USART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位
        USART_InitStructure.USART_Parity = USART_Parity_No;//无奇偶校验位
        USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件数据流控制
        USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;        //收发模式


  USART_Init(USART1, &USART_InitStructure); //初始化串口1
  USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启串口接受中断
  USART_Cmd(USART1, ENABLE);                    //使能串口1


}




void USART1_IRQHandler(void)           //串口1中断服务程序
        {

        static u8 Res_1;
#if SYSTEM_SUPPORT_OS                 //如果SYSTEM_SUPPORT_OS为真,则需要支持OS.
        OSIntEnter();   
#endif
        if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  //接收中断(接收到的数据必须是0x0d 0x0a结尾)


                {
                Res_1 =USART_ReceiveData(USART1);        //读取接收到的数据
               


                if((USART1_RX_STA&0x8000)==0)//接收未完成
                        {
                        if(USART1_RX_STA&0x4000)//接收到了0x0d
                                {
                                if(Res_1!=0x5a)USART1_RX_STA=0;//接收错误,重新开始
                                else USART1_RX_STA|=0x8000;        //接收完成了
                                }
                        else //还没收到0X0D
                                {       


                                        USART1_RX_BUF[USART1_RX_STA&0X3FFF]=Res_1 ;
                                        USART1_RX_STA++;
                                        if(USART1_RX_STA>(7))
                    {
                        USART_data_analyse(USART1_RX_BUF,8,First);
                        USART1_RX_STA=0;//接收数据错误,重新开始接收          
                    }


                                }            
                        }
        }
    }
数据处理函数


void USART_data_analyse(u8 *buf,u8 num,u8 state)  
{
        u8 i,sum=0;
       
        for(i=0; i<(num-1); i++)
        {
                sum += *(buf+i);
        }
       
        if(!(*(buf)==0X5A && *(buf+1)==0X5A && *(buf+2)==0X15))
        {
                return;
        }
       
        if(sum == (*(buf+7)))
        {                       
                switch(state)
        {
            case First:
            {
                Laser.Original_Dist[First] = *(buf+4)<<8 | *(buf+5);        /* 原始距离 */
                Laser.Measure_Mode[First] = *(buf+6);
                               
//                Laser.Relative_Dist[First] = Laser.Original_Dist[Left]*my_cos(my_abs(MOVE.Goal_Z-IMU.Ang.Z)/57.2957f);        /* 相对距离 */
            }break;
                        case Second:
            {
                Laser.Original_Dist[Second] = *(buf+4)<<8 | *(buf+5);        /* 原始距离 */
                Laser.Measure_Mode[Second] = *(buf+6);
                               
//                Laser.Relative_Dist[Second] = Laser.Original_Dist[Left]*my_cos(my_abs(MOVE.Goal_Z-IMU.Ang.Z)/57.2957f);        /* 相对距离 */
            }break;
                        case Third:
            {
                Laser.Original_Dist[Third] = *(buf+4)<<8 | *(buf+5);        /* 原始距离 */
                Laser.Measure_Mode[Third] = *(buf+6);
                               
//                Laser.Relative_Dist[Third] = Laser.Original_Dist[Left]*my_cos(my_abs(MOVE.Goal_Z-IMU.Ang.Z)/57.2957f);        /* 相对距离 */
            }break;
                        case Fourth:
            {
                Laser.Original_Dist[Fourth] = *(buf+4)<<8 | *(buf+5);        /* 原始距离 */
                Laser.Measure_Mode[Fourth] = *(buf+6);
                               
//                Laser.Relative_Dist[Fourth] = Laser.Original_Dist[Left]*my_cos(my_abs(MOVE.Goal_Z-IMU.Ang.Z)/57.2957f);        /* 相对距离 */
            }break;
        }
               
        }
}


该函数可用于4个串口的数据处理,使用时:
*buf 为缓冲区数组; num为数据字节;u8 state为不同串口
例如:


USART_data_analyse(USART1_RX_BUF,8,First);
1
处理后的数据
        float Original_Dist[direction];/* 原始距离 */
        float Relative_Dist[direction];/* 相对距离 */
1
2
仅为个人拙见,如有不当请指正,谢谢!!
若有疑问欢迎私信提问


附上GY-53使用手册 + 程序源码
链接:https://pan.baidu.com/s/1wqLGIj7PNv4QrRikggML3A
提取码:qdxx
复制这段内容后打开百度网盘手机App,操作更方便哦–来自百度网盘超级会员V1的分享


举报

更多回帖

发帖
×
20
完善资料,
赚取积分