2020-08-02 22:22:19 +00:00
|
|
|
/*
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
2015-04-08 08:25:51 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-FileCopyrightText: 2014 Martin Gräßlin <mgraesslin@kde.org>
|
2015-04-08 08:25:51 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
2015-04-08 08:25:51 +00:00
|
|
|
#include "udev.h"
|
2022-01-18 08:35:52 +00:00
|
|
|
#include "utils/common.h"
|
2015-04-08 08:45:52 +00:00
|
|
|
// Qt
|
|
|
|
#include <QByteArray>
|
2021-04-12 18:09:10 +00:00
|
|
|
#include <QDebug>
|
2015-04-08 08:45:52 +00:00
|
|
|
// system
|
2021-04-16 20:32:50 +00:00
|
|
|
#include <cerrno>
|
2022-03-23 10:13:38 +00:00
|
|
|
#include <functional>
|
|
|
|
#include <libudev.h>
|
2015-04-08 08:25:51 +00:00
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
|
|
|
Udev::Udev()
|
|
|
|
: m_udev(udev_new())
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
Udev::~Udev()
|
|
|
|
{
|
|
|
|
if (m_udev) {
|
|
|
|
udev_unref(m_udev);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-08 08:45:52 +00:00
|
|
|
class UdevEnumerate
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
UdevEnumerate(Udev *udev);
|
|
|
|
~UdevEnumerate();
|
|
|
|
|
|
|
|
enum class Match {
|
|
|
|
SubSystem,
|
|
|
|
SysName
|
|
|
|
};
|
|
|
|
void addMatch(Match match, const char *name);
|
|
|
|
void scan();
|
2023-08-10 16:58:58 +00:00
|
|
|
std::vector<std::unique_ptr<UdevDevice>> find();
|
2015-04-08 08:45:52 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
Udev *m_udev;
|
|
|
|
|
|
|
|
struct EnumerateDeleter
|
|
|
|
{
|
2022-08-01 21:29:02 +00:00
|
|
|
void operator()(udev_enumerate *e)
|
2015-04-08 08:45:52 +00:00
|
|
|
{
|
|
|
|
udev_enumerate_unref(e);
|
|
|
|
}
|
|
|
|
};
|
2022-08-01 21:29:02 +00:00
|
|
|
std::unique_ptr<udev_enumerate, EnumerateDeleter> m_enumerate;
|
2015-04-08 08:45:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
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)
|
|
|
|
{
|
2022-08-01 21:29:02 +00:00
|
|
|
if (!m_enumerate) {
|
2015-04-08 08:45:52 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
switch (match) {
|
|
|
|
case Match::SubSystem:
|
2022-08-01 21:29:02 +00:00
|
|
|
udev_enumerate_add_match_subsystem(m_enumerate.get(), name);
|
2015-04-08 08:45:52 +00:00
|
|
|
break;
|
|
|
|
case Match::SysName:
|
2022-08-01 21:29:02 +00:00
|
|
|
udev_enumerate_add_match_sysname(m_enumerate.get(), name);
|
2015-04-08 08:45:52 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
Q_UNREACHABLE();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void UdevEnumerate::scan()
|
|
|
|
{
|
2022-08-01 21:29:02 +00:00
|
|
|
if (!m_enumerate) {
|
2015-04-08 08:45:52 +00:00
|
|
|
return;
|
|
|
|
}
|
2022-08-01 21:29:02 +00:00
|
|
|
udev_enumerate_scan_devices(m_enumerate.get());
|
2015-04-08 08:45:52 +00:00
|
|
|
}
|
|
|
|
|
2023-08-10 16:58:58 +00:00
|
|
|
std::vector<std::unique_ptr<UdevDevice>> UdevEnumerate::find()
|
2015-04-08 08:45:52 +00:00
|
|
|
{
|
2022-08-01 21:29:02 +00:00
|
|
|
if (!m_enumerate) {
|
2021-04-26 13:37:09 +00:00
|
|
|
return {};
|
2015-04-08 08:45:52 +00:00
|
|
|
}
|
2023-08-10 16:58:58 +00:00
|
|
|
std::vector<std::unique_ptr<UdevDevice>> vect;
|
2022-08-01 21:29:02 +00:00
|
|
|
udev_list_entry *it = udev_enumerate_get_list_entry(m_enumerate.get());
|
2015-04-08 08:45:52 +00:00
|
|
|
while (it) {
|
|
|
|
auto current = it;
|
|
|
|
it = udev_list_entry_get_next(it);
|
|
|
|
auto device = m_udev->deviceFromSyspath(udev_list_entry_get_name(current));
|
2022-07-24 16:36:50 +00:00
|
|
|
if (device) {
|
|
|
|
vect.push_back(std::move(device));
|
2015-04-08 08:45:52 +00:00
|
|
|
}
|
|
|
|
}
|
2020-10-05 21:05:55 +00:00
|
|
|
return vect;
|
2015-04-08 08:45:52 +00:00
|
|
|
}
|
|
|
|
|
2023-08-10 16:58:58 +00:00
|
|
|
std::vector<std::unique_ptr<UdevDevice>> Udev::listGPUs()
|
2015-04-08 08:45:52 +00:00
|
|
|
{
|
|
|
|
if (!m_udev) {
|
2021-04-26 13:37:09 +00:00
|
|
|
return {};
|
2015-04-08 08:45:52 +00:00
|
|
|
}
|
2020-05-31 21:52:28 +00:00
|
|
|
#if defined(Q_OS_FREEBSD)
|
2023-08-10 16:58:58 +00:00
|
|
|
std::vector<std::unique_ptr<UdevDevice>> r;
|
2020-10-05 21:05:55 +00:00
|
|
|
r.push_back(deviceFromSyspath("/dev/dri/card0"));
|
|
|
|
return r;
|
2020-05-31 21:52:28 +00:00
|
|
|
#else
|
2015-04-08 08:45:52 +00:00
|
|
|
UdevEnumerate enumerate(this);
|
|
|
|
enumerate.addMatch(UdevEnumerate::Match::SubSystem, "drm");
|
2020-10-05 21:05:55 +00:00
|
|
|
enumerate.addMatch(UdevEnumerate::Match::SysName, "card[0-9]");
|
2015-04-08 08:45:52 +00:00
|
|
|
enumerate.scan();
|
2020-10-05 21:05:55 +00:00
|
|
|
auto vect = enumerate.find();
|
2023-08-10 16:58:58 +00:00
|
|
|
std::sort(vect.begin(), vect.end(), [](const auto &device1, const auto &device2) {
|
2023-07-18 08:44:09 +00:00
|
|
|
// prevent usb devices from becoming the primaryGpu
|
|
|
|
if (device1->isHotpluggable()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (device2->isHotpluggable()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-10-05 21:05:55 +00:00
|
|
|
// if set as boot GPU, prefer 1
|
2021-04-26 13:18:41 +00:00
|
|
|
if (device1->isBootVga()) {
|
|
|
|
return true;
|
2015-04-08 08:45:52 +00:00
|
|
|
}
|
2020-10-05 21:05:55 +00:00
|
|
|
// if set as boot GPU, prefer 2
|
2021-04-26 13:18:41 +00:00
|
|
|
if (device2->isBootVga()) {
|
|
|
|
return false;
|
2020-10-05 21:05:55 +00:00
|
|
|
}
|
2020-10-12 18:04:11 +00:00
|
|
|
return true;
|
2015-04-08 08:45:52 +00:00
|
|
|
});
|
2020-10-05 21:05:55 +00:00
|
|
|
return vect;
|
2020-05-31 21:52:28 +00:00
|
|
|
#endif
|
2015-04-08 08:45:52 +00:00
|
|
|
}
|
|
|
|
|
2023-08-10 16:58:58 +00:00
|
|
|
std::unique_ptr<UdevDevice> Udev::deviceFromSyspath(const char *syspath)
|
2015-04-08 08:45:52 +00:00
|
|
|
{
|
2021-04-12 18:09:10 +00:00
|
|
|
auto dev = udev_device_new_from_syspath(m_udev, syspath);
|
|
|
|
if (!dev) {
|
|
|
|
qCWarning(KWIN_CORE) << "failed to retrieve device for" << syspath << strerror(errno);
|
|
|
|
return {};
|
|
|
|
}
|
2023-08-10 16:53:37 +00:00
|
|
|
return std::make_unique<UdevDevice>(dev);
|
2015-04-08 08:45:52 +00:00
|
|
|
}
|
|
|
|
|
2022-05-17 10:36:34 +00:00
|
|
|
std::unique_ptr<UdevMonitor> Udev::monitor()
|
2015-04-23 13:08:24 +00:00
|
|
|
{
|
2022-05-17 10:36:34 +00:00
|
|
|
auto m = std::make_unique<UdevMonitor>(this);
|
|
|
|
if (m->isValid()) {
|
|
|
|
return m;
|
|
|
|
} else {
|
|
|
|
return nullptr;
|
2015-04-23 13:08:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-08 08:45:52 +00:00
|
|
|
UdevDevice::UdevDevice(udev_device *device)
|
|
|
|
: m_device(device)
|
|
|
|
{
|
2021-04-12 18:09:10 +00:00
|
|
|
Q_ASSERT(device);
|
2015-04-08 08:45:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
UdevDevice::~UdevDevice()
|
|
|
|
{
|
2021-04-12 18:09:10 +00:00
|
|
|
udev_device_unref(m_device);
|
2015-04-08 08:45:52 +00:00
|
|
|
}
|
|
|
|
|
2021-04-30 07:05:49 +00:00
|
|
|
QString UdevDevice::devNode() const
|
2015-04-08 08:45:52 +00:00
|
|
|
{
|
2021-04-30 07:05:49 +00:00
|
|
|
return QString::fromUtf8(udev_device_get_devnode(m_device));
|
2015-04-08 08:45:52 +00:00
|
|
|
}
|
|
|
|
|
2021-04-28 18:30:16 +00:00
|
|
|
dev_t UdevDevice::devNum() const
|
2015-04-09 12:49:32 +00:00
|
|
|
{
|
2021-04-28 18:30:16 +00:00
|
|
|
return udev_device_get_devnum(m_device);
|
2015-04-09 12:49:32 +00:00
|
|
|
}
|
|
|
|
|
2015-04-23 13:07:30 +00:00
|
|
|
const char *UdevDevice::property(const char *key)
|
|
|
|
{
|
|
|
|
return udev_device_get_property_value(m_device, key);
|
|
|
|
}
|
|
|
|
|
2021-04-12 17:53:50 +00:00
|
|
|
QMap<QByteArray, QByteArray> UdevDevice::properties() const
|
|
|
|
{
|
|
|
|
QMap<QByteArray, QByteArray> r;
|
|
|
|
auto it = udev_device_get_properties_list_entry(m_device);
|
|
|
|
auto current = it;
|
2022-03-23 10:13:38 +00:00
|
|
|
udev_list_entry_foreach(current, it)
|
|
|
|
{
|
2021-04-12 17:53:50 +00:00
|
|
|
r.insert(udev_list_entry_get_name(current), udev_list_entry_get_value(current));
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2015-04-23 13:07:30 +00:00
|
|
|
bool UdevDevice::hasProperty(const char *key, const char *value)
|
|
|
|
{
|
|
|
|
const char *p = property(key);
|
|
|
|
if (!p) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return qstrcmp(p, value) == 0;
|
|
|
|
}
|
|
|
|
|
2021-04-26 13:18:41 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2022-07-24 16:36:50 +00:00
|
|
|
QString UdevDevice::seat() const
|
|
|
|
{
|
|
|
|
QString deviceSeat = udev_device_get_property_value(m_device, "ID_SEAT");
|
|
|
|
if (deviceSeat.isEmpty()) {
|
|
|
|
deviceSeat = QStringLiteral("seat0");
|
|
|
|
}
|
|
|
|
return deviceSeat;
|
|
|
|
}
|
|
|
|
|
2021-04-27 11:49:53 +00:00
|
|
|
QString UdevDevice::action() const
|
|
|
|
{
|
|
|
|
return QString::fromLocal8Bit(udev_device_get_action(m_device));
|
|
|
|
}
|
|
|
|
|
2023-07-18 08:44:09 +00:00
|
|
|
bool UdevDevice::isHotpluggable() const
|
|
|
|
{
|
|
|
|
QString devPath = QString::fromUtf8(udev_device_get_devpath(m_device));
|
|
|
|
return devPath.toLower().contains("usb");
|
|
|
|
}
|
|
|
|
|
2015-04-23 13:08:24 +00:00
|
|
|
UdevMonitor::UdevMonitor(Udev *udev)
|
2017-07-31 16:22:14 +00:00
|
|
|
: m_monitor(udev_monitor_new_from_netlink(*udev, "udev"))
|
2015-04-23 13:08:24 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2023-08-10 16:58:58 +00:00
|
|
|
std::unique_ptr<UdevDevice> UdevMonitor::getDevice()
|
2015-04-23 13:08:24 +00:00
|
|
|
{
|
|
|
|
if (!m_monitor) {
|
2023-08-10 16:58:58 +00:00
|
|
|
return nullptr;
|
2015-04-23 13:08:24 +00:00
|
|
|
}
|
2021-04-12 17:52:42 +00:00
|
|
|
auto dev = udev_monitor_receive_device(m_monitor);
|
2023-08-10 16:53:37 +00:00
|
|
|
return dev ? std::make_unique<UdevDevice>(dev) : nullptr;
|
2015-04-23 13:08:24 +00:00
|
|
|
}
|
|
|
|
|
2015-04-08 08:25:51 +00:00
|
|
|
}
|