forked from qt-creator/qt-creator
		
	Replace hard distinction between 'out', 'err' with flexible OutputFormat. Also make sure that QmlProfiler shows application output. Change-Id: I130c71884321e4c59c9a75f0836c37a7f0e805c4 Reviewed-on: http://codereview.qt.nokia.com/458 Reviewed-by: Christiaan Janssen <christiaan.janssen@nokia.com>
		
			
				
	
	
		
			362 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			362 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/**************************************************************************
 | 
						|
**
 | 
						|
** This file is part of Qt Creator
 | 
						|
**
 | 
						|
** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies).
 | 
						|
**
 | 
						|
** Author: Milian Wolff, KDAB (milian.wolff@kdab.com)
 | 
						|
**
 | 
						|
** Contact: Nokia Corporation (info@qt.nokia.com)
 | 
						|
**
 | 
						|
**
 | 
						|
** GNU Lesser General Public License Usage
 | 
						|
**
 | 
						|
** 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.
 | 
						|
**
 | 
						|
** Other Usage
 | 
						|
**
 | 
						|
** Alternatively, this file may be used in accordance with the terms and
 | 
						|
** conditions contained in a signed written agreement between you and Nokia.
 | 
						|
**
 | 
						|
** If you have questions regarding the use of this file, please contact
 | 
						|
** Nokia at info@qt.nokia.com.
 | 
						|
**
 | 
						|
**************************************************************************/
 | 
						|
 | 
						|
#include "valgrindprocess.h"
 | 
						|
 | 
						|
#include <QtCore/QDebug>
 | 
						|
#include <QtCore/QEventLoop>
 | 
						|
#include <QtCore/QFileInfo>
 | 
						|
 | 
						|
#include <utils/qtcassert.h>
 | 
						|
 | 
						|
namespace Valgrind {
 | 
						|
 | 
						|
ValgrindProcess::ValgrindProcess(QObject *parent)
 | 
						|
    : QObject(parent)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
////////////////////////
 | 
						|
 | 
						|
LocalValgrindProcess::LocalValgrindProcess(QObject *parent)
 | 
						|
: ValgrindProcess(parent)
 | 
						|
{
 | 
						|
    connect(&m_process, SIGNAL(finished(int, QProcess::ExitStatus)),
 | 
						|
            this, SIGNAL(finished(int, QProcess::ExitStatus)));
 | 
						|
    connect(&m_process, SIGNAL(started()),
 | 
						|
            this, SIGNAL(started()));
 | 
						|
    connect(&m_process, SIGNAL(error(QProcess::ProcessError)),
 | 
						|
            this, SIGNAL(error(QProcess::ProcessError)));
 | 
						|
    connect(&m_process, SIGNAL(readyReadStandardError()),
 | 
						|
            this, SLOT(readyReadStandardError()));
 | 
						|
    connect(&m_process, SIGNAL(readyReadStandardOutput()),
 | 
						|
            this, SLOT(readyReadStandardOutput()));
 | 
						|
}
 | 
						|
 | 
						|
void LocalValgrindProcess::setProcessChannelMode(QProcess::ProcessChannelMode mode)
 | 
						|
{
 | 
						|
    m_process.setProcessChannelMode(mode);
 | 
						|
}
 | 
						|
 | 
						|
void LocalValgrindProcess::setWorkingDirectory(const QString &path)
 | 
						|
{
 | 
						|
    m_process.setWorkingDirectory(path);
 | 
						|
}
 | 
						|
 | 
						|
QString LocalValgrindProcess::workingDirectory() const
 | 
						|
{
 | 
						|
    return m_process.workingDirectory();
 | 
						|
}
 | 
						|
 | 
						|
bool LocalValgrindProcess::isRunning() const
 | 
						|
{
 | 
						|
    return m_process.state() != QProcess::NotRunning;
 | 
						|
}
 | 
						|
 | 
						|
void LocalValgrindProcess::setEnvironment(const Utils::Environment &environment)
 | 
						|
{
 | 
						|
    m_process.setEnvironment(environment);
 | 
						|
}
 | 
						|
 | 
						|
void LocalValgrindProcess::close()
 | 
						|
{
 | 
						|
    m_process.terminate();
 | 
						|
}
 | 
						|
 | 
						|
void LocalValgrindProcess::run(const QString &valgrindExecutable, const QStringList &valgrindArguments,
 | 
						|
                                const QString &debuggeeExecutable, const QString &debuggeeArguments)
 | 
						|
{
 | 
						|
    QString arguments;
 | 
						|
    Utils::QtcProcess::addArgs(&arguments, valgrindArguments);
 | 
						|
 | 
						|
    Utils::QtcProcess::addArg(&arguments, debuggeeExecutable);
 | 
						|
    Utils::QtcProcess::addArgs(&arguments, debuggeeArguments);
 | 
						|
 | 
						|
    m_process.setCommand(valgrindExecutable, arguments);
 | 
						|
    m_process.start();
 | 
						|
    m_process.waitForStarted();
 | 
						|
    m_pid = m_process.pid();
 | 
						|
}
 | 
						|
 | 
						|
QString LocalValgrindProcess::errorString() const
 | 
						|
{
 | 
						|
    return m_process.errorString();
 | 
						|
}
 | 
						|
 | 
						|
QProcess::ProcessError LocalValgrindProcess::error() const
 | 
						|
{
 | 
						|
    return m_process.error();
 | 
						|
}
 | 
						|
 | 
						|
Q_PID LocalValgrindProcess::pid() const
 | 
						|
{
 | 
						|
    return m_pid;
 | 
						|
}
 | 
						|
 | 
						|
void LocalValgrindProcess::readyReadStandardError()
 | 
						|
{
 | 
						|
    const QByteArray b = m_process.readAllStandardError();
 | 
						|
    if (!b.isEmpty())
 | 
						|
        emit processOutput(b, Utils::StdErrFormat);
 | 
						|
}
 | 
						|
 | 
						|
void LocalValgrindProcess::readyReadStandardOutput()
 | 
						|
{
 | 
						|
    const QByteArray b = m_process.readAllStandardOutput();
 | 
						|
    if (!b.isEmpty())
 | 
						|
        emit processOutput(b, Utils::StdOutFormat);
 | 
						|
}
 | 
						|
 | 
						|
////////////////////////
 | 
						|
 | 
						|
RemoteValgrindProcess::RemoteValgrindProcess(const Utils::SshConnectionParameters &sshParams,
 | 
						|
                                             QObject *parent)
 | 
						|
    : ValgrindProcess(parent)
 | 
						|
    , m_params(sshParams)
 | 
						|
    , m_error(QProcess::UnknownError)
 | 
						|
    , m_pid(0)
 | 
						|
{}
 | 
						|
 | 
						|
RemoteValgrindProcess::RemoteValgrindProcess(const Utils::SshConnection::Ptr &connection, QObject *parent)
 | 
						|
    : ValgrindProcess(parent)
 | 
						|
    , m_params(connection->connectionParameters())
 | 
						|
    , m_connection(connection)
 | 
						|
    , m_error(QProcess::UnknownError)
 | 
						|
    , m_pid(0)
 | 
						|
{}
 | 
						|
 | 
						|
bool RemoteValgrindProcess::isRunning() const
 | 
						|
{
 | 
						|
    return m_process && m_process->isRunning();
 | 
						|
}
 | 
						|
 | 
						|
void RemoteValgrindProcess::run(const QString &valgrindExecutable, const QStringList &valgrindArguments,
 | 
						|
                                const QString &debuggeeExecutable, const QString &debuggeeArguments)
 | 
						|
{
 | 
						|
    m_valgrindExe = valgrindExecutable;
 | 
						|
    m_debuggee = debuggeeExecutable;
 | 
						|
    m_debuggeeArgs = debuggeeArguments;
 | 
						|
    m_valgrindArgs = valgrindArguments;
 | 
						|
 | 
						|
    // connect to host and wait for connection
 | 
						|
    if (!m_connection)
 | 
						|
        m_connection = Utils::SshConnection::create(m_params);
 | 
						|
 | 
						|
    if (m_connection->state() != Utils::SshConnection::Connected) {
 | 
						|
        connect(m_connection.data(), SIGNAL(connected()),
 | 
						|
                this, SLOT(connected()));
 | 
						|
        connect(m_connection.data(), SIGNAL(error(Utils::SshError)),
 | 
						|
                this, SLOT(error(Utils::SshError)));
 | 
						|
        m_connection->connectToHost();
 | 
						|
    } else {
 | 
						|
        connected();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void RemoteValgrindProcess::connected()
 | 
						|
{
 | 
						|
    QTC_ASSERT(m_connection->state() == Utils::SshConnection::Connected, return);
 | 
						|
 | 
						|
    // connected, run command
 | 
						|
    QString cmd;
 | 
						|
 | 
						|
    if (!m_workingDir.isEmpty())
 | 
						|
        cmd += QString("cd '%1' && ").arg(m_workingDir);
 | 
						|
 | 
						|
    QString arguments;
 | 
						|
    Utils::QtcProcess::addArgs(&arguments, m_valgrindArgs);
 | 
						|
    Utils::QtcProcess::addArg(&arguments, m_debuggee);
 | 
						|
    Utils::QtcProcess::addArgs(&arguments, m_debuggeeArgs);
 | 
						|
    cmd += m_valgrindExe + ' ' + arguments;
 | 
						|
 | 
						|
    m_process = m_connection->createRemoteProcess(cmd.toUtf8());
 | 
						|
    connect(m_process.data(), SIGNAL(errorOutputAvailable(QByteArray)),
 | 
						|
            this, SLOT(standardError(QByteArray)));
 | 
						|
    connect(m_process.data(), SIGNAL(outputAvailable(QByteArray)),
 | 
						|
            this, SLOT(standardOutput(QByteArray)));
 | 
						|
    connect(m_process.data(), SIGNAL(closed(int)),
 | 
						|
            this, SLOT(closed(int)));
 | 
						|
    connect(m_process.data(), SIGNAL(started()),
 | 
						|
            this, SLOT(processStarted()));
 | 
						|
    m_process->start();
 | 
						|
}
 | 
						|
 | 
						|
Utils::SshConnection::Ptr RemoteValgrindProcess::connection() const
 | 
						|
{
 | 
						|
    return m_connection;
 | 
						|
}
 | 
						|
 | 
						|
void RemoteValgrindProcess::processStarted()
 | 
						|
{
 | 
						|
    QTC_ASSERT(m_connection->state() == Utils::SshConnection::Connected, return);
 | 
						|
 | 
						|
    // find out what PID our process has
 | 
						|
 | 
						|
    // NOTE: valgrind cloaks its name,
 | 
						|
    // e.g.: valgrind --tool=memcheck foobar
 | 
						|
    // => ps aux, pidof will see valgrind.bin
 | 
						|
    // => pkill/killall/top... will see memcheck-amd64-linux or similar
 | 
						|
    // hence we need to do something more complex...
 | 
						|
 | 
						|
    // plain path to exe, m_valgrindExe contains e.g. env vars etc. pp.
 | 
						|
    const QString proc = m_valgrindExe.split(QLatin1Char(' ')).last();
 | 
						|
    // sleep required since otherwise we might only match "bash -c..."
 | 
						|
    //  and not the actual valgrind run
 | 
						|
    const QString cmd = QString("sleep 1; ps ax" // list all processes with aliased name
 | 
						|
                                " | grep '\\b%1.*%2'" // find valgrind process
 | 
						|
                                " | tail -n 1" // limit to single process
 | 
						|
                                               // we pick the last one, first would be "bash -c ..."
 | 
						|
                                " | awk '{print $1;}'" // get pid
 | 
						|
                                ).arg(proc, QFileInfo(m_debuggee).fileName());
 | 
						|
 | 
						|
    m_findPID = m_connection->createRemoteProcess(cmd.toUtf8());
 | 
						|
    connect(m_findPID.data(), SIGNAL(errorOutputAvailable(QByteArray)),
 | 
						|
            this, SLOT(standardOutput(QByteArray)));
 | 
						|
    connect(m_findPID.data(), SIGNAL(outputAvailable(QByteArray)),
 | 
						|
            this, SLOT(findPIDOutputReceived(QByteArray)));
 | 
						|
    m_findPID->start();
 | 
						|
}
 | 
						|
 | 
						|
void RemoteValgrindProcess::findPIDOutputReceived(const QByteArray &output)
 | 
						|
{
 | 
						|
    bool ok;
 | 
						|
    m_pid = output.trimmed().toLongLong(&ok);
 | 
						|
    if (!ok) {
 | 
						|
        m_pid = 0;
 | 
						|
        m_errorString = tr("Could not determine remote PID.");
 | 
						|
        m_error = QProcess::FailedToStart;
 | 
						|
        emit ValgrindProcess::error(QProcess::FailedToStart);
 | 
						|
        close();
 | 
						|
    } else {
 | 
						|
        emit started();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void RemoteValgrindProcess::standardOutput(const QByteArray &output)
 | 
						|
{
 | 
						|
    emit processOutput(output, Utils::StdOutFormat);
 | 
						|
}
 | 
						|
 | 
						|
void RemoteValgrindProcess::standardError(const QByteArray &output)
 | 
						|
{
 | 
						|
    emit processOutput(output, Utils::StdErrFormat);
 | 
						|
}
 | 
						|
 | 
						|
void RemoteValgrindProcess::error(Utils::SshError error)
 | 
						|
{
 | 
						|
    switch (error) {
 | 
						|
        case Utils::SshTimeoutError:
 | 
						|
            m_error = QProcess::Timedout;
 | 
						|
            break;
 | 
						|
        default:
 | 
						|
            m_error = QProcess::FailedToStart;
 | 
						|
            break;
 | 
						|
    }
 | 
						|
    m_errorString = m_connection->errorString();
 | 
						|
    emit ValgrindProcess::error(m_error);
 | 
						|
}
 | 
						|
 | 
						|
void RemoteValgrindProcess::close()
 | 
						|
{
 | 
						|
    QTC_ASSERT(m_connection->state() == Utils::SshConnection::Connected, return);
 | 
						|
    if (m_process) {
 | 
						|
        if (m_pid) {
 | 
						|
            const QString killTemplate = QString("kill -%2 %1" // kill
 | 
						|
                                                ).arg(m_pid);
 | 
						|
 | 
						|
            const QString niceKill = killTemplate.arg("SIGTERM");
 | 
						|
            const QString brutalKill = killTemplate.arg("SIGKILL");
 | 
						|
            const QString remoteCall = niceKill + QLatin1String("; sleep 1; ") + brutalKill;
 | 
						|
 | 
						|
            Utils::SshRemoteProcess::Ptr cleanup = m_connection->createRemoteProcess(remoteCall.toUtf8());
 | 
						|
            cleanup->start();
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void RemoteValgrindProcess::closed(int status)
 | 
						|
{
 | 
						|
    QTC_ASSERT(m_process, return);
 | 
						|
 | 
						|
    m_errorString = m_process->errorString();
 | 
						|
    if (status == Utils::SshRemoteProcess::FailedToStart) {
 | 
						|
        m_error = QProcess::FailedToStart;
 | 
						|
        emit ValgrindProcess::error(QProcess::FailedToStart);
 | 
						|
    } else if (status == Utils::SshRemoteProcess::ExitedNormally) {
 | 
						|
        emit finished(m_process->exitCode(), QProcess::NormalExit);
 | 
						|
    } else if (status == Utils::SshRemoteProcess::KilledBySignal) {
 | 
						|
        m_error = QProcess::Crashed;
 | 
						|
        emit finished(m_process->exitCode(), QProcess::CrashExit);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
QString RemoteValgrindProcess::errorString() const
 | 
						|
{
 | 
						|
    return m_errorString;
 | 
						|
}
 | 
						|
 | 
						|
QProcess::ProcessError RemoteValgrindProcess::error() const
 | 
						|
{
 | 
						|
    return m_error;
 | 
						|
}
 | 
						|
 | 
						|
void RemoteValgrindProcess::setEnvironment(const Utils::Environment &environment)
 | 
						|
{
 | 
						|
    Q_UNUSED(environment);
 | 
						|
    ///TODO: anything that should/could be done here?
 | 
						|
}
 | 
						|
 | 
						|
void RemoteValgrindProcess::setProcessChannelMode(QProcess::ProcessChannelMode mode)
 | 
						|
{
 | 
						|
    Q_UNUSED(mode);
 | 
						|
    ///TODO: support this by handling the mode internally
 | 
						|
}
 | 
						|
 | 
						|
void RemoteValgrindProcess::setWorkingDirectory(const QString &path)
 | 
						|
{
 | 
						|
    m_workingDir = path;
 | 
						|
}
 | 
						|
 | 
						|
QString RemoteValgrindProcess::workingDirectory() const
 | 
						|
{
 | 
						|
    return m_workingDir;
 | 
						|
}
 | 
						|
 | 
						|
Q_PID RemoteValgrindProcess::pid() const
 | 
						|
{
 | 
						|
    return m_pid;
 | 
						|
}
 | 
						|
 | 
						|
} // namespace Valgrind
 |