//
// 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 .
//
#include "libusb1_base.hpp"
#include
#include
#include
#include
#include
#include
#include
using namespace uhd;
using namespace uhd::transport;
/***********************************************************************
* libusb session
**********************************************************************/
class libusb_session_impl : public libusb::session{
public:
libusb_session_impl(void){
UHD_ASSERT_THROW(libusb_init(&_context) == 0);
libusb_set_debug(_context, debug_level);
_mutex.lock();
_thread_group.create_thread(boost::bind(&libusb_session_impl::run_event_loop, this));
_mutex.lock();
}
~libusb_session_impl(void){
_running = false;
_thread_group.join_all();
libusb_exit(_context);
}
libusb_context *get_context(void) const{
return _context;
}
private:
libusb_context *_context;
boost::thread_group _thread_group;
bool _running;
boost::mutex _mutex;
void run_event_loop(void){
set_thread_priority_safe();
_running = true;
timeval tv;
_mutex.unlock();
while(_running){
tv.tv_sec = 0;
tv.tv_usec = 100000; //100ms
libusb_handle_events_timeout(this->get_context(), &tv);
}
}
};
libusb::session::sptr libusb::session::get_global_session(void){
static boost::weak_ptr global_session;
//not expired -> get existing session
if (not global_session.expired()) return global_session.lock();
//create a new global session
sptr new_global_session(new libusb_session_impl());
global_session = new_global_session;
return new_global_session;
}
/***********************************************************************
* libusb device
**********************************************************************/
class libusb_device_impl : public libusb::device{
public:
libusb_device_impl(libusb_device *dev){
_session = libusb::session::get_global_session();
_dev = dev;
}
~libusb_device_impl(void){
libusb_unref_device(this->get());
}
libusb_device *get(void) const{
return _dev;
}
private:
libusb::session::sptr _session; //always keep a reference to session
libusb_device *_dev;
};
/***********************************************************************
* libusb device list
**********************************************************************/
class libusb_device_list_impl : public libusb::device_list{
public:
libusb_device_list_impl(void){
libusb::session::sptr sess = libusb::session::get_global_session();
//allocate a new list of devices
libusb_device** dev_list;
ssize_t ret = libusb_get_device_list(sess->get_context(), &dev_list);
if (ret < 0) throw std::runtime_error("cannot enumerate usb devices");
//fill the vector of device references
for (size_t i = 0; i < size_t(ret); i++) _devs.push_back(
libusb::device::sptr(new libusb_device_impl(dev_list[i]))
);
//free the device list but dont unref (done in ~device)
libusb_free_device_list(dev_list, false/*dont unref*/);
}
size_t size(void) const{
return _devs.size();
}
libusb::device::sptr at(size_t i) const{
return _devs.at(i);
}
private:
std::vector _devs;
};
libusb::device_list::sptr libusb::device_list::make(void){
return sptr(new libusb_device_list_impl());
}
/***********************************************************************
* libusb device descriptor
**********************************************************************/
class libusb_device_descriptor_impl : public libusb::device_descriptor{
public:
libusb_device_descriptor_impl(libusb::device::sptr dev){
_dev = dev;
UHD_ASSERT_THROW(libusb_get_device_descriptor(_dev->get(), &_desc) == 0);
}
const libusb_device_descriptor &get(void) const{
return _desc;
}
std::string get_ascii_serial(void) const{
if (this->get().iSerialNumber == 0) return "";
libusb::device_handle::sptr handle(
libusb::device_handle::get_cached_handle(_dev)
);
unsigned char buff[512];
ssize_t ret = libusb_get_string_descriptor_ascii(
handle->get(), this->get().iSerialNumber, buff, sizeof(buff)
);
if (ret < 0) return ""; //on error, just return empty string
return std::string((char *)buff, ret);
}
private:
libusb::device::sptr _dev; //always keep a reference to device
libusb_device_descriptor _desc;
};
libusb::device_descriptor::sptr libusb::device_descriptor::make(device::sptr dev){
return sptr(new libusb_device_descriptor_impl(dev));
}
/***********************************************************************
* libusb device handle
**********************************************************************/
class libusb_device_handle_impl : public libusb::device_handle{
public:
libusb_device_handle_impl(libusb::device::sptr dev){
_dev = dev;
UHD_ASSERT_THROW(libusb_open(_dev->get(), &_handle) == 0);
}
~libusb_device_handle_impl(void){
//release all claimed interfaces
for (size_t i = 0; i < _claimed.size(); i++){
libusb_release_interface(this->get(), _claimed[i]);
}
libusb_close(_handle);
}
libusb_device_handle *get(void) const{
return _handle;
}
void claim_interface(int interface){
UHD_ASSERT_THROW(libusb_claim_interface(this->get(), interface) == 0);
_claimed.push_back(interface);
}
private:
libusb::device::sptr _dev; //always keep a reference to device
libusb_device_handle *_handle;
std::vector _claimed;
};
libusb::device_handle::sptr libusb::device_handle::get_cached_handle(device::sptr dev){
static uhd::dict > handles;
//not expired -> get existing handle
if (handles.has_key(dev->get()) and not handles[dev->get()].expired()){
return handles[dev->get()].lock();
}
//create a new cached handle
try{
sptr new_handle(new libusb_device_handle_impl(dev));
handles[dev->get()] = new_handle;
return new_handle;
}
catch(const std::exception &e){
std::cerr << "USB open failed: see the application notes for your device." << std::endl;
throw std::runtime_error(e.what());
}
}
/***********************************************************************
* libusb special handle
**********************************************************************/
class libusb_special_handle_impl : public libusb::special_handle{
public:
libusb_special_handle_impl(libusb::device::sptr dev){
_dev = dev;
}
libusb::device::sptr get_device(void) const{
return _dev;
}
std::string get_serial(void) const{
return libusb::device_descriptor::make(this->get_device())->get_ascii_serial();
}
boost::uint16_t get_vendor_id(void) const{
return libusb::device_descriptor::make(this->get_device())->get().idVendor;
}
boost::uint16_t get_product_id(void) const{
return libusb::device_descriptor::make(this->get_device())->get().idProduct;
}
private:
libusb::device::sptr _dev; //always keep a reference to device
};
libusb::special_handle::sptr libusb::special_handle::make(device::sptr dev){
return sptr(new libusb_special_handle_impl(dev));
}
/***********************************************************************
* list device handles implementations
**********************************************************************/
std::vector usb_device_handle::get_device_list(
boost::uint16_t vid, boost::uint16_t pid
){
std::vector handles;
libusb::device_list::sptr dev_list = libusb::device_list::make();
for (size_t i = 0; i < dev_list->size(); i++){
usb_device_handle::sptr handle = libusb::special_handle::make(dev_list->at(i));
if (handle->get_vendor_id() == vid and handle->get_product_id() == pid){
handles.push_back(handle);
}
}
return handles;
}