Make bluetooth payload upload work & asynchronous
Summary: Sharing files to the Android bluetooth backend now works. The code is asynchronous now too. Test Plan: I tested several files, and they all get transferred correctly. The socket gets destroyed correctly too. Reviewers: #kde_connect, apol Reviewed By: #kde_connect, apol Subscribers: apol, #kde_connect Tags: #kde_connect Differential Revision: https://phabricator.kde.org/D12264
This commit is contained in:
parent
fcfd1d7c69
commit
1b570d4719
3 changed files with 49 additions and 18 deletions
|
@ -51,12 +51,9 @@ QString BluetoothDeviceLink::name()
|
||||||
bool BluetoothDeviceLink::sendPacket(NetworkPacket& np)
|
bool BluetoothDeviceLink::sendPacket(NetworkPacket& np)
|
||||||
{
|
{
|
||||||
if (np.hasPayload()) {
|
if (np.hasPayload()) {
|
||||||
qCWarning(KDECONNECT_CORE) << "Sending packets with payload over bluetooth not yet supported";
|
|
||||||
/*
|
|
||||||
BluetoothUploadJob* uploadJob = new BluetoothUploadJob(np.payload(), mBluetoothSocket->peerAddress(), this);
|
BluetoothUploadJob* uploadJob = new BluetoothUploadJob(np.payload(), mBluetoothSocket->peerAddress(), this);
|
||||||
np.setPayloadTransferInfo(uploadJob->transferInfo());
|
np.setPayloadTransferInfo(uploadJob->transferInfo());
|
||||||
uploadJob->start();
|
uploadJob->start();
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
int written = mSocketReader->write(np.serialize());
|
int written = mSocketReader->write(np.serialize());
|
||||||
return (written != -1);
|
return (written != -1);
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2016 Saikrishna Arcot <saiarcot895@gmail.com>
|
* Copyright 2016 Saikrishna Arcot <saiarcot895@gmail.com>
|
||||||
|
* Copyright 2018 Matthijs TIjink <matthijstijink@gmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or
|
* This program is free software; you can redistribute it and/or
|
||||||
* modify it under the terms of the GNU General Public License as
|
* modify it under the terms of the GNU General Public License as
|
||||||
|
@ -43,43 +44,70 @@ QVariantMap BluetoothUploadJob::transferInfo() const
|
||||||
|
|
||||||
void BluetoothUploadJob::start()
|
void BluetoothUploadJob::start()
|
||||||
{
|
{
|
||||||
connect(mServer, SIGNAL(newConnection()), this, SLOT(newConnection()));
|
connect(mServer, &QBluetoothServer::newConnection, this, &BluetoothUploadJob::newConnection);
|
||||||
mServiceInfo = mServer->listen(mTransferUuid, "KDE Connect Transfer Job");
|
mServiceInfo = mServer->listen(mTransferUuid, "KDE Connect Transfer Job");
|
||||||
Q_ASSERT(mServiceInfo.isValid());
|
Q_ASSERT(mServiceInfo.isValid());
|
||||||
}
|
}
|
||||||
|
|
||||||
void BluetoothUploadJob::newConnection()
|
void BluetoothUploadJob::newConnection()
|
||||||
{
|
{
|
||||||
QBluetoothSocket* socket = mServer->nextPendingConnection();
|
m_socket = mServer->nextPendingConnection();
|
||||||
connect(socket, SIGNAL(disconnected()), socket, SLOT(deleteLater()));
|
Q_ASSERT(m_socket);
|
||||||
|
m_socket->setParent(this);
|
||||||
|
connect(m_socket, &QBluetoothSocket::disconnected, this, &BluetoothUploadJob::deleteLater);
|
||||||
|
|
||||||
if (socket->peerAddress() != mRemoteAddress) {
|
if (m_socket->peerAddress() != mRemoteAddress) {
|
||||||
socket->close();
|
m_socket->close();
|
||||||
} else {
|
} else {
|
||||||
mServer->close();
|
mServer->close();
|
||||||
disconnect(mServer, SIGNAL(newConnection()), this, SLOT(newConnection()));
|
disconnect(mServer, &QBluetoothServer::newConnection, this, &BluetoothUploadJob::newConnection);
|
||||||
mServiceInfo.unregisterService();
|
mServiceInfo.unregisterService();
|
||||||
|
|
||||||
if (!mData->open(QIODevice::ReadOnly)) {
|
if (!mData->open(QIODevice::ReadOnly)) {
|
||||||
qCWarning(KDECONNECT_CORE) << "error when opening the input to upload";
|
qCWarning(KDECONNECT_CORE) << "error when opening the input to upload";
|
||||||
socket->close();
|
m_socket->close();
|
||||||
deleteLater();
|
deleteLater();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
while (mData->bytesAvailable() > 0 && socket->isWritable()) {
|
connect(m_socket, &QBluetoothSocket::bytesWritten, this, &BluetoothUploadJob::writeSome);
|
||||||
qint64 bytes = qMin(mData->bytesAvailable(), (qint64)4096);
|
connect(m_socket, &QBluetoothSocket::disconnected, this, &BluetoothUploadJob::closeConnection);
|
||||||
int w = socket->write(mData->read(bytes));
|
writeSome();
|
||||||
if (w < 0) {
|
}
|
||||||
qCWarning(KDECONNECT_CORE()) << "error when writing data to upload" << bytes << mData->bytesAvailable();
|
|
||||||
|
void BluetoothUploadJob::writeSome() {
|
||||||
|
Q_ASSERT(m_socket);
|
||||||
|
|
||||||
|
bool errorOccurred = false;
|
||||||
|
while (m_socket->bytesToWrite() == 0 && mData->bytesAvailable() && m_socket->isWritable()) {
|
||||||
|
qint64 bytes = qMin<qint64>(mData->bytesAvailable(), 4096);
|
||||||
|
int bytesWritten = m_socket->write(mData->read(bytes));
|
||||||
|
|
||||||
|
if (bytesWritten < 0) {
|
||||||
|
qCWarning(KDECONNECT_CORE()) << "error when writing data to bluetooth upload" << bytes << mData->bytesAvailable();
|
||||||
|
errorOccurred = true;
|
||||||
break;
|
break;
|
||||||
} else {
|
|
||||||
QCoreApplication::processEvents(QEventLoop::ExcludeUserInputEvents, 2000);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (mData->atEnd() || errorOccurred) {
|
||||||
|
disconnect(m_socket, &QBluetoothSocket::bytesWritten, this, &BluetoothUploadJob::writeSome);
|
||||||
|
mData->close();
|
||||||
|
|
||||||
|
connect(m_socket, &QBluetoothSocket::bytesWritten, this, &BluetoothUploadJob::finishWrites);
|
||||||
|
finishWrites();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void BluetoothUploadJob::finishWrites() {
|
||||||
|
Q_ASSERT(m_socket);
|
||||||
|
if (m_socket->bytesToWrite() == 0) {
|
||||||
|
closeConnection();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void BluetoothUploadJob::closeConnection() {
|
||||||
mData->close();
|
mData->close();
|
||||||
socket->close();
|
|
||||||
deleteLater();
|
deleteLater();
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2016 Saikrishna Arcot <saiarcot895@gmail.com>
|
* Copyright 2016 Saikrishna Arcot <saiarcot895@gmail.com>
|
||||||
|
* Copyright 2018 Matthijs TIjink <matthijstijink@gmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or
|
* This program is free software; you can redistribute it and/or
|
||||||
* modify it under the terms of the GNU General Public License as
|
* modify it under the terms of the GNU General Public License as
|
||||||
|
@ -45,9 +46,14 @@ private:
|
||||||
QBluetoothUuid mTransferUuid;
|
QBluetoothUuid mTransferUuid;
|
||||||
QBluetoothServer* mServer;
|
QBluetoothServer* mServer;
|
||||||
QBluetoothServiceInfo mServiceInfo;
|
QBluetoothServiceInfo mServiceInfo;
|
||||||
|
QBluetoothSocket* m_socket;
|
||||||
|
|
||||||
|
void closeConnection();
|
||||||
|
|
||||||
private Q_SLOTS:
|
private Q_SLOTS:
|
||||||
void newConnection();
|
void newConnection();
|
||||||
|
void writeSome();
|
||||||
|
void finishWrites();
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // BLUETOOTHUPLOADJOB_H
|
#endif // BLUETOOTHUPLOADJOB_H
|
||||||
|
|
Loading…
Reference in a new issue