// // 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; }