QNAM: Fix previous HTTP upload CPU fix

My previous fix for CPU load issues between HTTP thread
and user thread  was fragile if the upload QIODevice
emitted readyRead() multiple times.

[ChangeLog][QtNetwork][QNetworkAccessManager] Fix behavior of upload QIODevice
that generate data on readyRead() for HTTP PUT/POST

Change-Id: Idb1c2d5a382a704d8cc08fe03c55c883bfc95aa7
Reviewed-by: Christian Kamm <kamm@incasoftware.de>
Reviewed-by: Richard J. Moore <rich@kde.org>
This commit is contained in:
Markus Goetz 2014-09-19 17:18:59 +02:00
parent 5c2d7b1635
commit 097b641c3e
3 changed files with 13 additions and 1 deletions

View File

@ -428,6 +428,7 @@ QNetworkReplyHttpImplPrivate::QNetworkReplyHttpImplPrivate()
, synchronous(false) , synchronous(false)
, state(Idle) , state(Idle)
, statusCode(0) , statusCode(0)
, uploadDeviceChoking(false)
, outgoingData(0) , outgoingData(0)
, bytesUploaded(-1) , bytesUploaded(-1)
, cacheLoadDevice(0) , cacheLoadDevice(0)
@ -1291,9 +1292,12 @@ void QNetworkReplyHttpImplPrivate::wantUploadDataSlot(qint64 maxSize)
char *data = const_cast<char*>(uploadByteDevice->readPointer(maxSize, currentUploadDataLength)); char *data = const_cast<char*>(uploadByteDevice->readPointer(maxSize, currentUploadDataLength));
if (currentUploadDataLength == 0) { if (currentUploadDataLength == 0) {
uploadDeviceChoking = true;
// No bytes from upload byte device. There will be bytes later, it will emit readyRead() // No bytes from upload byte device. There will be bytes later, it will emit readyRead()
// and our uploadByteDeviceReadyReadSlot() is called. // and our uploadByteDeviceReadyReadSlot() is called.
return; return;
} else {
uploadDeviceChoking = false;
} }
// Let's make a copy of this data // Let's make a copy of this data
@ -1306,8 +1310,13 @@ void QNetworkReplyHttpImplPrivate::wantUploadDataSlot(qint64 maxSize)
void QNetworkReplyHttpImplPrivate::uploadByteDeviceReadyReadSlot() void QNetworkReplyHttpImplPrivate::uploadByteDeviceReadyReadSlot()
{ {
// Start the flow between this thread and the HTTP thread again by triggering a upload. // Start the flow between this thread and the HTTP thread again by triggering a upload.
// However only do this when we were choking before, else the state in
// QNonContiguousByteDeviceThreadForwardImpl gets messed up.
if (uploadDeviceChoking) {
uploadDeviceChoking = false;
wantUploadDataSlot(1024); wantUploadDataSlot(1024);
} }
}
/* /*

View File

@ -212,6 +212,7 @@ public:
// upload // upload
QNonContiguousByteDevice* createUploadByteDevice(); QNonContiguousByteDevice* createUploadByteDevice();
QSharedPointer<QNonContiguousByteDevice> uploadByteDevice; QSharedPointer<QNonContiguousByteDevice> uploadByteDevice;
bool uploadDeviceChoking; // if we couldn't readPointer() any data at the moment
QIODevice *outgoingData; QIODevice *outgoingData;
QSharedPointer<QRingBuffer> outgoingDataBuffer; QSharedPointer<QRingBuffer> outgoingDataBuffer;
void emitReplyUploadProgress(qint64 bytesSent, qint64 bytesTotal); // dup? void emitReplyUploadProgress(qint64 bytesSent, qint64 bytesTotal); // dup?

View File

@ -7883,6 +7883,8 @@ protected slots:
//qDebug() << Q_FUNC_INFO; //qDebug() << Q_FUNC_INFO;
bandwidthQuota = 8*1024; // fill quota bandwidthQuota = 8*1024; // fill quota
emit readyRead(); emit readyRead();
// Emitting readyRead() several times triggers a bug ("QIODevice::read: Called with maxSize < 0") we fix with this commit
emit readyRead();
} }
}; };