美文网首页巨人的工具工作生活技术篇
单片机+北斗模块实现定位

单片机+北斗模块实现定位

作者: 氢立方 | 来源:发表于2019-06-30 18:46 被阅读310次

    本文原创,转载请注明出处。小编可能以后不会在简书上更新了,因为这里的markdown 编辑器太不好用了,没有CSDN 上面的好用,也希望粉丝们多多关注本人CSDN(一样的昵称呦)。点击友方蓝字即可浏览小编CSDN内容 csdn

    导航是继移动通信之后发展最快的信息产业之一,只要是设计到位置、速度、时间信息的领域斗鱼卫星导航技术相关。中国北斗卫星导航系统是我国自主研制的全球导航系统,是继美国定位系统(GPS)、俄罗斯格洛纳斯卫星导航系统之后的第三个成熟的卫星导航系统。随着北斗导航系统的不断完善,基于北斗导航系统的定位也应用的越来越广泛。小编采用的STC12C5A32S2单片机结合卫星接收模块UM220-III设计的北斗导航系统接收机,UM220-III是三模接收模块,能够同时接受GPS、北斗的信号。这次的设计主要完成了导航信息接收机的基本设计,实现了获取实时的位置经纬度、标准时间等相关信息的显示功能。

    1 原理介绍

    采用以单片机为核心,读取北斗导航系统模块的标准数据,并在 LCD 屏幕上显示当前的经纬度信息。具体的系统方案图如下:


    方案设计.PNG

    1.1北斗 UM220-III 模块简介

    (1)接口电路
    北斗模块芯片电路由北斗模块UM220-III N 和其附加电子器件组成。其中北斗模块外接4组排针,在芯片与电源之间串联电感的作用是起差模滤波作用,防止电流突变对芯片产生损坏,并联电感的作用是提高芯片运行的稳定性,防止产生噪声。
    模块输入端口(UM220-III N 包括:RXD、GPIO、SDA ,SCL 等 如下图)为防止输入端不定态对模块造成影响,模块内置上拉电阻至VCC,因此在模块未加电时,如果上述端口有数据输入,会在模块VCC上形成串电,又可能造成模块上电失败。


    内置电路图.PNG

    case1:设计中使用 nRESET功能
    在模块上电后,将nRESET拉低5 ms以上,即可确保模块正常启动。
    case2:设计中未使用nRESET
    在模块上电之前,保证模块已连接输入端口为高阻态或低电平,以避免串电。使用串口1的典型用户,需要吧RXD1设置为高阻态或低电平,未使用的其他PIN悬空。程序流程图如下:


    流程图.PNG
    (2)LCD 液晶显示器
    LCD液晶显示器数据由单片机p0口进行控制,p2口进行指令控制操作。LCD1602是一种专门用来显示字母,数字,符号等的点阵型液晶模块。1602:显示的内容主要是16*2,即可以显示两行,每行16个字符液晶显示模块。相应的管教功能,百度上都是可以查阅的,所以小编这里就不赘述了。
    (3)UM220-III 通信协议简介

    在Unicore 协议中,输入和输出的语句被称为消息。每条消息均为ASCII 字符组成的字符串。
    消息的基本格式为:GNAME,data1,data2,data3,……[*cc]\r\n 所有的消息都以 ''(0x24)开始,后面紧跟着的就是消息名。之后的跟的就是不定数目的参数和数据。消息名与数据之间均以逗号隔开(0x2c)进行分割。最后一个参数是可选的校验和,以 '*'(0x2A)与前面的数据分割最后,输入的消息以 ' \r\n' 结束。每条消息的总长度不超过256个字节,消息名和参数,校验和中的字母不区分大小写。
    某些输入命令的某些参数可以省略(在命令描述中被标记为可选)。这些参数可以为空,即在两个逗号之间没有任何字符。

    设定串口配置.PNG

    2 调试方法

    由1.1的原理简介可知,此次课题实现主要由 5 部分组成:系统初始化、设定显示模式、读取预显示内容、送扫描脉冲、送显示数据。
    这里对单片机与模块的连接做简要说明。
    UM220/um220-3-n 上带有两组 TTL 电平(2.85V),一组标准电平 RS232电平。当单片机的RS232电平接口接到UM220模块上的RS232上,正常通信。UM220模块上的RS232接口是DB9 母头,可以使用公头的连接线与RS230的接口相连,注意的是通信线需要交叉连接,就像TTL电平中的TXD - - RXD,RXD--TXD 一样,RS232电平通信中也是有2根通信电缆,一个是发送端(PCXD),一个是接收端(PCRXD)。若板子上的RS232 的接口是DB9 母头,那么 2 3针就是 PCRXD 和 PCTXD .第五针就是GND,若没有串口线来连接UM220模块的话,可以考虑在DB9下面的2 3 5 针 焊接出3跟线跟单片机的RS232连接。
    若单片机是3.3V时,可以将单片机的TTL 接口连接到模块上的TTL接口。连接好后就可以编写程序了。
    注意,我们使用的 52 单片机的驱动程序,使用的是12mhz的晶振,波特率为9600.

    3 程序编写

    程序主要是由5部分组成:系统初始化,设定显示模式。读取预显示内容,送扫描脉冲,送显示数据。小编只放主程序部分好了:

    3.1 定义端口及变量
    #include <REG52.H>
    #include <stdio.h>
    #include <intrins.h>
    #include <lcd1602.h>
    #include <uart.h>
    #include <delay.h>
    #include "string.h"
    #include <stdlib.h>
    
    unsigned char  flag_rec=0;    //接收数据标志
    unsigned char num_rec=0;      // 计数标志      
    //char code TIME_AREA= 8;       //时区,我们不需要它
    unsigned char flag_data;    //数据缓冲器
    //only displaty cmd $GPGGA information
    unsigned char JD[16];       //longitude
    unsigned char JD_a;     //经度方向
    unsigned char WD[15];       //latitude
    unsigned char WD_a;     // 纬度方向
    unsigned char date[6];      //data
    unsigned char time[6];      //data
    unsigned char time1[6];     //data
    unsigned char speed[5]={'0','0','0','0','0'};       // 速度
    unsigned char high[6];      // 高度
    unsigned char angle[5];     //方位角
    unsigned char use_sat[2];   // 卫星计数器
    unsigned char total_sat[2]; //卫星总数
    unsigned char lock;         //位置状态
    
    //date handing variable
    unsigned char seg_count;    // 逗号计数器
    unsigned char dot_count;    //小数点计数器
    unsigned char byte_count;   // 位计数器
    unsigned char cmd_number;   // 命令模式
    unsigned char mode;         
    unsigned char buf_full;     
    unsigned char cmd[5];       // 存储命令模式
    
    //serial disconnect timer
    unsigned  long int tt=0;
    
    主函数 系统初始化
    //main
    void main () 
    {   
      int jd_second,wd_second;  // 中间变量
        init_uart();         //初始化序列号
      lcd_init() ;       // 初始化 lcd1602
      delay(200);
      LCD_Write_String(0,0,"Please Waiting...");   //    "Please Waiting" when it is boot up
      delay(200);
      delay(200);
      delay(200);
      delay(200);
      delay(200);
      delay(200);
      delay(200);
      delay(200);
      delay(200);
      delay(200);                             // 延迟显示
      write_com(0x01);          // 清屏
    
    

    设置延时函数,以形成视觉暂留

    while(1)
    {
    tt++;
    if(tt>10000)
      {
        tt=10000;
        write_com(0x01);
        LCD_Write_String(3,0,"No Data!");
        LCD_Write_String(3,1,"No Data!");
        delay(200);
        delay(200);
        delay(200);
        delay(200);
        delay(200);
       }
    

    ···

    读取预显示内容,设置显示模式,转16进制为10进制
     if(flag_rec==1)             // 获取gps数据
      {
        flag_rec=0;               // 清除标志符
        if (lock==1)              //  获取位置信息
            {  
          //
          LCD_Write_String(0,0,"JD  :");    // 显示经度
          LCD_Write_String(6,0,JD);           // 显示数据
          LCD_Write_String(9,0,".");           // 进制转换
          LCD_Write_String(10,0,JD+3);           
          jd_second=60*atof((char *)(JD+5));     
          LCD_Write_Char(13,0,jd_second/10+'0');  // 将上一步转换得到浮点数据打印在lcd
          LCD_Write_Char(14,0,jd_second%10+'0');  // 将上面得到的数据分为两部分,分别打印在LCD 上
          LCD_Write_Char(15,0,' ');                //填充空间
    
          delay(200);                         // 保护lcd
          LCD_Write_String(0,1,"WD  :");     // 显示下一行
          LCD_Write_String(6,1,WD);
          LCD_Write_String(8,1,"."); 
          LCD_Write_String(9,1,WD+2);           // 小数点
          wd_second=60*atof((char *)(WD+4));// 将字符串转换成浮点数
          LCD_Write_Char(12,1,wd_second/10+'0');
          LCD_Write_Char(13,1,wd_second%10+'0');
          LCD_Write_String(14,1,"  ");
          delay(200);
       }
      }
     }
        }
    

    串口中断函数及模式判断

    判断的主要依据就是接收端接受消息与预结果匹配,通过设置数组[i,j]和if函数进行判断匹配(发送报文的消息内容见1.3的UM220的通信协议详解)

    
           //serial interruupt service function
           void ser_int (void) interrupt 4
          {
        unsigned char tmp;
        if(RI)
        {
        tt=0;
            RI=0;
            tmp=SBUF;            // 从缓冲区接收数据
            switch(tmp)   //if $GPGGA,$GNGSW,$GNRMC,get data then processing it
            {
          //date start with $
                case '$':
                    cmd_number=0;       // 清除命令模式
                    mode=1;             // 选项命令接收模式
                    byte_count=0;       //清除位计数器
                    flag_data=1;     // 设置数据标志
                    flag_rec=1;     // 设置数据接收标志
                break;
    
                case ',':         //Eg:$GNRMC,134645.000,A,2603.964436,N,11912.410232,E,0.000,15.744,030718,,E,A*0B
                    seg_count++;        // 计数器增加
                    byte_count=0;
                    break;
    
                case '*':
                    switch(cmd_number)
                    {
                        case 1:
                            buf_full|=0x01;   //00000001
                            break;
                        case 2:
                            buf_full|=0x02;  //00000010
                            break;
                        case 3:
                            buf_full|=0x04;  //00000100
                            break;
                    }
    
                    mode=0;         //clear mode
                    break;
                default:
    
                       // 接受数据
                    if(mode==1) 
                    {
                        cmd[byte_count]=tmp;    // 获取数据和存储缓冲区   
                            if(byte_count>=4)                                                   
                        {           
                            if(cmd[0]=='G')           // 第一个字符
                            {
                                if(cmd[1]=='N')
                                {
                                    if(cmd[2]=='G')
                                    {
                                        if(cmd[3]=='G')
                                        {
                                            if(cmd[4]=='A')//判断$GNGGA
                                            {
                                                cmd_number=1;      //数据类型
                                                mode=2;            //接收日期
                                                seg_count=0;       //逗号计数器置0
                                                byte_count=0;      //位计数器清除
                                            }
                                        }
                                        else if(cmd[3]=='S')       //命令模式$GNGSV
                                        {
                                            if(cmd[4]=='V')
                                            {
                                                cmd_number=2;
                                                mode=2;                //获取数据
                                                seg_count=0;
                                                byte_count=0;
                                            }
                                        }
                                    }
                                    else if(cmd[2]=='R')   //命令模式 $GNRMC
                                    {
                                        if(cmd[3]=='M')
                                        {
                                            if(cmd[4]=='C')
                                            {
                                                cmd_number=3;
                                                mode=2;         //存储数据
                                                seg_count=0;
                                                byte_count=0;
                                            }
                                        }
                                    }
                                }
                            }
                        }
                    }
                                   //日期处理
                    else if(mode==2)
                    {
                        
                        switch (cmd_number)  //if receive data
                        {
                            case 1:             //get and store data,$GPGGA,[],[],[],[],[],[],[],[],[].....
                                switch(seg_count)   //  comma 计数器
                                {
                                    case 2:     // 2rd逗号后的纬度
                                        if(byte_count<9)
                                        {
                                            WD[byte_count]=tmp;   //获取纬度
                                        }
                                        break;
                                    case 3:     //纬度方向
                                        if(byte_count<1)
                                        {
                                            WD_a=tmp;
                                        }
                                        break;
                                    case 4:     //经度
                                        if(byte_count<10)
                                        {
                                            JD[byte_count]=tmp; //存储
                                        }
                                        break;
                                    case 5:     //经度方向
                                        if(byte_count<1)
                                        {
                                            JD_a=tmp;
                                        }
                                        break;
                                    case 6:     //location
                                        if(byte_count<1)
                                        {
                                            lock=tmp;
                                        }
                                        break;
                                    case 7:     
                                        if(byte_count<2)
                                        {
                                            use_sat[byte_count]=tmp;
                                        }
                                        break;
                                    case 9:     // 高度
                                        if(byte_count<6)
                                        {
                                            high[byte_count]=tmp;
                                        }
                                        break;
                                }
                                break;
    
    
                            case 2: //命令模式  $GPGSV
                                switch(seg_count)
                                {
                                    case 3:     // 卫星总数
                                        if(byte_count<2)
                                        {
                                            total_sat[byte_count]=tmp;
                                        }
                                        break;
                                }
                                break;
    
    
                                                 //命令模式3:无SUE
                            case 3:             //$GPRMC
                                switch(seg_count)
                                {
                                    case 1:     //time
                                        if(byte_count<6)
                                        {               
                                            time[byte_count]=tmp;   
                                        }
                                        break;
                                    case 2:     // 位置           
                                        if(byte_count<1)
                                        {
                                          if (tmp=='V') {lock=0;}
                                          else
                                          {
                                            lock=1;
                                           }
                                        }
                                        break;
                                    case 3:     //lititude          
                                        if(byte_count<9)
                                        {
                                            WD[byte_count]=tmp;//我们只需要一次
                                        }
                                        break;
                                    case 4:                         
                                        if(byte_count<1)
                                        {
                                            WD_a=tmp;
                                        }
                                        break;
                                    case 5:     //          
                                        if(byte_count<10)
                                        {
                                            JD[byte_count]=tmp;  //do not get again
                                        }
                                        break;
                                    case 6:     // 直线方向 
                                        if(byte_count<1)
                                        {
                                            JD_a=tmp;
                                        }
                                        break;
                                    case 7:     // 速度处理     
                                        if(byte_count<5)
                                        {
                                            speed[byte_count]=tmp;
                                        }
                                        break;
                                    case 8:     // 方向角              
                                        if(byte_count<5)
                                        {
                                            angle[byte_count]=tmp;
                                        }
                                        break;
                                    case 9:     //other         
                                        if(byte_count<6)
                                        {
                                            date[byte_count]=tmp;
                                        }
                                        break;
    
    >                           }
                                break;
                        }
                    }
                    byte_count++;       // 位计数器++
                    break;
            }
        }
    

    4 结果显示

    1.PNG
    IMG_20190630_091924.jpg
    IMG_20190630_103508.jpg

    相关文章

      网友评论

        本文标题:单片机+北斗模块实现定位

        本文链接:https://www.haomeiwen.com/subject/amybcctx.html