aboutsummaryrefslogtreecommitdiffstats
path: root/host/lib/usrp/gpsd_iface.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'host/lib/usrp/gpsd_iface.cpp')
-rw-r--r--host/lib/usrp/gpsd_iface.cpp313
1 files changed, 0 insertions, 313 deletions
diff --git a/host/lib/usrp/gpsd_iface.cpp b/host/lib/usrp/gpsd_iface.cpp
deleted file mode 100644
index ab9ae4163..000000000
--- a/host/lib/usrp/gpsd_iface.cpp
+++ /dev/null
@@ -1,313 +0,0 @@
-//
-// Copyright 2015-2016 Ettus Research LLC
-// Copyright 2018 Ettus Research, a National Instruments Company
-//
-// SPDX-License-Identifier: GPL-3.0-or-later
-//
-
-#include <cmath>
-#include <stdint.h>
-
-#include <gps.h>
-
-#include <boost/bind.hpp>
-#include "boost/date_time/gregorian/gregorian.hpp"
-#include <boost/format.hpp>
-#include <boost/thread/shared_mutex.hpp>
-#include <boost/thread.hpp>
-#include <boost/math/special_functions/fpclassify.hpp>
-
-#include <uhd/exception.hpp>
-#include <uhd/usrp/gps_ctrl.hpp>
-#include <uhd/utils/log.hpp>
-#include <uhd/types/dict.hpp>
-
-#include "gpsd_iface.hpp"
-
-namespace uhd { namespace usrp {
-
-static const size_t TIMEOUT = 240;
-static const size_t CLICK_RATE = 250000;
-
-class gpsd_iface_impl : public virtual gpsd_iface {
-public:
- gpsd_iface_impl(const std::string &addr, uint16_t port)
- : _detected(false), _bthread(), _timeout_cnt(0)
- {
- boost::unique_lock<boost::shared_mutex> l(_d_mutex);
-
- if (gps_open(addr.c_str(),
- str(boost::format("%u") % port).c_str(),
- &_gps_data) < 0) {
- throw uhd::runtime_error(
- str((boost::format("Failed to connect to gpsd: %s")
- % gps_errstr(errno))));
- }
-
- // register for updates, we don't specify a specific device,
- // therefore no WATCH_DEVICE
- gps_stream(&_gps_data, WATCH_ENABLE, NULL);
-
- // create background thread talking to gpsd
- boost::thread t(boost::bind(&gpsd_iface_impl::_thread_fcn ,this));
- _bthread.swap(t);
-
- _sensors = {
- "gps_locked",
- "gps_time",
- "gps_position",
- "gps_gpgga",
- "gps_gprmc"
- };
- }
-
- virtual ~gpsd_iface_impl(void)
- {
- // interrupt the background thread and wait for it to finish
- _bthread.interrupt();
- _bthread.join();
-
- // clean up ...
- {
- boost::unique_lock<boost::shared_mutex> l(_d_mutex);
-
- gps_stream(&_gps_data, WATCH_DISABLE, NULL);
- gps_close(&_gps_data);
- }
- }
-
- uhd::sensor_value_t get_sensor(std::string key)
- {
- if (key == "gps_locked") {
- return sensor_value_t(
- "GPS lock status", _gps_locked(), "locked", "unlocked");
- } else if (key == "gps_time") {
- return sensor_value_t(
- "GPS epoch time", int(_epoch_time()), "seconds");
- } else if (key == "gps_gpgga") {
- return sensor_value_t(
- "GPGGA", _gps_gpgga(), "");
- } else if (key == "gps_gprmc") {
- return sensor_value_t(
- "GPRMC", _gps_gprmc(), "");
- } else if (key == "gps_position") {
- return sensor_value_t(
- "GPS Position", str(
- boost::format("%s %s %s")
- % _gps_position()["lat"]
- % _gps_position()["lon"]
- % _gps_position()["alt"]), "lat/lon/alt");
- } else
- throw uhd::key_error(
- str(boost::format("sensor %s unknown.") % key));
- }
-
- bool gps_detected(void) { return _detected; };
-
- std::vector<std::string> get_sensors(void) { return _sensors; };
-
-private: // member functions
- void _thread_fcn()
- {
- while (not boost::this_thread::interruption_requested()) {
- if (!gps_waiting(&_gps_data, CLICK_RATE)) {
- if (TIMEOUT < _timeout_cnt++)
- _detected = false;
- } else {
- boost::unique_lock<boost::shared_mutex> l(_d_mutex);
-
- _timeout_cnt = 0;
- _detected = true;
-
-#if GPSD_API_MAJOR_VERSION < 7
- if (gps_read(&_gps_data) < 0)
-#else
- if (gps_read(&_gps_data, NULL, 0) < 0)
-#endif
- throw std::runtime_error("error while reading");
- }
- }
- }
-
- bool _gps_locked(void)
- {
- boost::shared_lock<boost::shared_mutex> l(_d_mutex);
- return _gps_data.fix.mode >= MODE_2D;
- }
-
- int64_t _epoch_time(void)
- {
- boost::shared_lock<boost::shared_mutex> l(_d_mutex);
- return (boost::posix_time::from_time_t(_gps_data.fix.time)
- - boost::posix_time::from_time_t(0)).total_seconds();
- }
-
- boost::gregorian::date _gregorian_date(void)
- {
- boost::shared_lock<boost::shared_mutex> l(_d_mutex);
- return boost::posix_time::from_time_t(_gps_data.fix.time).date();
- }
-
- uhd::dict<std::string, std::string> _gps_position(void)
- {
- boost::shared_lock<boost::shared_mutex> l(_d_mutex);
-
- uhd::dict<std::string, std::string> tmp;
- if (_gps_data.fix.mode >= MODE_2D) {
- tmp["lon"] = str(boost::format("%f deg")
- % _gps_data.fix.longitude);
- tmp["lat"] = str(boost::format("%f deg")
- % _gps_data.fix.latitude);
- tmp["alt"] = str(boost::format("%fm")
- % _gps_data.fix.altitude);
- } else {
- tmp["lon"] = "n/a";
- tmp["lat"] = "n/a";
- tmp["alt"] = "n/a";
- }
- return tmp;
- }
-
- float _zeroize(float x)
- {
- return boost::math::isnan(x) ? 0.0 : x;
- }
-
- int _nmea_checksum(const std::string &s)
- {
- if ((s.at(0) != '$'))
- return 0;
-
- uint8_t sum = '\0';
- for (size_t i = 1; i < s.size(); i++)
- sum ^= static_cast<uint8_t>(s.at(i));
-
- return sum;
- }
-
- double _deg_to_dm(double angle)
- {
- double fraction, integer;
- fraction = std::modf(angle, &integer);
- return std::floor(angle) * 100 + fraction * 60;
- }
-
- std::string _gps_gprmc(void)
- {
- struct tm tm;
- time_t intfixtime;
-
- boost::shared_lock<boost::shared_mutex> l(_d_mutex);
-
- tm.tm_mday = tm.tm_mon = tm.tm_year = 0;
- tm.tm_hour = tm.tm_min = tm.tm_sec = 0;
-
- if (boost::math::isnan(_gps_data.fix.time) == 0) {
- intfixtime = (time_t) _gps_data.fix.time;
- (void)gmtime_r(&intfixtime, &tm);
- tm.tm_mon++;
- tm.tm_year %= 100;
- }
- std::string string = str(boost::format(
- "$GPRMC,%02d%02d%02d,%c,%09.4f,%c,%010.4f,%c,%.4f,%.3f,%02d%02d%02d,,")
- % tm.tm_hour
- % tm.tm_min
- % tm.tm_sec
- % (_gps_data.status ? 'A' : 'V')
- % _zeroize(_deg_to_dm(std::abs(_gps_data.fix.latitude)))
- % ((_gps_data.fix.latitude > 0) ? 'N' : 'S')
- % _zeroize(_deg_to_dm(std::abs(_gps_data.fix.longitude)))
- % ((_gps_data.fix.longitude > 0) ? 'E' : 'W')
- % _zeroize(_gps_data.fix.speed * MPS_TO_KNOTS)
- % _zeroize(_gps_data.fix.track)
- % tm.tm_mday % tm.tm_mon % tm.tm_year);
-
- string.append(str(
- boost::format("*%02X") % _nmea_checksum(string)));
-
- return string;
- }
-
- std::string _gps_gpgga(void)
- {
- struct tm tm;
- time_t intfixtime;
-
- // currently not supported, make it blank
- float mag_var = NAN;
-
- boost::shared_lock<boost::shared_mutex> l(_d_mutex);
-
- intfixtime = (time_t) _gps_data.fix.time;
- (void) gmtime_r(&intfixtime, &tm);
-
- std::string string = str(boost::format(
- "$GPGGA,%02d%02d%02d,%09.4f,%c,%010.4f,%c,%d,%02d,")
- % tm.tm_hour
- % tm.tm_min
- % tm.tm_sec
- % _zeroize(_deg_to_dm(std::abs(_gps_data.fix.latitude)))
- % ((_gps_data.fix.latitude > 0) ? 'N' : 'S')
- % _zeroize(_deg_to_dm(std::abs(_gps_data.fix.longitude)))
- % ((_gps_data.fix.longitude > 0) ? 'E' : 'W')
- % _gps_data.status
- % _gps_data.satellites_used);
-
- if (boost::math::isnan(_gps_data.dop.hdop))
- string.append(",");
- else
- string.append(
- str(boost::format("%.2f,") % _gps_data.dop.hdop));
-
- if (boost::math::isnan(_gps_data.fix.altitude))
- string.append(",,");
- else
- string.append(
- str(boost::format("%.2f,M,") % _gps_data.fix.altitude));
-
-#if GPSD_API_MAJOR_VERSION < 9
- if (boost::math::isnan(_gps_data.separation))
- string.append(",,");
- else
- string.append(
- str(boost::format("%.3f,M,") % _gps_data.separation));
-#else
- if (boost::math::isnan(_gps_data.fix.geoid_sep))
- string.append(",,");
- else
- string.append(
- str(boost::format("%.3f,M,") % _gps_data.fix.geoid_sep));
-#endif
-
- if (boost::math::isnan(mag_var))
- string.append(",");
- else {
- string.append(
- str(boost::format("%3.2f,%s") % std::abs(mag_var)
- % (mag_var > 0 ? "E" : "W")));
- }
-
- string.append(str(
- boost::format("*%02X") % _nmea_checksum(string)));
-
- return string;
- }
-
-private: // members
- std::vector<std::string> _sensors;
- bool _detected;
-
- gps_data_t _gps_data;
- boost::shared_mutex _d_mutex;
- boost::thread _bthread;
- size_t _timeout_cnt;
-};
-
-}} //namespace
-
-using namespace uhd::usrp;
-
-gpsd_iface::sptr gpsd_iface::make(const std::string &addr, const uint16_t port)
-{
- return gpsd_iface::sptr(new gpsd_iface_impl(addr, port));
-}