diff options
author | Benoit Dumas <benoit.dumas.rim@kdab.com> | 2012-08-15 07:18:11 (GMT) |
---|---|---|
committer | Qt by Nokia <qt-info@nokia.com> | 2012-08-30 07:37:27 (GMT) |
commit | e83148a61a6c677b44337f828018a8eb43ea0e74 (patch) | |
tree | ead847367a3f570fe30b93bd93f2b72d85ab6c9f /src/plugins | |
parent | 919f939757e37527780cf53566d43dd13f86c225 (diff) | |
download | Qt-e83148a61a6c677b44337f828018a8eb43ea0e74.zip Qt-e83148a61a6c677b44337f828018a8eb43ea0e74.tar.gz Qt-e83148a61a6c677b44337f828018a8eb43ea0e74.tar.bz2 |
Implementation of the BlackBerry Bearer engine
An implementation for BlackBerry devices based on the BPS netstatus APIs.
cherry-picked from qtbase df7a1a28aaab31
Change-Id: I979f594aca64315af1f39e68886b819f7f215f6b
Reviewed-by: Rafael Roquetto <rafael.roquetto@kdab.com>
Reviewed-by: Vladimir Minenko <vminenko@rim.com>
Reviewed-by: Sean Harmer <sean.harmer@kdab.com>
Diffstat (limited to 'src/plugins')
-rw-r--r-- | src/plugins/bearer/bearer.pro | 1 | ||||
-rw-r--r-- | src/plugins/bearer/blackberry/blackberry.pro | 21 | ||||
-rw-r--r-- | src/plugins/bearer/blackberry/main.cpp | 86 | ||||
-rw-r--r-- | src/plugins/bearer/blackberry/qbbengine.cpp | 424 | ||||
-rw-r--r-- | src/plugins/bearer/blackberry/qbbengine.h | 112 |
5 files changed, 644 insertions, 0 deletions
diff --git a/src/plugins/bearer/bearer.pro b/src/plugins/bearer/bearer.pro index d1d75f0..c8c73fc 100644 --- a/src/plugins/bearer/bearer.pro +++ b/src/plugins/bearer/bearer.pro @@ -15,5 +15,6 @@ win32:!wince*:SUBDIRS += nativewifi macx:contains(QT_CONFIG, corewlan):SUBDIRS += corewlan macx:SUBDIRS += generic symbian:SUBDIRS += symbian +blackberry:SUBDIRS += blackberry isEmpty(SUBDIRS):SUBDIRS = generic diff --git a/src/plugins/bearer/blackberry/blackberry.pro b/src/plugins/bearer/blackberry/blackberry.pro new file mode 100644 index 0000000..4b24caa --- /dev/null +++ b/src/plugins/bearer/blackberry/blackberry.pro @@ -0,0 +1,21 @@ +TARGET = qbbbearer +include(../../qpluginbase.pri) + +QT = core network + +# Uncomment this to enable debugging output for the plugin +#DEFINES += QBBENGINE_DEBUG + +HEADERS += qbbengine.h \ + ../qnetworksession_impl.h \ + ../qbearerengine_impl.h + +SOURCES += qbbengine.cpp \ + ../qnetworksession_impl.cpp \ + main.cpp + +OTHER_FILES += blackberry.json + +QTDIR_build:DESTDIR = $$QT_BUILD_TREE/plugins/bearer +target.path += $$[QT_INSTALL_PLUGINS]/bearer +INSTALLS += target diff --git a/src/plugins/bearer/blackberry/main.cpp b/src/plugins/bearer/blackberry/main.cpp new file mode 100644 index 0000000..db9814c --- /dev/null +++ b/src/plugins/bearer/blackberry/main.cpp @@ -0,0 +1,86 @@ +/**************************************************************************** +** +** Copyright (C) 2012 Research In Motion +** Contact: http://www.qt-project.org/ +** +** This file is part of the plugins of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:LGPL$ +** GNU Lesser General Public License Usage +** 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, Nokia gives you certain additional +** rights. These rights are described in the Nokia 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. +** +** Other Usage +** Alternatively, this file may be used in accordance with the terms and +** conditions contained in a signed written agreement between you and Nokia. +** +** +** +** +** +** +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#include "qbbengine.h" + +#include <QtNetwork/private/qbearerplugin_p.h> + +#ifndef QT_NO_BEARERMANAGEMENT + +QT_BEGIN_NAMESPACE + +class QBBEnginePlugin : public QBearerEnginePlugin +{ +public: + QBBEnginePlugin(); + ~QBBEnginePlugin(); + + QStringList keys() const; + QBearerEngine *create(const QString &key) const; +}; + +QBBEnginePlugin::QBBEnginePlugin() +{ +} + +QBBEnginePlugin::~QBBEnginePlugin() +{ +} + +QStringList QBBEnginePlugin::keys() const +{ + return QStringList() << QLatin1String("blackberry"); +} + +QBearerEngine *QBBEnginePlugin::create(const QString &key) const +{ + if (key == QLatin1String("blackberry")) + return new QBBEngine; + + return 0; +} + +Q_EXPORT_STATIC_PLUGIN(QBBEnginePlugin) +Q_EXPORT_PLUGIN2(qbbbearer, QBBEnginePlugin) + +QT_END_NAMESPACE + +#endif // QT_NO_BEARERMANAGEMENT diff --git a/src/plugins/bearer/blackberry/qbbengine.cpp b/src/plugins/bearer/blackberry/qbbengine.cpp new file mode 100644 index 0000000..7faef8b --- /dev/null +++ b/src/plugins/bearer/blackberry/qbbengine.cpp @@ -0,0 +1,424 @@ +/**************************************************************************** +** +** Copyright (C) 2012 Research In Motion +** Contact: http://www.qt-project.org/ +** +** This file is part of the plugins of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:LGPL$ +** GNU Lesser General Public License Usage +** 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, Nokia gives you certain additional +** rights. These rights are described in the Nokia 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. +** +** Other Usage +** Alternatively, this file may be used in accordance with the terms and +** conditions contained in a signed written agreement between you and Nokia. +** +** +** +** +** +** +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#include "qbbengine.h" +#include "../qnetworksession_impl.h" + +#include <QDebug> +#include <QThreadStorage> +#include <QStringList> + +#include <bps/netstatus.h> + +#ifndef QT_NO_BEARERMANAGEMENT + +#ifdef QBBENGINE_DEBUG +#define qBearerDebug qDebug +#else +#define qBearerDebug QT_NO_QDEBUG_MACRO +#endif + +struct NetstatusInterfaceListCleanupHelper +{ + static inline void cleanup(netstatus_interface_list_t *list) + { + netstatus_free_interfaces(list); + } +}; + +struct NetstatusInterfaceCleanupHelper +{ + static inline void cleanup(char *interface) + { + bps_free(interface); + } +}; + +struct EngineInstanceHolder +{ + EngineInstanceHolder(QBBEngine *engine) : + instance(engine) {} + + QBBEngine *instance; +}; + +Q_GLOBAL_STATIC(QThreadStorage<EngineInstanceHolder *>, instanceStorage); + +static QNetworkConfiguration::BearerType +interfaceType(netstatus_interface_type_t type) +{ + switch (type) { + case NETSTATUS_INTERFACE_TYPE_USB: + case NETSTATUS_INTERFACE_TYPE_WIRED: + return QNetworkConfiguration::BearerEthernet; + + case NETSTATUS_INTERFACE_TYPE_WIFI: + return QNetworkConfiguration::BearerWLAN; + + case NETSTATUS_INTERFACE_TYPE_BLUETOOTH_DUN: + return QNetworkConfiguration::BearerBluetooth; + + case NETSTATUS_INTERFACE_TYPE_CELLULAR: + //### TODO not sure which BearerType would be the best + //to return here. We need to be able to get more + //information on the bearer type in order to return + //the exact match. + return QNetworkConfiguration::Bearer2G; + + case NETSTATUS_INTERFACE_TYPE_VPN: + case NETSTATUS_INTERFACE_TYPE_BB: + case NETSTATUS_INTERFACE_TYPE_UNKNOWN: + break; + } + + return QNetworkConfiguration::BearerUnknown; +} + +static QString idForName(const QString &name) +{ + return QLatin1String("bps:") + name; +} + +QT_BEGIN_NAMESPACE + + +QBBEngine::QBBEngine(QObject *parent) : + QBearerEngineImpl(parent), + previousEventFilter(0), + pollingRequired(false), + initialized(false) +{ +} + +QBBEngine::~QBBEngine() +{ + QAbstractEventDispatcher::instance()->setEventFilter(previousEventFilter); +} + + +QString QBBEngine::getInterfaceFromId(const QString &id) +{ + const QMutexLocker locker(&mutex); + + return configurationInterface.value(id); +} + +bool QBBEngine::hasIdentifier(const QString &id) +{ + const QMutexLocker locker(&mutex); + + return configurationInterface.contains(id); +} + +void QBBEngine::connectToId(const QString &id) +{ + Q_EMIT connectionError(id, OperationNotSupported); +} + +void QBBEngine::disconnectFromId(const QString &id) +{ + Q_EMIT connectionError(id, OperationNotSupported); +} + +void QBBEngine::initialize() +{ + if (initialized) { + qWarning() << Q_FUNC_INFO << "called, but instance already initialized."; + return; + } + + instanceStorage()->setLocalData(new EngineInstanceHolder(this)); + + if (netstatus_request_events(0) != BPS_SUCCESS) { + qWarning() << Q_FUNC_INFO << "cannot register for network events. Polling enabled."; + + const QMutexLocker locker(&pollingMutex); + pollingRequired = true; + } else { + previousEventFilter = + QAbstractEventDispatcher::instance()->setEventFilter(filterEvent); + } + + doRequestUpdate(); +} + +void QBBEngine::requestUpdate() +{ + doRequestUpdate(); +} + +void QBBEngine::doRequestUpdate() +{ + qBearerDebug() << Q_FUNC_INFO << "entered method"; + + netstatus_interface_list_t interfaceList; + + if ((netstatus_get_interfaces(&interfaceList)) != BPS_SUCCESS) { + qBearerDebug() << Q_FUNC_INFO << "cannot retrieve interface list"; + return; + } + + const QScopedPointer<netstatus_interface_list_t, + NetstatusInterfaceListCleanupHelper> holder(&interfaceList); + + QSet<QString> currentConfigurations; + + for (int i = 0; i < interfaceList.num_interfaces; i++) { + const char *interface = interfaceList.interfaces[i]; + + qBearerDebug() << Q_FUNC_INFO << "discovered interface" << interface; + + updateConfiguration(interface); + + currentConfigurations << idForName(QString::fromLatin1(interface)); + } + + QMutexLocker locker(&mutex); + + const QStringList keys = accessPointConfigurations.uniqueKeys(); + + locker.unlock(); + + Q_FOREACH (const QString &id, keys) { + if (!currentConfigurations.contains(id)) + removeConfiguration(id); + } + + Q_EMIT updateCompleted(); +} + +QNetworkSession::State QBBEngine::sessionStateForId(const QString &id) +{ + const QMutexLocker locker(&mutex); + + QNetworkConfigurationPrivatePointer ptr = accessPointConfigurations.value(id); + + if (!ptr || !ptr->isValid) + return QNetworkSession::Invalid; + + if ((ptr->state & QNetworkConfiguration::Active) == QNetworkConfiguration::Active) + return QNetworkSession::Connected; + else if ((ptr->state & QNetworkConfiguration::Discovered) == QNetworkConfiguration::Discovered) + return QNetworkSession::Disconnected; + else if ((ptr->state & QNetworkConfiguration::Defined) == QNetworkConfiguration::Defined) + return QNetworkSession::NotAvailable; + else if ((ptr->state & QNetworkConfiguration::Undefined) == QNetworkConfiguration::Undefined) + return QNetworkSession::NotAvailable; + + return QNetworkSession::Invalid; +} + +QNetworkConfigurationManager::Capabilities QBBEngine::capabilities() const +{ + return QNetworkConfigurationManager::ForcedRoaming; +} + +QNetworkSessionPrivate *QBBEngine::createSessionBackend() +{ + return new QNetworkSessionPrivateImpl; +} + +QNetworkConfigurationPrivatePointer QBBEngine::defaultConfiguration() +{ + char *interface = 0; + + if (netstatus_get_default_interface(&interface) != BPS_SUCCESS) + return QNetworkConfigurationPrivatePointer(); + + if (!interface) + return QNetworkConfigurationPrivatePointer(); + + const QScopedPointer<char, NetstatusInterfaceCleanupHelper> holder(interface); + + const QString id = idForName(QString::fromLatin1(interface)); + + const QMutexLocker locker(&mutex); + + if (accessPointConfigurations.contains(id)) { + qBearerDebug() << Q_FUNC_INFO << "found default interface:" << id; + + return accessPointConfigurations.value(id); + } + + return QNetworkConfigurationPrivatePointer(); +} + +bool QBBEngine::requiresPolling() const +{ + const QMutexLocker locker(&pollingMutex); + + return pollingRequired; +} + +bool QBBEngine::filterEvent(void *message) +{ + bps_event_t * const event = static_cast<bps_event_t *>(message); + + Q_ASSERT(event); + + QBBEngine *self = instanceStorage()->localData()->instance; + + Q_ASSERT(self); + + if (bps_event_get_domain(event) == netstatus_get_domain()) + self->filterEvent(event); + + if (self->previousEventFilter) + return self->previousEventFilter(message); + + return false; +} + +void QBBEngine::filterEvent(bps_event_t *event) +{ + Q_UNUSED(event); + + qBearerDebug() << Q_FUNC_INFO << "got update request."; + + doRequestUpdate(); +} + +void QBBEngine::updateConfiguration(const char *interface) +{ + netstatus_interface_details_t *details = 0; + + if (netstatus_get_interface_details(interface, &details) != BPS_SUCCESS) { + qBearerDebug() << Q_FUNC_INFO << "cannot retrieve details for interface" << interface; + + return; + } + + const QString name = QString::fromLatin1(netstatus_interface_get_name(details)); + const QString id = idForName(name); + + + const int numberOfIpAddresses = netstatus_interface_get_num_ip_addresses(details); + const bool isConnected = netstatus_interface_is_connected(details); + const netstatus_interface_type_t type = netstatus_interface_get_type(details); + + netstatus_free_interface_details(&details); + + QNetworkConfiguration::StateFlags state = QNetworkConfiguration::Defined; + + if (isConnected && (numberOfIpAddresses > 0)) + state |= QNetworkConfiguration::Active; + + QMutexLocker locker(&mutex); + + if (accessPointConfigurations.contains(id)) { + QNetworkConfigurationPrivatePointer ptr = accessPointConfigurations.value(id); + + bool changed = false; + + QMutexLocker ptrLocker(&ptr->mutex); + + if (!ptr->isValid) { + ptr->isValid = true; + changed = true; + } + + if (ptr->name != name) { + ptr->name = name; + changed = true; + } + + if (ptr->id != id) { + ptr->id = id; + changed = true; + } + + if (ptr->state != state) { + ptr->state = state; + changed = true; + } + + ptrLocker.unlock(); + + locker.unlock(); + + if (changed) { + qBearerDebug() << Q_FUNC_INFO << "configuration changed:" << interface; + + Q_EMIT configurationChanged(ptr); + } else { + qBearerDebug() << Q_FUNC_INFO << "configuration has not changed."; + } + + return; + } + + QNetworkConfigurationPrivatePointer ptr(new QNetworkConfigurationPrivate); + + ptr->name = name; + ptr->isValid = true; + ptr->id = id; + ptr->state = state; + ptr->type = QNetworkConfiguration::InternetAccessPoint; + ptr->bearerType = interfaceType(type); + + accessPointConfigurations.insert(id, ptr); + configurationInterface.insert(id, name); + + locker.unlock(); + + qBearerDebug() << Q_FUNC_INFO << "configuration added:" << interface; + + Q_EMIT configurationAdded(ptr); +} + +void QBBEngine::removeConfiguration(const QString &id) +{ + QMutexLocker locker(&mutex); + + QNetworkConfigurationPrivatePointer ptr = + accessPointConfigurations.take(id); + + configurationInterface.remove(ptr->id); + + locker.unlock(); + + Q_EMIT configurationRemoved(ptr); +} + +QT_END_NAMESPACE + +#endif // QT_NO_BEARERMANAGEMENT + diff --git a/src/plugins/bearer/blackberry/qbbengine.h b/src/plugins/bearer/blackberry/qbbengine.h new file mode 100644 index 0000000..c44a855 --- /dev/null +++ b/src/plugins/bearer/blackberry/qbbengine.h @@ -0,0 +1,112 @@ +/**************************************************************************** +** +** Copyright (C) 2012 Research In Motion +** Contact: http://www.qt-project.org/ +** +** This file is part of the plugins of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:LGPL$ +** GNU Lesser General Public License Usage +** 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, Nokia gives you certain additional +** rights. These rights are described in the Nokia 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. +** +** Other Usage +** Alternatively, this file may be used in accordance with the terms and +** conditions contained in a signed written agreement between you and Nokia. +** +** +** +** +** +** +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#ifndef QBBENGINE_H +#define QBBENGINE_H + +#include "../qbearerengine_impl.h" + +#include <QAbstractEventDispatcher> + +#ifndef QT_NO_BEARERMANAGEMENT + +struct bps_event_t; + +QT_BEGIN_NAMESPACE + +class QNetworkConfigurationPrivate; +class QNetworkSessionPrivate; + +class QBBEngine : public QBearerEngineImpl +{ + Q_OBJECT + +public: + explicit QBBEngine(QObject *parent = 0); + ~QBBEngine(); + + QString getInterfaceFromId(const QString &id); + bool hasIdentifier(const QString &id); + + void connectToId(const QString &id); + void disconnectFromId(const QString &id); + + Q_INVOKABLE void initialize(); + Q_INVOKABLE void requestUpdate(); + + QNetworkSession::State sessionStateForId(const QString &id); + + QNetworkConfigurationManager::Capabilities capabilities() const; + + QNetworkSessionPrivate *createSessionBackend(); + + QNetworkConfigurationPrivatePointer defaultConfiguration(); + + bool requiresPolling() const; + +protected: + void updateConfiguration(const char *interface); + void removeConfiguration(const QString &id); + +private Q_SLOTS: + void doRequestUpdate(); + +private: + static bool filterEvent(void *message); + + void filterEvent(bps_event_t *event); + + QHash<QString, QString> configurationInterface; + + QAbstractEventDispatcher::EventFilter previousEventFilter; + + mutable QMutex pollingMutex; + + bool pollingRequired; + bool initialized; +}; + + +QT_END_NAMESPACE + +#endif // QT_NO_BEARERMANAGEMENT + +#endif // QBBENGINE_H |