diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/serialport/qserialportinfo_unix.cpp | 14 | ||||
-rw-r--r-- | src/serialport/qtudev_p.h | 144 | ||||
-rw-r--r-- | src/serialport/serialport-lib.pri | 4 |
3 files changed, 154 insertions, 8 deletions
diff --git a/src/serialport/qserialportinfo_unix.cpp b/src/serialport/qserialportinfo_unix.cpp index ea5a1b3..44a4a2c 100644 --- a/src/serialport/qserialportinfo_unix.cpp +++ b/src/serialport/qserialportinfo_unix.cpp @@ -49,11 +49,8 @@ #ifndef Q_OS_MAC -#ifdef HAVE_LIBUDEV -extern "C" -{ -#include <libudev.h> -} +#if defined(LINK_LIBUDEV) || defined(LOAD_LIBUDEV) +#include "qtudev_p.h" #else #include <QtCore/qdir.h> #include <QtCore/qstringlist.h> @@ -65,7 +62,7 @@ QT_BEGIN_NAMESPACE #ifndef Q_OS_MAC -#ifndef HAVE_LIBUDEV +#if !defined(LINK_LIBUDEV) && !defined(LOAD_LIBUDEV) static inline const QStringList& filtersOfDevices() { @@ -218,6 +215,11 @@ QList<QSerialPortInfo> QSerialPortInfo::availablePorts() QList<QSerialPortInfo> QSerialPortInfo::availablePorts() { +#ifdef LOAD_LIBUDEV + static bool symbolsResolved = resolveSymbols(); + if (!symbolsResolved) + return QList<QSerialPortInfo>(); +#endif QList<QSerialPortInfo> serialPortInfoList; // White list for devices without a parent diff --git a/src/serialport/qtudev_p.h b/src/serialport/qtudev_p.h new file mode 100644 index 0000000..2619f8e --- /dev/null +++ b/src/serialport/qtudev_p.h @@ -0,0 +1,144 @@ +/**************************************************************************** +** +** Copyright (C) 2013 Laszlo Papp <lpapp@kde.org> +** Contact: http://www.qt-project.org/legal +** +** This file is part of the QtSerialPort module of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:LGPL$ +** Commercial License Usage +** Licensees holding valid commercial Qt licenses may use this file in +** accordance with the commercial license agreement provided with the +** Software or, alternatively, in accordance with the terms contained in +** a written agreement between you and Digia. For licensing terms and +** conditions see http://qt.digia.com/licensing. For further information +** use the contact form at http://qt.digia.com/contact-us. +** +** GNU Lesser General Public License Usage +** Alternatively, this file may be used under the terms of the GNU Lesser +** General Public License version 2.1 as published by the Free Software +** Foundation and appearing in the file LICENSE.LGPL included in the +** packaging of this file. Please review the following information to +** ensure the GNU Lesser General Public License version 2.1 requirements +** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. +** +** In addition, as a special exception, Digia gives you certain additional +** rights. These rights are described in the Digia Qt LGPL Exception +** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. +** +** GNU General Public License Usage +** Alternatively, this file may be used under the terms of the GNU +** General Public License version 3.0 as published by the Free Software +** Foundation and appearing in the file LICENSE.GPL included in the +** packaging of this file. Please review the following information to +** ensure the GNU General Public License version 3.0 requirements will be +** met: http://www.gnu.org/copyleft/gpl.html. +** +** +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#ifndef QTUDEV_P_H +#define QTUDEV_P_H + +#ifdef LINK_LIBUDEV +extern "C" +{ +#include <libudev.h> +} +#elif defined(LOAD_LIBUDEV) +#include <QtCore/qlibrary.h> +#include <QtCore/qstring.h> +#include <QtCore/qdebug.h> + +#define GENERATE_SYMBOL_VARIABLE(returnType, symbolName, ...) \ + typedef returnType (*fp_##symbolName)(__VA_ARGS__); \ + fp_##symbolName symbolName; + +#define RESOLVE_SYMBOL(symbolName) \ + symbolName = (fp_##symbolName)resolveSymbol(#symbolName); \ + if (!symbolName) \ + return false; + +struct udev; + +#define udev_list_entry_foreach(list_entry, first_entry) \ + for (list_entry = first_entry; \ + list_entry != NULL; \ + list_entry = udev_list_entry_get_next(list_entry)) + +struct udev_device; +struct udev_enumerate; +struct udev_list_entry; + +GENERATE_SYMBOL_VARIABLE(struct ::udev *, udev_new); +GENERATE_SYMBOL_VARIABLE(struct ::udev_enumerate *, udev_enumerate_new, struct ::udev *) +GENERATE_SYMBOL_VARIABLE(int, udev_enumerate_add_match_subsystem, struct udev_enumerate *, const char *) +GENERATE_SYMBOL_VARIABLE(int, udev_enumerate_scan_devices, struct udev_enumerate *) +GENERATE_SYMBOL_VARIABLE(struct udev_list_entry *, udev_enumerate_get_list_entry, struct udev_enumerate *) +GENERATE_SYMBOL_VARIABLE(struct udev_list_entry *, udev_list_entry_get_next, struct udev_list_entry *) +GENERATE_SYMBOL_VARIABLE(struct udev_device *, udev_device_new_from_syspath, struct udev *udev, const char *syspath) +GENERATE_SYMBOL_VARIABLE(const char *, udev_list_entry_get_name, struct udev_list_entry *) +GENERATE_SYMBOL_VARIABLE(const char *, udev_device_get_devnode, struct udev_device *) +GENERATE_SYMBOL_VARIABLE(const char *, udev_device_get_sysname, struct udev_device *) +GENERATE_SYMBOL_VARIABLE(struct udev_device *, udev_device_get_parent, struct udev_device *) +GENERATE_SYMBOL_VARIABLE(const char *, udev_device_get_subsystem, struct udev_device *) +GENERATE_SYMBOL_VARIABLE(const char *, udev_device_get_property_value, struct udev_device *, const char *) +GENERATE_SYMBOL_VARIABLE(void, udev_device_unref, struct udev_device *) +GENERATE_SYMBOL_VARIABLE(void, udev_enumerate_unref, struct udev_enumerate *) +GENERATE_SYMBOL_VARIABLE(void, udev_unref, struct udev *) + +QLibrary udevLibrary; + +#if (QT_VERSION >= QT_VERSION_CHECK(5, 0, 0)) +inline QFunctionPointer resolveSymbol(const char *symbolName) +{ + QFunctionPointer symbolFunctionPointer = udevLibrary.resolve(symbolName); +#else +inline void *resolveSymbol(const char *symbolName) +{ + void *symbolFunctionPointer = udevLibrary.resolve(symbolName); +#endif + if (!symbolFunctionPointer) + qWarning("Failed to resolve the udev symbol: %s", symbolName); + + return symbolFunctionPointer; +} + +inline bool resolveSymbols() +{ + if (!udevLibrary.isLoaded()) { + udevLibrary.setFileNameAndVersion(QLatin1String("udev"), 1); + if (!udevLibrary.load()) { + udevLibrary.setFileNameAndVersion(QLatin1String("udev"), 0); + if (!udevLibrary.load()) { + qWarning("Failed to load the library: %s, supported version(s): %i and %i", qPrintable(udevLibrary.fileName()), 1, 0); + return false; + } + } + } + + RESOLVE_SYMBOL(udev_new) + RESOLVE_SYMBOL(udev_enumerate_new) + RESOLVE_SYMBOL(udev_enumerate_add_match_subsystem) + RESOLVE_SYMBOL(udev_enumerate_scan_devices) + RESOLVE_SYMBOL(udev_enumerate_get_list_entry) + RESOLVE_SYMBOL(udev_list_entry_get_next) + RESOLVE_SYMBOL(udev_device_new_from_syspath) + RESOLVE_SYMBOL(udev_list_entry_get_name) + RESOLVE_SYMBOL(udev_device_get_devnode) + RESOLVE_SYMBOL(udev_device_get_sysname) + RESOLVE_SYMBOL(udev_device_get_parent) + RESOLVE_SYMBOL(udev_device_get_subsystem) + RESOLVE_SYMBOL(udev_device_get_property_value) + RESOLVE_SYMBOL(udev_device_unref) + RESOLVE_SYMBOL(udev_enumerate_unref) + RESOLVE_SYMBOL(udev_unref) + + return true; +} + +#endif + +#endif diff --git a/src/serialport/serialport-lib.pri b/src/serialport/serialport-lib.pri index 78f4139..f9d0af3 100644 --- a/src/serialport/serialport-lib.pri +++ b/src/serialport/serialport-lib.pri @@ -1,10 +1,10 @@ INCLUDEPATH += $$PWD -unix { +!contains(DEFINES, LOAD_LIBUDEV): unix { CONFIG += link_pkgconfig packagesExist(libudev) { - DEFINES += HAVE_LIBUDEV + DEFINES += LINK_LIBUDEV PKGCONFIG += libudev } } |