/****************************************************************************
**
** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies).
** All rights reserved.
** Contact: Nokia Corporation (qt-info@nokia.com)
**
** This file is part of the Qt3Support module of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:LGPL$
** No Commercial Usage
** This file contains pre-release code and may not be distributed.
** You may use this file in accordance with the terms and conditions
** contained in the Technology Preview License Agreement accompanying
** this package.
**
** GNU Lesser General Public License Usage
** Alternatively, this file may be used under the terms of the GNU Lesser
** General Public License version 2.1 as published by the Free Software
** Foundation and appearing in the file LICENSE.LGPL included in the
** packaging of this file. Please review the following information to
** ensure the GNU Lesser General Public License version 2.1 requirements
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** In addition, as a special exception, Nokia gives you certain additional
** rights. These rights are described in the Nokia Qt LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
** If you have questions regarding the use of this file, please contact
** Nokia at qt-info@nokia.com.
**
**
**
**
**
**
**
**
** $QT_END_LICENSE$
**
****************************************************************************/
#include "qplatformdefs.h"
#include "q3process.h"
#ifndef QT_NO_PROCESS
#include "qapplication.h"
#include "q3cstring.h"
#include "q3ptrqueue.h"
#include "qtimer.h"
#include "qregexp.h"
#include "private/q3membuf_p.h"
#include "qt_windows.h"
#ifdef Q_OS_WINCE
#define STARTF_USESTDHANDLES 1
#endif
QT_BEGIN_NAMESPACE
//#define QT_Q3PROCESS_DEBUG
/***********************************************************************
*
* Q3ProcessPrivate
*
**********************************************************************/
class Q3ProcessPrivate
{
public:
Q3ProcessPrivate( Q3Process *proc )
{
stdinBufRead = 0;
pipeStdin[0] = 0;
pipeStdin[1] = 0;
pipeStdout[0] = 0;
pipeStdout[1] = 0;
pipeStderr[0] = 0;
pipeStderr[1] = 0;
exitValuesCalculated = false;
lookup = new QTimer( proc );
qApp->connect( lookup, SIGNAL(timeout()),
proc, SLOT(timeout()) );
pid = 0;
}
~Q3ProcessPrivate()
{
reset();
}
void reset()
{
while ( !stdinBuf.isEmpty() ) {
delete stdinBuf.dequeue();
}
closeHandles();
stdinBufRead = 0;
pipeStdin[0] = 0;
pipeStdin[1] = 0;
pipeStdout[0] = 0;
pipeStdout[1] = 0;
pipeStderr[0] = 0;
pipeStderr[1] = 0;
exitValuesCalculated = false;
deletePid();
}
void closeHandles()
{
if( pipeStdin[1] != 0 ) {
CloseHandle( pipeStdin[1] );
pipeStdin[1] = 0;
}
if( pipeStdout[0] != 0 ) {
CloseHandle( pipeStdout[0] );
pipeStdout[0] = 0;
}
if( pipeStderr[0] != 0 ) {
CloseHandle( pipeStderr[0] );
pipeStderr[0] = 0;
}
}
void deletePid()
{
if ( pid ) {
CloseHandle( pid->hProcess );
CloseHandle( pid->hThread );
delete pid;
pid = 0;
}
}
void newPid()
{
deletePid();
pid = new PROCESS_INFORMATION;
memset( pid, 0, sizeof(PROCESS_INFORMATION) );
}
Q3Membuf bufStdout;
Q3Membuf bufStderr;
Q3PtrQueue<QByteArray> stdinBuf;
HANDLE pipeStdin[2];
HANDLE pipeStdout[2];
HANDLE pipeStderr[2];
QTimer *lookup;
PROCESS_INFORMATION *pid;
uint stdinBufRead;
bool exitValuesCalculated;
};
/***********************************************************************
*
* Q3Process
*
**********************************************************************/
void Q3Process::init()
{
d = new Q3ProcessPrivate( this );
exitStat = 0;
exitNormal = false;
}
void Q3Process::reset()
{
d->reset();
exitStat = 0;
exitNormal = false;
d->bufStdout.clear();
d->bufStderr.clear();
}
Q3Membuf* Q3Process::membufStdout()
{
if( d->pipeStdout[0] != 0 )
socketRead( 1 );
return &d->bufStdout;
}
Q3Membuf* Q3Process::membufStderr()
{
if( d->pipeStderr[0] != 0 )
socketRead( 2 );
return &d->bufStderr;
}
Q3Process::~Q3Process()
{
delete d;
}
bool Q3Process::start( QStringList *env )
{
#if defined(QT_Q3PROCESS_DEBUG)
qDebug( "Q3Process::start()" );
#endif
reset();
if ( _arguments.isEmpty() )
return false;
// Open the pipes. Make non-inheritable copies of input write and output
// read handles to avoid non-closable handles (this is done by the
// DuplicateHandle() call).
SECURITY_ATTRIBUTES secAtt = { sizeof( SECURITY_ATTRIBUTES ), NULL, TRUE };
#ifndef Q_OS_WINCE
// I guess there is no stdin stdout and stderr on Q_OS_WINCE to dup
// CreatePipe and DupilcateHandle aren't available for Q_OS_WINCE
HANDLE tmpStdin, tmpStdout, tmpStderr;
if ( comms & Stdin ) {
if ( !CreatePipe( &d->pipeStdin[0], &tmpStdin, &secAtt, 0 ) ) {
d->closeHandles();
return false;
}
if ( !DuplicateHandle( GetCurrentProcess(), tmpStdin, GetCurrentProcess(), &d->pipeStdin[1], 0, FALSE, DUPLICATE_SAME_ACCESS ) ) {
d->closeHandles();
return false;
}
if ( !CloseHandle( tmpStdin ) ) {
d->closeHandles();
return false;
}
}
if ( comms & Stdout ) {
if ( !CreatePipe( &tmpStdout, &d->pipeStdout[1], &secAtt, 0 ) ) {
d->closeHandles();
return false;
}
if ( !DuplicateHandle( GetCurrentProcess(), tmpStdout, GetCurrentProcess(), &d->pipeStdout[0], 0, FALSE, DUPLICATE_SAME_ACCESS ) ) {
d->closeHandles();
return false;
}
if ( !CloseHandle( tmpStdout ) ) {
d->closeHandles();
return false;
}
}
if ( comms & Stderr ) {
if ( !CreatePipe( &tmpStderr, &d->pipeStderr[1], &secAtt, 0 ) ) {
d->closeHandles();
return false;
}
if ( !DuplicateHandle( GetCurrentProcess(), tmpStderr, GetCurrentProcess(), &d->pipeStderr[0], 0, FALSE, DUPLICATE_SAME_ACCESS ) ) {
d->closeHandles();
return false;
}
if ( !CloseHandle( tmpStderr ) ) {
d->closeHandles();
return false;
}
}
if ( comms & DupStderr ) {
CloseHandle( d->pipeStderr[1] );
d->pipeStderr[1] = d->pipeStdout[1];
}
#endif
// construct the arguments for CreateProcess()
QString args;
QString appName;
QStringList::Iterator it = _arguments.begin();
args = *it;
++it;
if ( args.endsWith( QLatin1String(".bat") ) && args.contains( QLatin1Char(' ') ) ) {
// CreateProcess() seems to have a strange semantics (see also
// http://www.experts-exchange.com/Programming/Programming_Platforms/Win_Prog/Q_11138647.html):
// If you start a batch file with spaces in the filename, the first
// argument to CreateProcess() must be the name of the batchfile
// without quotes, but the second argument must start with the same
// argument with quotes included. But if the same approach is used for
// .exe files, it doesn't work.
appName = args;
args = QLatin1Char('"') + args + QLatin1Char('"');
}
for ( ; it != _arguments.end(); ++it ) {
QString tmp = *it;
// escape a single " because the arguments will be parsed
tmp.replace( QLatin1Char('\"'), QLatin1String("\\\"") );
if ( tmp.isEmpty() || tmp.contains( QLatin1Char(' ') ) || tmp.contains( QLatin1Char('\t') ) ) {
// The argument must not end with a \ since this would be interpreted
// as escaping the quote -- rather put the \ behind the quote: e.g.
// rather use "foo"\ than "foo\"
QString endQuote( QLatin1String("\"") );
int i = tmp.length();
while ( i>0 && tmp.at( i-1 ) == QLatin1Char('\\') ) {
--i;
endQuote += QLatin1Char('\\');
}
args += QLatin1String(" \"") + tmp.left( i ) + endQuote;
} else {
args += QLatin1Char(' ') + tmp;
}
}
#if defined(QT_Q3PROCESS_DEBUG)
qDebug( "Q3Process::start(): args [%s]", args.latin1() );
#endif
// CreateProcess()
bool success;
d->newPid();
STARTUPINFOW startupInfo = {
sizeof( STARTUPINFO ), 0, 0, 0,
(ulong)CW_USEDEFAULT, (ulong)CW_USEDEFAULT, (ulong)CW_USEDEFAULT, (ulong)CW_USEDEFAULT,
0, 0, 0,
STARTF_USESTDHANDLES,
0, 0, 0,
d->pipeStdin[0], d->pipeStdout[1], d->pipeStderr[1]
};
wchar_t *applicationName;
if ( appName.isNull() )
applicationName = 0;
else
applicationName = _wcsdup( (wchar_t*)appName.utf16() );
wchar_t *commandLine = _wcsdup( (wchar_t*)args.utf16() );
QByteArray envlist;
if ( env != 0 ) {
int pos = 0;
// add PATH if necessary (for DLL loading)
QByteArray path = qgetenv( "PATH" );
if ( env->grep( QRegExp(QLatin1String("^PATH="),FALSE) ).empty() && !path.isNull() ) {
QString tmp = QString::fromLatin1("PATH=%1").arg(QLatin1String(path.constData()));
uint tmpSize = sizeof(wchar_t) * (tmp.length() + 1);
envlist.resize( envlist.size() + tmpSize );
memcpy( envlist.data() + pos, tmp.utf16(), tmpSize );
pos += tmpSize;
}
// add the user environment
for ( QStringList::Iterator it = env->begin(); it != env->end(); it++ ) {
QString tmp = *it;
uint tmpSize = sizeof(wchar_t) * (tmp.length() + 1);
envlist.resize( envlist.size() + tmpSize );
memcpy( envlist.data() + pos, tmp.utf16(), tmpSize );
pos += tmpSize;
}
// add the 2 terminating 0 (actually 4, just to be on the safe side)
envlist.resize( envlist.size()+4 );
envlist[pos++] = 0;
envlist[pos++] = 0;
envlist[pos++] = 0;
envlist[pos++] = 0;
}
success = CreateProcess( applicationName, commandLine,
0, 0, TRUE, ( comms == 0 ? CREATE_NEW_CONSOLE : CREATE_NO_WINDOW )
#ifndef Q_OS_WINCE
| CREATE_UNICODE_ENVIRONMENT
#endif
, env == 0 ? 0 : envlist.data(),
(wchar_t*)QDir::toNativeSeparators(workingDir.absPath()).utf16(),
&startupInfo, d->pid );
free( applicationName );
free( commandLine );
if ( !success ) {
d->deletePid();
return false;
}
#ifndef Q_OS_WINCE
if ( comms & Stdin )
CloseHandle( d->pipeStdin[0] );
if ( comms & Stdout )
CloseHandle( d->pipeStdout[1] );
if ( (comms & Stderr) && !(comms & DupStderr) )
CloseHandle( d->pipeStderr[1] );
#endif
if ( ioRedirection || notifyOnExit ) {
d->lookup->start( 100 );
}
// cleanup and return
return true;
}
static BOOL CALLBACK qt_terminateApp( HWND hwnd, LPARAM procId )
{
DWORD procId_win;
GetWindowThreadProcessId( hwnd, &procId_win );
if( procId_win == (DWORD)procId )
PostMessage( hwnd, WM_CLOSE, 0, 0 );
return TRUE;
}
void Q3Process::tryTerminate() const
{
if ( d->pid )
EnumWindows( qt_terminateApp, (LPARAM)d->pid->dwProcessId );
}
void Q3Process::kill() const
{
if ( d->pid )
TerminateProcess( d->pid->hProcess, 0xf291 );
}
bool Q3Process::isRunning() const
{
if ( !d->pid )
return false;
if ( WaitForSingleObject( d->pid->hProcess, 0) == WAIT_OBJECT_0 ) {
// there might be data to read
Q3Process *that = (Q3Process*)this;
that->socketRead( 1 ); // try stdout
that->socketRead( 2 ); // try stderr
// compute the exit values
if ( !d->exitValuesCalculated ) {
DWORD exitCode;
if ( GetExitCodeProcess( d->pid->hProcess, &exitCode ) ) {
if ( exitCode != STILL_ACTIVE ) { // this should ever be true?
that->exitNormal = exitCode != 0xf291;
that->exitStat = exitCode;
}
}
d->exitValuesCalculated = true;
}
d->deletePid();
d->closeHandles();
return false;
} else {
return true;
}
}
bool Q3Process::canReadLineStdout() const
{
if( !d->pipeStdout[0] )
return d->bufStdout.size() != 0;
Q3Process *that = (Q3Process*)this;
return that->membufStdout()->scanNewline( 0 );
}
bool Q3Process::canReadLineStderr() const
{
if( !d->pipeStderr[0] )
return d->bufStderr.size() != 0;
Q3Process *that = (Q3Process*)this;
return that->membufStderr()->scanNewline( 0 );
}
void Q3Process::writeToStdin( const QByteArray& buf )
{
d->stdinBuf.enqueue( new QByteArray(buf) );
socketWrite( 0 );
}
void Q3Process::closeStdin( )
{
if ( d->pipeStdin[1] != 0 ) {
CloseHandle( d->pipeStdin[1] );
d->pipeStdin[1] = 0;
}
}
void Q3Process::socketRead( int fd )
{
// fd == 1: stdout, fd == 2: stderr
HANDLE dev;
if ( fd == 1 ) {
dev = d->pipeStdout[0];
} else if ( fd == 2 ) {
dev = d->pipeStderr[0];
} else {
return;
}
#ifndef Q_OS_WINCE
// get the number of bytes that are waiting to be read
unsigned long i, r;
char dummy;
if ( !PeekNamedPipe( dev, &dummy, 1, &r, &i, 0 ) ) {
return; // ### is it worth to dig for the reason of the error?
}
#else
unsigned long i = 1000;
#endif
if ( i > 0 ) {
Q3Membuf *buffer;
if ( fd == 1 )
buffer = &d->bufStdout;
else
buffer = &d->bufStderr;
QByteArray *ba = new QByteArray( i );
uint sz = readStddev( dev, ba->data(), i );
if ( sz != i )
ba->resize( i );
if ( sz == 0 ) {
delete ba;
return;
}
buffer->append( ba );
if ( fd == 1 )
emit readyReadStdout();
else
emit readyReadStderr();
}
}
void Q3Process::socketWrite( int )
{
DWORD written;
while ( !d->stdinBuf.isEmpty() && isRunning() ) {
if ( !WriteFile( d->pipeStdin[1],
d->stdinBuf.head()->data() + d->stdinBufRead,
qMin( 8192, int(d->stdinBuf.head()->size() - d->stdinBufRead) ),
&written, 0 ) ) {
d->lookup->start( 100 );
return;
}
d->stdinBufRead += written;
if ( d->stdinBufRead == (DWORD)d->stdinBuf.head()->size() ) {
d->stdinBufRead = 0;
delete d->stdinBuf.dequeue();
if ( wroteToStdinConnected && d->stdinBuf.isEmpty() )
emit wroteToStdin();
}
}
}
void Q3Process::flushStdin()
{
socketWrite( 0 );
}
/*
Use a timer for polling misc. stuff.
*/
void Q3Process::timeout()
{
// Disable the timer temporary since one of the slots that are connected to
// the readyRead...(), etc. signals might trigger recursion if
// processEvents() is called.
d->lookup->stop();
// try to write pending data to stdin
if ( !d->stdinBuf.isEmpty() )
socketWrite( 0 );
if ( ioRedirection ) {
socketRead( 1 ); // try stdout
socketRead( 2 ); // try stderr
}
if ( isRunning() ) {
// enable timer again, if needed
if ( !d->stdinBuf.isEmpty() || ioRedirection || notifyOnExit )
d->lookup->start( 100 );
} else if ( notifyOnExit ) {
emit processExited();
}
}
/*
read on the pipe
*/
uint Q3Process::readStddev( HANDLE dev, char *buf, uint bytes )
{
if ( bytes > 0 ) {
ulong r;
if ( ReadFile( dev, buf, bytes, &r, 0 ) )
return r;
}
return 0;
}
/*
Used by connectNotify() and disconnectNotify() to change the value of
ioRedirection (and related behaviour)
*/
void Q3Process::setIoRedirection( bool value )
{
ioRedirection = value;
if ( !ioRedirection && !notifyOnExit )
d->lookup->stop();
if ( ioRedirection ) {
if ( isRunning() )
d->lookup->start( 100 );
}
}
/*
Used by connectNotify() and disconnectNotify() to change the value of
notifyOnExit (and related behaviour)
*/
void Q3Process::setNotifyOnExit( bool value )
{
notifyOnExit = value;
if ( !ioRedirection && !notifyOnExit )
d->lookup->stop();
if ( notifyOnExit ) {
if ( isRunning() )
d->lookup->start( 100 );
}
}
/*
Used by connectNotify() and disconnectNotify() to change the value of
wroteToStdinConnected (and related behaviour)
*/
void Q3Process::setWroteStdinConnected( bool value )
{
wroteToStdinConnected = value;
}
Q3Process::PID Q3Process::processIdentifier()
{
return d->pid;
}
QT_END_NAMESPACE
#endif // QT_NO_PROCESS