// // Copyright 2015 Ettus Research LLC // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . // #include #include #include #include #include #include "boost/date_time/gregorian/gregorian.hpp" #include #include #include #include #include #include #include #include #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, boost::uint16_t port) : _detected(false), _bthread(), _timeout_cnt(0) { boost::unique_lock 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 = boost::assign::list_of("gps_locked")("gps_time")("gps_position"); } virtual ~gpsd_iface_impl(void) { // interrupt the background thread and wait for it to finish _bthread.interrupt(); _bthread.join(); // clean up ... { boost::unique_lock 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_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 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 l(_d_mutex); _timeout_cnt = 0; _detected = true; if (gps_read(&_gps_data) < 0) throw std::runtime_error("error while reading"); } } } bool _gps_locked(void) { boost::shared_lock l(_d_mutex); return _gps_data.fix.mode >= MODE_2D; } std::time_t _epoch_time(void) { boost::shared_lock 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 l(_d_mutex); return boost::posix_time::from_time_t(_gps_data.fix.time).date(); } uhd::dict _gps_position(void) { boost::shared_lock l(_d_mutex); uhd::dict 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; } private: // members std::vector _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 boost::uint16_t port) { return gpsd_iface::sptr(new gpsd_iface_impl(addr, port)); }