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

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

一文解析C#中的StringSplitOptions枚举

《一文解析C#中的StringSplitOptions枚举》StringSplitOptions是C#中的一个枚举类型,用于控制string.Split()方法分割字符串时的行为,核心作用是处理分割后... 目录C#的StringSplitOptions枚举1.StringSplitOptions枚举的常用

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

Python实现字典转字符串的五种方法

《Python实现字典转字符串的五种方法》本文介绍了在Python中如何将字典数据结构转换为字符串格式的多种方法,首先可以通过内置的str()函数进行简单转换;其次利用ison.dumps()函数能够... 目录1、使用json模块的dumps方法:2、使用str方法:3、使用循环和字符串拼接:4、使用字符