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 "ui/events/ozone/device/udev/device_manager_udev.h"
9 #include "base/debug/trace_event.h"
10 #include "base/strings/stringprintf.h"
11 #include "ui/events/ozone/device/device_event.h"
12 #include "ui/events/ozone/device/device_event_observer.h"
18 const char* kSubsystems
[] = {
23 // Severity levels from syslog.h. We can't include it directly as it
24 // conflicts with base/logging.h
36 // Log handler for messages generated from libudev.
37 void UdevLog(struct udev
* udev
,
44 if (priority
<= SYS_LOG_ERR
)
45 LOG(ERROR
) << "libudev: " << fn
<< ": " << base::StringPrintV(format
, args
);
46 else if (priority
<= SYS_LOG_INFO
)
47 VLOG(1) << "libudev: " << fn
<< ": " << base::StringPrintV(format
, args
);
49 VLOG(2) << "libudev: " << fn
<< ": " << base::StringPrintV(format
, args
);
52 // Create libudev context.
53 device::ScopedUdevPtr
UdevCreate() {
54 struct udev
* udev
= udev_new();
56 udev_set_log_fn(udev
, UdevLog
);
57 udev_set_log_priority(udev
, SYS_LOG_DEBUG
);
59 return device::ScopedUdevPtr(udev
);
62 // Start monitoring input device changes.
63 device::ScopedUdevMonitorPtr
UdevCreateMonitor(struct udev
* udev
) {
64 struct udev_monitor
* monitor
= udev_monitor_new_from_netlink(udev
, "udev");
66 for (size_t i
= 0; i
< arraysize(kSubsystems
); ++i
)
67 udev_monitor_filter_add_match_subsystem_devtype(
68 monitor
, kSubsystems
[i
], NULL
);
70 if (udev_monitor_enable_receiving(monitor
))
71 LOG(ERROR
) << "Failed to start receiving events from udev";
73 LOG(ERROR
) << "Failed to create udev monitor";
76 return device::ScopedUdevMonitorPtr(monitor
);
81 DeviceManagerUdev::DeviceManagerUdev() : udev_(UdevCreate()) {
84 DeviceManagerUdev::~DeviceManagerUdev() {
87 void DeviceManagerUdev::CreateMonitor() {
90 monitor_
= UdevCreateMonitor(udev_
.get());
92 int fd
= udev_monitor_get_fd(monitor_
.get());
94 base::MessageLoopForUI::current()->WatchFileDescriptor(
95 fd
, true, base::MessagePumpLibevent::WATCH_READ
, &controller_
, this);
99 void DeviceManagerUdev::ScanDevices(DeviceEventObserver
* observer
) {
102 device::ScopedUdevEnumeratePtr
enumerate(udev_enumerate_new(udev_
.get()));
106 for (size_t i
= 0; i
< arraysize(kSubsystems
); ++i
)
107 udev_enumerate_add_match_subsystem(enumerate
.get(), kSubsystems
[i
]);
108 udev_enumerate_scan_devices(enumerate
.get());
110 struct udev_list_entry
* devices
=
111 udev_enumerate_get_list_entry(enumerate
.get());
112 struct udev_list_entry
* entry
;
114 udev_list_entry_foreach(entry
, devices
) {
115 device::ScopedUdevDevicePtr
device(udev_device_new_from_syspath(
116 udev_
.get(), udev_list_entry_get_name(entry
)));
120 scoped_ptr
<DeviceEvent
> event
= ProcessMessage(device
.get());
122 observer
->OnDeviceEvent(*event
.get());
126 void DeviceManagerUdev::AddObserver(DeviceEventObserver
* observer
) {
127 observers_
.AddObserver(observer
);
130 void DeviceManagerUdev::RemoveObserver(DeviceEventObserver
* observer
) {
131 observers_
.RemoveObserver(observer
);
134 void DeviceManagerUdev::OnFileCanReadWithoutBlocking(int fd
) {
135 // The netlink socket should never become disconnected. There's no need
136 // to handle broken connections here.
137 TRACE_EVENT1("ozone", "UdevDeviceChange", "socket", fd
);
139 device::ScopedUdevDevicePtr
device(
140 udev_monitor_receive_device(monitor_
.get()));
144 scoped_ptr
<DeviceEvent
> event
= ProcessMessage(device
.get());
147 DeviceEventObserver
, observers_
, OnDeviceEvent(*event
.get()));
150 void DeviceManagerUdev::OnFileCanWriteWithoutBlocking(int fd
) {
154 scoped_ptr
<DeviceEvent
> DeviceManagerUdev::ProcessMessage(udev_device
* device
) {
155 const char* path
= udev_device_get_devnode(device
);
156 const char* action
= udev_device_get_action(device
);
157 const char* hotplug
= udev_device_get_property_value(device
, "HOTPLUG");
158 const char* subsystem
= udev_device_get_property_value(device
, "SUBSYSTEM");
160 if (!path
|| !subsystem
)
161 return scoped_ptr
<DeviceEvent
>();
163 DeviceEvent::DeviceType device_type
;
164 if (!strcmp(subsystem
, "input") &&
165 StartsWithASCII(path
, "/dev/input/event", true))
166 device_type
= DeviceEvent::INPUT
;
167 else if (!strcmp(subsystem
, "drm") && hotplug
&& !strcmp(hotplug
, "1"))
168 device_type
= DeviceEvent::DISPLAY
;
170 return scoped_ptr
<DeviceEvent
>();
172 DeviceEvent::ActionType action_type
;
173 if (!action
|| !strcmp(action
, "add"))
174 action_type
= DeviceEvent::ADD
;
175 else if (!strcmp(action
, "remove"))
176 action_type
= DeviceEvent::REMOVE
;
177 else if (!strcmp(action
, "change"))
178 action_type
= DeviceEvent::CHANGE
;
180 return scoped_ptr
<DeviceEvent
>();
182 return scoped_ptr
<DeviceEvent
>(
183 new DeviceEvent(device_type
, action_type
, base::FilePath(path
)));