NEO—6M/7M GPS模块,具有高灵敏度、低功耗、小型化、高追踪灵敏度,大大扩大了其定位的覆盖面,在普通GPS接收模块不能定位的地方,如狭窄都市天空下、密集的丛林环境,NEO-6M都能高精度定位。模块的高灵敏度、小静态漂移、低功耗及轻巧的体积,适用于车载、手持设备如PDA,车辆监控、手机、摄像机及其他移动定位系统的应用,是GPS产品应用的好选择。
一、模块来源
模块实物展示:
资料链接:https://pan.baidu.com/s/1QvwMg9JbkzFauYmwExWcnQ
资料提取码:8888
二、规格参数
工作电压:3.3V-5V
工作电流:10-26mA
模块尺寸:
控制方式:SPI
三、移植过程
我们的目标是在立创·CW32F030C8T6开发板上能够获取到定位信息的功能。首先要获取资料,查看数据手册应如何实现,再移植至我们的工程。
3.1查看资料
注:在主电源断开后,后备电池可以维持半小时左右的GPS星历数据的保存,以支持温启动或热启动,从而实现快速定位。
首次定位时间较长,请确保是在室外进行定位。
3.2引脚选择
想要使用uart串口,需要确定使用的引脚是否有串口外设功能,可以通过用户手册进行查看。在用户手册的第146页。
这里选择使用PA2和PA3的附加串口2功能。
有串口功能的引脚
模块接线图
3.3移植至工程
移植步骤中的导入.c和.h文件与【CW32模块使用】DHT11温湿度传感器相同,只是将.c和.h文件更改为bsp_gps.c与bsp_gps.h。这里不再过多讲述,移植完成后面修改相关代码。
在文件bsp_gps.c中,编写如下代码。
/* * Change Logs: * Date Author Notes * 2024-06-21 LCKFB-LP first version */ #include "bsp_gps.h" #include "stdio.h" #include "stdlib.h" #include "string.h" #define GPSRX_LEN_MAX 255 unsigned char GPSRX_BUFF[GPSRX_LEN_MAX]; unsigned char GPSRX_LEN = 0; _SaveData Save_Data; /****************************************************************** * 函 数 名 称:GPS_GPIO_Init * 函 数 说 明:GPS引脚初始化 * 函 数 形 参:band_rate GPS通信波特率 * 函 数 返 回:无 * 作 者:LC * 备 注:默认波特率为9600 ******************************************************************/ void GPS_GPIO_Init(uint32_t band_rate) { GPIO_InitTypeDef GPIO_InitStruct; // GPIO初始化结构体 BSP_GPS_GPIO_RCC_ENABLE(); // 使能GPIO时钟 BSP_GPS_UART_RCC_ENABLE(); // 使能UART时钟 GPIO_InitStruct.Pins = BSP_GPS_TX_PIN; // GPIO引脚 GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; // 推挽输出 GPIO_InitStruct.Speed = GPIO_SPEED_HIGH; // 输出速度高 GPIO_Init(BSP_GPS_GPIO_PORT, &GPIO_InitStruct); // 初始化 GPIO_InitStruct.Pins = BSP_GPS_RX_PIN; // GPIO引脚 GPIO_InitStruct.Mode = GPIO_MODE_INPUT_PULLUP; // 上拉输入 GPIO_Init(BSP_GPS_GPIO_PORT, &GPIO_InitStruct); // 初始化 BSP_GPS_AF_UART_TX(); // UART_TX复用 BSP_GPS_AF_UART_RX(); // UART_RX复用 // 配置UART USART_InitTypeDef USART_InitStructure; USART_InitStructure.USART_BaudRate = band_rate; // 波特率 USART_InitStructure.USART_Over = USART_Over_16; // 配置USART的过采样率。 USART_InitStructure.USART_Source = USART_Source_PCLK; // 设置时钟源 USART_InitStructure.USART_UclkFreq = 64000000; //设置USART时钟频率(和主频一致即可) USART_InitStructure.USART_StartBit = USART_StartBit_FE; //RXD下降沿开始 USART_InitStructure.USART_StopBits = USART_StopBits_1; // 停止位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(BSP_GPS_USART, &USART_InitStructure); // 初始化串口2 // 优先级,无优先级分组 NVIC_SetPriority(BSP_GPS_IRQn, 0); // UARTx中断使能 NVIC_EnableIRQ(BSP_GPS_IRQn); // 使能UARTx RC中断 USART_ITConfig(BSP_GPS_USART, USART_IT_RC, ENABLE); } /****************************************************************** * 函 数 名 称:GPS_Send_Bit * 函 数 说 明:向GPS发送单个字符 * 函 数 形 参:ch发送的字符 * 函 数 返 回:无 * 作 者:LC * 备 注:无 ******************************************************************/ void GPS_Send_Bit(unsigned char ch) { // 发送一个字节 USART_SendData_8bit(BSP_GPS_USART, (uint8_t)ch); // 等待发送完成 while( RESET == USART_GetFlagStatus(BSP_GPS_USART, USART_FLAG_TXE) ){} } /****************************************************************** * 函 数 名 称:GPS_send_String * 函 数 说 明:GPS发送字符串 * 函 数 形 参:str要发送的字符串 * 函 数 返 回:无 * 作 者:LC * 备 注:无 ******************************************************************/ void GPS_send_String(unsigned char *str) { while( str && *str ) // 地址为空或者值为空跳出 { GPS_Send_Bit(*str++); } } /****************************************************************** * 函 数 名 称:Hand * 函 数 说 明:在GPS数据中,识别是否有想要的串口命令 * 函 数 形 参:需要识别的命令 * 函 数 返 回:1识别成功 0识别失败 * 作 者:LC * 备 注:无 ******************************************************************/ uint8_t Hand(char *a) { if(strstr((const char*)GPSRX_BUFF,a)!=NULL) return 1; else return 0; } /****************************************************************** * 函 数 名 称:CLR_Buf * 函 数 说 明:清除串口接收的数据 * 函 数 形 参:无 * 函 数 返 回:无 * 作 者:LC * 备 注:无 ******************************************************************/ void CLR_Buf(void) { memset(GPSRX_BUFF, 0, GPSRX_LEN_MAX); //清空 GPSRX_LEN = 0; } /****************************************************************** * 函 数 名 称:clrStruct * 函 数 说 明:清除GPS结构体数据 * 函 数 形 参:无 * 函 数 返 回:无 * 作 者:LC * 备 注:无 ******************************************************************/ void clrStruct(void) { Save_Data.isGetData = 0; Save_Data.isParseData = 0; Save_Data.isUsefull = 0; memset(Save_Data.GPS_Buffer, 0, GPS_Buffer_Length); //清空 memset(Save_Data.UTCTime, 0, UTCTime_Length); memset(Save_Data.latitude, 0, latitude_Length); memset(Save_Data.N_S, 0, N_S_Length); memset(Save_Data.longitude, 0, longitude_Length); memset(Save_Data.E_W, 0, E_W_Length); } /****************************************************************** * 函 数 名 称:BSP_GPS_IRQHandler * 函 数 说 明:串口中断服务函数 * 函 数 形 参:无 * 函 数 返 回:无 * 作 者:LC * 备 注:无 ******************************************************************/ void BSP_GPS_IRQHandler(void) { uint8_t Res; if(USART_GetITStatus(BSP_GPS_USART,USART_IT_RC) != RESET) // 接收缓冲区不为空 { Res = USART_ReceiveData(BSP_GPS_USART); if(Res == '$') { GPSRX_LEN = 0; } GPSRX_BUFF[GPSRX_LEN++] = Res; if(GPSRX_BUFF[0] == '$' && GPSRX_BUFF[4] == 'M' && GPSRX_BUFF[5] == 'C')//确定是否收到"GPRMC/GNRMC"这一帧数据 { if(Res == 'n') { memset(Save_Data.GPS_Buffer, 0, GPS_Buffer_Length); //清空 memcpy(Save_Data.GPS_Buffer, GPSRX_BUFF, GPSRX_LEN); //保存数据 Save_Data.isGetData = 1; GPSRX_LEN = 0; memset(GPSRX_BUFF, 0, GPSRX_LEN_MAX); //清空 } } if(GPSRX_LEN >= GPSRX_LEN_MAX) { GPSRX_LEN = GPSRX_LEN_MAX; } } USART_ClearITPendingBit(BSP_GPS_USART, USART_IT_RC); //已经处理就清楚标志位 }
在文件bsp_gps.h中,编写如下代码。
/* * Change Logs: * Date Author Notes * 2024-06-21 LCKFB-LP first version */ #ifndef _BSP_GPS_H #define _BSP_GPS_H #include "board.h" #define BSP_GPS_GPIO_RCC_ENABLE() __RCC_GPIOA_CLK_ENABLE() // 串口TX的端口时钟 #define BSP_GPS_UART_RCC_ENABLE() __RCC_UART2_CLK_ENABLE() // 串口2的时钟 #define BSP_GPS_AF_UART_TX() PA02_AFx_UART2TXD() // GPIO的引脚复用 #define BSP_GPS_AF_UART_RX() PA03_AFx_UART2RXD() // GPIO的引脚复用 #define BSP_GPS_GPIO_PORT CW_GPIOA // 串口TX的端口 #define BSP_GPS_TX_PIN GPIO_PIN_2 // 串口TX的引脚 #define BSP_GPS_RX_PIN GPIO_PIN_3 // 串口RX的引脚 #define BSP_GPS_USART CW_UART2 // 串口2 #define BSP_GPS_IRQn UART2_IRQn #define BSP_GPS_IRQHandler UART2_IRQHandler //定义数组长度 #define GPS_Buffer_Length 80 #define UTCTime_Length 11 #define latitude_Length 11 #define N_S_Length 2 #define longitude_Length 12 #define E_W_Length 2 typedef struct SaveData { char GPS_Buffer[GPS_Buffer_Length]; char isGetData; //是否获取到GPS数据 char isParseData; //是否解析完成 char UTCTime[UTCTime_Length]; //UTC时间 char latitude[latitude_Length]; //纬度 char N_S[N_S_Length]; //N/S char longitude[longitude_Length]; //经度 char E_W[E_W_Length]; //E/W char isUsefull; //定位信息是否有效 } _SaveData; extern _SaveData Save_Data; void GPS_GPIO_Init(uint32_t band_rate); void CLR_Buf(void); uint8_t Hand(char *a); void clrStruct(void); #endif
四、移植验证
在自己工程中的main主函数中,编写如下。
/* * Change Logs: * Date Author Notes * 2024-06-21 LCKFB-LP first version */ #include "board.h" #include "stdio.h" #include "bsp_uart.h" #include "bsp_gps.h" void parseGpsBuffer(void); void printGpsBuffer(void); int32_t main(void) { board_init(); uart1_init(115200U); GPS_GPIO_Init(9600U); clrStruct(); printf("startrn"); while(1) { parseGpsBuffer(); printGpsBuffer(); } } /****************************************************************** * 函 数 名 称:errorLog * 函 数 说 明:错误日志打印 * 函 数 形 参:num 要输出的错误码 * 函 数 返 回:无 * 作 者:LC * 备 注:无 ******************************************************************/ void errorLog(int num) { while (1) { printf("ERROR%drn",num); } } /****************************************************************** * 函 数 名 称:parseGpsBuffer * 函 数 说 明:解析GPS发送过来的数据 * 函 数 形 参:无 * 函 数 返 回:无 * 作 者:LC * 备 注:无 ******************************************************************/ void parseGpsBuffer(void) { char *subString; char *subStringNext; char i = 0; if (Save_Data.isGetData) { Save_Data.isGetData = 0; printf("**************rn"); printf("%srn",Save_Data.GPS_Buffer); for (i = 0 ; i <= 6 ; i++) { if (i == 0) { if ((subString = strstr(Save_Data.GPS_Buffer, ",")) == NULL) errorLog(1); //解析错误 } else { subString++; if ((subStringNext = strstr(subString, ",")) != NULL) { char usefullBuffer[2]; switch(i) { case 1:memcpy(Save_Data.UTCTime, subString, subStringNext - subString);break; //获取UTC时间 case 2:memcpy(usefullBuffer, subString, subStringNext - subString);break; //获取UTC时间 case 3:memcpy(Save_Data.latitude, subString, subStringNext - subString);break; //获取纬度信息 case 4:memcpy(Save_Data.N_S, subString, subStringNext - subString);break; //获取N/S case 5:memcpy(Save_Data.longitude, subString, subStringNext - subString);break; //获取经度信息 case 6:memcpy(Save_Data.E_W, subString, subStringNext - subString);break; //获取E/W default:break; } subString = subStringNext; Save_Data.isParseData = 1; if(usefullBuffer[0] == 'A') Save_Data.isUsefull = 1; else if(usefullBuffer[0] == 'V') Save_Data.isUsefull = 0; } else { errorLog(2); //解析错误 } } } } } /****************************************************************** * 函 数 名 称:printGpsBuffer * 函 数 说 明:输出解析后的数据 * 函 数 形 参:无 * 函 数 返 回:无 * 作 者:LC * 备 注:无 ******************************************************************/ void printGpsBuffer(void) { // uint8_t buff[100]={0}; if (Save_Data.isParseData) { Save_Data.isParseData = 0; //在广东深圳进行测量,发现UTC时间存在8小时误差 printf("Save_Data.UTCTime = "); printf("%s",Save_Data.UTCTime); printf("rn"); if(Save_Data.isUsefull) { Save_Data.isUsefull = 0; //串口显示纬度 printf("Save_Data.latitude = "); printf("%s",Save_Data.latitude); //串口显示 printf("Save_Data.N_S = "); printf("%srn",Save_Data.N_S); //串口显示经度 printf("Save_Data.longitude = "); printf("%s",Save_Data.longitude); printf("rn"); //串口显示 printf("Save_Data.E_W = "); printf("%s",Save_Data.E_W); printf("rn"); } else { printf("GPS DATA is not usefull!rn"); } } }
移植现象:
注意室内大概率无法定位,所以最好外接屏幕去空旷地带测试!
模块移植成功案例代码:
链接:https://pan.baidu.com/s/1Zqe1B8ncdZA2cpboYM-kgA?pwd=LCKF
提取码:LCKF
审核编辑 黄宇
全部0条评论
快来发表一下你的评论吧 !