Skip to content
Snippets Groups Projects
makestep.cpp 13.4 KiB
Newer Older
/**************************************************************************
con's avatar
con committed
**
** This file is part of Qt Creator
**
hjk's avatar
hjk committed
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
con's avatar
con committed
**
** Contact: Nokia Corporation (qt-info@nokia.com)
con's avatar
con committed
**
** Commercial Usage
** Licensees holding valid Qt Commercial licenses may use this file in
** accordance with the Qt Commercial License Agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and Nokia.
** 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.
** If you are unsure which license is appropriate for your use, please
hjk's avatar
hjk committed
** contact the sales department at http://qt.nokia.com/contact.
con's avatar
con committed
**
**************************************************************************/
hjk's avatar
hjk committed

con's avatar
con committed
#include "makestep.h"
hjk's avatar
hjk committed

con's avatar
con committed
#include "qt4project.h"
Tobias Hunger's avatar
Tobias Hunger committed
#include "qt4target.h"
#include "qt4buildconfiguration.h"
con's avatar
con committed
#include "qt4projectmanagerconstants.h"
#include "qtparser.h"
#include "qt-s60/abldparser.h"
#include "qt-s60/sbsv2parser.h"
con's avatar
con committed

Tobias Hunger's avatar
Tobias Hunger committed
#include <projectexplorer/buildsteplist.h>
#include <projectexplorer/gnumakeparser.h>
#include <projectexplorer/projectexplorer.h>
#include <extensionsystem/pluginmanager.h>
hjk's avatar
hjk committed
#include <QtCore/QDir>
#include <QtCore/QFileInfo>
con's avatar
con committed

using ExtensionSystem::PluginManager;
using namespace Qt4ProjectManager;
using namespace Qt4ProjectManager::Internal;

namespace {
const char * const MAKESTEP_BS_ID("Qt4ProjectManager.MakeStep");

const char * const MAKE_ARGUMENTS_KEY("Qt4ProjectManager.MakeStep.MakeArguments");
const char * const MAKE_COMMAND_KEY("Qt4ProjectManager.MakeStep.MakeCommand");
const char * const CLEAN_KEY("Qt4ProjectManager.MakeStep.Clean");
}

Tobias Hunger's avatar
Tobias Hunger committed
MakeStep::MakeStep(ProjectExplorer::BuildStepList *bsl) :
    AbstractProcessStep(bsl, QLatin1String(MAKESTEP_BS_ID)),
Tobias Hunger's avatar
Tobias Hunger committed
MakeStep::MakeStep(ProjectExplorer::BuildStepList *bsl, MakeStep *bs) :
    AbstractProcessStep(bsl, bs),
    m_clean(bs->m_clean),
    m_userArgs(bs->m_userArgs),
    m_makeCmd(bs->m_makeCmd)
con's avatar
con committed
{
con's avatar
con committed

Tobias Hunger's avatar
Tobias Hunger committed
MakeStep::MakeStep(ProjectExplorer::BuildStepList *bsl, const QString &id) :
    AbstractProcessStep(bsl, id),
con's avatar
con committed
}

void MakeStep::ctor()
con's avatar
con committed
{
    setDefaultDisplayName(tr("Make", "Qt4 MakeStep display name."));
con's avatar
con committed

con's avatar
con committed
}

dt's avatar
dt committed
Qt4BuildConfiguration *MakeStep::qt4BuildConfiguration() const
{
    return static_cast<Qt4BuildConfiguration *>(buildConfiguration());
}

void MakeStep::setClean(bool clean)
{
    m_clean = clean;
}

QVariantMap MakeStep::toMap() const
    QVariantMap map(ProjectExplorer::AbstractProcessStep::toMap());
    map.insert(QLatin1String(MAKE_ARGUMENTS_KEY), m_userArgs);
    map.insert(QLatin1String(MAKE_COMMAND_KEY), m_makeCmd);
    map.insert(QLatin1String(CLEAN_KEY), m_clean);
    return map;
bool MakeStep::fromMap(const QVariantMap &map)
    m_makeCmd = map.value(QLatin1String(MAKE_COMMAND_KEY)).toString();
    m_userArgs = map.value(QLatin1String(MAKE_ARGUMENTS_KEY)).toStringList();
    m_clean = map.value(QLatin1String(CLEAN_KEY)).toBool();
    return ProjectExplorer::AbstractProcessStep::fromMap(map);
bool MakeStep::init()
con's avatar
con committed
{
dt's avatar
dt committed
    Qt4BuildConfiguration *bc = qt4BuildConfiguration();
    Utils::Environment environment = bc->environment();
    setEnvironment(environment);
con's avatar
con committed

    QString workingDirectory;
    if (bc->subNodeBuild())
        workingDirectory = bc->subNodeBuild()->buildDir();
    else
        workingDirectory = bc->buildDirectory();
    setWorkingDirectory(workingDirectory);
con's avatar
con committed

    QString makeCmd = bc->makeCommand();
    if (!m_makeCmd.isEmpty())
        makeCmd = m_makeCmd;
con's avatar
con committed
    if (!QFileInfo(makeCmd).isAbsolute()) {
        // Try to detect command in environment
        const QString tmp = environment.searchInPath(makeCmd);
        if (tmp.isEmpty()) {
            emit addOutput(tr("Could not find make command: %1 in the build environment").arg(makeCmd), BuildStep::ErrorOutput);
con's avatar
con committed
            return false;
        }
        makeCmd = tmp;
    }
    setCommand(makeCmd);
con's avatar
con committed

    // If we are cleaning, then make can fail with a error code, but that doesn't mean
    // we should stop the clean queue
Tobias Hunger's avatar
Tobias Hunger committed
    // That is mostly so that rebuild works on a already clean project
    setIgnoreReturnValue(m_clean);
    QStringList args = m_userArgs;
        if (!bc->defaultMakeTarget().isEmpty())
            args << bc->defaultMakeTarget();
con's avatar
con committed
    }
    // -w option enables "Enter"/"Leaving directory" messages, which we need for detecting the
    // absolute file path
    // FIXME doing this without the user having a way to override this is rather bad
    // so we only do it for unix and if the user didn't override the make command
    // but for now this is the least invasive change
    ProjectExplorer::ToolChain *toolchain = bc->toolChain();
    if (toolchain) {
        if (toolchain->type() != ProjectExplorer::ToolChain::MSVC &&
            toolchain->type() != ProjectExplorer::ToolChain::WINCE) {
            if (m_makeCmd.isEmpty())
                args << "-w";
        }
con's avatar
con committed
    }

    setEnabled(true);
    setArguments(args);
con's avatar
con committed

    if (bc->qtVersion()->supportsTargetId(Qt4ProjectManager::Constants::S60_DEVICE_TARGET_ID) ||
        bc->qtVersion()->supportsTargetId(Qt4ProjectManager::Constants::S60_EMULATOR_TARGET_ID)) {
        if (bc->qtVersion()->isBuildWithSymbianSbsV2()) {
            setOutputParser(new SbsV2Parser);
        } else {
            setOutputParser(new AbldParser);
            m_gnuMakeParser = new ProjectExplorer::GnuMakeParser(workingDirectory);
            appendOutputParser(m_gnuMakeParser);
        }
    } else {
        setOutputParser(new ProjectExplorer::GnuMakeParser(workingDirectory));
    }
    appendOutputParser(new QtParser);

    if (toolchain)
        appendOutputParser(toolchain->outputParser());

    return AbstractProcessStep::init();
con's avatar
con committed
}

void MakeStep::run(QFutureInterface<bool> & fi)
{
Tobias Hunger's avatar
Tobias Hunger committed
    if (qt4BuildConfiguration()->qt4Target()->qt4Project()->rootProjectNode()->projectType() == ScriptTemplate) {
con's avatar
con committed
        fi.reportResult(true);
        return;
    }

    AbstractProcessStep::run(fi);
con's avatar
con committed
}

bool MakeStep::processSucceeded(int exitCode, QProcess::ExitStatus status)
{
    // Symbian does retun 0, even on failed makes! So we check for fatal make errors here.
    if (m_gnuMakeParser)
        return m_gnuMakeParser->fatalErrors() == 0;

    return AbstractProcessStep::processSucceeded(exitCode, status);
}

con's avatar
con committed
bool MakeStep::immutable() const
{
    return false;
con's avatar
con committed
}

ProjectExplorer::BuildStepConfigWidget *MakeStep::createConfigWidget()
{
    return new MakeStepConfigWidget(this);
}

QStringList MakeStep::userArguments()
void MakeStep::setUserArguments(const QStringList &arguments)
    m_userArgs = arguments;
    emit userArgumentsChanged();
con's avatar
con committed
MakeStepConfigWidget::MakeStepConfigWidget(MakeStep *makeStep)
    : BuildStepConfigWidget(), m_ui(new Ui::MakeStep), m_makeStep(makeStep), m_ignoreChange(false)
con's avatar
con committed
{

    m_ui->makePathChooser->setExpectedKind(Utils::PathChooser::ExistingCommand);
    m_ui->makePathChooser->setBaseDirectory(Utils::PathChooser::homePath());

    connect(m_ui->makePathChooser, SIGNAL(changed(QString)),
            this, SLOT(makeEdited()));
    connect(m_ui->makeArgumentsLineEdit, SIGNAL(textEdited(QString)),
            this, SLOT(makeArgumentsLineEdited()));
    connect(makeStep, SIGNAL(userArgumentsChanged()),
            this, SLOT(userArgumentsChanged()));
dt's avatar
dt committed
    connect(makeStep->buildConfiguration(), SIGNAL(buildDirectoryChanged()),
            this, SLOT(updateDetails()));

    connect(makeStep->qt4BuildConfiguration(), SIGNAL(qtVersionChanged()),
            this, SLOT(qtVersionChanged()));

    connect(ProjectExplorer::ProjectExplorerPlugin::instance(), SIGNAL(settingsChanged()),
            this, SLOT(updateMakeOverrideLabel()));
    connect(ProjectExplorer::ProjectExplorerPlugin::instance(), SIGNAL(settingsChanged()),
            this, SLOT(updateDetails()));
MakeStepConfigWidget::~MakeStepConfigWidget()
{
    delete m_ui;
}

void MakeStepConfigWidget::qtVersionChanged()
{
    updateMakeOverrideLabel();
    updateDetails();
}

void MakeStepConfigWidget::updateMakeOverrideLabel()
{
dt's avatar
dt committed
    Qt4BuildConfiguration *qt4bc = m_makeStep->qt4BuildConfiguration();
    m_ui->makeLabel->setText(tr("Override %1:").arg(qt4bc->makeCommand()));
dt's avatar
dt committed
void MakeStepConfigWidget::updateDetails()
dt's avatar
dt committed
{
dt's avatar
dt committed
    Qt4BuildConfiguration *bc = m_makeStep->qt4BuildConfiguration();
    QString workingDirectory = bc->buildDirectory();
    QString makeCmd = bc->makeCommand();
    if (!m_makeStep->m_makeCmd.isEmpty())
        makeCmd = m_makeStep->m_makeCmd;
dt's avatar
dt committed
    if (!QFileInfo(makeCmd).isAbsolute()) {
        Utils::Environment environment = bc->environment();
dt's avatar
dt committed
        // Try to detect command in environment
        const QString tmp = environment.searchInPath(makeCmd);
        if (tmp.isEmpty()) {
            m_summaryText = tr("<b>Make:</b> %1 not found in the environment.").arg(makeCmd);
dt's avatar
dt committed
            emit updateSummary();
            return;
        }
        makeCmd = tmp;
    }
    // -w option enables "Enter"/"Leaving directory" messages, which we need for detecting the
    // absolute file path
    // FIXME doing this without the user having a way to override this is rather bad
dt's avatar
dt committed
    // so we only do it for unix and if the user didn't override the make command
    // but for now this is the least invasive change
    QStringList args = m_makeStep->userArguments();
dt's avatar
dt committed
    ProjectExplorer::ToolChain::ToolChainType t = ProjectExplorer::ToolChain::UNKNOWN;
    ProjectExplorer::ToolChain *toolChain = bc->toolChain();
dt's avatar
dt committed
    if (toolChain)
        t = toolChain->type();
dt's avatar
dt committed
    if (t != ProjectExplorer::ToolChain::MSVC && t != ProjectExplorer::ToolChain::WINCE) {
        if (m_makeStep->m_makeCmd.isEmpty())
dt's avatar
dt committed
            args << "-w";
    }
    m_summaryText = tr("<b>Make:</b> %1 %2 in %3").arg(QFileInfo(makeCmd).fileName(), args.join(" "),
                                                       QDir::toNativeSeparators(workingDirectory));
dt's avatar
dt committed
    emit updateSummary();
}

QString MakeStepConfigWidget::summaryText() const
{
    return m_summaryText;
con's avatar
con committed
}

QString MakeStepConfigWidget::displayName() const
{
    return m_makeStep->displayName();
}

void MakeStepConfigWidget::userArgumentsChanged()
    if (m_ignoreChange)
        return;
    const QStringList &makeArguments = m_makeStep->userArguments();
    m_ui->makeArgumentsLineEdit->setText(Utils::Environment::joinArgumentList(makeArguments));
void MakeStepConfigWidget::init()
con's avatar
con committed
{
    updateMakeOverrideLabel();
    const QString &makeCmd = m_makeStep->m_makeCmd;
    m_ui->makePathChooser->setPath(makeCmd);
    const QStringList &makeArguments = m_makeStep->userArguments();
    m_ui->makeArgumentsLineEdit->setText(Utils::Environment::joinArgumentList(makeArguments));
dt's avatar
dt committed
    updateDetails();
con's avatar
con committed
}

void MakeStepConfigWidget::makeEdited()
con's avatar
con committed
{
    m_makeStep->m_makeCmd = m_ui->makePathChooser->rawPath();
dt's avatar
dt committed
    updateDetails();
con's avatar
con committed
}

void MakeStepConfigWidget::makeArgumentsLineEdited()
con's avatar
con committed
{
    m_ignoreChange = true;
    m_makeStep->setUserArguments(
            Utils::Environment::parseCombinedArgString(m_ui->makeArgumentsLineEdit->text()));
    m_ignoreChange = false;
dt's avatar
dt committed
    updateDetails();
con's avatar
con committed
}
MakeStepFactory::MakeStepFactory(QObject *parent) :
    ProjectExplorer::IBuildStepFactory(parent)
{
}

MakeStepFactory::~MakeStepFactory()
{
}

Tobias Hunger's avatar
Tobias Hunger committed
bool MakeStepFactory::canCreate(ProjectExplorer::BuildStepList *parent, const QString &id) const
Tobias Hunger's avatar
Tobias Hunger committed
    if (parent->target()->project()->id() != QLatin1String(Constants::QT4PROJECT_ID))
        return false;
    return (id == QLatin1String(MAKESTEP_BS_ID));
Tobias Hunger's avatar
Tobias Hunger committed
ProjectExplorer::BuildStep *MakeStepFactory::create(ProjectExplorer::BuildStepList *parent, const QString &id)
Tobias Hunger's avatar
Tobias Hunger committed
    if (!canCreate(parent, id))
        return 0;
    return new MakeStep(parent);
Tobias Hunger's avatar
Tobias Hunger committed
bool MakeStepFactory::canClone(ProjectExplorer::BuildStepList *parent, ProjectExplorer::BuildStep *source) const
Tobias Hunger's avatar
Tobias Hunger committed
    return canCreate(parent, source->id());
Tobias Hunger's avatar
Tobias Hunger committed
ProjectExplorer::BuildStep *MakeStepFactory::clone(ProjectExplorer::BuildStepList *parent, ProjectExplorer::BuildStep *source)
Tobias Hunger's avatar
Tobias Hunger committed
    if (!canClone(parent, source))
        return 0;
    return new MakeStep(parent, static_cast<MakeStep *>(source));
}

Tobias Hunger's avatar
Tobias Hunger committed
bool MakeStepFactory::canRestore(ProjectExplorer::BuildStepList *parent, const QVariantMap &map) const
{
    QString id(ProjectExplorer::idFromMap(map));
Tobias Hunger's avatar
Tobias Hunger committed
    return canCreate(parent, id);
Tobias Hunger's avatar
Tobias Hunger committed
ProjectExplorer::BuildStep *MakeStepFactory::restore(ProjectExplorer::BuildStepList *parent, const QVariantMap &map)
Tobias Hunger's avatar
Tobias Hunger committed
    if (!canRestore(parent, map))
        return 0;
    MakeStep *bs(new MakeStep(parent));
    if (bs->fromMap(map))
        return bs;
    delete bs;
    return 0;
}

Tobias Hunger's avatar
Tobias Hunger committed
QStringList MakeStepFactory::availableCreationIds(ProjectExplorer::BuildStepList *parent) const
Tobias Hunger's avatar
Tobias Hunger committed
    if (parent->target()->project()->id() == QLatin1String(Constants::QT4PROJECT_ID))
        return QStringList() << QLatin1String(MAKESTEP_BS_ID);
    return QStringList();
QString MakeStepFactory::displayNameForId(const QString &id) const
    if (id == QLatin1String(MAKESTEP_BS_ID))
        return tr("Make");
    return QString();