You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
310 lines
6.8 KiB
310 lines
6.8 KiB
/* |
|
KWin - the KDE window manager |
|
This file is part of the KDE project. |
|
|
|
SPDX-FileCopyrightText: 2014 Martin Gräßlin <mgraesslin@kde.org> |
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later |
|
*/ |
|
#include "udev.h" |
|
#include "main.h" |
|
#include "platform.h" |
|
#include "session.h" |
|
#include "utils.h" |
|
// Qt |
|
#include <QByteArray> |
|
#include <QScopedPointer> |
|
#include <QDebug> |
|
// system |
|
#include <libudev.h> |
|
#include <functional> |
|
#include <cerrno> |
|
|
|
namespace KWin |
|
{ |
|
|
|
Udev::Udev() |
|
: m_udev(udev_new()) |
|
{ |
|
} |
|
|
|
Udev::~Udev() |
|
{ |
|
if (m_udev) { |
|
udev_unref(m_udev); |
|
} |
|
} |
|
|
|
class UdevEnumerate |
|
{ |
|
public: |
|
UdevEnumerate(Udev *udev); |
|
~UdevEnumerate(); |
|
|
|
enum class Match { |
|
SubSystem, |
|
SysName |
|
}; |
|
void addMatch(Match match, const char *name); |
|
void scan(); |
|
std::vector<UdevDevice::Ptr> find(); |
|
|
|
private: |
|
Udev *m_udev; |
|
|
|
struct EnumerateDeleter |
|
{ |
|
static inline void cleanup(udev_enumerate *e) |
|
{ |
|
udev_enumerate_unref(e); |
|
} |
|
}; |
|
QScopedPointer<udev_enumerate, EnumerateDeleter> m_enumerate; |
|
}; |
|
|
|
UdevEnumerate::UdevEnumerate(Udev *udev) |
|
: m_udev(udev) |
|
, m_enumerate(udev_enumerate_new(*m_udev)) |
|
{ |
|
} |
|
|
|
UdevEnumerate::~UdevEnumerate() = default; |
|
|
|
void UdevEnumerate::addMatch(UdevEnumerate::Match match, const char *name) |
|
{ |
|
if (m_enumerate.isNull()) { |
|
return; |
|
} |
|
switch (match) { |
|
case Match::SubSystem: |
|
udev_enumerate_add_match_subsystem(m_enumerate.data(), name); |
|
break; |
|
case Match::SysName: |
|
udev_enumerate_add_match_sysname(m_enumerate.data(), name); |
|
break; |
|
default: |
|
Q_UNREACHABLE(); |
|
break; |
|
} |
|
} |
|
|
|
void UdevEnumerate::scan() |
|
{ |
|
if (m_enumerate.isNull()) { |
|
return; |
|
} |
|
udev_enumerate_scan_devices(m_enumerate.data()); |
|
} |
|
|
|
std::vector<UdevDevice::Ptr> UdevEnumerate::find() |
|
{ |
|
if (m_enumerate.isNull()) { |
|
return {}; |
|
} |
|
std::vector<UdevDevice::Ptr> vect; |
|
QString defaultSeat = QStringLiteral("seat0"); |
|
udev_list_entry *it = udev_enumerate_get_list_entry(m_enumerate.data()); |
|
while (it) { |
|
auto current = it; |
|
it = udev_list_entry_get_next(it); |
|
auto device = m_udev->deviceFromSyspath(udev_list_entry_get_name(current)); |
|
if (!device) { |
|
continue; |
|
} |
|
QString deviceSeat = device->property("ID_SEAT"); |
|
if (deviceSeat.isEmpty()) { |
|
deviceSeat = defaultSeat; |
|
} |
|
if (deviceSeat != kwinApp()->platform()->session()->seat()) { |
|
continue; |
|
} |
|
vect.push_back(std::move(device)); |
|
} |
|
return vect; |
|
} |
|
|
|
std::vector<UdevDevice::Ptr> Udev::listGPUs() |
|
{ |
|
if (!m_udev) { |
|
return {}; |
|
} |
|
#if defined(Q_OS_FREEBSD) |
|
std::vector<UdevDevice::Ptr> r; |
|
r.push_back(deviceFromSyspath("/dev/dri/card0")); |
|
return r; |
|
#else |
|
UdevEnumerate enumerate(this); |
|
enumerate.addMatch(UdevEnumerate::Match::SubSystem, "drm"); |
|
enumerate.addMatch(UdevEnumerate::Match::SysName, "card[0-9]"); |
|
enumerate.scan(); |
|
auto vect = enumerate.find(); |
|
std::sort(vect.begin(), vect.end(), [](const UdevDevice::Ptr &device1, const UdevDevice::Ptr &device2) { |
|
// if set as boot GPU, prefer 1 |
|
if (device1->isBootVga()) { |
|
return true; |
|
} |
|
// if set as boot GPU, prefer 2 |
|
if (device2->isBootVga()) { |
|
return false; |
|
} |
|
return true; |
|
}); |
|
return vect; |
|
#endif |
|
} |
|
|
|
std::vector<UdevDevice::Ptr> Udev::listFramebuffers() |
|
{ |
|
if (!m_udev) { |
|
return {}; |
|
} |
|
UdevEnumerate enumerate(this); |
|
enumerate.addMatch(UdevEnumerate::Match::SubSystem, "graphics"); |
|
enumerate.addMatch(UdevEnumerate::Match::SysName, "fb[0-9]"); |
|
enumerate.scan(); |
|
auto vect = enumerate.find(); |
|
std::sort(vect.begin(), vect.end(), [](const UdevDevice::Ptr &device1, const UdevDevice::Ptr &device2) { |
|
// if set as boot GPU, prefer 1 |
|
if (device1->isBootVga()) { |
|
return true; |
|
} |
|
// if set as boot GPU, prefer 2 |
|
if (device2->isBootVga()) { |
|
return false; |
|
} |
|
return true; |
|
}); |
|
return vect; |
|
} |
|
|
|
UdevDevice::Ptr Udev::deviceFromSyspath(const char *syspath) |
|
{ |
|
auto dev = udev_device_new_from_syspath(m_udev, syspath); |
|
if (!dev) { |
|
qCWarning(KWIN_CORE) << "failed to retrieve device for" << syspath << strerror(errno); |
|
return {}; |
|
} |
|
return UdevDevice::Ptr(new UdevDevice(dev)); |
|
} |
|
|
|
UdevMonitor *Udev::monitor() |
|
{ |
|
UdevMonitor *m = new UdevMonitor(this); |
|
if (!m->isValid()) { |
|
delete m; |
|
m = nullptr; |
|
} |
|
return m; |
|
} |
|
|
|
UdevDevice::UdevDevice(udev_device *device) |
|
: m_device(device) |
|
{ |
|
Q_ASSERT(device); |
|
} |
|
|
|
UdevDevice::~UdevDevice() |
|
{ |
|
udev_device_unref(m_device); |
|
} |
|
|
|
QString UdevDevice::devNode() const |
|
{ |
|
return QString::fromUtf8(udev_device_get_devnode(m_device)); |
|
} |
|
|
|
dev_t UdevDevice::devNum() const |
|
{ |
|
return udev_device_get_devnum(m_device); |
|
} |
|
|
|
const char *UdevDevice::property(const char *key) |
|
{ |
|
return udev_device_get_property_value(m_device, key); |
|
} |
|
|
|
QMap<QByteArray, QByteArray> UdevDevice::properties() const |
|
{ |
|
QMap<QByteArray, QByteArray> r; |
|
auto it = udev_device_get_properties_list_entry(m_device); |
|
auto current = it; |
|
udev_list_entry_foreach (current, it) { |
|
r.insert(udev_list_entry_get_name(current), udev_list_entry_get_value(current)); |
|
} |
|
return r; |
|
} |
|
|
|
bool UdevDevice::hasProperty(const char *key, const char *value) |
|
{ |
|
const char *p = property(key); |
|
if (!p) { |
|
return false; |
|
} |
|
return qstrcmp(p, value) == 0; |
|
} |
|
|
|
bool UdevDevice::isBootVga() const |
|
{ |
|
auto pci = udev_device_get_parent_with_subsystem_devtype(m_device, "pci", nullptr); |
|
if (!pci) { |
|
return false; |
|
} |
|
const char *systAttrValue = udev_device_get_sysattr_value(pci, "boot_vga"); |
|
return systAttrValue && qstrcmp(systAttrValue, "1") == 0; |
|
} |
|
|
|
QString UdevDevice::action() const |
|
{ |
|
return QString::fromLocal8Bit(udev_device_get_action(m_device)); |
|
} |
|
|
|
UdevMonitor::UdevMonitor(Udev *udev) |
|
: m_monitor(udev_monitor_new_from_netlink(*udev, "udev")) |
|
{ |
|
} |
|
|
|
UdevMonitor::~UdevMonitor() |
|
{ |
|
if (m_monitor) { |
|
udev_monitor_unref(m_monitor); |
|
} |
|
} |
|
|
|
int UdevMonitor::fd() const |
|
{ |
|
if (m_monitor) { |
|
return udev_monitor_get_fd(m_monitor); |
|
} |
|
return -1; |
|
} |
|
|
|
void UdevMonitor::filterSubsystemDevType(const char *subSystem, const char *devType) |
|
{ |
|
if (!m_monitor) { |
|
return; |
|
} |
|
udev_monitor_filter_add_match_subsystem_devtype(m_monitor, subSystem, devType); |
|
} |
|
|
|
void UdevMonitor::enable() |
|
{ |
|
if (!m_monitor) { |
|
return; |
|
} |
|
udev_monitor_enable_receiving(m_monitor); |
|
} |
|
|
|
UdevDevice::Ptr UdevMonitor::getDevice() |
|
{ |
|
if (!m_monitor) { |
|
return UdevDevice::Ptr(); |
|
} |
|
auto dev = udev_monitor_receive_device(m_monitor); |
|
if (!dev) { |
|
return {}; |
|
} |
|
|
|
return UdevDevice::Ptr(new UdevDevice(dev)); |
|
} |
|
|
|
}
|
|
|