aboutsummaryrefslogtreecommitdiffstats
path: root/lib
diff options
context:
space:
mode:
Diffstat (limited to 'lib')
-rw-r--r--lib/device.cpp5
-rw-r--r--lib/device_addr.cpp18
-rw-r--r--lib/usrp/mboard/Makefile.am5
-rw-r--r--lib/usrp/mboard/usrp2.cpp113
-rw-r--r--lib/usrp/mboard/usrp2_fw_common.h55
5 files changed, 177 insertions, 19 deletions
diff --git a/lib/device.cpp b/lib/device.cpp
index a314da3a8..5718c71e7 100644
--- a/lib/device.cpp
+++ b/lib/device.cpp
@@ -16,6 +16,7 @@
//
#include <usrp_uhd/usrp/usrp.hpp>
+#include <usrp_uhd/usrp/mboard/usrp2.hpp>
#include <usrp_uhd/device.hpp>
#include <boost/format.hpp>
#include <stdexcept>
@@ -24,6 +25,10 @@ using namespace usrp_uhd;
std::vector<device_addr_t> device::discover(const device_addr_t & hint = device_addr_t()){
std::vector<device_addr_t> device_addrs;
+ if (hint.type == DEVICE_ADDR_TYPE_UDP){
+ std::vector<device_addr_t> usrp2_addrs = usrp::mboard::usrp2::discover(hint);
+ device_addrs.insert(device_addrs.begin(), usrp2_addrs.begin(), usrp2_addrs.end());
+ }
if (hint.type == DEVICE_ADDR_TYPE_VIRTUAL){
//make a copy of the hint for virtual testing
device_addr_t virtual_device_addr = hint;
diff --git a/lib/device_addr.cpp b/lib/device_addr.cpp
index 784795444..e11d51580 100644
--- a/lib/device_addr.cpp
+++ b/lib/device_addr.cpp
@@ -70,24 +70,6 @@ std::ostream& operator<<(std::ostream &os, const usrp_uhd::mac_addr_t &x){
return os;
}
-//----------------------- u2 ipv4 wrapper ----------------------------//
-usrp_uhd::ip_addr_t::ip_addr_t(const std::string &ip_addr_str_){
- std::string ip_addr_str = (ip_addr_str_ == "")? "255.255.255.255" : ip_addr_str_;
- int ret = inet_pton(AF_INET, ip_addr_str.c_str(), &ip_addr);
- if (ret == 0) throw std::runtime_error("Invalid ip address: " + ip_addr_str);
-}
-
-std::string usrp_uhd::ip_addr_t::to_string(void) const{
- char addr_buf[128];
- inet_ntop(AF_INET, &ip_addr, addr_buf, INET_ADDRSTRLEN);
- return std::string(addr_buf);
-}
-
-std::ostream& operator<<(std::ostream &os, const usrp_uhd::ip_addr_t &x){
- os << x.to_string();
- return os;
-}
-
//----------------------- usrp device_addr_t wrapper -------------------------//
usrp_uhd::device_addr_t::device_addr_t(device_addr_type_t device_addr_type){
type = device_addr_type;
diff --git a/lib/usrp/mboard/Makefile.am b/lib/usrp/mboard/Makefile.am
index 33782556d..0cb258e93 100644
--- a/lib/usrp/mboard/Makefile.am
+++ b/lib/usrp/mboard/Makefile.am
@@ -25,6 +25,9 @@ noinst_LTLIBRARIES = lib.la
lib_la_SOURCES = \
base.cpp \
- test.cpp
+ test.cpp \
+ usrp2.cpp
lib_la_LIBADD = $(GENERAL_LDDFLAGS)
+
+noinst_HEADERS = usrp2_fw_common.h
diff --git a/lib/usrp/mboard/usrp2.cpp b/lib/usrp/mboard/usrp2.cpp
new file mode 100644
index 000000000..7064f1460
--- /dev/null
+++ b/lib/usrp/mboard/usrp2.cpp
@@ -0,0 +1,113 @@
+//
+// Copyright 2010 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 <http://www.gnu.org/licenses/>.
+//
+
+#include <usrp_uhd/usrp/mboard/usrp2.hpp>
+#include "usrp2_fw_common.h"
+#include <usrp_uhd/device.hpp>
+#include <boost/asio.hpp>
+#include <boost/thread.hpp>
+#include <boost/format.hpp>
+#include <boost/utility.hpp>
+#include <boost/lexical_cast.hpp>
+#include <netinet/in.h>
+
+using namespace usrp_uhd;
+using namespace usrp_uhd::usrp::mboard;
+using boost::asio::ip::udp;
+
+/***********************************************************************
+ * Wrapper for the udp transport
+ **********************************************************************/
+class udp_transport : boost::noncopyable{
+public:
+ udp_transport(const std::string &addr, const std::string &port, bool bcast = false){
+ //std::cout << boost::format("Creating udp transport for %s %s") % addr % port << std::endl;
+
+ // resolve the address
+ udp::resolver resolver(_io_service);
+ udp::resolver::query query(udp::v4(), addr, port);
+ _receiver_endpoint = *resolver.resolve(query);
+
+ // Create and open the socket
+ _socket = new udp::socket(_io_service);
+ _socket->open(udp::v4());
+
+ if (bcast){
+ // Allow broadcasting
+ boost::asio::socket_base::broadcast option(true);
+ _socket->set_option(option);
+ }
+
+ }
+
+ ~udp_transport(void){
+ delete _socket;
+ }
+
+ void send(const device::send_args_t &buffs){
+ _socket->send_to(buffs, _receiver_endpoint);
+ }
+
+ void send(const void *buff, size_t len){
+ _socket->send_to(boost::asio::buffer(buff, len), _receiver_endpoint);
+ }
+
+ void recv(const device::recv_args_t &handler){
+ // make sure that bytes are available (crappy timeout 100 ms)
+ for (size_t i = 0; i < 10; i++){
+ if (_socket->available()) break;
+ boost::this_thread::sleep(boost::posix_time::milliseconds(10));
+ }
+
+ // receive the bytes and call the handler
+ udp::endpoint sender_endpoint;
+ while (_socket->available()){
+ size_t len = _socket->receive_from(
+ boost::asio::buffer(_recv_buff, sizeof(_recv_buff)),
+ sender_endpoint
+ );
+ bool done = handler(boost::asio::buffer(_recv_buff, len));
+ if (done) return;
+ }
+ }
+
+private:
+ udp::socket *_socket;
+ udp::endpoint _receiver_endpoint;
+ boost::asio::io_service _io_service;
+ uint8_t _recv_buff[1500];
+};
+
+/***********************************************************************
+ * Discovery over the udp transport
+ **********************************************************************/
+std::vector<device_addr_t> usrp2::discover(const device_addr_t &hint){
+ std::vector<device_addr_t> usrp2_addrs;
+
+ //create a udp transport to communicate
+ std::string ctrl_port = boost::lexical_cast<std::string>(USRP2_UDP_CTRL_PORT);
+ udp_transport trans(hint.udp_args.addr, ctrl_port, true);
+
+ //send a hello control packet
+ usrp2_ctrl_data_t data;
+ data.id = htonl(USRP2_CTRL_ID_HELLO);
+ trans.send(&data, sizeof(data));
+
+ //TODO start a thread to listen and sleep for timeout
+
+ return usrp2_addrs;
+}
diff --git a/lib/usrp/mboard/usrp2_fw_common.h b/lib/usrp/mboard/usrp2_fw_common.h
new file mode 100644
index 000000000..eb097c3fe
--- /dev/null
+++ b/lib/usrp/mboard/usrp2_fw_common.h
@@ -0,0 +1,55 @@
+//
+// Copyright 2010 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 <http://www.gnu.org/licenses/>.
+//
+
+#ifndef INCLUDED_USRP2_FW_COMMON_H
+#define INCLUDED_USRP2_FW_COMMON_H
+
+/*!
+ * Structs and constants for usrp2 communication.
+ * This header is shared by the firmware and host code.
+ * Therefore, this header may only contain valid C code.
+ */
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+// udp ports for the usrp2 communication
+// Dynamic and/or private ports: 49152–65535
+#define USRP2_UDP_CTRL_PORT 49152
+#define USRP2_UDP_DATA_PORT 49153
+
+typedef enum{
+ USRP2_CTRL_ID_HELLO
+} usrp2_ctrl_id_t;
+
+typedef struct{
+ uint32_t id;
+ uint32_t seq;
+ union data{
+ struct discovery_addrs{
+ uint32_t ip_addr;
+ uint8_t mac_addr[6];
+ uint8_t _padding[2];
+ };
+ };
+} usrp2_ctrl_data_t;
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* INCLUDED_USRP2_FW_COMMON_H */