/* KWin - the KDE window manager This file is part of the KDE project. SPDX-FileCopyrightText: 2014 Martin Gräßlin SPDX-License-Identifier: GPL-2.0-or-later */ #include "udev.h" #include "utils/common.h" // Qt #include #include // system #include #include #include 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> find(); private: Udev *m_udev; struct EnumerateDeleter { void operator()(udev_enumerate *e) { udev_enumerate_unref(e); } }; std::unique_ptr 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) { return; } switch (match) { case Match::SubSystem: udev_enumerate_add_match_subsystem(m_enumerate.get(), name); break; case Match::SysName: udev_enumerate_add_match_sysname(m_enumerate.get(), name); break; default: Q_UNREACHABLE(); break; } } void UdevEnumerate::scan() { if (!m_enumerate) { return; } udev_enumerate_scan_devices(m_enumerate.get()); } std::vector> UdevEnumerate::find() { if (!m_enumerate) { return {}; } std::vector> vect; udev_list_entry *it = udev_enumerate_get_list_entry(m_enumerate.get()); 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) { vect.push_back(std::move(device)); } } return vect; } std::vector> Udev::listGPUs() { if (!m_udev) { return {}; } #if defined(Q_OS_FREEBSD) std::vector> 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 auto &device1, const auto &device2) { // prevent usb devices from becoming the primaryGpu if (device1->isHotpluggable()) { return false; } if (device2->isHotpluggable()) { return true; } // 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::unique_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 std::make_unique(dev); } std::unique_ptr Udev::monitor() { auto m = std::make_unique(this); if (m->isValid()) { return m; } else { return nullptr; } } 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 UdevDevice::properties() const { QMap 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::seat() const { QString deviceSeat = udev_device_get_property_value(m_device, "ID_SEAT"); if (deviceSeat.isEmpty()) { deviceSeat = QStringLiteral("seat0"); } return deviceSeat; } QString UdevDevice::action() const { return QString::fromLocal8Bit(udev_device_get_action(m_device)); } bool UdevDevice::isHotpluggable() const { QString devPath = QString::fromUtf8(udev_device_get_devpath(m_device)); return devPath.toLower().contains("usb"); } 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); } std::unique_ptr UdevMonitor::getDevice() { if (!m_monitor) { return nullptr; } auto dev = udev_monitor_receive_device(m_monitor); return dev ? std::make_unique(dev) : nullptr; } }