aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--host/include/uhd/usrp/gps_ctrl.hpp30
-rw-r--r--host/lib/usrp/b200/b200_impl.cpp15
-rw-r--r--host/lib/usrp/gps_ctrl.cpp205
3 files changed, 195 insertions, 55 deletions
diff --git a/host/include/uhd/usrp/gps_ctrl.hpp b/host/include/uhd/usrp/gps_ctrl.hpp
index 6fda04f21..0db1d5317 100644
--- a/host/include/uhd/usrp/gps_ctrl.hpp
+++ b/host/include/uhd/usrp/gps_ctrl.hpp
@@ -54,10 +54,40 @@ public:
*/
virtual bool gps_detected(void) = 0;
+ /*!
+ * Tell if the GPSDO is a LEA-M8F
+ * \return true if the LEA-M8F has been detected
+ */
+ virtual bool gps_detected_lea_m8f(void) = 0;
+
//TODO: other fun things you can do with a GPS.
};
+/*! The UBX-NAV-SOL message structure from the
+ * u-blox UBX protocol
+ */
+struct ubx_nav_sol_t
+{
+ uint32_t itow; // ms GPS Millisecond Time of Week
+ int32_t frac; // ns remainder of rounded ms above
+ int16_t week; // GPS week
+ uint8_t GPSfix; // GPSfix Type, range 0..6
+ uint8_t Flags; // Navigation Status Flags
+ int32_t ECEF_X; // cm ECEF X coordinate
+ int32_t ECEF_Y; // cm ECEF Y coordinate
+ int32_t ECEF_Z; // cm ECEF Z coordinate
+ int32_t PAcc; // cm 3D Position Accuracy Estimate
+ int32_t ECEFVX; // cm/s ECEF X velocity
+ int32_t ECEFVY; // cm/s ECEF Y velocity
+ int32_t ECEFVZ; // cm/s ECEF Z velocity
+ uint32_t SAcc; // cm/s Speed Accuracy Estimate
+ uint16_t PDOP; // 0.01 Position DOP
+ uint8_t res1; // reserved
+ uint8_t numSV; // Number of SVs used in navigation solution
+ uint32_t res2; // reserved
+} __attribute__((packed));
+
} //namespace uhd
#endif /* INCLUDED_GPS_CTRL_HPP */
diff --git a/host/lib/usrp/b200/b200_impl.cpp b/host/lib/usrp/b200/b200_impl.cpp
index ffce08567..3da402f13 100644
--- a/host/lib/usrp/b200/b200_impl.cpp
+++ b/host/lib/usrp/b200/b200_impl.cpp
@@ -704,12 +704,27 @@ b200_impl::b200_impl(const uhd::device_addr_t& device_addr, usb_device_handle::s
_radio_perifs[i].ddc->set_host_rate(default_tick_rate / ad936x_manager::DEFAULT_DECIM);
_radio_perifs[i].duc->set_host_rate(default_tick_rate / ad936x_manager::DEFAULT_INTERP);
}
+
// We can automatically choose a master clock rate, but not if the user specifies one
_tree->access<bool>(mb_path / "auto_tick_rate").set(not device_addr.has_key("master_clock_rate"));
if (not device_addr.has_key("master_clock_rate")) {
UHD_MSG(status) << "Setting master clock rate selection to 'automatic'." << std::endl;
}
+ //GPS installed: use external ref, time, and init time spec
+ if (_gps and _gps->gps_detected())
+ {
+ UHD_MSG(status) << "Setting references to the internal GPSDO" << std::endl;
+ _tree->access<std::string>(mb_path / "time_source" / "value").set("gpsdo");
+ _tree->access<std::string>(mb_path / "clock_source" / "value").set("gpsdo");
+
+ if (not _gps->gps_detected_lea_m8f()) {
+ UHD_MSG(status) << "Initializing time to the internal GPSDO" << std::endl;
+ const time_t tp = time_t(_gps->get_sensor("gps_time").to_int()+1);
+ _tree->access<time_spec_t>(mb_path / "time" / "pps").set(time_spec_t(tp));
+ }
+ }
+
}
b200_impl::~b200_impl(void)
diff --git a/host/lib/usrp/gps_ctrl.cpp b/host/lib/usrp/gps_ctrl.cpp
index 1cee8e566..b6019efbd 100644
--- a/host/lib/usrp/gps_ctrl.cpp
+++ b/host/lib/usrp/gps_ctrl.cpp
@@ -62,7 +62,7 @@ private:
return update_cached_sensors(sensor);
}
} catch(std::exception &e) {
- UHD_MSG(warning) << "get_cached_sensor: " << e.what();
+ UHD_MSG(warning) << "get_cached_sensor: " << e.what() << std::endl;
}
return std::string();
}
@@ -89,64 +89,144 @@ private:
}
std::string update_cached_sensors(const std::string sensor) {
- if(not gps_detected() || !(gps_type == GPS_TYPE_INTERNAL_GPSDO || gps_type == GPS_TYPE_LEA_M8F)) {
- UHD_MSG(error) << "get_stat(): unsupported GPS or no GPS detected";
- return std::string();
- }
+ if (gps_detected() && gps_type == GPS_TYPE_INTERNAL_GPSDO) {
+
+ const std::list<std::string> list = boost::assign::list_of("GPGGA")("GPRMC")("SERVO");
+
+ static const boost::regex status_regex("\\d\\d-\\d\\d-\\d\\d");
+ static const boost::regex gp_msg_regex("^\\$GP.*,\\*[0-9A-F]{2}$");
+ std::map<std::string,std::string> msgs;
+
+ // Get all GPSDO messages available
+ // Creating a map here because we only want the latest of each message type
+ for (std::string msg = _recv(); msg.length(); msg = _recv())
+ {
+ // Strip any end of line characters
+ erase_all(msg, "\r");
+ erase_all(msg, "\n");
+
+ if (msg.length() < 6)
+ {
+ UHD_LOGV(regularly) << __FUNCTION__ << ": Short NMEA string: " << msg << std::endl;
+ continue;
+ }
+
+ // Look for SERVO message
+ if (boost::regex_search(msg, status_regex, boost::regex_constants::match_continuous))
+ {
+ msgs["SERVO"] = msg;
+ }
+ else if (boost::regex_match(msg, gp_msg_regex) and is_nmea_checksum_ok(msg))
+ {
+ msgs[msg.substr(1,5)] = msg;
+ }
+ else
+ {
+ UHD_LOGV(regularly) << __FUNCTION__ << ": Malformed NMEA string: " << msg << std::endl;
+ }
+ }
+
+ boost::system_time time = boost::get_system_time();
+
+ // Update sensors with newly read data
+ BOOST_FOREACH(std::string key, list) {
+ if (msgs[key].length())
+ sensors[key] = boost::make_tuple(msgs[key], time, !sensor.compare(key));
+ }
- std::list<std::string> list = boost::assign::list_of("GPGGA")("GPRMC")("SERVO");
- const std::list<std::string> list_lea_m8f = boost::assign::list_of("GNGGA")("GNRMC");
+ // Return requested sensor if it was updated
+ if (msgs[sensor].length())
+ return msgs[sensor];
- if (gps_type == GPS_TYPE_LEA_M8F) {
- list = list_lea_m8f;
+ return std::string();
}
+ else if (gps_detected() && gps_type == GPS_TYPE_LEA_M8F) {
+ const std::list<std::string> list = boost::assign::list_of("GNGGA")("GNRMC")("FIXTYPE");
- static const boost::regex status_regex("\\d\\d-\\d\\d-\\d\\d");
- static const boost::regex gp_msg_regex("^\\$GP.*,\\*[0-9A-F]{2}$");
- std::map<std::string,std::string> msgs;
+ // We try to receive the some UBX messages to find out if the clock is disciplined
+ std::string msg = _recv();
- // Get all GPSDO messages available
- // Creating a map here because we only want the latest of each message type
- for (std::string msg = _recv(); msg.length(); msg = _recv())
- {
- // Strip any end of line characters
- erase_all(msg, "\r");
- erase_all(msg, "\n");
+ std::map<std::string,std::string> msgs;
- if (msg.length() < 6)
- {
- UHD_LOGV(regularly) << __FUNCTION__ << ": Short NMEA string: " << msg << std::endl;
- continue;
+ // Get all GPSDO messages available
+ // Creating a map here because we only want the latest of each message type
+ for (std::string msg = _recv(); msg.length() > 6; msg = _recv())
+ {
+ std::stringstream ss;
+ ss << "Got message ";
+ for (size_t m = 0; m < msg.size(); m++) {
+ ss << std::hex << (unsigned int)(unsigned char)msg[m] << " " << std::dec;
}
+ UHD_MSG(warning) << ss.str() << ":" << std::endl << msg << std::endl;
+ // Get UBX-NAV-SOL
+ const uint8_t nav_sol_head[4] = {0xb5, 0x62, 0x01, 0x06};
+ const std::string nav_sol_head_str(reinterpret_cast<const char *>(nav_sol_head), 4);
+ if (msg.find(nav_sol_head_str) == 0) {
+ if (msg.length() > 52 + 8) {
+ ubx_nav_sol_t nav_sol;
+ memcpy(&nav_sol, msg.c_str(), sizeof(nav_sol));
+
+ UHD_MSG(warning) << "Got NAV-SOL " << nav_sol.itow << ", "
+ << (int)nav_sol.numSV << " SVs, flags "
+ << (int)nav_sol.Flags << std::endl;
+
+ std::string fixtype;
+
+ if (nav_sol.GPSfix == 0) {
+ fixtype = "no fix";
+ }
+ else if (nav_sol.GPSfix == 1) {
+ fixtype = "dead reckoning";
+ }
+ else if (nav_sol.GPSfix == 2) {
+ fixtype = "2d fix";
+ }
+ else if (nav_sol.GPSfix == 3) {
+ fixtype = "3d fix";
+ }
+ else if (nav_sol.GPSfix == 4) {
+ fixtype = "combined fix";
+ }
+ else if (nav_sol.GPSfix == 5) {
+ fixtype = "time-only fix";
+ }
- // Look for SERVO message
- if (boost::regex_search(msg, status_regex, boost::regex_constants::match_continuous))
- {
- msgs["SERVO"] = msg;
- }
- else if (boost::regex_match(msg, gp_msg_regex) and is_nmea_checksum_ok(msg))
- {
- msgs[msg.substr(1,5)] = msg;
+ if (not fixtype.empty()) {
+ msgs["FIXTYPE"] = fixtype;
+ }
+
+ std::string next_msg = msg.substr(52+8);
+
+ UHD_MSG(warning) << "Next message " << next_msg << std::endl;
+ if (next_msg.find("$") == 0) {
+ msgs[next_msg.substr(1,5)] = next_msg;
+ }
+ }
}
- else
- {
- UHD_LOGV(regularly) << __FUNCTION__ << ": Malformed NMEA string: " << msg << std::endl;
+ else {
+ msgs[msg.substr(1,5)] = msg;
}
- }
+ }
- boost::system_time time = boost::get_system_time();
+ boost::system_time time = boost::get_system_time();
- // Update sensors with newly read data
- BOOST_FOREACH(std::string key, list) {
- if (msgs[key].length())
- sensors[key] = boost::make_tuple(msgs[key], time, !sensor.compare(key));
- }
+ // Update sensors with newly read data
+ BOOST_FOREACH(std::string key, list) {
+ if (msgs[key].length()) {
+ sensors[key] = boost::make_tuple(msgs[key], time, !sensor.compare(key));
+ }
+ }
- // Return requested sensor if it was updated
- if (msgs[sensor].length())
+ // Return requested sensor if it was updated
+ if (msgs[sensor].length())
return msgs[sensor];
- return std::string();
+ return std::string();
+ }
+ else {
+ UHD_MSG(error) << "get_stat(): unsupported GPS or no GPS detected" << std::endl;
+ return std::string();
+ }
}
public:
@@ -162,6 +242,9 @@ public:
_flush(); //get whatever junk is in the rx buffer right now, and throw it away
_send("HAAAY GUYYYYS\n"); //to elicit a response from the GPSDO
+ // try to init LEA-M8F
+ init_lea_m8f();
+
//wait for _send(...) to return
sleep(milliseconds(GPSDO_COMMAND_DELAY_MS));
@@ -169,14 +252,16 @@ public:
const boost::system_time comm_timeout = boost::get_system_time() + milliseconds(GPS_COMM_TIMEOUT_MS);
while(boost::get_system_time() < comm_timeout) {
reply = _recv();
+ UHD_MSG(warning) << "Received " << reply << std::endl;
if(reply.find("Command Error") != std::string::npos) {
gps_type = GPS_TYPE_INTERNAL_GPSDO;
break;
}
else if(reply.substr(0, 3) == "$GP") i_heard_some_nmea = true; //but keep looking for that "Command Error" response
- else if(reply.substr(0, 3) == "$GN") {
- // The u-blox LEA-M8F outputs $GNGGA and $GNRMC messages
+ else if(reply.substr(0, 2) == "\xB5""\x62") {
+ // The u-blox LEA-M8F outputs UBX protocol messages
gps_type = GPS_TYPE_LEA_M8F;
+ i_heard_some_nmea = false;
break;
}
else if(reply.length() != 0) i_heard_something_weird = true; //probably wrong baud rate
@@ -224,7 +309,8 @@ public:
("gps_gnrmc")
("gps_time")
("gps_locked")
- ("gps_servo");
+ ("gps_servo")
+ ("gps_fixtype");
return ret;
}
@@ -232,7 +318,8 @@ public:
if(key == "gps_gpgga"
or key == "gps_gprmc"
or key == "gps_gngga"
- or key == "gps_gnrmc") {
+ or key == "gps_gnrmc"
+ or key == "gps_fixtype" ) {
return sensor_value_t(
boost::to_upper_copy(key),
get_cached_sensor(boost::to_upper_copy(key.substr(4,8)), GPS_NMEA_NORMAL_FRESHNESS, false, false),
@@ -273,7 +360,9 @@ private:
}
void init_lea_m8f(void) {
- // Nothing much to do yet
+ // Enable the UBX-NAV-SOL message:
+ const uint8_t en_nav_sol[11] = {0xb5, 0x62, 0x06, 0x01, 0x03, 0x00, 0x01, 0x06, 0x01, 0x12, 0x4f};
+ _send(std::string(reinterpret_cast<const char *>(en_nav_sol), 11));
}
//retrieve a raw NMEA sentence
@@ -343,7 +432,7 @@ private:
return gps_time;
} catch(std::exception &e) {
- UHD_MSG(warning) << "get_time: " << e.what();
+ UHD_MSG(warning) << "get_time: " << e.what() << std::endl;
_flush();
error_cnt++;
}
@@ -357,6 +446,10 @@ private:
return (get_time() - from_time_t(0)).total_seconds();
}
+ bool gps_detected_lea_m8f(void) {
+ return (gps_type == GPS_TYPE_LEA_M8F);
+ }
+
bool gps_detected(void) {
return (gps_type != GPS_TYPE_NONE);
}
@@ -367,16 +460,18 @@ private:
try {
std::string reply;
if (gps_type == GPS_TYPE_LEA_M8F) {
- reply = get_cached_sensor("GNGGA", GPS_LOCK_FRESHNESS, false, false);
+ reply = get_cached_sensor("FIXTYPE", GPS_LOCK_FRESHNESS, false, false);
+ UHD_MSG(warning) << "FIXTYPE is " << reply << std::endl;
+ return reply == "3d fix";
}
else {
reply = get_cached_sensor("GPGGA", GPS_LOCK_FRESHNESS, false, false);
+ if(reply.size() <= 1) return false;
+ return (get_token(reply, 6) != "0");
}
- if(reply.size() <= 1) return false;
- return (get_token(reply, 6) != "0");
} catch(std::exception &e) {
- UHD_MSG(warning) << "locked: " << e.what();
+ UHD_MSG(warning) << "locked: " << e.what() << std::endl;
error_cnt++;
}
}
@@ -429,7 +524,7 @@ private:
GPS_TYPE_NONE
} gps_type;
- static const int GPS_COMM_TIMEOUT_MS = 1300;
+ static const int GPS_COMM_TIMEOUT_MS = 2300;
static const int GPS_NMEA_FRESHNESS = 10;
static const int GPS_NMEA_LOW_FRESHNESS = 2500;
static const int GPS_NMEA_NORMAL_FRESHNESS = 1000;