Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(86)

Side by Side Diff: device/hid/hid_service_linux.cc

Issue 143883005: HID backend (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@hid-impl-base
Patch Set: Really fix tests for lack of testing hardware; adapt to ancient libudev in the cros distro Created 6 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
1 // Copyright (c) 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include <libudev.h>
6 #include <string>
7 #include <vector>
8
9 #include "base/basictypes.h"
10 #include "base/bind.h"
11 #include "base/callback.h"
12 #include "base/logging.h"
13 #include "base/memory/scoped_vector.h"
14 #include "base/platform_file.h"
15 #include "base/strings/string_number_conversions.h"
16 #include "base/strings/string_piece.h"
17 #include "base/strings/string_split.h"
18 #include "base/threading/thread_restrictions.h"
19 #include "device/hid/hid_connection.h"
20 #include "device/hid/hid_connection_linux.h"
21 #include "device/hid/hid_device_info.h"
22 #include "device/hid/hid_service_linux.h"
23
24 namespace device {
25
26 namespace {
27
28 const char kUdevName[] = "udev";
29 const char kUdevActionAdd[] = "add";
30 const char kUdevActionRemove[] = "remove";
31 const char kHIDSubSystem[] = "hid";
32
33 const char kHIDID[] = "HID_ID";
34 const char kHIDName[] = "HID_NAME";
35 const char kHIDUnique[] = "HID_UNIQ";
36
37 } // namespace
38
39 HidServiceLinux::HidServiceLinux() {
40 udev_.reset(udev_new());
41 if (!udev_) {
42 LOG(ERROR) << "Failed to create udev.";
43 return;
44 }
45 monitor_.reset(udev_monitor_new_from_netlink(udev_.get(), kUdevName));
46 if (!monitor_) {
47 LOG(ERROR) << "Failed to create udev monitor.";
48 return;
49 }
50 int ret = udev_monitor_filter_add_match_subsystem_devtype(
51 monitor_.get(),
52 kHIDSubSystem,
53 NULL);
54 if (ret != 0) {
55 LOG(ERROR) << "Failed to add udev monitor filter.";
56 return;
57 }
58
59 ret = udev_monitor_enable_receiving(monitor_.get());
60 if (ret != 0) {
61 LOG(ERROR) << "Failed to start udev monitoring.";
62 return;
63 }
64
65 monitor_fd_ = udev_monitor_get_fd(monitor_.get());
66 if (monitor_fd_ <= 0) {
67 LOG(ERROR) << "Failed to start udev monitoring.";
68 return;
69 }
70
71 if (!base::MessageLoopForIO::current()->WatchFileDescriptor(
72 monitor_fd_,
73 true,
74 base::MessageLoopForIO::WATCH_READ,
75 &monitor_watcher_,
76 this))
77 return;
78
79 Enumerate();
80 }
81
82 HidServiceLinux::~HidServiceLinux() {
83 monitor_watcher_.StopWatchingFileDescriptor();
84 close(monitor_fd_);
85 }
86
87 void HidServiceLinux::Enumerate() {
88 scoped_ptr<udev_enumerate, UdevEnumerateDeleter> enumerate(
89 udev_enumerate_new(udev_.get()));
90
91 if (!enumerate) {
92 LOG(ERROR) << "Failed to enumerate devices.";
93 return;
94 }
95
96 if (udev_enumerate_add_match_subsystem(enumerate.get(), kHIDSubSystem)) {
97 LOG(ERROR) << "Failed to enumerate devices.";
98 return;
99 }
100
101 if (udev_enumerate_scan_devices(enumerate.get()) != 0) {
102 LOG(ERROR) << "Failed to enumerate devices.";
103 return;
104 }
105
106 // This list is managed by |enumerate|.
107 udev_list_entry* devices = udev_enumerate_get_list_entry(enumerate.get());
108 for (udev_list_entry* i = devices; i != NULL;
109 i = udev_list_entry_get_next(i)) {
110 ScopedUdevDevicePtr hid_dev(
111 udev_device_new_from_syspath(udev_.get(), udev_list_entry_get_name(i)));
112 if (hid_dev) {
113 PlatformDeviceAdd(hid_dev.get());
114 }
115 }
116
117 initialized_ = true;
118 }
119
120 void HidServiceLinux::PlatformDeviceAdd(udev_device* device) {
121 if (!device)
122 return;
123
124 const char* device_id = udev_device_get_syspath(device);
125 if (!device_id)
126 return;
127
128
129 HidDeviceInfo device_info;
130 device_info.device_id = device_id;
131
132 uint32 int_property = 0;
133 const char* str_property = NULL;
134
135 const char* hid_id = udev_device_get_property_value(device, kHIDID);
136 if (!hid_id)
137 return;
138
139 std::vector<std::string> parts;
140 base::SplitString(hid_id, ':', &parts);
141 if (parts.size() != 3) {
142 return;
143 }
144
145 if (HexStringToUInt(base::StringPiece(parts[1]), &int_property)) {
146 device_info.vendor_id = int_property;
147 }
148
149 if (HexStringToUInt(base::StringPiece(parts[2]), &int_property)) {
150 device_info.product_id = int_property;
151 }
152
153 str_property = udev_device_get_property_value(device, kHIDUnique);
154 if (str_property != NULL)
155 device_info.serial_number = str_property;
156
157 str_property = udev_device_get_property_value(device, kHIDName);
158 if (str_property != NULL)
159 device_info.product_name = str_property;
160
161 AddDevice(device_info);
162 }
163
164 void HidServiceLinux::PlatformDeviceRemove(udev_device* raw_dev) {
165 // The returned the device is not referenced.
166 udev_device* hid_dev =
167 udev_device_get_parent_with_subsystem_devtype(raw_dev, "hid", NULL);
168
169 if (!hid_dev)
170 return;
171
172 const char* device_id = NULL;
173 device_id = udev_device_get_syspath(hid_dev);
174 if (device_id == NULL)
175 return;
176
177 RemoveDevice(device_id);
178 }
179
180 scoped_refptr<HidConnection> HidServiceLinux::Connect(std::string device_id) {
181 if (!ContainsKey(devices_, device_id))
182 return NULL;
183 ScopedUdevDevicePtr hid_device(
184 udev_device_new_from_syspath(udev_.get(), device_id.c_str()));
185 if (hid_device) {
186 scoped_refptr<HidConnectionLinux> connection =
187 new HidConnectionLinux(devices_[device_id], hid_device.Pass());
188 if (connection->initialized())
189 return connection;
190 }
191 return NULL;
192 }
193
194 void HidServiceLinux::OnFileCanReadWithoutBlocking(int fd) {
195 DCHECK_EQ(monitor_fd_, fd);
196
197 ScopedUdevDevicePtr dev(udev_monitor_receive_device(monitor_.get()));
198 if (!dev)
199 return;
200
201 std::string action(udev_device_get_action(dev.get()));
202 if (action == kUdevActionAdd) {
203 PlatformDeviceAdd(dev.get());
204 } else if (action == kUdevActionRemove) {
205 PlatformDeviceRemove(dev.get());
206 }
207 }
208
209 void HidServiceLinux::OnFileCanWriteWithoutBlocking(int fd) {}
210
211 } // namespace dev
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698