| /* |
| * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved. |
| * |
| * Use of this source code is governed by a BSD-style license |
| * that can be found in the LICENSE file in the root of the source |
| * tree. An additional intellectual property rights grant can be found |
| * in the file PATENTS. All contributing project authors may |
| * be found in the AUTHORS file in the root of the source tree. |
| */ |
| |
| #include "webrtc/media/devices/deviceinfo.h" |
| |
| #include "webrtc/base/common.h" // for ASSERT |
| #include "webrtc/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() { |
| return libudev_.Load() && |
| !IsWrongLibUDevAbiVersion(libudev_.GetDllHandle()); |
| } |
| |
| 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) { |
| rtc::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 |