NMEA(xxGGA)报文解析(FPGA实现)

2023-10-13 19:50

本文主要是介绍NMEA(xxGGA)报文解析(FPGA实现),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近接触GPS,需要使用FPGA进行NMEA报文的解析,以获得经纬度和时间信息,我选用的报文是xxGGA,包含GPGGA(GPS系统的)、GBGGA(北斗系统的)、GLGGA(GLONASS系统的)、GAGGA(伽利略系统的),GNGGA(任意GNSS系统组合)。他们的格式完全相同,不同之处仅在于报文头,xxGGA报文格式如下

$xxGGA,time,lat,NS,lon,EW,quality,numSV,HDOP,alt,altUnit,sep,sepUnit,diffAge,diffStation*cs<CR><LF>

我们需要关注的数据域如下

time:UTC时间,hhmmss.ss格式,如132253.27表示UTC时间 13时22分53.27秒,需要注意的是.ss表示秒的小数域(2位小数),而非毫秒

lat:纬度,ddmm.mmmm格式,如3124.73251表示 31度24.73251分,1度=60分

NS:指示南北半球,北半球为‘N’,南半球为‘S’

lon:经度,dddmm.mmmmm格式,如13424.73251表示 134度24.73251分

EW:指示东西半球,东半球为‘E’,西半球为‘W’

alt:海拔,-9999.9~9999.9

altUnit:海拔单位,‘M’表示以 米 为单位

例如:

$GPGGA,092725.00,4717.11399,N,00833.91590,E,1,08,1.01,499.6,M,48.0,M,,*5B

表示 UTC时间 09 时 27 分 25.00 秒,北纬 47 度 17.11399 分,东经 8 度 33.91590 分,海拔 499.6 米

------------------------------------------------------------------------------分割线---------------------------------------------------------------------------------------------

本xxGGA解析模块使用UART串口数据,以UART模块发出的rx_done信号驱动。

xxGGA报文解析模块:

/******************************FILE HEAD*********************************** file_name         : parseGGA.v* function          : 解析xxGGA报文,获取UTC时间、经纬度、海拔* author            : 今朝无言* version & date    : 2021/10/14 & v1.0*************************************************************************/
module parseGGA(input				rx_done_toUart,	//整个模块由rx_done_toUart驱动input		[7:0]	rddat_toUart,output	reg 		rx_done,		//接收指令结束,上升沿对齐'\n'字符出现时刻,下降沿对齐$xxGGA后面的','的出现时刻的下一时刻output	reg	[4:0]	hh,				//UTC时间,整数,时  0~24output	reg	[5:0]	mm,				//UTC时间,整数,分  0~59output	reg	[5:0]	ss,				//UTC时间,整数,秒  0~59output	reg	[6:0]	ss2,			//UTC时间,小数,秒  2位小数,0~99output	reg	[6:0]	lat,			//纬度  整数部分,度  0~90output	reg	[5:0]	lat2,			//纬度  整数部分,分  0~59output	reg	[16:0]	lat3,			//纬度  小数部分,分  5位小数,0~99999output	reg			NS,				//区分南北纬,北纬标为1,南纬标为0output	reg	[7:0]	lon,			//经度  整数部分,度  0~180output	reg	[5:0]	lon2,			//经度  整数部分,分  0~59output	reg	[16:0]	lon3,			//经度  小数部分,分  5位小数,0~99999output	reg			EW,				//区分东西经,东经标为1,西经标为0output	reg [13:0]	alt,			//海拔  整数部分,moutput	reg [3:0]	alt2			//海拔  小数部分,一位小数  0~9
);
//xxGGA格式: $xxGGA,time,lat,NS,lon,EW,quality,numSV,HDOP,alt,altUnit,sep,sepUnit,diffAge,diffStation*cs<CR><LF>
//time格式: hhmmss.ss
//lat格式: ddmm.mmmmm
//lon格式: dddmm.mmmmm
//alt格式: numeric,一位小数reg		[4:0]	cntField;		//当前读取第几个域,以','分隔
reg 	[3:0]	cntChar;		//当前读取域中的第几个字符reg				start	= 1'b0;	//NMEA报文的接收标志,以$开始,到\n结束
reg		[7:0]	charBuffer;
reg		[1:0]	corrNum	= 2'd0;	//比对是否为xxGGA,当corrNum=3时,表示"GGA"字符通过测试,该条报文即xxGGAwire	[3:0]	num;			//若charBuffer为字符0~9,则将之转换为数字0~9
wire			isnum;reg				afterDot;		//判断是否是"."后面的数字,在解析海拔时用到//将字符0~9转换为数字0~9
Char2Num Char2Num_inst(.Char(charBuffer),.Num(num),.isNum(isnum)
);always @(posedge rx_done_toUart) begincharBuffer	<= rddat_toUart;//-----------------------接收NMEA报文数据-----------------------------if(rddat_toUart == "$") begin			//接收到$,标志着NMEA数据的起始start		<= 1'b1;cntField	<= 5'd0;cntChar		<= 4'd0;corrNum		<= 2'd0;endelse if(start) beginif(rddat_toUart == "\n") begin		//收到\n,标志NMEA报文结束start		<= 1'b0;rx_done		<= 1'b1;endelse if(rddat_toUart == "," || rddat_toUart == "*") begin	//收到','或'*',为域的分隔符cntField	<= cntField + 1'b1;cntChar		<= 4'd0;endelse begin							//收到其他字符cntChar		<= cntChar + 1'b1;endendelse beginstart		<= 1'b0;cntField	<= 5'd0;cntChar		<= 4'd0;corrNum		<= 2'd0;end//------------------------判断是否为xxGGA----------------------------if(cntField == 5'd0) beginif(cntChar == 4'd3 && charBuffer == "G") begincorrNum <= corrNum + 1'b1;endelse if(cntChar == 4'd4 && charBuffer == "G") begincorrNum <= corrNum + 1'b1;endelse if(cntChar == 4'd5 && charBuffer == "A") begincorrNum <= corrNum + 1'b1;endendif(corrNum == 2'd3) begin	//检测到是"xxGGA",开启解析rx_done <= 1'b0;corrNum	<= 2'b0;end//---------------------------解析xxGGA------------------------------if(rx_done == 1'b0) begin//解析UTC时间if(cntField == 5'd1) beginif(cntChar == 4'd1) begin		//UTC-hhhh	<= num*4'd10;endelse if(cntChar == 4'd2) beginhh	<= hh + num;endelse if(cntChar == 4'd3) begin	//UTC-mmmm	<= num*4'd10;endelse if(cntChar == 4'd4) beginmm	<= mm + num;endelse if(cntChar == 4'd5) begin	//UTC-ssss	<= num*4'd10;endelse if(cntChar == 4'd6) beginss	<= ss + num;endelse if(cntChar == 4'd8) begin	//UTC-.ssss2	<= num*4'd10;endelse if(cntChar == 4'd9) beginss2	<= ss2 + num;endend//解析纬度if(cntField == 5'd2) beginif(cntChar == 4'd1) begin		//lat-ddlat		<= num*4'd10;endelse if(cntChar == 4'd2) beginlat		<= lat + num;endelse if(cntChar == 4'd3) begin	//lat-mmlat2	<= num*4'd10;endelse if(cntChar == 4'd4) beginlat2	<= lat2 + num;endelse if(cntChar == 4'd6) begin	//lat-.mmmmmlat3	<= num;endelse if(cntChar == 4'd7 || cntChar == 4'd8 ||cntChar == 4'd9 || cntChar == 4'd10) beginlat3	<= lat3*4'd10 + num;endendif(cntField == 5'd3 && cntChar == 4'd1) begin	//NSif(charBuffer == "N") beginNS	<= 1'b1;endelse beginNS	<= 1'b0;endend//解析经度if(cntField == 5'd4) beginif(cntChar == 4'd1) begin		//lon-dddlon		<= num;endelse if(cntChar == 4'd2 || cntChar == 4'd3) beginlon		<= lon*4'd10 + num;endelse if(cntChar == 4'd4) begin	//lon-mmlon2	<= num*4'd10;endelse if(cntChar == 4'd5) beginlon2	<= lon2 + num;endelse if(cntChar == 4'd7) begin	//lon-.mmmmmlon3	<= num;endelse if(cntChar == 4'd8 || cntChar == 4'd9 ||cntChar == 4'd10 || cntChar == 4'd11) beginlon3	<= lon3*4'd10 + num;endendif(cntField == 5'd5 && cntChar == 4'd1) begin	//EWif(charBuffer == "E") beginEW	<= 1'b1;endelse beginEW	<= 1'b0;endend//解析海拔if(cntField == 5'd9) beginif(cntChar == 4'd1) beginalt			<= num;afterDot	<= 1'b0;endelse if(charBuffer==".") beginafterDot	<= 1'b1;alt2		<= 4'd0;endelse beginif(~afterDot) beginalt		<= alt*4'd10 + num;		//alt-MMMendelse beginalt2	<= alt2*4'd10 +num;		//alt-.Mendendendend
endendmodule
//END OF parseGGA.v FILE***************************************************

字符-数字转换模块:

/******************************FILE HEAD*********************************** file_name         : Char2Num.v* function          : 若Char为字符0~9,将之转化为数字0~9* author            : 今朝无言* version & date    : 2021/10/14 & v1.0*************************************************************************/
module Char2Num(input 		[7:0]	Char,output		[3:0]	Num,output	reg			isNum
);always@(*)begincase(Char)"0": isNum <= 1;"1": isNum <= 1;"2": isNum <= 1;"3": isNum <= 1;"4": isNum <= 1;"5": isNum <= 1;"6": isNum <= 1;"7": isNum <= 1;"8": isNum <= 1;"9": isNum <= 1;default: isNum <= 0;endcase
endassign Num = isNum? Char - "0" : 4'hff;endmodule
//END OF Char2Num.v FILE***************************************************

testbench:

/******************************FILE HEAD*********************************** file_name         : parseGGA_tb.v* function          : 解析xxGGA报文,获取UTC时间、经纬度、海拔* author            : 今朝无言* version & date    : 2021/10/14 & v1.0*************************************************************************/
`default_nettype none
`timescale 1ns/1psmodule parseGGA_tb;reg	[0:75*8-1]data = {"$GNGGA,092725.00,4717.11399,N,00833.91590,E,1,08,1.01,499.6,M,48.0,M,,*5B",8'd13,8'd10}; //\r\n, \r=13,\n=10reg				rx_done_toUart;	//整个模块由rx_done_toUart驱动
reg		[7:0]	rddat_toUart;wire 			rx_done;		//接收指令结束,上升沿对齐'\n'字符出现时刻,下降沿对齐$xxGGA后面的','的出现时刻wire	[4:0]	hh;				//UTC时间,整数,时  0~24
wire	[5:0]	mm;				//UTC时间,整数,分  0~59
wire	[5:0]	ss;				//UTC时间,整数,秒  0~59
wire	[6:0]	ss2;			//UTC时间,小数,秒  2位小数,0~99wire	[6:0]	lat;			//纬度  整数部分,度  0~90
wire	[5:0]	lat2;			//纬度  整数部分,分  0~59
wire	[16:0]	lat3;			//纬度  小数部分,分  5位小数,0~99999
wire			NS;				//区分南北纬,北纬标为1,南纬标为0wire	[7:0]	lon;			//经度  整数部分,度  0~180
wire	[5:0]	lon2;			//经度  整数部分,分  0~59
wire	[16:0]	lon3;			//经度  小数部分,分  5位小数,0~99999
wire			EW;				//区分东西经,东经标为1,西经标为0wire	[13:0]	alt;			//海拔  整数部分,m
wire	[3:0]	alt2;			//海拔  小数部分,一位小数  0~9reg		[9:0]	i;
initial beginrx_done_toUart	<= 0;#50;for(i=0;i<=74*8;i=i+8)beginrddat_toUart	<= {data[i],data[i+1],data[i+2],data[i+3],data[i+4],data[i+5],data[i+6],data[i+7]};#5;rx_done_toUart	<= 1;#50;rx_done_toUart	<=0;#50;end#200;$stop;
end//解析xxGGA报文
parseGGA parseGGA_inst(.rx_done_toUart	(rx_done_toUart),.rddat_toUart	(rddat_toUart),.rx_done		(rx_done),.hh				(hh),.mm				(mm),.ss				(ss),.ss2			(ss2),.lat			(lat),.lat2			(lat2),.lat3			(lat3),.NS				(NS),.lon			(lon),.lon2			(lon2),.lon3			(lon3),.EW				(EW),.alt			(alt),.alt2			(alt2)
);endmodule
//END OF parseGGA_tb.v FILE***************************************************

ModelSim仿真结果:

在这里插入图片描述

这篇关于NMEA(xxGGA)报文解析(FPGA实现)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/205498

相关文章

基于Python实现一个图片拆分工具

《基于Python实现一个图片拆分工具》这篇文章主要为大家详细介绍了如何基于Python实现一个图片拆分工具,可以根据需要的行数和列数进行拆分,感兴趣的小伙伴可以跟随小编一起学习一下... 简单介绍先自己选择输入的图片,默认是输出到项目文件夹中,可以自己选择其他的文件夹,选择需要拆分的行数和列数,可以通过

Python中将嵌套列表扁平化的多种实现方法

《Python中将嵌套列表扁平化的多种实现方法》在Python编程中,我们常常会遇到需要将嵌套列表(即列表中包含列表)转换为一个一维的扁平列表的需求,本文将给大家介绍了多种实现这一目标的方法,需要的朋... 目录python中将嵌套列表扁平化的方法技术背景实现步骤1. 使用嵌套列表推导式2. 使用itert

Python使用pip工具实现包自动更新的多种方法

《Python使用pip工具实现包自动更新的多种方法》本文深入探讨了使用Python的pip工具实现包自动更新的各种方法和技术,我们将从基础概念开始,逐步介绍手动更新方法、自动化脚本编写、结合CI/C... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核

在Linux中改变echo输出颜色的实现方法

《在Linux中改变echo输出颜色的实现方法》在Linux系统的命令行环境下,为了使输出信息更加清晰、突出,便于用户快速识别和区分不同类型的信息,常常需要改变echo命令的输出颜色,所以本文给大家介... 目python录在linux中改变echo输出颜色的方法技术背景实现步骤使用ANSI转义码使用tpu

Python使用python-can实现合并BLF文件

《Python使用python-can实现合并BLF文件》python-can库是Python生态中专注于CAN总线通信与数据处理的强大工具,本文将使用python-can为BLF文件合并提供高效灵活... 目录一、python-can 库:CAN 数据处理的利器二、BLF 文件合并核心代码解析1. 基础合

Python使用OpenCV实现获取视频时长的小工具

《Python使用OpenCV实现获取视频时长的小工具》在处理视频数据时,获取视频的时长是一项常见且基础的需求,本文将详细介绍如何使用Python和OpenCV获取视频时长,并对每一行代码进行深入解析... 目录一、代码实现二、代码解析1. 导入 OpenCV 库2. 定义获取视频时长的函数3. 打开视频文

golang版本升级如何实现

《golang版本升级如何实现》:本文主要介绍golang版本升级如何实现问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录golanwww.chinasem.cng版本升级linux上golang版本升级删除golang旧版本安装golang最新版本总结gola

PostgreSQL的扩展dict_int应用案例解析

《PostgreSQL的扩展dict_int应用案例解析》dict_int扩展为PostgreSQL提供了专业的整数文本处理能力,特别适合需要精确处理数字内容的搜索场景,本文给大家介绍PostgreS... 目录PostgreSQL的扩展dict_int一、扩展概述二、核心功能三、安装与启用四、字典配置方法

SpringBoot中SM2公钥加密、私钥解密的实现示例详解

《SpringBoot中SM2公钥加密、私钥解密的实现示例详解》本文介绍了如何在SpringBoot项目中实现SM2公钥加密和私钥解密的功能,通过使用Hutool库和BouncyCastle依赖,简化... 目录一、前言1、加密信息(示例)2、加密结果(示例)二、实现代码1、yml文件配置2、创建SM2工具

Mysql实现范围分区表(新增、删除、重组、查看)

《Mysql实现范围分区表(新增、删除、重组、查看)》MySQL分区表的四种类型(范围、哈希、列表、键值),主要介绍了范围分区的创建、查询、添加、删除及重组织操作,具有一定的参考价值,感兴趣的可以了解... 目录一、mysql分区表分类二、范围分区(Range Partitioning1、新建分区表:2、分