From 5098256a258623b2d0bb6aa54b923f22e92a178f Mon Sep 17 00:00:00 2001
From: Josh Blum <josh@joshknows.com>
Date: Wed, 28 Sep 2011 12:04:08 -0700
Subject: e100: added support for internal gpsdo

---
 host/lib/usrp/e100/e100_ctrl.cpp | 76 ++++++++++++++++++++++++++++++++++++++++
 host/lib/usrp/e100/e100_ctrl.hpp |  3 ++
 host/lib/usrp/e100/e100_impl.cpp | 25 +++++++++++++
 host/lib/usrp/e100/e100_impl.hpp |  3 ++
 4 files changed, 107 insertions(+)

(limited to 'host/lib/usrp')

diff --git a/host/lib/usrp/e100/e100_ctrl.cpp b/host/lib/usrp/e100/e100_ctrl.cpp
index 87f7855d3..028e734e3 100644
--- a/host/lib/usrp/e100/e100_ctrl.cpp
+++ b/host/lib/usrp/e100/e100_ctrl.cpp
@@ -18,12 +18,14 @@
 #include "e100_ctrl.hpp"
 #include "e100_regs.hpp"
 #include <uhd/exception.hpp>
+#include <uhd/utils/log.hpp>
 #include <uhd/utils/msg.hpp>
 #include <sys/ioctl.h> //ioctl
 #include <fcntl.h> //open, close
 #include <linux/usrp_e.h> //ioctl structures and constants
 #include <boost/thread/thread.hpp> //sleep
 #include <boost/thread/mutex.hpp>
+#include <boost/foreach.hpp>
 #include <boost/format.hpp>
 #include <fstream>
 
@@ -174,6 +176,80 @@ uhd::i2c_iface::sptr e100_ctrl::make_dev_i2c_iface(const std::string &node){
     return uhd::i2c_iface::sptr(new i2c_dev_iface(node));
 }
 
+/***********************************************************************
+ * UART control implementation
+ **********************************************************************/
+#include <termios.h>
+#include <cstring>
+class uart_dev_iface : public uart_iface{
+public:
+    uart_dev_iface(const std::string &node){
+        if ((_node_fd = ::open(node.c_str(), O_RDWR | O_NONBLOCK)) < 0){
+            throw uhd::io_error("Failed to open " + node);
+        }
+
+        //init the tty settings w/ termios
+        termios tio;
+        std::memset(&tio,0,sizeof(tio));
+        tio.c_iflag=0;
+        tio.c_oflag=0;
+        tio.c_cflag=CS8|CREAD|CLOCAL;           // 8n1, see termios.h for more information
+        tio.c_lflag=0;
+        tio.c_cc[VMIN]=1;
+        tio.c_cc[VTIME]=5;
+
+        cfsetospeed(&tio,B115200);            // 115200 baud
+        cfsetispeed(&tio,B115200);            // 115200 baud
+
+        tcsetattr(_node_fd,TCSANOW,&tio);
+    }
+
+    void write_uart(const std::string &buf){
+        std::string out_str;
+        BOOST_FOREACH(const char &ch, buf){
+            if (ch == '\n') out_str += "\r\n";
+            else out_str += std::string(1, ch);
+        }
+        const ssize_t ret = ::write(_node_fd, out_str.c_str(), out_str.size());
+        if (size_t(ret) != out_str.size()) UHD_LOG << ret;
+    }
+
+    std::string read_uart(double timeout){
+        const boost::system_time exit_time = boost::get_system_time() + boost::posix_time::milliseconds(long(timeout*1000));
+
+        std::string line;
+        while(true){
+            char ch;
+            const ssize_t ret = ::read(_node_fd, &ch, 1);
+
+            //got a character -> process it
+            if (ret == 1){
+                const bool flush  = ch == '\n' or ch == '\r';
+                if (flush and line.empty()) continue; //avoid flushing on empty lines
+                line += std::string(1, ch);
+                if (flush) break;
+            }
+
+            //didnt get a character, check the timeout
+            if (boost::get_system_time() > exit_time){
+                break;
+            }
+
+            //otherwise sleep for a bit
+            else{
+                boost::this_thread::sleep(boost::posix_time::milliseconds(10));
+            }
+        }
+        return line;
+    }
+
+private: int _node_fd;
+};
+
+uhd::uart_iface::sptr e100_ctrl::make_gps_uart_iface(const std::string &node){
+    return uhd::uart_iface::sptr(new uart_dev_iface(node));
+}
+
 /***********************************************************************
  * USRP-E100 control implementation
  **********************************************************************/
diff --git a/host/lib/usrp/e100/e100_ctrl.hpp b/host/lib/usrp/e100/e100_ctrl.hpp
index 8520ea595..fd66791d4 100644
--- a/host/lib/usrp/e100/e100_ctrl.hpp
+++ b/host/lib/usrp/e100/e100_ctrl.hpp
@@ -36,6 +36,9 @@ public:
     //! Make an i2c iface for the i2c device node
     static uhd::spi_iface::sptr make_aux_spi_iface(void);
 
+    //! Make a uart iface for the uart device node
+    static uhd::uart_iface::sptr make_gps_uart_iface(const std::string &node);
+
     virtual void ioctl(int request, void *mem) = 0;
 
     virtual int get_file_descriptor(void) = 0;
diff --git a/host/lib/usrp/e100/e100_impl.cpp b/host/lib/usrp/e100/e100_impl.cpp
index b988de9a1..564a05a7e 100644
--- a/host/lib/usrp/e100/e100_impl.cpp
+++ b/host/lib/usrp/e100/e100_impl.cpp
@@ -229,6 +229,22 @@ e100_impl::e100_impl(const uhd::device_addr_t &device_addr){
     _tree->create<sensor_value_t>(mb_path / "sensors/ref_locked")
         .publish(boost::bind(&e100_impl::get_ref_locked, this));
 
+    ////////////////////////////////////////////////////////////////////
+    // Create the GPSDO control
+    ////////////////////////////////////////////////////////////////////
+    try{
+        _gps = gps_ctrl::make(e100_ctrl::make_gps_uart_iface(E100_UART_DEV_NODE));
+    }
+    catch(std::exception &e){
+        UHD_MSG(error) << "An error occurred making GPSDO control: " << e.what() << std::endl;
+    }
+    if (_gps.get() != NULL and _gps->gps_detected()){
+        BOOST_FOREACH(const std::string &name, _gps->get_sensors()){
+            _tree->create<sensor_value_t>(mb_path / "sensors" / name)
+                .publish(boost::bind(&gps_ctrl::get_sensor, _gps, name));
+        }
+    }
+
     ////////////////////////////////////////////////////////////////////
     // create frontend control objects
     ////////////////////////////////////////////////////////////////////
@@ -376,6 +392,15 @@ e100_impl::e100_impl(const uhd::device_addr_t &device_addr){
     _tree->access<std::string>(mb_path / "clock_source/value").set("internal");
     _tree->access<std::string>(mb_path / "time_source/value").set("none");
 
+    //GPS installed: use external ref, time, and init time spec
+    if (_gps.get() != NULL 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("external");
+        _tree->access<std::string>(mb_path / "clock_source/value").set("external");
+        UHD_MSG(status) << "Initializing time to the internal GPSDO" << std::endl;
+        _time64->set_time_next_pps(time_spec_t(time_t(_gps->get_sensor("gps_time").to_int()+1)));
+    }
+
 }
 
 e100_impl::~e100_impl(void){
diff --git a/host/lib/usrp/e100/e100_impl.hpp b/host/lib/usrp/e100/e100_impl.hpp
index 99c8481e3..536eba040 100644
--- a/host/lib/usrp/e100/e100_impl.hpp
+++ b/host/lib/usrp/e100/e100_impl.hpp
@@ -31,6 +31,7 @@
 #include <uhd/usrp/subdev_spec.hpp>
 #include <uhd/usrp/dboard_eeprom.hpp>
 #include <uhd/usrp/mboard_eeprom.hpp>
+#include <uhd/usrp/gps_ctrl.hpp>
 #include <uhd/types/sensors.hpp>
 #include <uhd/types/otw_type.hpp>
 #include <uhd/types/clock_config.hpp>
@@ -47,6 +48,7 @@ uhd::transport::zero_copy_if::sptr e100_make_mmap_zero_copy(e100_ctrl::sptr ifac
 static const double          E100_RX_LINK_RATE_BPS = 166e6/3/2*2;
 static const double          E100_TX_LINK_RATE_BPS = 166e6/3/1*2;
 static const std::string     E100_I2C_DEV_NODE = "/dev/i2c-3";
+static const std::string     E100_UART_DEV_NODE = "/dev/ttyO1";
 static const boost::uint16_t E100_FPGA_COMPAT_NUM = 0x06;
 static const boost::uint32_t E100_RX_SID_BASE = 2;
 static const boost::uint32_t E100_TX_ASYNC_SID = 1;
@@ -98,6 +100,7 @@ private:
     e100_ctrl::sptr _fpga_ctrl;
     uhd::i2c_iface::sptr _dev_i2c_iface;
     uhd::spi_iface::sptr _aux_spi_iface;
+    uhd::gps_ctrl::sptr _gps;
 
     //transports
     uhd::transport::zero_copy_if::sptr _data_transport;
-- 
cgit v1.2.3