2013-07-10 00:45:36 +00:00
|
|
|
/*
|
|
|
|
* libjingle
|
|
|
|
* Copyright 2012 Google Inc.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
|
|
* and/or other materials provided with the distribution.
|
|
|
|
* 3. The name of the author may not be used to endorse or promote products
|
|
|
|
* derived from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
|
|
|
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
|
|
|
|
* EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
|
|
|
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
|
|
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
|
|
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "talk/media/devices/deviceinfo.h"
|
|
|
|
|
|
|
|
#include "talk/base/common.h" // for ASSERT
|
|
|
|
#include "talk/media/devices/libudevsymboltable.h"
|
|
|
|
|
|
|
|
namespace cricket {
|
|
|
|
|
|
|
|
class ScopedLibUdev {
|
|
|
|
public:
|
|
|
|
static ScopedLibUdev* Create() {
|
|
|
|
ScopedLibUdev* ret_val = new ScopedLibUdev();
|
|
|
|
if (!ret_val->Init()) {
|
|
|
|
delete ret_val;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return ret_val;
|
|
|
|
}
|
|
|
|
~ScopedLibUdev() {
|
|
|
|
libudev_.Unload();
|
|
|
|
}
|
|
|
|
|
|
|
|
LibUDevSymbolTable* instance() { return &libudev_; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
ScopedLibUdev() {}
|
|
|
|
|
|
|
|
bool Init() {
|
2013-10-07 23:32:02 +00:00
|
|
|
return libudev_.Load() &&
|
|
|
|
!IsWrongLibUDevAbiVersion(libudev_.GetDllHandle());
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
LibUDevSymbolTable libudev_;
|
|
|
|
};
|
|
|
|
|
|
|
|
class ScopedUdev {
|
|
|
|
public:
|
|
|
|
explicit ScopedUdev(LibUDevSymbolTable* libudev) : libudev_(libudev) {
|
|
|
|
udev_ = libudev_->udev_new()();
|
|
|
|
}
|
|
|
|
~ScopedUdev() {
|
|
|
|
if (udev_) libudev_->udev_unref()(udev_);
|
|
|
|
}
|
|
|
|
|
|
|
|
udev* instance() { return udev_; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
LibUDevSymbolTable* libudev_;
|
|
|
|
udev* udev_;
|
|
|
|
};
|
|
|
|
|
|
|
|
class ScopedUdevEnumerate {
|
|
|
|
public:
|
|
|
|
ScopedUdevEnumerate(LibUDevSymbolTable* libudev, udev* udev)
|
|
|
|
: libudev_(libudev) {
|
|
|
|
enumerate_ = libudev_->udev_enumerate_new()(udev);
|
|
|
|
}
|
|
|
|
~ScopedUdevEnumerate() {
|
|
|
|
if (enumerate_) libudev_->udev_enumerate_unref()(enumerate_);
|
|
|
|
}
|
|
|
|
|
|
|
|
udev_enumerate* instance() { return enumerate_; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
LibUDevSymbolTable* libudev_;
|
|
|
|
udev_enumerate* enumerate_;
|
|
|
|
};
|
|
|
|
|
|
|
|
bool GetUsbProperty(const Device& device, const char* property_name,
|
|
|
|
std::string* property) {
|
|
|
|
talk_base::scoped_ptr<ScopedLibUdev> libudev_context(ScopedLibUdev::Create());
|
|
|
|
if (!libudev_context) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
ScopedUdev udev_context(libudev_context->instance());
|
|
|
|
if (!udev_context.instance()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
ScopedUdevEnumerate enumerate_context(libudev_context->instance(),
|
|
|
|
udev_context.instance());
|
|
|
|
if (!enumerate_context.instance()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
libudev_context->instance()->udev_enumerate_add_match_subsystem()(
|
|
|
|
enumerate_context.instance(), "video4linux");
|
|
|
|
libudev_context->instance()->udev_enumerate_scan_devices()(
|
|
|
|
enumerate_context.instance());
|
|
|
|
udev_list_entry* devices =
|
|
|
|
libudev_context->instance()->udev_enumerate_get_list_entry()(
|
|
|
|
enumerate_context.instance());
|
|
|
|
if (!devices) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
udev_list_entry* dev_list_entry = NULL;
|
|
|
|
const char* property_value = NULL;
|
|
|
|
// Macro that expands to a for-loop over the devices.
|
|
|
|
for (dev_list_entry = devices; dev_list_entry != NULL;
|
|
|
|
dev_list_entry = libudev_context->instance()->
|
|
|
|
udev_list_entry_get_next()(dev_list_entry)) {
|
|
|
|
const char* path = libudev_context->instance()->udev_list_entry_get_name()(
|
|
|
|
dev_list_entry);
|
|
|
|
if (!path) continue;
|
|
|
|
udev_device* dev =
|
|
|
|
libudev_context->instance()->udev_device_new_from_syspath()(
|
|
|
|
udev_context.instance(), path);
|
|
|
|
if (!dev) continue;
|
|
|
|
const char* device_node =
|
|
|
|
libudev_context->instance()->udev_device_get_devnode()(dev);
|
|
|
|
if (!device_node || device.id.compare(device_node) != 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
dev = libudev_context->instance()->
|
|
|
|
udev_device_get_parent_with_subsystem_devtype()(
|
|
|
|
dev, "usb", "usb_device");
|
|
|
|
if (!dev) continue;
|
|
|
|
property_value = libudev_context->instance()->
|
|
|
|
udev_device_get_sysattr_value()(
|
|
|
|
dev, property_name);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!property_value) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
property->assign(property_value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GetUsbId(const Device& device, std::string* usb_id) {
|
|
|
|
std::string id_vendor;
|
|
|
|
std::string id_product;
|
|
|
|
if (!GetUsbProperty(device, "idVendor", &id_vendor)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!GetUsbProperty(device, "idProduct", &id_product)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
usb_id->clear();
|
|
|
|
usb_id->append(id_vendor);
|
|
|
|
usb_id->append(":");
|
|
|
|
usb_id->append(id_product);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GetUsbVersion(const Device& device, std::string* usb_version) {
|
|
|
|
return GetUsbProperty(device, "version", usb_version);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace cricket
|