GPS模块(GPS-NEO-6M)

2023-05-16

ATK-NEO-6M GPS 模块简介

    ATK-NEO-6M-V23 模块,是 ALIENTEK 生产的一款高性能 GPS 
    模块,模块核心采用 UBLOX公司的 NEO-6M 模组,具有 50 个
    通道,追踪灵敏度高达-161dBm,测量输出频率最高可达5Hz。
    ATK-NEO-6M-V23 模块具有以下特点:

  1. 模块采用 U-BLOX NEO-6M 模组,体积小巧,性能优异。
  2. 模块自带陶瓷天线及 MAXIM 公司 20.5dB 高增益 LNA 芯片,搜星能力强。
  3. 模块可通过串口进行各种参数设置,并可保存在 EEPROM,使用方便。
  4. 模块自带 IPX 接口,可以连接各种有源天线,适应能力强。
  5. 模块兼容 3.3V/5V 电平,方便连接各种单片机系统。
  6. 模块自带可充电后备电池,可以掉电保持星历数据
  7. 注 :在主电源断开后,后备电池可以维持半小时左右的 GPS 星历数据的保存,以支持温启动或热启动,从而实现快速定位。

在这里插入图片描述

电路图

在这里插入图片描述

数据显示

在这里插入图片描述

代码实现

gps.c

#include "gps.h"
#include "led.h"
#include "delay.h"
#include "usart3.h"
#include "stdio.h"
#include "stdarg.h"
#include "string.h"
#include "math.h"

//从 buf 里面得到第 cx 个逗号所在的位置
//返回值:0~0XFE,代表逗号所在位置的偏移.
// 0XFF,代表不存在第 cx 个逗号
u8 NMEA_Comma_Pos(u8 *buf,u8 cx)
{
	u8 *p=buf;
	while(cx)
	{
		if(*buf=='*'||*buf<' '||*buf>'z')return 0XFF//遇到'*'或者非法字符,则不存在第 cx 个逗号
		if(*buf==',')cx--;
		buf++;
	}
	return buf-p;
}

//m^n 函数
//返回值:m^n 次方.
u32 NMEA_Pow(u8 m,u8 n)
{
	u32 result=1;
	while(n--)result*=m;
	return result;
}

//str 转换为数字,以','或者'*'结束
//buf:数字存储区
//dx:小数点位数,返回给调用函数
//返回值:转换后的数值
int NMEA_Str2num(u8 *buf,u8*dx)
{
	u8 *p=buf;	
	u32 ires=0,fres=0;
	u8 ilen=0,flen=0,i;
	u8 mask=0;
	int res;
	while(1) //得到整数和小数的长度
	{
		if(*p=='-'){mask|=0X02;p++;}//是负数
		if(*p==','||(*p=='*'))break;//遇到结束了
		if(*p=='.'){mask|=0X01;p++;}//遇到小数点了
		else if(*p>'9'||(*p<'0')) //有非法字符
		{
			ilen=0;
			flen=0;
			break;
		}
		if(mask&0X01)flen++;
		else ilen++;
		p++;
	}
	if(mask&0X02)buf++; //去掉负号
	for(i=0;i<ilen;i++) //得到整数部分数据
	{
		ires+=NMEA_Pow(10,ilen-1-i)*(buf[i]-'0');
	}
	if(flen>5)flen=5; //最多取 5 位小数
	*dx=flen; //小数点位数
	for(i=0;i<flen;i++) //得到小数部分数据
	{
		fres+=NMEA_Pow(10,flen-1-i)*(buf[ilen+1+i]-'0');
	}
	res=ires*NMEA_Pow(10,flen)+fres;
	if(mask&0X02)res=-res;
	return res;
}

//分析 GPGSV 信息
//gpsx:nmea 信息结构体
//buf:接收到的 GPS 数据缓冲区首地址
void NMEA_GPGSV_Analysis(nmea_msg *gpsx,u8 *buf)
{
	u8 *p,*p1,dx;
	u8 len,i,j,slx=0;
	u8 posx;
	p=buf;
	p1=(u8*)strstr((const char *)p,"$GPGSV");
	len=p1[7]-'0'; //得到 GPGSV 的条数
	posx=NMEA_Comma_Pos(p1,3); //得到可见卫星总数
	if(posx!=0XFF)gpsx->svnum=NMEA_Str2num(p1+posx,&dx);
	for(i=0;i<len;i++)
	{
		p1=(u8*)strstr((const char *)p,"$GPGSV");
		for(j=0;j<4;j++)
		{
			posx=NMEA_Comma_Pos(p1,4+j*4);
			if(posx!=0XFF)gpsx->slmsg[slx].num=NMEA_Str2num(p1+posx,&dx)
			//得到卫星编号
			else break;
			posx=NMEA_Comma_Pos(p1,5+j*4);
			if(posx!=0XFF)gpsx->slmsg[slx].eledeg=NMEA_Str2num(p1+posx,&dx);
			//得到卫星仰角
			else break;
			posx=NMEA_Comma_Pos(p1,6+j*4);
			if(posx!=0XFF)gpsx->slmsg[slx].azideg=NMEA_Str2num(p1+posx,&dx);
			//得到卫星方位角
			else break;
			posx=NMEA_Comma_Pos(p1,7+j*4);
			if(posx!=0XFF)gpsx->slmsg[slx].sn=NMEA_Str2num(p1+posx,&dx);
			//得到卫星信噪比
			else break;
			slx++;
		}
		p=p1+1;//切换到下一个 GPGSV 信息
	}
}

//分析 GPGSA 信息
//gpsx:nmea 信息结构体
//buf:接收到的 GPS 数据缓冲区首地址
void NMEA_GPGSA_Analysis(nmea_msg *gpsx,u8 *buf)
{
	u8 *p1,dx;
	u8 posx;
	u8 i;
	p1=(u8*)strstr((const char *)buf,"$GPGSA");
	posx=NMEA_Comma_Pos(p1,2); //得到定位类型
	if(posx!=0XFF)gpsx->fixmode=NMEA_Str2num(p1+posx,&dx);
	for(i=0;i<12;i++) //得到定位卫星编号
	{
		posx=NMEA_Comma_Pos(p1,3+i);
		if(posx!=0XFF)gpsx->possl[i]=NMEA_Str2num(p1+posx,&dx);
		else break;
	}
	posx=NMEA_Comma_Pos(p1,15); //得到 PDOP 位置精度因子
	if(posx!=0XFF)gpsx->pdop=NMEA_Str2num(p1+posx,&dx);
	posx=NMEA_Comma_Pos(p1,16); //得到 HDOP 位置精度因子
	if(posx!=0XFF)gpsx->hdop=NMEA_Str2num(p1+posx,&dx);
	posx=NMEA_Comma_Pos(p1,17); //得到 VDOP 位置精度因子
	if(posx!=0XFF)gpsx->vdop=NMEA_Str2num(p1+posx,&dx);
}
	
//分析 GPRMC 信息
//gpsx:nmea 信息结构体
//buf:接收到的 GPS 数据缓冲区首地址
void NMEA_GPRMC_Analysis(nmea_msg *gpsx,u8 *buf)
{
	u8 *p1,dx;
	u8 posx;
	u32 temp;
	float rs;
	p1=(u8*)strstr((const char *)buf,"GPRMC");
	//"$GPRMC",经常有&和 GPRMC 分开的情况,故只判断 GPRMC.
	posx=NMEA_Comma_Pos(p1,1); //得到 UTC 时间
	if(posx!=0XFF)
	{
		temp=NMEA_Str2num(p1+posx,&dx)/NMEA_Pow(10,dx);//得到 UTC 时间
		gpsx->utc.hour=temp/10000;
		gpsx->utc.min=(temp/100)%100;
		gpsx->utc.sec=temp%100;
	}
	posx=NMEA_Comma_Pos(p1,3); //得到纬度
	if(posx!=0XFF)
	{
		temp=NMEA_Str2num(p1+posx,&dx);
		gpsx->latitude=temp/NMEA_Pow(10,dx+2); //得到°
		rs=temp%NMEA_Pow(10,dx+2); //得到'
		gpsx->latitude=gpsx->latitude*NMEA_Pow(10,5)+(rs*NMEA_Pow(10,5-dx))/60;
		//转换为°
	}
	posx=NMEA_Comma_Pos(p1,4); //南纬还是北纬
	if(posx!=0XFF)gpsx->nshemi=*(p1+posx);
	posx=NMEA_Comma_Pos(p1,5); //得到经度
	if(posx!=0XFF)
	{
		temp=NMEA_Str2num(p1+posx,&dx);
		gpsx->longitude=temp/NMEA_Pow(10,dx+2); //得到°
		rs=temp%NMEA_Pow(10,dx+2); //得到'
		gpsx->longitude=gpsx->longitude*NMEA_Pow(10,5)+(rs*NMEA_Pow(10,5-dx))
		/60;//转换为°
	}
	posx=NMEA_Comma_Pos(p1,6); //东经还是西经
	if(posx!=0XFF)gpsx->ewhemi=*(p1+posx);
	posx=NMEA_Comma_Pos(p1,9); //得到 UTC 日期
	if(posx!=0XFF)
	{
		temp=NMEA_Str2num(p1+posx,&dx); //得到 UTC 日期
		gpsx->utc.date=temp/10000;
		gpsx->utc.month=(temp/100)%100;
		gpsx->utc.year=2000+temp%100;
	}
}

//分析 GPVTG 信息
//gpsx:nmea 信息结构体
//buf:接收到的 GPS 数据缓冲区首地址
void NMEA_GPVTG_Analysis(nmea_msg *gpsx,u8 *buf)
{
	u8 *p1,dx;
	u8 posx;
	p1=(u8*)strstr((const char *)buf,"$GPVTG");
	posx=NMEA_Comma_Pos(p1,7); //得到地面速率
	if(posx!=0XFF)
	{
		gpsx->speed=NMEA_Str2num(p1+posx,&dx);
		if(dx<3)gpsx->speed*=NMEA_Pow(10,3-dx); //确保扩大 1000 倍
	}
}

//提取 NMEA-0183 信息
//gpsx:nmea 信息结构体
//buf:接收到的 GPS 数据缓冲区首地址
void GPS_Analysis(nmea_msg *gpsx,u8 *buf)
{
	NMEA_GPGSV_Analysis(gpsx,buf); //GPGSV 解析
	NMEA_GPGGA_Analysis(gpsx,buf); //GPGGA 解析
	NMEA_GPGSA_Analysis(gpsx,buf); //GPGSA 解析
	NMEA_GPRMC_Analysis(gpsx,buf); //GPRMC 解析
	NMEA_GPVTG_Analysis(gpsx,buf); //GPVTG 解析
}

//GPS 校验和计算
//buf:数据缓存区首地址
//len:数据长度
//cka,ckb:两个校验结果.
void Ublox_CheckSum(u8 *buf,u16 len,u8* cka,u8*ckb)
{
	u16 i;
	*cka=0;*ckb=0;
	for(i=0;i<len;i++)
	{
		*cka=*cka+buf[i];
		*ckb=*ckb+*cka;
	}
}

///UBLOX 配置代码/
//检查 CFG 配置执行情况
//返回值:0,ACK 成功
// 1,接收超时错误
// 2,没有找到同步字符
// 3,接收到 NACK 应答
u8 Ublox_Cfg_Ack_Check(void)
{
	u16 len=0,i;
	u8 rval=0;
	while((USART3_RX_STA&0X8000)==0 && len<100)//等待接收到应答
	{
	len++;
	delay_ms(5);
	}
	if(len<250) //超时错误.
	{
		len=USART3_RX_STA&0X7FFF; //此次接收到的数据长度
		for(i=0;i<len;i++)if(USART3_RX_BUF[i]==0XB5)break;//查找同步字符 0XB5
		if(i==len)rval=2; //没有找到同步字符
		else if(USART3_RX_BUF[i+3]==0X00)rval=3;//接收到 NACK 应答
		else rval=0; //接收到 ACK 应答
	}
	else rval=1; //接收超时错误
	USART3_RX_STA=0; //清除接收
	return rval;
}

//配置保存
//将当前配置保存在外部 EEPROM 里面
//返回值:0,执行成功;1,执行失败.
u8 Ublox_Cfg_Cfg_Save(void)
{
	u8 i;
	_ublox_cfg_cfg *cfg_cfg=(_ublox_cfg_cfg *)USART3_TX_BUF;
	cfg_cfg->header=0X62B5; //cfg header
	cfg_cfg->id=0X0906; //cfg cfg id
	cfg_cfg->dlength=13; //数据区长度为 13 个字节.
	cfg_cfg->clearmask=0; //清除掩码为 0
	cfg_cfg->savemask=0XFFFF; //保存掩码为 0XFFFF
	cfg_cfg->loadmask=0; //加载掩码为 0
	cfg_cfg->devicemask=4; //保存在 EEPROM 里面
	Ublox_CheckSum((u8*)(&cfg_cfg->id),sizeof(_ublox_cfg_cfg)-4,&cfg_cfg->cka,
	&cfg_cfg->ckb);
	Ublox_Send_Date((u8*)cfg_cfg,sizeof(_ublox_cfg_cfg));//发送数据给 NEO-6M
	for(i=0;i<6;i++)if(Ublox_Cfg_Ack_Check()==0)break;
	//EEPROM 写入需要比较久时间,所以连续判断多次
	return i==6?1:0;
}

//配置 NMEA 输出信息格式
//msgid:要操作的 NMEA 消息条目,具体见下面的参数表
// 00,GPGGA;01,GPGLL;02,GPGSA;
// 03,GPGSV;04,GPRMC;05,GPVTG;
// 06,GPGRS;07,GPGST;08,GPZDA;
// 09,GPGBS;0A,GPDTM;0D,GPGNS;
//uart1set:0,输出关闭;1,输出开启.
//返回值:0,执行成功;其他,执行失败.
u8 Ublox_Cfg_Msg(u8 msgid,u8 uart1set)
{
	_ublox_cfg_msg *cfg_msg=(_ublox_cfg_msg *)USART3_TX_BUF;
	cfg_msg->header=0X62B5; //cfg header
	cfg_msg->id=0X0106; //cfg msg id
	cfg_msg->dlength=8; //数据区长度为 8 个字节.
	cfg_msg->msgclass=0XF0; //NMEA 消息
	cfg_msg->msgid=msgid; //要操作的 NMEA 消息条目
	cfg_msg->iicset=1; //默认开启
	cfg_msg->uart1set=uart1set; //开关设置
	cfg_msg->uart2set=1; //默认开启
	cfg_msg->usbset=1; //默认开启
	cfg_msg->spiset=1; //默认开启
	cfg_msg->ncset=1; //默认开启
	Ublox_CheckSum((u8*)(&cfg_msg->id),sizeof(_ublox_cfg_msg)-4,&cfg_msg->cka,
	&cfg_msg->ckb);
	Ublox_Send_Date((u8*)cfg_msg,sizeof(_ublox_cfg_msg));//发送数据给 NEO-6M
	return Ublox_Cfg_Ack_Check();
}	

//配置 NMEA 输出信息格式
//baudrate:波特率,4800/9600/19200/38400/57600/115200/230400
//返回值:0,执行成功;其他,执行失败(这里不会返回 0 了)
u8 Ublox_Cfg_Prt(u32 baudrate)
{
_ublox_cfg_prt *cfg_prt=(_ublox_cfg_prt *)USART3_TX_BUF;
cfg_prt->header=0X62B5; //cfg header
cfg_prt->id=0X0006; //cfg prt id
cfg_prt->dlength=20; //数据区长度为 20 个字节.
cfg_prt->portid=1; //操作串口 1
cfg_prt->reserved=0; //保留字节,设置为 0
cfg_prt->txready=0; //TX Ready 设置为 0
cfg_prt->mode=0X08D0; //8 位,1 个停止位,无校验位
cfg_prt->baudrate=baudrate; //波特率设置
cfg_prt->inprotomask=0X0007;//0+1+2
cfg_prt->outprotomask=0X0007;//0+1+2
cfg_prt->reserved4=0; //保留字节,设置为 0
cfg_prt->reserved5=0; //保留字节,设置为 0
Ublox_CheckSum((u8*)(&cfg_prt->id),sizeof(_ublox_cfg_prt)-4,&cfg_prt->cka,
&cfg_prt->ckb);
Ublox_Send_Date((u8*)cfg_prt,sizeof(_ublox_cfg_prt));//发送数据给 NEO-6M
delay_ms(200); //等待发送完成
usart3_init(42,baudrate); //重新初始化串口 3
return Ublox_Cfg_Ack_Check();
//这里不会反回 0,因为 UBLOX 发回来的应答在串口重新初始化时已经被丢弃了.
}
//配置 UBLOX NEO-6 的时钟脉冲输出
//interval:脉冲间隔(us)
//length:脉冲宽度(us)
//status:脉冲配置:1,高电平有效;0,关闭;-1,低电平有效.
//返回值:0,发送成功;其他,发送失败.
u8 Ublox_Cfg_Tp(u32 interval,u32 length,signed char status)
{
	_ublox_cfg_tp *cfg_tp=(_ublox_cfg_tp *)USART3_TX_BUF;
	cfg_tp->header=0X62B5; //cfg header
	cfg_tp->id=0X0706; //cfg tp id
	cfg_tp->dlength=20; //数据区长度为 20 个字节.
	cfg_tp->interval=interval; //脉冲间隔,us
	cfg_tp->length=length; //脉冲宽度,us
	cfg_tp->status=status; //时钟脉冲配置
	cfg_tp->timeref=0; //参考 UTC 时间
	cfg_tp->flags=0; //flags 为 0
	cfg_tp->reserved=0; //保留位为 0
	cfg_tp->antdelay=820; //天线延时为 820ns
	cfg_tp->rfdelay=0; //RF 延时为 0ns
	cfg_tp->userdelay=0; //用户延时为 0ns
	Ublox_CheckSum((u8*)(&cfg_tp->id),sizeof(_ublox_cfg_tp)-4,&cfg_tp->cka,
	&cfg_tp->ckb);
	Ublox_Send_Date((u8*)cfg_tp,sizeof(_ublox_cfg_tp));//发送数据给 NEO-6M
	return Ublox_Cfg_Ack_Check();
}

//配置 UBLOX NEO-6 的更新速率
//measrate:测量时间间隔,单位为 ms,最少不能小于 200ms(5Hz)
//reftime:参考时间,0=UTC Time;1=GPS Time(一般设置为 1)
//返回值:0,发送成功;其他,发送失败.
u8 Ublox_Cfg_Rate(u16 measrate,u8 reftime)
{
	_ublox_cfg_rate *cfg_rate=(_ublox_cfg_rate *)USART3_TX_BUF;
	if(measrate<200)return 1; //小于 200ms,直接退出
	cfg_rate->header=0X62B5; //cfg header
	cfg_rate->id=0X0806; //cfg rate id
	cfg_rate->dlength=6; //数据区长度为 6 个字节.
	cfg_rate->measrate=measrate;//脉冲间隔,us
	cfg_rate->navrate=1; //导航速率(周期),固定为 1
	cfg_rate->timeref=reftime; //参考时间为 GPS 时间
	Ublox_CheckSum((u8*)(&cfg_rate->id),sizeof(_ublox_cfg_rate)-4,&cfg_rate->cka,
	&cfg_rate->ckb);
	Ublox_Send_Date((u8*)cfg_rate,sizeof(_ublox_cfg_rate));//发送数据给 NEO-6M
	return Ublox_Cfg_Ack_Check();
}
//发送一批数据给 Ublox NEO-6M,这里通过串口 3 发送
//dbuf:数据缓存首地址
//len:要发送的字节数
void Ublox_Send_Date(u8* dbuf,u16 len)
{
	u16 j;
	for(j=0;j<len;j++)//循环发送数据
	{
		while((USART3->SR&0X40)==0);//循环发送,直到发送完毕
		USART3->DR=dbuf[j];
	}
}						

gps.h

#ifndef __GPS_H
#define __GPS_H
#include "sys.h"
//GPS NMEA-0183 协议重要参数结构体定义
//卫星信息
__packed typedef struct
{
	u8 num; //卫星编号
	u8 eledeg; //卫星仰角
	u16 azideg; //卫星方位角
	u8 sn; //信噪比
}nmea_slmsg;

//UTC 时间信息
__packed typedef struct
{
	u16 year; //年份
	u8 month; //月份
	u8 date; //日期
	u8 hour; //小时
	u8 min; //分钟
	u8 sec; //秒钟
}nmea_utc_time;

//NMEA 0183 协议解析后数据存放结构体
__packed typedef struct
{
	u8 svnum; //可见卫星数
	nmea_slmsg slmsg[12]; //最多 12 颗卫星
	nmea_utc_time utc; //UTC 时间
	u32 latitude; //纬度 分扩大 100000 倍,实际要除以 100000
	u8 nshemi; //北纬/南纬,N:北纬;S:南纬
	u32 longitude; //经度 分扩大 100000 倍,实际要除以 100000
	u8 ewhemi; //东经/西经,E:东经;W:西经
	u8 gpssta; //GPS 状态:0,未定位;1,非差分定位;2,差分定位;6,正在估算.
	u8 posslnum; //用于定位的卫星数,0~12.
	u8 possl[12]; //用于定位的卫星编号
	u8 fixmode; //定位类型:1,没有定位;2,2D 定位;3,3D 定位
	u16 pdop; //位置精度因子 0~500,对应实际值 0~50.0
	u16 hdop; //水平精度因子 0~500,对应实际值 0~50.0
	u16 vdop; //垂直精度因子 0~500,对应实际值 0~50.0
	int altitude; //海拔高度,放大了 10 倍,实际除以 10.单位:0.1m
	u16 speed; //地面速率,放大了 1000 倍,实际除以 10.单位:0.001 公里/小时
}nmea_msg;

//UBLOX NEO-6M 配置(清除,保存,加载等)结构体
__packed typedef struct
{
	u16 header; //cfg header,固定为 0X62B5(小端模式)
	u16 id; //CFG CFG ID:0X0906 (小端模式)
	u16 dlength; //数据长度 12/13
	u32 clearmask; //子区域清除掩码(1 有效)
	u32 savemask; //子区域保存掩码
	u32 loadmask; //子区域加载掩码
	u8 devicemask;
	//目标器件选择掩码 b0:BK RAM;b1:FLASH;b2,EEPROM;b4,SPI FLASH
	u8 cka; //校验 CK_A
	u8 ckb; //校验 CK_B
}_ublox_cfg_cfg;

//UBLOX NEO-6M 消息设置结构体
__packed typedef struct
{
	u16 header; //cfg header,固定为 0X62B5(小端模式)
	u16 id; //CFG MSG ID:0X0106 (小端模式)
	u16 dlength; //数据长度 8
	u8 msgclass; //消息类型(F0 代表 NMEA 消息格式)
	u8 msgid; //消息 ID
	//00,GPGGA;01,GPGLL;02,GPGSA;
	//03,GPGSV;04,GPRMC;05,GPVTG;
	//06,GPGRS;07,GPGST;08,GPZDA;
	//09,GPGBS;0A,GPDTM;0D,GPGNS;
	u8 iicset; //IIC 消输出设置 0,关闭;1,使能.
	u8 uart1set; //UART1 输出设置 0,关闭;1,使能.
	u8 uart2set; //UART2 输出设置 0,关闭;1,使能.
	u8 usbset; //USB 输出设置 0,关闭;1,使能.
	u8 spiset; //SPI 输出设置 0,关闭;1,使能.
	u8 ncset; //未知输出设置 默认为 1 即可.
	u8 cka; //校验 CK_A
	u8 ckb; //校验 CK_B
}_ublox_cfg_msg;

//UBLOX NEO-6M UART 端口设置结构体
__packed typedef struct
{
	u16 header; //cfg header,固定为 0X62B5(小端模式)
	u16 id; //CFG PRT ID:0X0006 (小端模式)
	u16 dlength; //数据长度 20
	u8 portid; //端口号,0=IIC;1=UART1;2=UART2;3=USB;4=SPI;
	u8 reserved; //保留,设置为 0
	u16 txready; //TX Ready 引脚设置,默认为 0
	u32 mode; //串口工作模式设置,奇偶校验,停止位,字节长度等的设置.
	u32 baudrate; //波特率设置
	u16 inprotomask; //输入协议激活屏蔽位 默认设置为 0X07 0X00 即可.
	u16 outprotomask; //输出协议激活屏蔽位 默认设置为 0X07 0X00 即可.
	u16 reserved4; //保留,设置为 0
	u16 reserved5; //保留,设置为 0
	u8 cka; //校验 CK_A
	u8 ckb; //校验 CK_B
}_ublox_cfg_prt;
//UBLOX NEO-6M 时钟脉冲配置结构体
__packed typedef struct
{
	u16 header; //cfg header,固定为 0X62B5(小端模式)
	u16 id; //CFG TP ID:0X0706 (小端模式)
	u16 dlength; //数据长度
	u32 interval; //时钟脉冲间隔,单位为 us
	u32 length; //脉冲宽度,单位为 us
	signed char status; //时钟脉冲配置:1,高电平有效;0,关闭;-1,低电平有效.
	u8 timeref; //参考时间:0,UTC 时间;1,GPS 时间;2,当地时间.
	u8 flags; //时间脉冲设置标志
	u8 reserved; //保留
	signed short antdelay; //天线延时
	signed short rfdelay; //RF 延时
	signed int userdelay; //用户延时
	u8 cka; //校验 CK_A
	u8 ckb; //校验 CK_B
}_ublox_cfg_tp;

//UBLOX NEO-6M 刷新速率配置结构体
__packed typedef struct
{
	u16 header; //cfg header,固定为 0X62B5(小端模式)
	u16 id; //CFG RATE ID:0X0806 (小端模式)
	u16 dlength; //数据长度
	u16 measrate; //测量时间间隔,单位为 ms,最少不能小于 200ms (5Hz)
	u16 navrate; //导航速率(周期),固定为 1
	u16 timeref; //参考时间:0=UTC Time;1=GPS Time;
	u8 cka; //校验 CK_A
	u8 ckb; //校验 CK_B
}_ublox_cfg_rate;

int NMEA_Str2num(u8 *buf,u8*dx);
void GPS_Analysis(nmea_msg *gpsx,u8 *buf);
void NMEA_GPGSV_Analysis(nmea_msg *gpsx,u8 *buf);
void NMEA_GPGGA_Analysis(nmea_msg *gpsx,u8 *buf);
void NMEA_GPGSA_Analysis(nmea_msg *gpsx,u8 *buf);
void NMEA_GPGSA_Analysis(nmea_msg *gpsx,u8 *buf);
void NMEA_GPRMC_Analysis(nmea_msg *gpsx,u8 *buf);
void NMEA_GPVTG_Analysis(nmea_msg *gpsx,u8 *buf);
u8 Ublox_Cfg_Cfg_Save(void);
u8 Ublox_Cfg_Msg(u8 msgid,u8 uart1set);
u8 Ublox_Cfg_Prt(u32 baudrate);
u8 Ublox_Cfg_Tp(u32 interval,u32 length,signed char status);
u8 Ublox_Cfg_Rate(u16 measrate,u8 reftime);
void Ublox_Send_Date(u8* dbuf,u16 len);

#endif

test.c

u8 USART1_TX_BUF[USART3_MAX_RECV_LEN]; //串口 1,发送缓存区
nmea_msg gpsx; //GPS 信息
__align(4) u8 dtbuf[50]; //打印缓存器
const u8*fixmode_tbl[4]={"Fail","Fail"," 2D "," 3D "}; //fix mode 字符串
//显示 GPS 定位信息
void Gps_Msg_Show(void)
{
	float tp;
	POINT_COLOR=BLUE;
	tp=gpsx.longitude;
	sprintf((char *)dtbuf,"Longitude:%.5f %1c ",tp/=100000,gpsx.ewhemi);//得到经度
	LCD_ShowString(30,130,200,16,16,dtbuf);
	tp=gpsx.latitude;
	sprintf((char *)dtbuf,"Latitude:%.5f %1c ",tp/=100000,gpsx.nshemi); //得到纬度
	LCD_ShowString(30,150,200,16,16,dtbuf);
	tp=gpsx.altitude;
	sprintf((char *)dtbuf,"Altitude:%.1fm ",tp/=10);//得到高度字符串
	LCD_ShowString(30,170,200,16,16,dtbuf);
	tp=gpsx.speed;
	sprintf((char *)dtbuf,"Speed:%.3fkm/h ",tp/=1000);//得到速度字符串
	LCD_ShowString(30,190,200,16,16,dtbuf);
	if(gpsx.fixmode<=3)//定位状态
	{
		sprintf((char *)dtbuf,"Fix Mode:%s",fixmode_tbl[gpsx.fixmode]);
		LCD_ShowString(30,210,200,16,16,dtbuf);
	}
		sprintf((char *)dtbuf,"Valid satellite:%02d",gpsx.posslnum);//用于定位的卫星数
	LCD_ShowString(30,230,200,16,16,dtbuf);
	sprintf((char *)dtbuf,"Visible satellite:%02d",gpsx.svnum%100);//可见卫星数
	LCD_ShowString(30,250,200,16,16,dtbuf);
	sprintf((char *)dtbuf,"UTC Date:%04d/%02d/%02d ",gpsx.utc.year,gpsx.utc.month,
	gpsx.utc.date); //显示 UTC 日期
	//printf("year2:%d\r\n",gpsx.utc.year);
	LCD_ShowString(30,270,200,16,16,dtbuf);
	sprintf((char *)dtbuf,"UTC Time:%02d:%02d:%02d ",gpsx.utc.hour,gpsx.utc.min,
	gpsx.utc.sec); //显示 UTC 时间
	LCD_ShowString(30,290,200,16,16,dtbuf);
}

int main(void)
{
	u16 i,rxlen;
	u16 lenx;
	u8 key=0XFF;
	u8 upload=0;
	Stm32_Clock_Init(336,8,2,7);//设置时钟,168Mhz
	delay_init(168); //延时初始化
	uart_init(84,115200); //初始化串口波特率为 115200
	usart3_init(42,38400); //初始化串口 3 波特率为 38400
	usmart_dev.init(84); //初始化 USMART
	LED_Init(); //初始化 LED
	KEY_Init(); //初始化按键
	LCD_Init(); //初始化 LCD
	usmart_dev.init(72); //初始化 USMART
	POINT_COLOR=RED;
	LCD_ShowString(30,20,200,16,16,"ALIENTEK STM32F4 ^_^");
	LCD_ShowString(30,40,200,16,16,"NE0-6M GPS TEST");
	LCD_ShowString(30,60,200,16,16,"ATOM@ALIENTEK");
	LCD_ShowString(30,80,200,16,16,"KEY0:Upload NMEA Data SW");
	LCD_ShowString(30,100,200,16,16,"NMEA Data Upload:OFF");
	if(Ublox_Cfg_Rate(1000,1)!=0)
	//设置定位信息更新速度为 1000ms,顺便判断 GPS 模块是否在位.
	{
		LCD_ShowString(30,120,200,16,16,"NEO-6M Setting...");
		while((Ublox_Cfg_Rate(1000,1)!=0)&&key)
		//持续判断,直到可以检查到 NEO-6M,且数据保存成功
		{
			usart3_init(42,9600);
			//初始化串口 3,9600(EEPROM 没有保存数据的时候,波特率为 9600.)
			Ublox_Cfg_Prt(38400); //重新设置模块的波特率为 38400
			usart3_init(42,38400); //重新设置 38400
			Ublox_Cfg_Tp(1000000,100000,1);//设置 PPS 为 1 秒钟输出,脉宽为 100ms
			key=Ublox_Cfg_Cfg_Save(); //保存配置
		}
		LCD_ShowString(30,120,200,16,16,"NEO-6M Set Done!!");
		delay_ms(500);
		LCD_Fill(30,120,30+200,120+16,WHITE);//清除显示
	}
	while(1)
	{
		delay_ms(1);
		if(USART3_RX_STA&0X8000) //接收到一次数据了
		{
			rxlen=USART3_RX_STA&0X7FFF; //得到数据长度
			for(i=0;i<rxlen;i++)USART1_TX_BUF[i]=USART3_RX_BUF[i];
			USART3_RX_STA=0; //启动下一次接收
			USART1_TX_BUF[i]=0; //自动添加结束符
			GPS_Analysis(&gpsx,(u8*)USART1_TX_BUF);//分析字符串
			Gps_Msg_Show(); //显示信息
			if(upload)printf("\r\n%s\r\n",USART1_TX_BUF);//发送数据到串口 1
		}
		key=KEY_Scan(0);
		if(key==KEY0_PRES)
		{
		upload=!upload;
		POINT_COLOR=RED;
		if(upload)LCD_ShowString(30,100,200,16,16,"NMEA Data Upload:ON ");
		else LCD_ShowString(30,100,200,16,16,"NMEA Data Upload:OFF");
		}
		if((lenx%500)==0)LED0=!LED0;
		lenx++;
	}
}


---------------------
作者:R_Z_Q
来源:CSDN
原文:https://blog.csdn.net/R_Z_Q/article/details/104464836
版权声明:本文为作者原创文章,转载请附上博文链接!
内容解析By:CSDN,CNBLOG博客文章一键转载插件

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

GPS模块(GPS-NEO-6M) 的相关文章

  • 在android中以编程方式打开GPS [重复]

    这个问题在这里已经有答案了 可能重复 如何以编程方式在android中启用 禁用gps和移动数据 https stackoverflow com questions 16752859 how to enable disable gps an
  • 传递给 onLocationChanged 的​​位置对象是否为 null?

    我有一个在我的应用程序中实现 LocationListener 的活动 并且我的 onLocationChanged 方法直到最近为止一直运行良好 由于某种原因 传递给该方法的 Location 对象为 null 我的问题是 为什么它是空的
  • 使用 JavaScript 获取 GPS 位置

    我正在使用平板电脑 正在处理 html 应用程序 希望在不使用任何网络连接的情况下获取 GPS 位置 请建议我简单的方法 使用JS你可以这样得到它
  • 我无法在 Android 真实手机上获取位置

    public class LocationService extends Service private Handler mHandler new Handler private Timer mTimer null private int
  • 同时有多个位置提供商

    我的定位系统有一些问题 我有一个实现位置侦听器的服务 我希望尽可能使用网络获得最佳位置 如果网络不够准确 精度大于 300mt 则使用 GPS 问题是这样的 我每 5 分钟需要一次位置信息 如果可能的话准确 否则不准确 我从一个开始 Loc
  • GPS 转换 - 像素坐标到 GPS 坐标

    我正在根据视频数据进行一些运动跟踪 使用一些视频处理 即转换为自上而下的视图 我获得了移动路径 我现在需要将路径的像素坐标 x y 转换为世界坐标 纬度 经度 我在图像中有四个参考点及其相关的纬度和经度点 纬度 经度 gt 像素坐标 51
  • Android 手机和模拟器中的mapView不同

    关于应用程序 这是一个简单的应用程序 可以查找用户当前位置 问题 该应用程序在模拟器上运行良好 请参见图片 但在手机中它没有显示MapView 请看图片 请告诉我手机出了什么问题 在手机中 它只下载巨大的 20 MB 数据 但不显示实际地图
  • 通过js获取WebView中的位置

    我正在尝试创建 WebView 它将通过 js 获取 GPS 本地化 但是当我单击应该显示本地化的按钮时 在android 4 1 1 模拟器 中 错误代码2 无法启动地理定位服务 在 android 4 1 2 phone 中什么也没有发
  • 动态更改 GPS LocationListener 的 minTime

    我正在编写的应用程序使用 GPS 位置管理器服务 requestLocationUpdates 但我希望能够更改整个程序中的最小时间和最小距离参数 我在创建时初始化位置侦听器 但我不知道如何更改这些参数 甚至不知道是否可以这样做 这样做的主
  • GPS定位无服务

    我是一名新开发人员 有一个简单的问题 我已经四处搜索 但尚未找到明确的答案 简而言之 我正在开发一个需要能够使用 GPS 的应用程序 然而 诀窍是我想使用 GPS 来获取手机的位置 即使它们没有运营商服务 话虽如此 我有两个问题 是否可以通
  • 在设备所有者应用程序中启用 GPS

    根据API文档 https developer android com reference android app admin DevicePolicyManager html setSecureSetting android conten
  • React Native Android 位置请求超时

    在 IOS 中查找 GPS 坐标时没有问题 效果很好 Android 端不如 IOS 稳定 在真机和模拟器中都会出现这个问题 有时它可以找到位置 但有时却找不到 寻找了3天 但没有找到解决方案 当我的应用程序无法找到我的位置时 我尝试通过谷
  • 寻找另一部智能手机的笛卡尔坐标?

    考虑到我有两部智能手机 A 和 B 如果我拿着智能手机 A 有没有办法确定 B 相对于我自己的位置 所以如果我们有这张图片的情况 它会告诉我 B 位于位置 2 1 利用 WiFi 信号强度来获取位置等创造性方法更受欢迎 我还可以确定两部手机
  • GPS 对比加速度计计算距离

    我正在尝试实现一个健身应用程序 可以在Android 中跟踪跑步速度和跑步距离 看起来我可以使用 GPS 或加速度计来计算这些信息 由于跑步者可能会将手机放在手里 放在肩膀上或放在口袋里 所以我的第一直觉是使用 GPS 获取位置并计算跑步速
  • 如何从 Android 手机获取 GPS 数据?

    有没有办法将 Android 手机的 GPS 数据连接 USB 有线 到 PC 我目前正在使用基于 gpsd 项目的 GPSTether 应用程序 我正在寻找比该应用程序提供更多控制且错误更少的替代方案 另外 是否有另一种方法可以在不使用任
  • 使用 Google 电子表格中的脚本从手机获取我的当前位置

    有没有办法使用 Google Apps 脚本从手机的 GPS 数据中获取我的当前位置 纬度和经度 最好是十进制形式 另外 是否可以打开和关闭 GPS 或者至少检测它是否打开或关闭 这是我尝试做的 我带着电动助力车去一些地方 在每个地方我都会
  • Android 中的 GPS 超时

    在黑莓中 我们使用超时来获取位置 这样如果它在这么长时间内没有重新调整位置 我们就会知道 但是在Android中 没有超时的概念 任何人都可以告诉我们替代方案 我们可以发现 在这么长时间之后 GPS没有位置更新 您可以使用两个线程来实现此超
  • 向 tk103 GPS 跟踪器发送命令

    我正在使用 php 开发实时 GPS 跟踪器 Web 应用程序 跟踪器参考号是tk103 我可以从跟踪器接收信息并将其存储到数据库中 设备的 GPRS 模式已启用 我的问题是 如何使用 php ini 将命令从服务器发送到设备 提前致谢 这
  • 如何创建在 React-Native 中检测自动位置的地图

    我已经在react native中创建了地图 参考https github com lelandrichardson react native maps https github com lelandrichardson react nat
  • Ruby on Rails:如何使用 TCP 套接字连接 GPS 设备

    ruby 2 3 0p0 2015 12 25 修订版 53290 x86 64 linux 轨道 4 2 4 我正在使用 cloud9 IDE 和 webrick 服务器 我的项目是实时跟踪GPS 我想使用TCP连接与GPS跟踪设备进行通

随机推荐

  • java 简单员工管理系统

    package cn test import java io import java util class Emp 64 Override public String toString return 34 Emp empNo 61 34 4
  • 关闭虚拟机vmware自动挂起

    1 xff0c 桌面右键 属性 xff0c 里的屏幕保护程序 xff0c 选 无 2 xff0c 控制面板的电源选项 xff0c 方案为 一直开着 xff0c 具体选项选 从不关机 3 xff0c 在Windows 2003 Server点
  • GTK+2.0之初始学习篇(二)—— g_signal_connect宏解释及HelloWorld

    GTK 43 2 0中利用信号 回调函数机制来处理窗口外部传来的事件 消息或信号 以下实现的是单击窗口关闭按钮 xff0c 窗口自动关闭 在程序中调用了gtk main quit 函数实现退出主循环 include lt gtk gtk h
  • linux-----页、页表、页框(块)

    基本介绍 我们知道 xff0c 在linux操作系统中 xff0c CPU在执行一个进程的时候 xff0c 都会访问内存 但CPU并不是直接访问物理内存地址 xff0c 而是通过虚拟地址空间来间接的访问物理内存地址 所谓的虚拟地址空间 xf
  • CMake交叉编译配置

    罗列一下cmake常用的命令 CMake支持大写 小写 混合大小写的命令 1 添加头文件目录INCLUDE DIRECTORIES 语法 xff1a include directories AFTER BEFORE SYSTEM dir1
  • 8086CPU的14个寄存器全称

    通用寄存器 xff1a ax accumulate register 累加器 bx based register 基地址寄存器 cx count register 计数器 dx data registered 数据寄存器 段寄存器 xff1
  • 项目开发-后台管理框架

    开发中几乎的平台都需要一个后台管理 xff0c 但是从零开发一套后台控制面板并不容易 xff0c 幸运的是有很多开源免费的后台控制面板可以给开发者使用 xff0c 那么有哪些优秀的开源免费的控制面板呢 xff1f 我在 Github 上收集
  • 项目经理必备工具-个人推荐

    点击链接 项目经理必备工具 进入详细说明
  • 如何绕过CDN找源站ip?

    如何绕过CDN找源站ip xff1f 这是一个总结帖 xff0c 查了一下关于这个问题的国内外大大小小的网站 xff0c 对其中说的一些方法总结归纳形成 xff0c 里面具体发现ip的方法不是原创 xff0c 所有参考的原贴都也贴在了后面
  • Navicat修改MySQL数据库密码就是这么简单

    方法1 xff1a 用SET PASSWORD命令 首先登录MySQL 格式 xff1a mysql gt set password for 用户名 64 localhost 61 password 39 新密码 39 例子 xff1a m
  • Axure 元件库-原型

    元件名称 xff1a 1 Axure交互原型设计指南 rp 2 PC和移动原型常用元件 rp 3 后台模板 rp 4 全局说明 rp https pan baidu com s 1vmac 08MZAKj6qsdjwIZlg 提取码 xff
  • Node.js 通过http调用外部接口

    通过http request发送带参数的post请求 data xff1a 发送的内容 opt xff1a 描述将要发出的请求 data xff1a 事件在数据到达时被触发 end xff1a 请求结束时触发 error xff1a 发生错
  • 想成为出色的 CTO,你要具备这七大能力

    编者按 xff1a 首席技术官这一职位在20世纪80年代出现于美国 起于做很多技术研发的大公司 xff0c 如General Electric 主要职能是将科研成果转化实际效益 简单地说 xff0c 就是一个企业中技术的最高负责人 要扮演好
  • Keil中添加自己的头文件

    xfeff xfeff 方法一 在keil的开发环境下添加 xff1a 请注意上面的系统生成的头文件目录是 xff1a xff0c 即 dd jj pp kk xff0c 其中 是相对于项目文件 uvproj 而言的 其中 表示项目文件所在
  • http authorization 基本认证

    最近做的一个项目需要与其它系统对接接口 xff0c 对方提供的是webservice的接口 xff0c 并且需要Basic Authorization基本认证 xff0c 一开始都是用postman请求 xff0c 用户名和密码需要填在Ba
  • 终端命令安装 chrome for linux

    终端里安装chrome for linux 备注 xff1a 我是在Linux Mint 17 1 64位系统下安装的chrome for linux xff0c 其它Debian衍生版应该也是一样的 1 在终端里输入下载命令 xff1a
  • Arduino串口函数详解

    本文总结了Arduino常用串口操作函数 xff0c 函数说明部分来源于Arduino 官网串口使用指南 xff0c 示例与实验部分为自编 本文是对Arduino串口操作函数的较全面总结 xff0c 可作为工具贴查找使用 1 串口设置 xf
  • STM32CubeMX——霍尔编码器、L298N驱动电机

    前言 人生如逆旅 xff0c 我亦是行人 苏轼 临江仙 送钱穆父 目录 xff1a L298N电机驱动介绍编码器介绍电机介绍新建工程编写代码实验结果 一 L298N电机驱动介绍 B站 视频讲解 xff1a l298n电机驱动模块 电机正反转
  • AT命令拨电话,如何判断手机的状态?

    我使用AT命令拨电话 xff0c 如 xff1a ATD10086 我怎么知道我拨打的电话是否成功了呢 xff1f 比如SIM卡欠费了 xff0c 那么肯定算是没有拨通 xff1b 再比如网络有问题 xff0c 被叫方没有收到来电 xff0
  • GPS模块(GPS-NEO-6M)

    ATK NEO 6M GPS 模块简介 ATK NEO 6M V23 模块 xff0c 是 ALIENTEK 生产的一款高性能 GPS 模块 xff0c 模块核心采用 UBLOX公司的 NEO 6M 模组 xff0c 具有 50 个 通道