/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Author: Milian Wolff, KDAB (milian.wolff@kdab.com) ** Contact: https://www.qt.io/licensing/ ** ** This file is part of Qt Creator. ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ****************************************************************************/ #include "valgrindrunner.h" #include "valgrindprocess.h" #include #include #include #include #include #include #include using namespace ProjectExplorer; namespace Valgrind { class ValgrindRunner::Private { public: Private(ValgrindRunner *runner) : q(runner) {} void run(ValgrindProcess *process); ValgrindRunner *q; ValgrindProcess *process = 0; QProcess::ProcessChannelMode channelMode = QProcess::SeparateChannels; bool finished = false; QString valgrindExecutable; QStringList valgrindArguments; StandardRunnable debuggee; bool useStartupProject = true; QSsh::SshConnectionParameters connParams; }; void ValgrindRunner::Private::run(ValgrindProcess *_process) { if (process && process->isRunning()) { process->close(); process->disconnect(q); process->deleteLater(); } QTC_ASSERT(_process, return); process = _process; process->setProcessChannelMode(channelMode); // consider appending our options last so they override any interfering user-supplied options // -q as suggested by valgrind manual process->setValgrindExecutable(valgrindExecutable); process->setValgrindArguments(q->fullValgrindArguments()); process->setDebuggee(debuggee); QObject::connect(process, &ValgrindProcess::processOutput, q, &ValgrindRunner::processOutputReceived); QObject::connect(process, &ValgrindProcess::started, q, &ValgrindRunner::started); QObject::connect(process, &ValgrindProcess::finished, q, &ValgrindRunner::processFinished); QObject::connect(process, &ValgrindProcess::error, q, &ValgrindRunner::processError); QObject::connect(process, &ValgrindProcess::localHostAddressRetrieved, q, &ValgrindRunner::localHostAddressRetrieved); process->run(); } ValgrindRunner::ValgrindRunner(QObject *parent) : QObject(parent), d(new Private(this)) { } ValgrindRunner::~ValgrindRunner() { if (d->process && d->process->isRunning()) { // make sure we don't delete the thread while it's still running waitForFinished(); } delete d; d = 0; } void ValgrindRunner::setValgrindExecutable(const QString &executable) { d->valgrindExecutable = executable; } QString ValgrindRunner::valgrindExecutable() const { return d->valgrindExecutable; } void ValgrindRunner::setValgrindArguments(const QStringList &toolArguments) { d->valgrindArguments = toolArguments; } QStringList ValgrindRunner::valgrindArguments() const { return d->valgrindArguments; } QStringList ValgrindRunner::fullValgrindArguments() const { QStringList fullArgs = valgrindArguments(); fullArgs << QString::fromLatin1("--tool=%1").arg(tool()); if (Utils::HostOsInfo::isMacHost()) // May be slower to start but without it we get no filenames for symbols. fullArgs << QLatin1String("--dsymutil=yes"); return fullArgs; } void ValgrindRunner::setDebuggee(const StandardRunnable &debuggee) { d->debuggee = debuggee; } const QSsh::SshConnectionParameters &ValgrindRunner::connectionParameters() const { return d->connParams; } void ValgrindRunner::setConnectionParameters(const QSsh::SshConnectionParameters &connParams) { d->connParams = connParams; } void ValgrindRunner::setProcessChannelMode(QProcess::ProcessChannelMode mode) { d->channelMode = mode; } void ValgrindRunner::setUseStartupProject(bool useStartupProject) { d->useStartupProject = useStartupProject; } bool ValgrindRunner::useStartupProject() const { return d->useStartupProject; } void ValgrindRunner::waitForFinished() const { if (d->finished || !d->process) return; QEventLoop loop; connect(this, &ValgrindRunner::finished, &loop, &QEventLoop::quit); loop.exec(); } bool ValgrindRunner::start() { // FIXME: This wrongly uses "useStartupProject" for a Local/Remote decision. d->run(new ValgrindProcess(d->useStartupProject, d->connParams, 0, this)); return true; } void ValgrindRunner::processError(QProcess::ProcessError e) { if (d->finished) return; d->finished = true; // make sure we don't wait for the connection anymore emit processErrorReceived(errorString(), e); emit finished(); } void ValgrindRunner::processFinished(int ret, QProcess::ExitStatus status) { if (d->finished) return; d->finished = true; // make sure we don't wait for the connection anymore emit finished(); if (ret != 0 || status == QProcess::CrashExit) emit processErrorReceived(errorString(), d->process->processError()); } void ValgrindRunner::localHostAddressRetrieved(const QHostAddress &localHostAddress) { Q_UNUSED(localHostAddress); } QString ValgrindRunner::errorString() const { if (d->process) return d->process->errorString(); else return QString(); } void ValgrindRunner::stop() { if (d->process) d->process->close(); } ValgrindProcess *ValgrindRunner::valgrindProcess() const { return d->process; } } // namespace Valgrind