1 // Copyright 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.
5 #include "device/hid/hid_service_mac.h"
7 #include <CoreFoundation/CoreFoundation.h>
8 #include <IOKit/hid/IOHIDDevice.h>
14 #include "base/bind.h"
15 #include "base/location.h"
16 #include "base/logging.h"
17 #include "base/mac/foundation_util.h"
18 #include "base/single_thread_task_runner.h"
19 #include "base/stl_util.h"
20 #include "base/strings/stringprintf.h"
21 #include "base/strings/sys_string_conversions.h"
22 #include "base/thread_task_runner_handle.h"
23 #include "base/threading/thread_restrictions.h"
24 #include "components/device_event_log/device_event_log.h"
25 #include "device/hid/hid_connection_mac.h"
31 std::string
HexErrorCode(IOReturn error_code
) {
32 return base::StringPrintf("0x%04x", error_code
);
35 bool TryGetHidIntProperty(IOHIDDeviceRef device
,
39 base::mac::CFCast
<CFNumberRef
>(IOHIDDeviceGetProperty(device
, key
));
40 return ref
&& CFNumberGetValue(ref
, kCFNumberSInt32Type
, result
);
43 int32_t GetHidIntProperty(IOHIDDeviceRef device
, CFStringRef key
) {
45 if (TryGetHidIntProperty(device
, key
, &value
))
50 bool TryGetHidStringProperty(IOHIDDeviceRef device
,
52 std::string
* result
) {
54 base::mac::CFCast
<CFStringRef
>(IOHIDDeviceGetProperty(device
, key
));
58 *result
= base::SysCFStringRefToUTF8(ref
);
62 std::string
GetHidStringProperty(IOHIDDeviceRef device
, CFStringRef key
) {
64 TryGetHidStringProperty(device
, key
, &value
);
68 bool TryGetHidDataProperty(IOHIDDeviceRef device
,
70 std::vector
<uint8
>* result
) {
72 base::mac::CFCast
<CFDataRef
>(IOHIDDeviceGetProperty(device
, key
));
76 STLClearObject(result
);
77 const uint8
* bytes
= CFDataGetBytePtr(ref
);
78 result
->insert(result
->begin(), bytes
, bytes
+ CFDataGetLength(ref
));
84 HidServiceMac::HidServiceMac(
85 scoped_refptr
<base::SingleThreadTaskRunner
> file_task_runner
)
86 : file_task_runner_(file_task_runner
) {
87 task_runner_
= base::ThreadTaskRunnerHandle::Get();
88 DCHECK(task_runner_
.get());
90 notify_port_
= IONotificationPortCreate(kIOMasterPortDefault
);
91 CFRunLoopAddSource(CFRunLoopGetMain(),
92 IONotificationPortGetRunLoopSource(notify_port_
),
93 kCFRunLoopDefaultMode
);
95 io_iterator_t iterator
;
97 IOServiceAddMatchingNotification(notify_port_
,
98 kIOFirstMatchNotification
,
99 IOServiceMatching(kIOHIDDeviceKey
),
103 if (result
!= kIOReturnSuccess
) {
104 HID_LOG(ERROR
) << "Failed to listen for device arrival: "
105 << HexErrorCode(result
);
109 // Drain the iterator to arm the notification.
110 devices_added_iterator_
.reset(iterator
);
112 iterator
= IO_OBJECT_NULL
;
114 result
= IOServiceAddMatchingNotification(notify_port_
,
115 kIOTerminatedNotification
,
116 IOServiceMatching(kIOHIDDeviceKey
),
120 if (result
!= kIOReturnSuccess
) {
121 HID_LOG(ERROR
) << "Failed to listen for device removal: "
122 << HexErrorCode(result
);
126 // Drain devices_added_iterator_ to arm the notification.
127 devices_removed_iterator_
.reset(iterator
);
129 FirstEnumerationComplete();
132 void HidServiceMac::Connect(const HidDeviceId
& device_id
,
133 const ConnectCallback
& callback
) {
134 DCHECK(thread_checker_
.CalledOnValidThread());
136 const auto& map_entry
= devices().find(device_id
);
137 if (map_entry
== devices().end()) {
138 task_runner_
->PostTask(FROM_HERE
, base::Bind(callback
, nullptr));
141 scoped_refptr
<HidDeviceInfo
> device_info
= map_entry
->second
;
143 base::ScopedCFTypeRef
<CFDictionaryRef
> matching_dict(
144 IORegistryEntryIDMatching(device_id
));
145 if (!matching_dict
.get()) {
146 HID_LOG(EVENT
) << "Failed to create matching dictionary for ID: "
148 task_runner_
->PostTask(FROM_HERE
, base::Bind(callback
, nullptr));
152 // IOServiceGetMatchingService consumes a reference to the matching dictionary
154 base::mac::ScopedIOObject
<io_service_t
> service(IOServiceGetMatchingService(
155 kIOMasterPortDefault
, matching_dict
.release()));
156 if (!service
.get()) {
157 HID_LOG(EVENT
) << "IOService not found for ID: " << device_id
;
158 task_runner_
->PostTask(FROM_HERE
, base::Bind(callback
, nullptr));
162 base::ScopedCFTypeRef
<IOHIDDeviceRef
> hid_device(
163 IOHIDDeviceCreate(kCFAllocatorDefault
, service
));
165 HID_LOG(EVENT
) << "Unable to create IOHIDDevice object.";
166 task_runner_
->PostTask(FROM_HERE
, base::Bind(callback
, nullptr));
170 IOReturn result
= IOHIDDeviceOpen(hid_device
, kIOHIDOptionsTypeNone
);
171 if (result
!= kIOReturnSuccess
) {
172 HID_LOG(EVENT
) << "Failed to open device: " << HexErrorCode(result
);
173 task_runner_
->PostTask(FROM_HERE
, base::Bind(callback
, nullptr));
177 task_runner_
->PostTask(
178 FROM_HERE
, base::Bind(callback
, make_scoped_refptr(new HidConnectionMac(
179 hid_device
.release(), device_info
,
180 file_task_runner_
))));
183 HidServiceMac::~HidServiceMac() {
187 void HidServiceMac::FirstMatchCallback(void* context
, io_iterator_t iterator
) {
188 DCHECK_EQ(CFRunLoopGetMain(), CFRunLoopGetCurrent());
189 HidServiceMac
* service
= static_cast<HidServiceMac
*>(context
);
190 DCHECK_EQ(service
->devices_added_iterator_
, iterator
);
191 service
->AddDevices();
195 void HidServiceMac::TerminatedCallback(void* context
, io_iterator_t iterator
) {
196 DCHECK_EQ(CFRunLoopGetMain(), CFRunLoopGetCurrent());
197 HidServiceMac
* service
= static_cast<HidServiceMac
*>(context
);
198 DCHECK_EQ(service
->devices_removed_iterator_
, iterator
);
199 service
->RemoveDevices();
202 void HidServiceMac::AddDevices() {
203 DCHECK(thread_checker_
.CalledOnValidThread());
206 while ((device
= IOIteratorNext(devices_added_iterator_
)) != IO_OBJECT_NULL
) {
207 scoped_refptr
<HidDeviceInfo
> device_info
= CreateDeviceInfo(device
);
209 AddDevice(device_info
);
210 // The reference retained by IOIteratorNext is released below in
211 // RemoveDevices when the device is removed.
213 IOObjectRelease(device
);
218 void HidServiceMac::RemoveDevices() {
219 DCHECK(thread_checker_
.CalledOnValidThread());
222 while ((device
= IOIteratorNext(devices_removed_iterator_
)) !=
225 IOReturn result
= IORegistryEntryGetRegistryEntryID(device
, &entry_id
);
226 if (result
== kIOReturnSuccess
) {
227 RemoveDevice(entry_id
);
230 // Release reference retained by AddDevices above.
231 IOObjectRelease(device
);
232 // Release the reference retained by IOIteratorNext.
233 IOObjectRelease(device
);
238 scoped_refptr
<HidDeviceInfo
> HidServiceMac::CreateDeviceInfo(
239 io_service_t service
) {
241 IOReturn result
= IORegistryEntryGetRegistryEntryID(service
, &entry_id
);
242 if (result
!= kIOReturnSuccess
) {
243 HID_LOG(EVENT
) << "Failed to get IORegistryEntry ID: "
244 << HexErrorCode(result
);
248 base::ScopedCFTypeRef
<IOHIDDeviceRef
> hid_device(
249 IOHIDDeviceCreate(kCFAllocatorDefault
, service
));
251 HID_LOG(EVENT
) << "Unable to create IOHIDDevice object for new device.";
255 std::vector
<uint8
> report_descriptor
;
256 if (!TryGetHidDataProperty(hid_device
, CFSTR(kIOHIDReportDescriptorKey
),
257 &report_descriptor
)) {
258 HID_LOG(EVENT
) << "Unable to get report descriptor for new device.";
262 return new HidDeviceInfo(
263 entry_id
, GetHidIntProperty(hid_device
, CFSTR(kIOHIDVendorIDKey
)),
264 GetHidIntProperty(hid_device
, CFSTR(kIOHIDProductIDKey
)),
265 GetHidStringProperty(hid_device
, CFSTR(kIOHIDProductKey
)),
266 GetHidStringProperty(hid_device
, CFSTR(kIOHIDSerialNumberKey
)),
267 kHIDBusTypeUSB
, // TODO(reillyg): Detect Bluetooth. crbug.com/443335
271 } // namespace device