diff --git a/targets/ARCH/USRP/USERSPACE/LIB/Makefile.inc b/targets/ARCH/USRP/USERSPACE/LIB/Makefile.inc
new file mode 100644
index 0000000000000000000000000000000000000000..d50783d6657e53b1f1839874376f01f3d4faa021
--- /dev/null
+++ b/targets/ARCH/USRP/USERSPACE/LIB/Makefile.inc
@@ -0,0 +1,4 @@
+USRP_OBJ += $(OPENAIR_TARGETS)/ARCH/USRP/USERSPACE/LIB/usrp_lib.o
+USRP_FILE_OBJ += $(OPENAIR_TARGETS)/ARCH/USRP/USERSPACE/LIB/usrp_lib.cpp
+USRP_CFLAGS += -I$(OPENAIR_TARGETS)/ARCH/USRP/USERSPACE/LIB/
+
diff --git a/targets/ARCH/USRP/USERSPACE/LIB/usrp_lib.cpp b/targets/ARCH/USRP/USERSPACE/LIB/usrp_lib.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..27e0ba15764e9b7add8575566fab429c8450e4ba
--- /dev/null
+++ b/targets/ARCH/USRP/USERSPACE/LIB/usrp_lib.cpp
@@ -0,0 +1,214 @@
+
+/** usrp_lib.cpp
+ *
+ * Author: HongliangXU : hong-liang-xu@agilent.com
+ */
+
+#include <string.h>
+#include <pthread.h>
+#include <unistd.h>
+#include <stdio.h>
+#include <uhd/utils/thread_priority.hpp>
+#include <uhd/usrp/multi_usrp.hpp>
+#include <boost/lexical_cast.hpp>
+#include <boost/algorithm/string.hpp>
+#include <iostream>
+#include <complex>
+#include <fstream>
+#include <cmath>
+#include "usrp_lib.h"
+
+typedef struct
+{
+
+  // --------------------------------
+  // variables for USRP configuration
+  // --------------------------------
+  uhd::usrp::multi_usrp::sptr usrp;
+  //uhd::usrp::multi_usrp::sptr rx_usrp;
+
+  //create a send streamer and a receive streamer
+  uhd::tx_streamer::sptr tx_stream;
+  uhd::rx_streamer::sptr rx_stream;
+
+  uhd::tx_metadata_t tx_md;
+  uhd::rx_metadata_t rx_md;
+
+  uhd::time_spec_t tm_spec;
+  //setup variables and allocate buffer
+  uhd::async_metadata_t async_md;
+
+  double sample_rate;
+  // time offset between transmiter timestamp and receiver timestamp;
+  double tdiff;
+  // use usrp_time_offset to get this value
+  int tx_forward_nsamps; //166 for 20Mhz
+
+
+  // --------------------------------
+  // Debug and output control
+  // --------------------------------
+  int num_underflows;
+  int num_overflows;
+  int num_seq_errors;
+
+  int64_t tx_count;
+  int64_t rx_count;
+  openair0_timestamp rx_timestamp;
+
+} usrp_state_t;
+
+
+static int trx_usrp_start(openair0_device *device)
+{
+  usrp_state_t *s = (usrp_state_t*)device->priv;
+
+  // init recv and send streaming
+  uhd::stream_cmd_t cmd(uhd::stream_cmd_t::STREAM_MODE_START_CONTINUOUS);
+  cmd.time_spec = s->usrp->get_time_now() + uhd::time_spec_t(0.01);
+  cmd.stream_now = false; // start at constant delay
+  s->rx_stream->issue_stream_cmd(cmd);
+
+  s->tx_md.time_spec = s->usrp->get_time_now() + uhd::time_spec_t(1-(double)s->tx_forward_nsamps/s->sample_rate);
+  s->tx_md.has_time_spec = true;
+  s->tx_md.start_of_burst = true;
+  s->tx_md.end_of_burst = false;
+
+
+  s->rx_count = 0;
+  s->tx_count = 0;
+  s->rx_timestamp = 0;
+}
+
+static void trx_usrp_end(openair0_device *device)
+{
+  usrp_state_t *s = (usrp_state_t*)device->priv;
+
+  s->rx_stream->issue_stream_cmd(uhd::stream_cmd_t::STREAM_MODE_STOP_CONTINUOUS);
+
+	//send a mini EOB packet
+	s->tx_md.end_of_burst = true;
+	s->tx_stream->send("", 0, s->tx_md);
+	s->tx_md.end_of_burst = false;
+}
+static void trx_usrp_write(openair0_device *device, openair0_timestamp timestamp, const void *buff, int nsamps, int flags)
+{
+  usrp_state_t *s = (usrp_state_t*)device->priv;
+
+  s->tx_md.time_spec = uhd::time_spec_t::from_ticks(timestamp, s->sample_rate);
+  if(flags)
+    s->tx_md.has_time_spec = true;
+  else
+    s->tx_md.has_time_spec = false;
+  s->tx_stream->send(buff, nsamps, s->tx_md);
+  s->tx_md.start_of_burst = false;
+}
+
+static int trx_usrp_read(openair0_device *device, openair0_timestamp *ptimestamp, void *buff, int nsamps)
+{
+
+  usrp_state_t *s = (usrp_state_t*)device->priv;
+
+  int samples_received;
+  //TODO: only one channel is supported now
+  samples_received = s->rx_stream->recv(buff, nsamps, s->rx_md);
+
+	//handle the error code
+	switch(s->rx_md.error_code){
+		case uhd::rx_metadata_t::ERROR_CODE_NONE:
+			break;
+		case uhd::rx_metadata_t::ERROR_CODE_OVERFLOW:
+			printf("[recv] USRP RX OVERFLOW!\n");
+			s->num_overflows++;
+			break;
+		case uhd::rx_metadata_t::ERROR_CODE_TIMEOUT:
+			printf("[recv] USRP RX TIMEOUT!\n");
+			break;
+		default:
+			printf("[recv] Unexpected error on RX, Error code: 0x%x\n",s->rx_md.error_code);
+			break;
+	}
+  s->rx_count += nsamps;
+  s->rx_timestamp = s->rx_md.time_spec.to_ticks(s->sample_rate);
+  *ptimestamp = s->rx_timestamp;
+  return samples_received;
+}
+
+static bool is_equal(double a, double b)
+{
+  return std::fabs(a-b) < std::numeric_limits<double>::epsilon();
+}
+
+int openair0_device_init(openair0_device* device, openair0_config_t *openair0_cfg)
+{
+  usrp_state_t *s = (usrp_state_t*)malloc(sizeof(usrp_state_t));
+  memset(s, 0, sizeof(usrp_state_t));
+
+  // Initialize USRP device
+  std::string args = "";
+  uhd::device_addrs_t device_adds = uhd::device::find(args);
+  if(device_adds.size() == 0)
+  {
+    std::cerr<<"No USRP Device Found. " << std::endl;
+    free(s);
+    return -1;
+  }
+  s->usrp = uhd::usrp::multi_usrp::make(args);
+
+  // lock mboard clocks
+  s->usrp->set_clock_source("internal");
+  // set master clock rate and sample rate for tx & rx for streaming
+  s->usrp->set_master_clock_rate(30.72e6);
+  s->usrp->set_rx_rate(openair0_cfg->sample_rate);
+  s->usrp->set_tx_rate(openair0_cfg->sample_rate);
+
+  s->usrp->set_tx_freq(openair0_cfg->tx_freq);
+  s->usrp->set_rx_freq(openair0_cfg->rx_freq);
+  s->usrp->set_tx_gain(openair0_cfg->tx_gain);
+  s->usrp->set_rx_gain(openair0_cfg->rx_gain);
+  s->usrp->set_tx_bandwidth(openair0_cfg->tx_bw);
+  s->usrp->set_rx_bandwidth(openair0_cfg->rx_bw);
+
+  // create tx & rx streamer
+  uhd::stream_args_t stream_args("sc16", "sc16");
+  s->tx_stream = s->usrp->get_tx_stream(stream_args);
+  s->rx_stream = s->usrp->get_rx_stream(stream_args);
+
+  s->usrp->set_time_now(uhd::time_spec_t(0.0));
+
+  // display USRP settings
+	std::cout << std::endl<<boost::format("Actual TX sample rate: %fMSps...") % (s->usrp->get_tx_rate()/1e6) << std::endl;
+	std::cout << boost::format("Actual RX sample rate: %fMSps...") % (s->usrp->get_rx_rate()/1e6) << std::endl;
+
+	std::cout << boost::format("Actual TX frequency: %fGHz...") % (s->usrp->get_tx_freq()/1e9) << std::endl;
+	std::cout << boost::format("Actual RX frequency: %fGHz...") % (s->usrp->get_rx_freq()/1e9) << std::endl;
+
+	std::cout << boost::format("Actual TX gain: %f...") % (s->usrp->get_tx_gain()) << std::endl;
+	std::cout << boost::format("Actual RX gain: %f...") % (s->usrp->get_rx_gain()) << std::endl;
+
+	std::cout << boost::format("Actual TX bandwidth: %fM...") % (s->usrp->get_tx_bandwidth()/1e6) << std::endl;
+	std::cout << boost::format("Actual RX bandwidth: %fM...") % (s->usrp->get_rx_bandwidth()/1e6) << std::endl;
+
+	std::cout << boost::format("Actual TX antenna: %s...") % (s->usrp->get_tx_antenna()) << std::endl;
+	std::cout << boost::format("Actual RX antenna: %s...") % (s->usrp->get_rx_antenna()) << std::endl;
+
+	std::cout << boost::format("Device timestamp: %f...") % (s->usrp->get_time_now().get_real_secs()) << std::endl;
+
+  device->priv = s;
+  device->trx_start_func = trx_usrp_start;
+  device->trx_end_func   = trx_usrp_end;
+  device->trx_read_func  = trx_usrp_read;
+  device->trx_write_func = trx_usrp_write;
+
+  s->sample_rate = openair0_cfg->sample_rate;
+  // TODO:
+  // init tx_forward_nsamps based usrp_time_offset ex
+  if(is_equal(s->sample_rate, (double)30.72e6))
+    s->tx_forward_nsamps  = 176;
+  if(is_equal(s->sample_rate, (double)15.36e6))
+    s->tx_forward_nsamps = 90;
+  if(is_equal(s->sample_rate, (double)7.68e6))
+    s->tx_forward_nsamps = 50;
+
+  return 0;
+}
diff --git a/targets/ARCH/USRP/USERSPACE/LIB/usrp_lib.h b/targets/ARCH/USRP/USERSPACE/LIB/usrp_lib.h
new file mode 100644
index 0000000000000000000000000000000000000000..2f2ba098884aea28eaf6a07a4cf2e6e25af6b455
--- /dev/null
+++ b/targets/ARCH/USRP/USERSPACE/LIB/usrp_lib.h
@@ -0,0 +1,70 @@
+/** usrp_lib.h
+ *
+ * Author: HongliangXU : hong-liang-xu@agilent.com
+ */
+#ifndef USRP_LIB_H
+#define USRP_LIB_H
+
+typedef int64_t openair0_timestamp;
+typedef struct openair0_device_t openair0_device;
+/* structrue holds the parameters to configure USRP devices
+ */
+typedef struct {
+  /* the sample rate for both transmit and receive. */
+  double sample_rate;
+  /* number of RX channels (=RX antennas) */
+  int rx_num_channels;
+  /* number of TX channels (=TX antennas) */
+  int tx_num_channels;
+  /* center frequency in Hz for RX */
+  double rx_freq;
+  /* center frequency in Hz for TX */
+  double tx_freq;
+  /* gain for RX in dB */
+  double rx_gain;
+  /* gain for TX in dB */
+  double tx_gain;
+  /* RX bandwidth in Hz */
+  double rx_bw;
+  /* TX bandwidth in Hz */
+  double tx_bw;
+} openair0_config_t;
+
+struct openair0_device_t {
+  /* USRP RF frontend parameters set by application */
+  openair0_config_t openair0_cfg;
+
+  /* Can be used by driver to hold internal structure*/
+  void *priv;
+
+  /* Functions API, which are called by the application*/
+
+  /* Called to start the transceiver. Return 0 if OK, < 0 if error */
+  int (*trx_start_func)(openair0_device *device);
+
+  /* Write 'nsamps' samples on each channel from buffers. buff[0] is the array for
+   * the first channel. timestamp if the time (in samples) at which the first sample
+   * MUST be sent
+   * use flags = 1 to send as timestamp specfied*/
+  void (*trx_write_func)(openair0_device *device, openair0_timestamp timestamp, const void *buff, int nsamps, int flags);
+
+  /* Read 'nsamps' samples from each channel to buffers. buff[0] is the array for
+   * the first channel. *ptimestamp is the time at which the first sample
+   * was received.
+   * Return the number of sample read */
+  int (*trx_read_func)(openair0_device *device, openair0_timestamp *ptimestamp, void *buff, int nsamps);
+
+  /* Terminate operation of the transceiver -- free all associated resources */
+  void (*trx_end_func)(openair0_device *device);
+};
+
+
+#ifdef __cplusplus
+extern "C"
+{
+/* return 0 if OK, < 0 if error */
+int openair0_device_init(openair0_device* device, openair0_config_t *openair0_cfg);
+}
+#endif
+
+#endif // USRP_LIB_H