/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtBluetooth 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 The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/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 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include #include #include "android/serveracceptancethread_p.h" Q_DECLARE_LOGGING_CATEGORY(QT_BT_ANDROID) ServerAcceptanceThread::ServerAcceptanceThread(QObject *parent) : QObject(parent), maxPendingConnections(1) { qRegisterMetaType(); } ServerAcceptanceThread::~ServerAcceptanceThread() { Q_ASSERT(!isRunning()); QMutexLocker lock(&m_mutex); shutdownPendingConnections(); } void ServerAcceptanceThread::setServiceDetails(const QBluetoothUuid &uuid, const QString &serviceName, QBluetooth::SecurityFlags securityFlags) { QMutexLocker lock(&m_mutex); m_uuid = uuid; m_serviceName = serviceName; secFlags = securityFlags; } bool ServerAcceptanceThread::hasPendingConnections() const { QMutexLocker lock(&m_mutex); return (pendingSockets.count() > 0); } /* * Returns the next pending connection or an invalid JNI object. * Note that even a stopped thread may still have pending * connections. Pending connections are only terminated upon * thread restart or destruction. */ QAndroidJniObject ServerAcceptanceThread::nextPendingConnection() { QMutexLocker lock(&m_mutex); if (pendingSockets.isEmpty()) return QAndroidJniObject(); else return pendingSockets.takeFirst(); } void ServerAcceptanceThread::setMaxPendingConnections(int maximumCount) { QMutexLocker lock(&m_mutex); maxPendingConnections = maximumCount; } void ServerAcceptanceThread::run() { QMutexLocker lock(&m_mutex); if (!validSetup()) { qCWarning(QT_BT_ANDROID) << "Invalid Server Socket setup"; return; } if (isRunning()) { stop(); shutdownPendingConnections(); } javaThread = QAndroidJniObject("org/qtproject/qt5/android/bluetooth/QtBluetoothSocketServer"); if (!javaThread.isValid()) return; javaThread.setField("qtObject", reinterpret_cast(this)); javaThread.setField("logEnabled", QT_BT_ANDROID().isDebugEnabled()); QString tempUuid = m_uuid.toString(); tempUuid.chop(1); //remove trailing '}' tempUuid.remove(0,1); //remove first '{' QAndroidJniObject uuidString = QAndroidJniObject::fromString(tempUuid); QAndroidJniObject serviceNameString = QAndroidJniObject::fromString(m_serviceName); bool isSecure = !(secFlags == QBluetooth::NoSecurity); javaThread.callMethod("setServiceDetails", "(Ljava/lang/String;Ljava/lang/String;Z)V", uuidString.object(), serviceNameString.object(), isSecure); javaThread.callMethod("start"); } void ServerAcceptanceThread::stop() { if (javaThread.isValid()) { qCDebug(QT_BT_ANDROID) << "Closing server socket"; javaThread.callMethod("close"); } } bool ServerAcceptanceThread::isRunning() const { if (javaThread.isValid()) return javaThread.callMethod("isAlive"); return false; } //Runs inside the java thread void ServerAcceptanceThread::javaThreadErrorOccurred(int errorCode) { qCDebug(QT_BT_ANDROID) << "JavaThread error:" << errorCode; emit error(QBluetoothServer::InputOutputError); } //Runs inside the Java thread void ServerAcceptanceThread::javaNewSocket(jobject s) { QMutexLocker lock(&m_mutex); QAndroidJniObject socket(s); if (!socket.isValid()) return; if (pendingSockets.count() < maxPendingConnections) { qCDebug(QT_BT_ANDROID) << "New incoming java socket detected"; pendingSockets.append(socket); emit newConnection(); } else { QAndroidJniEnvironment env; qCWarning(QT_BT_ANDROID) << "Refusing connection due to limited pending socket queue"; socket.callMethod("close"); if (env->ExceptionCheck()) { qCWarning(QT_BT_ANDROID) << "Error during refusal of new socket"; env->ExceptionDescribe(); env->ExceptionClear(); } } } bool ServerAcceptanceThread::validSetup() const { return (!m_uuid.isNull() && !m_serviceName.isEmpty()); } void ServerAcceptanceThread::shutdownPendingConnections() { while (!pendingSockets.isEmpty()) { QAndroidJniObject socket = pendingSockets.takeFirst(); socket.callMethod("close"); } }