177 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			177 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|   Send UBX binary commands to enable RTCM sentences on Ublox ZED-F9P module
 | |
|   By: Nathan Seidle
 | |
|   SparkFun Electronics
 | |
|   Date: January 9th, 2019
 | |
|   License: MIT. See license file for more information but you can
 | |
|   basically do whatever you want with this code.
 | |
| 
 | |
|   This example does all steps to configure and enable a ZED-F9P as a base station:
 | |
|     Begin Survey-In
 | |
|     Once we've achieved 2m accuracy and 300s have passed, survey is complete
 | |
|     Enable six RTCM messages
 | |
|     Begin outputting RTCM bytes
 | |
| 
 | |
|   Feel like supporting open source hardware?
 | |
|   Buy a board from SparkFun!
 | |
|   ZED-F9P RTK2: https://www.sparkfun.com/products/15136
 | |
|   NEO-M8P RTK: https://www.sparkfun.com/products/15005
 | |
|   SAM-M8Q: https://www.sparkfun.com/products/15106
 | |
| 
 | |
|   Hardware Connections:
 | |
|   Plug a Qwiic cable into the GPS and a BlackBoard
 | |
|   If you don't have a platform with a Qwiic connection use the SparkFun Qwiic Breadboard Jumper (https://www.sparkfun.com/products/14425)
 | |
|   Open the serial monitor at 115200 baud to see the output
 | |
| */
 | |
| 
 | |
| #include <Wire.h> //Needed for I2C to GPS
 | |
| 
 | |
| #include "SparkFun_Ublox_Arduino_Library.h" //http://librarymanager/All#SparkFun_u-blox_GNSS
 | |
| SFE_UBLOX_GPS myGPS;
 | |
| 
 | |
| //#define USE_SERIAL1 // Uncomment this line to push the RTCM data to Serial1
 | |
| 
 | |
| void setup()
 | |
| {
 | |
|   Serial.begin(115200);
 | |
|   while (!Serial); //Wait for user to open terminal
 | |
|   Serial.println("Ublox Base station example");
 | |
| 
 | |
| #ifdef USE_SERIAL1
 | |
|   // If our board supports it, we can output the RTCM data on Serial1
 | |
|   Serial1.begin(115200);
 | |
| #endif
 | |
| 
 | |
|   Wire.begin();
 | |
|   Wire.setClock(400000); //Increase I2C clock speed to 400kHz
 | |
| 
 | |
|   if (myGPS.begin() == false) //Connect to the Ublox module using Wire port
 | |
|   {
 | |
|     Serial.println(F("Ublox GPS not detected at default I2C address. Please check wiring. Freezing."));
 | |
|     while (1);
 | |
|   }
 | |
| 
 | |
|   // Uncomment the next line if you want to reset your module back to the default settings with 1Hz navigation rate
 | |
|   //myGPS.factoryDefault(); delay(5000);
 | |
| 
 | |
|   myGPS.setI2COutput(COM_TYPE_UBX); //Set the I2C port to output UBX only (turn off NMEA noise)
 | |
|   myGPS.saveConfigSelective(VAL_CFG_SUBSEC_IOPORT); //Save the communications port settings to flash and BBR
 | |
| 
 | |
|   while (Serial.available()) Serial.read(); //Clear any latent chars in serial buffer
 | |
|   Serial.println("Press any key to send commands to begin Survey-In");
 | |
|   while (Serial.available() == 0) ; //Wait for user to press a key
 | |
| 
 | |
|   boolean response = true;
 | |
|   response &= myGPS.enableRTCMmessage(UBX_RTCM_1005, COM_PORT_I2C, 1); //Enable message 1005 to output through I2C port, message every second
 | |
|   response &= myGPS.enableRTCMmessage(UBX_RTCM_1074, COM_PORT_I2C, 1);
 | |
|   response &= myGPS.enableRTCMmessage(UBX_RTCM_1084, COM_PORT_I2C, 1);
 | |
|   response &= myGPS.enableRTCMmessage(UBX_RTCM_1094, COM_PORT_I2C, 1);
 | |
|   response &= myGPS.enableRTCMmessage(UBX_RTCM_1124, COM_PORT_I2C, 1);
 | |
|   response &= myGPS.enableRTCMmessage(UBX_RTCM_1230, COM_PORT_I2C, 10); //Enable message every 10 seconds
 | |
| 
 | |
|   //Use COM_PORT_UART1 for the above six messages to direct RTCM messages out UART1
 | |
|   //COM_PORT_UART2, COM_PORT_USB, COM_PORT_SPI are also available
 | |
|   //For example: response &= myGPS.enableRTCMmessage(UBX_RTCM_1005, COM_PORT_UART1, 10);
 | |
| 
 | |
|   if (response == true)
 | |
|   {
 | |
|     Serial.println("RTCM messages enabled");
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     Serial.println("RTCM failed to enable. Are you sure you have an ZED-F9P?");
 | |
|     while (1); //Freeze
 | |
|   }
 | |
| 
 | |
|   //Check if Survey is in Progress before initiating one
 | |
|   response = myGPS.getSurveyStatus(2000); //Query module for SVIN status with 2000ms timeout (request can take a long time)
 | |
|   if (response == false)
 | |
|   {
 | |
|     Serial.println("Failed to get Survey In status");
 | |
|     while (1); //Freeze
 | |
|   }
 | |
| 
 | |
|   if (myGPS.svin.active == true)
 | |
|   {
 | |
|     Serial.print("Survey already in progress.");
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     //Start survey
 | |
|     //The ZED-F9P is slightly different than the NEO-M8P. See the Integration manual 3.5.8 for more info.
 | |
|     //response = myGPS.enableSurveyMode(300, 2.000); //Enable Survey in on NEO-M8P, 300 seconds, 2.0m
 | |
|     response = myGPS.enableSurveyMode(60, 5.000); //Enable Survey in, 60 seconds, 5.0m
 | |
|     if (response == false)
 | |
|     {
 | |
|       Serial.println("Survey start failed");
 | |
|       while (1);
 | |
|     }
 | |
|     Serial.println("Survey started. This will run until 60s has passed and less than 5m accuracy is achieved.");
 | |
|   }
 | |
| 
 | |
|   while(Serial.available()) Serial.read(); //Clear buffer
 | |
|   
 | |
|   //Begin waiting for survey to complete
 | |
|   while (myGPS.svin.valid == false)
 | |
|   {
 | |
|     if(Serial.available())
 | |
|     {
 | |
|       byte incoming = Serial.read();
 | |
|       if(incoming == 'x')
 | |
|       {
 | |
|         //Stop survey mode
 | |
|         response = myGPS.disableSurveyMode(); //Disable survey
 | |
|         Serial.println("Survey stopped");
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|     
 | |
|     response = myGPS.getSurveyStatus(2000); //Query module for SVIN status with 2000ms timeout (req can take a long time)
 | |
|     if (response == true)
 | |
|     {
 | |
|       Serial.print("Press x to end survey - ");
 | |
|       Serial.print("Time elapsed: ");
 | |
|       Serial.print((String)myGPS.svin.observationTime);
 | |
| 
 | |
|       Serial.print(" Accuracy: ");
 | |
|       Serial.print((String)myGPS.svin.meanAccuracy);
 | |
|       Serial.println();
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|       Serial.println("SVIN request failed");
 | |
|     }
 | |
| 
 | |
|     delay(1000);
 | |
|   }
 | |
|   Serial.println("Survey valid!");
 | |
| 
 | |
|   Serial.println("Base survey complete! RTCM now broadcasting.");
 | |
| 
 | |
|   myGPS.setI2COutput(COM_TYPE_UBX | COM_TYPE_RTCM3); //Set the I2C port to output UBX and RTCM sentences (not really an option, turns on NMEA as well)
 | |
| }
 | |
| 
 | |
| void loop()
 | |
| {
 | |
|   myGPS.checkUblox(); //See if new data is available. Process bytes as they come in.
 | |
| 
 | |
|   delay(250); //Don't pound too hard on the I2C bus
 | |
| }
 | |
| 
 | |
| //This function gets called from the SparkFun Ublox Arduino Library.
 | |
| //As each RTCM byte comes in you can specify what to do with it
 | |
| //Useful for passing the RTCM correction data to a radio, Ntrip broadcaster, etc.
 | |
| void SFE_UBLOX_GPS::processRTCM(uint8_t incoming)
 | |
| {
 | |
| #ifdef USE_SERIAL1
 | |
|   //Push the RTCM data to Serial1
 | |
|   Serial1.write(incoming);
 | |
| #endif
 | |
| 
 | |
|   //Pretty-print the HEX values to Serial
 | |
|   if (myGPS.rtcmFrameCounter % 16 == 0) Serial.println();
 | |
|   Serial.print(" ");
 | |
|   if (incoming < 0x10) Serial.print("0");
 | |
|   Serial.print(incoming, HEX);
 | |
| }
 |