blob: 0daae61f586f934d907392ed51d66e66d83aa7e1 [file] [log] [blame]
/****************************************************************************
**
** 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 "qbluetoothserver.h"
#include "qbluetoothserver_p.h"
#include "qbluetoothsocket.h"
#include "qbluetoothsocket_bluez_p.h"
#include "qbluetoothlocaldevice.h"
#include "bluez/bluez_data_p.h"
#include <QtCore/QLoggingCategory>
#include <QtCore/QSocketNotifier>
#include <errno.h>
QT_BEGIN_NAMESPACE
Q_DECLARE_LOGGING_CATEGORY(QT_BT_BLUEZ)
QBluetoothSocket *QBluetoothServerPrivate::createSocketForServer(
QBluetoothServiceInfo::Protocol socketType)
{
// QBluetoothServer does not work with the BluetoothSocket implementation for DBus.
// Fall back to the raw socket implementation.
// Usually the private implementation is picked based on detected BlueZ version.
// ownership of these objects is taken care of inside QBluetoothSocket and QBluetoothServer
QBluetoothSocketPrivateBluez *rawSocketPrivate = new QBluetoothSocketPrivateBluez();
QBluetoothSocket *socket = new QBluetoothSocket(rawSocketPrivate, socketType);
return socket;
}
QBluetoothServerPrivate::QBluetoothServerPrivate(QBluetoothServiceInfo::Protocol sType,
QBluetoothServer *parent)
: maxPendingConnections(1), securityFlags(QBluetooth::Authorization), serverType(sType),
q_ptr(parent), m_lastError(QBluetoothServer::NoError)
{
if (sType == QBluetoothServiceInfo::RfcommProtocol)
socket = createSocketForServer(QBluetoothServiceInfo::RfcommProtocol);
else
socket = createSocketForServer(QBluetoothServiceInfo::L2capProtocol);
}
QBluetoothServerPrivate::~QBluetoothServerPrivate()
{
delete socketNotifier;
delete socket;
}
void QBluetoothServerPrivate::_q_newConnection()
{
// disable socket notifier until application calls nextPendingConnection().
socketNotifier->setEnabled(false);
emit q_ptr->newConnection();
}
void QBluetoothServerPrivate::setSocketSecurityLevel(
QBluetooth::SecurityFlags requestedSecLevel, int *errnoCode)
{
if (requestedSecLevel == QBluetooth::NoSecurity) {
qCWarning(QT_BT_BLUEZ) << "Cannot set NoSecurity on server socket";
return;
}
struct bt_security security;
memset(&security, 0, sizeof(security));
// ignore QBluetooth::Authentication -> not used anymore
if (requestedSecLevel & QBluetooth::Authorization)
security.level = BT_SECURITY_LOW;
if (requestedSecLevel & QBluetooth::Encryption)
security.level = BT_SECURITY_MEDIUM;
if (requestedSecLevel & QBluetooth::Secure)
security.level = BT_SECURITY_HIGH;
if (setsockopt(socket->socketDescriptor(), SOL_BLUETOOTH, BT_SECURITY,
&security, sizeof(security)) != 0) {
if (errnoCode)
*errnoCode = errno;
}
}
QBluetooth::SecurityFlags QBluetoothServerPrivate::socketSecurityLevel() const
{
struct bt_security security;
memset(&security, 0, sizeof(security));
socklen_t length = sizeof(security);
if (getsockopt(socket->socketDescriptor(), SOL_BLUETOOTH, BT_SECURITY,
&security, &length) != 0) {
qCWarning(QT_BT_BLUEZ) << "Failed to get security flags" << qt_error_string(errno);
return QBluetooth::NoSecurity;
}
switch (security.level) {
case BT_SECURITY_LOW:
return QBluetooth::Authorization;
case BT_SECURITY_MEDIUM:
return QBluetooth::Encryption;
case BT_SECURITY_HIGH:
return QBluetooth::Secure;
default:
qCWarning(QT_BT_BLUEZ) << "Unknown server socket security level" << security.level;
return QBluetooth::NoSecurity;
}
}
void QBluetoothServer::close()
{
Q_D(QBluetoothServer);
delete d->socketNotifier;
d->socketNotifier = nullptr;
d->socket->close();
}
bool QBluetoothServer::listen(const QBluetoothAddress &address, quint16 port)
{
Q_D(QBluetoothServer);
if (d->socket->state() == QBluetoothSocket::ListeningState) {
qCWarning(QT_BT_BLUEZ) << "Socket already in listen mode, close server first";
return false; //already listening, nothing to do
}
QBluetoothLocalDevice device(address);
if (!device.isValid()) {
qCWarning(QT_BT_BLUEZ) << "Device does not support Bluetooth or"
<< address.toString() << "is not a valid local adapter";
d->m_lastError = QBluetoothServer::UnknownError;
emit error(d->m_lastError);
return false;
}
QBluetoothLocalDevice::HostMode hostMode = device.hostMode();
if (hostMode == QBluetoothLocalDevice::HostPoweredOff) {
d->m_lastError = QBluetoothServer::PoweredOffError;
emit error(d->m_lastError);
qCWarning(QT_BT_BLUEZ) << "Bluetooth device is powered off";
return false;
}
int sock = d->socket->socketDescriptor();
if (sock < 0) {
/* Negative socket descriptor is not always an error case
* Another cause could be a call to close()/abort()
* Check whether we can recover by re-creating the socket
* we should really call Bluez::QBluetoothSocketPrivate::ensureNativeSocket
* but a re-creation of the socket will do as well.
*/
delete d->socket;
if (serverType() == QBluetoothServiceInfo::RfcommProtocol)
d->socket = QBluetoothServerPrivate::createSocketForServer(QBluetoothServiceInfo::RfcommProtocol);
else
d->socket = QBluetoothServerPrivate::createSocketForServer(QBluetoothServiceInfo::L2capProtocol);
sock = d->socket->socketDescriptor();
if (sock < 0) {
d->m_lastError = InputOutputError;
emit error(d->m_lastError);
return false;
}
}
if (d->serverType == QBluetoothServiceInfo::RfcommProtocol) {
sockaddr_rc addr;
addr.rc_family = AF_BLUETOOTH;
addr.rc_channel = port;
if (!address.isNull())
convertAddress(address.toUInt64(), addr.rc_bdaddr.b);
else
convertAddress(device.address().toUInt64(), addr.rc_bdaddr.b);
if (::bind(sock, reinterpret_cast<sockaddr *>(&addr), sizeof(sockaddr_rc)) < 0) {
if (errno == EADDRINUSE)
d->m_lastError = ServiceAlreadyRegisteredError;
else
d->m_lastError = InputOutputError;
emit error(d->m_lastError);
return false;
}
} else {
sockaddr_l2 addr;
memset(&addr, 0, sizeof(sockaddr_l2));
addr.l2_family = AF_BLUETOOTH;
addr.l2_psm = port;
if (!address.isNull())
convertAddress(address.toUInt64(), addr.l2_bdaddr.b);
else
convertAddress(Q_UINT64_C(0), addr.l2_bdaddr.b);
if (::bind(sock, reinterpret_cast<sockaddr *>(&addr), sizeof(sockaddr_l2)) < 0) {
d->m_lastError = InputOutputError;
emit error(d->m_lastError);
return false;
}
}
d->setSocketSecurityLevel(d->securityFlags, nullptr);
if (::listen(sock, d->maxPendingConnections) < 0) {
d->m_lastError = InputOutputError;
emit error(d->m_lastError);
return false;
}
d->socket->setSocketState(QBluetoothSocket::ListeningState);
if (!d->socketNotifier) {
d->socketNotifier = new QSocketNotifier(d->socket->socketDescriptor(),
QSocketNotifier::Read);
connect(d->socketNotifier, &QSocketNotifier::activated,
this, [d](){
d->_q_newConnection();
});
}
return true;
}
void QBluetoothServer::setMaxPendingConnections(int numConnections)
{
Q_D(QBluetoothServer);
if (d->socket->state() == QBluetoothSocket::UnconnectedState)
d->maxPendingConnections = numConnections;
}
bool QBluetoothServer::hasPendingConnections() const
{
Q_D(const QBluetoothServer);
if (!d || !d->socketNotifier)
return false;
// if the socket notifier is disabled there is a pending connection waiting for us to accept.
return !d->socketNotifier->isEnabled();
}
QBluetoothSocket *QBluetoothServer::nextPendingConnection()
{
Q_D(QBluetoothServer);
if (!hasPendingConnections())
return nullptr;
int pending;
if (d->serverType == QBluetoothServiceInfo::RfcommProtocol) {
sockaddr_rc addr;
socklen_t length = sizeof(sockaddr_rc);
pending = ::accept(d->socket->socketDescriptor(),
reinterpret_cast<sockaddr *>(&addr), &length);
} else {
sockaddr_l2 addr;
socklen_t length = sizeof(sockaddr_l2);
pending = ::accept(d->socket->socketDescriptor(),
reinterpret_cast<sockaddr *>(&addr), &length);
}
if (pending >= 0) {
QBluetoothSocket *newSocket = QBluetoothServerPrivate::createSocketForServer();
if (d->serverType == QBluetoothServiceInfo::RfcommProtocol)
newSocket->setSocketDescriptor(pending, QBluetoothServiceInfo::RfcommProtocol);
else
newSocket->setSocketDescriptor(pending, QBluetoothServiceInfo::L2capProtocol);
d->socketNotifier->setEnabled(true);
return newSocket;
} else {
d->socketNotifier->setEnabled(true);
}
return nullptr;
}
QBluetoothAddress QBluetoothServer::serverAddress() const
{
Q_D(const QBluetoothServer);
return d->socket->localAddress();
}
quint16 QBluetoothServer::serverPort() const
{
Q_D(const QBluetoothServer);
return d->socket->localPort();
}
void QBluetoothServer::setSecurityFlags(QBluetooth::SecurityFlags security)
{
Q_D(QBluetoothServer);
if (d->socket->state() == QBluetoothSocket::UnconnectedState) {
// nothing to set beyond the fact to remember the sec level for the next listen()
d->securityFlags = security;
return;
}
int errorCode = 0;
d->setSocketSecurityLevel(security, &errorCode);
if (errorCode) {
qCWarning(QT_BT_BLUEZ) << "Failed to set socket option, closing socket for safety" << errorCode;
qCWarning(QT_BT_BLUEZ) << "Error: " << qt_error_string(errorCode);
d->m_lastError = InputOutputError;
emit error(d->m_lastError);
d->socket->close();
}
}
QBluetooth::SecurityFlags QBluetoothServer::securityFlags() const
{
Q_D(const QBluetoothServer);
if (d->socket->state() == QBluetoothSocket::UnconnectedState)
return d->securityFlags;
return d->socketSecurityLevel();
}
QT_END_NAMESPACE