From 38286174dd6b3df4e460977326423a6bbda59700 Mon Sep 17 00:00:00 2001 From: Denis Shienkov Date: Fri, 28 Nov 2014 21:25:36 +0000 Subject: Improve QSPI algorithm on Unix QSPI uses the wrong algorithm of devices search. In case no any devices are detected with udev (or with sysfs), starts a search with a filters. In this case a user gets a list of devices that are not a serial ports, that is wrong. The main idea should be in that udev or sysfs are the main sources of obtaining information. In case they returns an empty list this means that system has no serial ports. Algorithm is: Try to find devices through udev. If it fails (system error or udev is not present), then try to find using sysfs; otherwise return udev result, even if it has an empty list. If sysfs is fails (permission denied or sysfs is not present), then try to find devices in /dev/ as last attempt; otherwise return sysfs result, even if it has an empty list. Tested on Linux (with udev, sysfs, filters) using FTDI serial ports. Change-Id: I0132e27f720b007ea3f4861e9cd7ed77833cff8c Reviewed-by: Sergey Belyashov --- src/serialport/qserialportinfo.h | 6 +++--- src/serialport/qserialportinfo_unix.cpp | 30 +++++++++++++++++++----------- 2 files changed, 22 insertions(+), 14 deletions(-) diff --git a/src/serialport/qserialportinfo.h b/src/serialport/qserialportinfo.h index 0c7a0f1..714708d 100644 --- a/src/serialport/qserialportinfo.h +++ b/src/serialport/qserialportinfo.h @@ -82,9 +82,9 @@ public: private: QSerialPortInfo(const QSerialPortInfoPrivate &dd); - friend QList availablePortsByUdev(); - friend QList availablePortsBySysfs(); - friend QList availablePortsByFiltersOfDevices(); + friend QList availablePortsByUdev(bool &ok); + friend QList availablePortsBySysfs(bool &ok); + friend QList availablePortsByFiltersOfDevices(bool &ok); QScopedPointer d_ptr; }; diff --git a/src/serialport/qserialportinfo_unix.cpp b/src/serialport/qserialportinfo_unix.cpp index 9637581..8bf5692 100644 --- a/src/serialport/qserialportinfo_unix.cpp +++ b/src/serialport/qserialportinfo_unix.cpp @@ -93,7 +93,7 @@ static QStringList filteredDeviceFilePaths() return result; } -QList availablePortsByFiltersOfDevices() +QList availablePortsByFiltersOfDevices(bool &ok) { QList serialPortInfoList; @@ -104,15 +104,18 @@ QList availablePortsByFiltersOfDevices() serialPortInfoList.append(priv); } + ok = true; return serialPortInfoList; } -QList availablePortsBySysfs() +QList availablePortsBySysfs(bool &ok) { QDir ttySysClassDir(QStringLiteral("/sys/class/tty")); - if (!(ttySysClassDir.exists() && ttySysClassDir.isReadable())) + if (!(ttySysClassDir.exists() && ttySysClassDir.isReadable())) { + ok = false; return QList(); + } QList serialPortInfoList; ttySysClassDir.setFilter(QDir::Dirs | QDir::NoDotAndDotDot); @@ -206,6 +209,7 @@ QList availablePortsBySysfs() serialPortInfoList.append(priv); } + ok = true; return serialPortInfoList; } @@ -282,8 +286,10 @@ QString getUdevSerialNumber(struct ::udev_device *dev) return getUdevPropertyValue(dev,"ID_SERIAL_SHORT"); } -QList availablePortsByUdev() +QList availablePortsByUdev(bool &ok) { + ok = false; + #ifndef LINK_LIBUDEV static bool symbolsResolved = resolveSymbols(udevLibrary()); if (!symbolsResolved) @@ -311,6 +317,8 @@ QList availablePortsByUdev() udev_list_entry *dev_list_entry; udev_list_entry_foreach(dev_list_entry, devices) { + ok = true; + QScopedPointer dev(::udev_device_new_from_syspath( udev.data(), ::udev_list_entry_get_name(dev_list_entry))); @@ -352,17 +360,17 @@ QList availablePortsByUdev() QList QSerialPortInfo::availablePorts() { - QList serialPortInfoList = availablePortsByUdev(); + bool ok; + + QList serialPortInfoList = availablePortsByUdev(ok); #ifdef Q_OS_LINUX - if (serialPortInfoList.isEmpty()) - serialPortInfoList = availablePortsBySysfs(); - else - return serialPortInfoList; + if (!ok) + serialPortInfoList = availablePortsBySysfs(ok); #endif - if (serialPortInfoList.isEmpty()) - serialPortInfoList = availablePortsByFiltersOfDevices(); + if (!ok) + serialPortInfoList = availablePortsByFiltersOfDevices(ok); return serialPortInfoList; } -- cgit v1.2.1