summaryrefslogtreecommitdiffstats
path: root/src/corelib/io/qprocess_win.cpp
diff options
context:
space:
mode:
authorAlex Trotsenko <alex1973tr@gmail.com>2021-06-09 12:04:56 +0300
committerAlex Trotsenko <alex1973tr@gmail.com>2021-06-18 22:20:47 +0300
commit79490d2a4c1208d54104e3eed69739db700897c4 (patch)
tree37c3cf0a98660416bb5e4074ec4fe47407dfc0da /src/corelib/io/qprocess_win.cpp
parentbccfb92507dc23f7629ce9aa5c3aef3754cb1d8f (diff)
QProcess/Win: avoid double buffering on write
As QWindowsPipeWriter now maintains a chunk queue, there is no need to use the internal QIODevice buffer and wait for the previous operation to complete. This also allows us to get rid of the stdinWriteTrigger timer; however, as a trade-off, QWindowsPipeWriter now needs to accept data even before a handle is assigned. Change-Id: I17fe0e36a6165fe05100bfab3fe01fc0d880d617 Reviewed-by: Oswald Buddenhagen <oswald.buddenhagen@gmx.de>
Diffstat (limited to 'src/corelib/io/qprocess_win.cpp')
-rw-r--r--src/corelib/io/qprocess_win.cpp74
1 files changed, 21 insertions, 53 deletions
diff --git a/src/corelib/io/qprocess_win.cpp b/src/corelib/io/qprocess_win.cpp
index e557d10453..24031b9163 100644
--- a/src/corelib/io/qprocess_win.cpp
+++ b/src/corelib/io/qprocess_win.cpp
@@ -53,7 +53,6 @@
#include <qrandom.h>
#include <qwineventnotifier.h>
#include <qscopedvaluerollback.h>
-#include <qtimer.h>
#include <private/qsystemlibrary_p.h>
#include <private/qthread_p.h>
@@ -376,8 +375,6 @@ void QProcessPrivate::cleanup()
q_func()->setProcessState(QProcess::NotRunning);
closeChannels();
- delete stdinWriteTrigger;
- stdinWriteTrigger = nullptr;
delete processFinishedNotifier;
processFinishedNotifier = nullptr;
if (pid) {
@@ -618,6 +615,12 @@ void QProcessPrivate::startProcess()
return;
}
+ // The pipe writer may have already been created before we had
+ // the pipe handle, specifically if the user wrote data from the
+ // stateChanged() slot.
+ if (stdinChannel.writer)
+ stdinChannel.writer->setHandle(stdinChannel.pipe[1]);
+
q->setProcessState(QProcess::Running);
// User can call kill()/terminate() from the stateChanged() slot
// so check before proceeding
@@ -713,9 +716,6 @@ bool QProcessPrivate::drainOutputPipes()
bool QProcessPrivate::waitForReadyRead(const QDeadlineTimer &deadline)
{
forever {
- if (!writeBuffer.isEmpty() && !_q_canWrite())
- return false;
-
QProcessPoller poller(*this);
int ret = poller.poll(deadline);
if (ret < 0)
@@ -748,10 +748,11 @@ bool QProcessPrivate::waitForReadyRead(const QDeadlineTimer &deadline)
bool QProcessPrivate::waitForBytesWritten(const QDeadlineTimer &deadline)
{
forever {
- // If no write is pending, try to start one. However, at entry into
- // the loop the write buffer can be empty to start with, in which
- // case _q_caWrite() fails immediately.
- if (pipeWriterBytesToWrite() == 0 && !_q_canWrite())
+ // At entry into the loop the pipe writer's buffer can be empty to
+ // start with, in which case we fail immediately. Also, if the input
+ // pipe goes down somewhere in the code below, we avoid waiting for
+ // a full timeout.
+ if (pipeWriterBytesToWrite() == 0)
return false;
QProcessPoller poller(*this);
@@ -797,9 +798,6 @@ bool QProcessPrivate::waitForFinished(const QDeadlineTimer &deadline)
#endif
forever {
- if (!writeBuffer.isEmpty() && !_q_canWrite())
- return false;
-
QProcessPoller poller(*this);
int ret = poller.poll(deadline);
if (ret < 0)
@@ -854,16 +852,16 @@ qint64 QProcess::writeData(const char *data, qint64 len)
return 0;
}
- if (!d->stdinWriteTrigger) {
- d->stdinWriteTrigger = new QTimer;
- d->stdinWriteTrigger->setSingleShot(true);
- QObjectPrivate::connect(d->stdinWriteTrigger, &QTimer::timeout,
- d, &QProcessPrivate::_q_canWrite);
+ if (!d->stdinChannel.writer) {
+ d->stdinChannel.writer = new QWindowsPipeWriter(d->stdinChannel.pipe[1], this);
+ QObjectPrivate::connect(d->stdinChannel.writer, &QWindowsPipeWriter::bytesWritten,
+ d, &QProcessPrivate::_q_bytesWritten);
}
- d->write(data, len);
- if (!d->stdinWriteTrigger->isActive())
- d->stdinWriteTrigger->start();
+ if (d->isWriteChunkCached(data, len))
+ d->stdinChannel.writer->write(*(d->currentWriteChunk));
+ else
+ d->stdinChannel.writer->write(data, len);
#if defined QPROCESS_DEBUG
qDebug("QProcess::writeData(%p \"%s\", %lld) == %lld (written to buffer)",
@@ -885,38 +883,8 @@ void QProcessPrivate::_q_bytesWritten(qint64 bytes)
QScopedValueRollback<bool> guard(emittedBytesWritten, true);
emit q->bytesWritten(bytes);
}
- _q_canWrite();
-}
-
-bool QProcessPrivate::_q_canWrite()
-{
- if (writeBuffer.isEmpty()) {
- if (stdinChannel.closed && pipeWriterBytesToWrite() == 0)
- closeWriteChannel();
-#if defined QPROCESS_DEBUG
- qDebug("QProcessPrivate::canWrite(), not writing anything (empty write buffer).");
-#endif
- return false;
- }
-
- return writeToStdin();
-}
-
-bool QProcessPrivate::writeToStdin()
-{
- Q_Q(QProcess);
-
- if (!stdinChannel.writer) {
- stdinChannel.writer = new QWindowsPipeWriter(stdinChannel.pipe[1], q);
- QObjectPrivate::connect(stdinChannel.writer, &QWindowsPipeWriter::bytesWritten,
- this, &QProcessPrivate::_q_bytesWritten);
- } else {
- if (stdinChannel.writer->isWriteOperationActive())
- return true;
- }
-
- stdinChannel.writer->write(writeBuffer.read());
- return true;
+ if (stdinChannel.closed && pipeWriterBytesToWrite() == 0)
+ closeWriteChannel();
}
// Use ShellExecuteEx() to trigger an UAC prompt when CreateProcess()fails