forked from qt-creator/qt-creator
CMake: Implement different backends to run cmake
Only the original one is implemented so far, but at least in theory backends for retrieving data from cmake can now be switched at runtime. Change-Id: Id73a81c7d40f078be95defd30a38511dca3a3720 Reviewed-by: Tim Jenssen <tim.jenssen@qt.io> Reviewed-by: Eike Ziller <eike.ziller@qt.io>
This commit is contained in:
678
src/plugins/cmakeprojectmanager/tealeafreader.cpp
Normal file
678
src/plugins/cmakeprojectmanager/tealeafreader.cpp
Normal file
@@ -0,0 +1,678 @@
|
||||
/****************************************************************************
|
||||
**
|
||||
** Copyright (C) 2016 The Qt Company Ltd.
|
||||
** 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 "tealeafreader.h"
|
||||
|
||||
#include "cmakebuildconfiguration.h"
|
||||
#include "cmakecbpparser.h"
|
||||
#include "cmakekitinformation.h"
|
||||
#include "cmakeparser.h"
|
||||
#include "cmakeprojectconstants.h"
|
||||
#include "cmakeprojectmanager.h"
|
||||
#include "cmakeprojectnodes.h"
|
||||
|
||||
#include <coreplugin/documentmanager.h>
|
||||
#include <coreplugin/editormanager/editormanager.h>
|
||||
#include <coreplugin/idocument.h>
|
||||
#include <coreplugin/messagemanager.h>
|
||||
#include <coreplugin/progressmanager/progressmanager.h>
|
||||
#include <coreplugin/reaper.h>
|
||||
#include <cpptools/projectpartbuilder.h>
|
||||
#include <projectexplorer/headerpath.h>
|
||||
#include <projectexplorer/ioutputparser.h>
|
||||
#include <projectexplorer/kitinformation.h>
|
||||
#include <projectexplorer/projectexplorerconstants.h>
|
||||
#include <projectexplorer/target.h>
|
||||
#include <projectexplorer/task.h>
|
||||
#include <projectexplorer/taskhub.h>
|
||||
#include <projectexplorer/toolchain.h>
|
||||
#include <projectexplorer/toolchainmanager.h>
|
||||
|
||||
#include <utils/algorithm.h>
|
||||
#include <utils/qtcassert.h>
|
||||
#include <utils/qtcprocess.h>
|
||||
|
||||
#include <QDateTime>
|
||||
#include <QFileInfo>
|
||||
|
||||
using namespace ProjectExplorer;
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// Helper:
|
||||
// --------------------------------------------------------------------
|
||||
|
||||
namespace CMakeProjectManager {
|
||||
namespace Internal {
|
||||
|
||||
class CMakeFile : public Core::IDocument
|
||||
{
|
||||
public:
|
||||
CMakeFile(TeaLeafReader *r, const Utils::FileName &fileName);
|
||||
|
||||
ReloadBehavior reloadBehavior(ChangeTrigger state, ChangeType type) const override;
|
||||
bool reload(QString *errorString, ReloadFlag flag, ChangeType type) override;
|
||||
|
||||
private:
|
||||
TeaLeafReader *m_reader;
|
||||
};
|
||||
|
||||
CMakeFile::CMakeFile(TeaLeafReader *r, const Utils::FileName &fileName) : m_reader(r)
|
||||
{
|
||||
setId("Cmake.ProjectFile");
|
||||
setMimeType(Constants::CMAKEPROJECTMIMETYPE);
|
||||
setFilePath(fileName);
|
||||
}
|
||||
|
||||
Core::IDocument::ReloadBehavior CMakeFile::reloadBehavior(ChangeTrigger state, ChangeType type) const
|
||||
{
|
||||
Q_UNUSED(state)
|
||||
Q_UNUSED(type)
|
||||
return BehaviorSilent;
|
||||
}
|
||||
|
||||
bool CMakeFile::reload(QString *errorString, Core::IDocument::ReloadFlag flag, Core::IDocument::ChangeType type)
|
||||
{
|
||||
Q_UNUSED(errorString);
|
||||
Q_UNUSED(flag);
|
||||
|
||||
if (type != TypePermissions)
|
||||
emit m_reader->dirty();
|
||||
return true;
|
||||
}
|
||||
|
||||
static QString lineSplit(const QString &rest, const QByteArray &array, std::function<void(const QString &)> f)
|
||||
{
|
||||
QString tmp = rest + Utils::SynchronousProcess::normalizeNewlines(QString::fromLocal8Bit(array));
|
||||
int start = 0;
|
||||
int end = tmp.indexOf(QLatin1Char('\n'), start);
|
||||
while (end >= 0) {
|
||||
f(tmp.mid(start, end - start));
|
||||
start = end + 1;
|
||||
end = tmp.indexOf(QLatin1Char('\n'), start);
|
||||
}
|
||||
return tmp.mid(start);
|
||||
}
|
||||
|
||||
static QStringList toArguments(const CMakeConfig &config, const Utils::MacroExpander *expander) {
|
||||
return Utils::transform(config, [expander](const CMakeConfigItem &i) -> QString {
|
||||
return i.toArgument(expander);
|
||||
});
|
||||
}
|
||||
|
||||
static QByteArray trimCMakeCacheLine(const QByteArray &in) {
|
||||
int start = 0;
|
||||
while (start < in.count() && (in.at(start) == ' ' || in.at(start) == '\t'))
|
||||
++start;
|
||||
|
||||
return in.mid(start, in.count() - start - 1);
|
||||
}
|
||||
|
||||
static QByteArrayList splitCMakeCacheLine(const QByteArray &line) {
|
||||
const int colonPos = line.indexOf(':');
|
||||
if (colonPos < 0)
|
||||
return QByteArrayList();
|
||||
|
||||
const int equalPos = line.indexOf('=', colonPos + 1);
|
||||
if (equalPos < colonPos)
|
||||
return QByteArrayList();
|
||||
|
||||
return QByteArrayList() << line.mid(0, colonPos)
|
||||
<< line.mid(colonPos + 1, equalPos - colonPos - 1)
|
||||
<< line.mid(equalPos + 1);
|
||||
}
|
||||
|
||||
static CMakeConfigItem::Type fromByteArray(const QByteArray &type) {
|
||||
if (type == "BOOL")
|
||||
return CMakeConfigItem::BOOL;
|
||||
if (type == "STRING")
|
||||
return CMakeConfigItem::STRING;
|
||||
if (type == "FILEPATH")
|
||||
return CMakeConfigItem::FILEPATH;
|
||||
if (type == "PATH")
|
||||
return CMakeConfigItem::PATH;
|
||||
QTC_CHECK(type == "INTERNAL" || type == "STATIC");
|
||||
|
||||
return CMakeConfigItem::INTERNAL;
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// TeaLeafReader:
|
||||
// --------------------------------------------------------------------
|
||||
|
||||
TeaLeafReader::TeaLeafReader()
|
||||
{
|
||||
connect(Core::EditorManager::instance(), &Core::EditorManager::aboutToSave,
|
||||
this, [this](const Core::IDocument *document) {
|
||||
if (m_cmakeFiles.contains(document->filePath()))
|
||||
emit dirty();
|
||||
});
|
||||
}
|
||||
|
||||
TeaLeafReader::~TeaLeafReader()
|
||||
{
|
||||
stop();
|
||||
resetData();
|
||||
}
|
||||
|
||||
bool TeaLeafReader::isCompatible(const BuildDirReader::Parameters &p)
|
||||
{
|
||||
Q_UNUSED(p);
|
||||
return true; // FIXME: Needs to take server mode into account!
|
||||
}
|
||||
|
||||
void TeaLeafReader::resetData()
|
||||
{
|
||||
m_hasData = false;
|
||||
|
||||
qDeleteAll(m_watchedFiles);
|
||||
m_watchedFiles.clear();
|
||||
|
||||
m_cmakeCache.clear();
|
||||
m_projectName.clear();
|
||||
m_buildTargets.clear();
|
||||
qDeleteAll(m_files);
|
||||
m_files.clear();
|
||||
}
|
||||
|
||||
void TeaLeafReader::parse(bool force)
|
||||
{
|
||||
const QString cbpFile = CMakeManager::findCbpFile(QDir(m_parameters.buildDirectory.toString()));
|
||||
const QFileInfo cbpFileFi = cbpFile.isEmpty() ? QFileInfo() : QFileInfo(cbpFile);
|
||||
if (!cbpFileFi.exists()) {
|
||||
// Initial create:
|
||||
startCMake(toArguments(m_parameters.configuration, m_parameters.expander));
|
||||
return;
|
||||
}
|
||||
|
||||
const bool mustUpdate = force
|
||||
|| m_cmakeFiles.isEmpty()
|
||||
|| Utils::anyOf(m_cmakeFiles, [&cbpFileFi](const Utils::FileName &f) {
|
||||
return f.toFileInfo().lastModified() > cbpFileFi.lastModified();
|
||||
});
|
||||
if (mustUpdate) {
|
||||
startCMake(QStringList());
|
||||
} else {
|
||||
extractData();
|
||||
m_hasData = true;
|
||||
emit dataAvailable();
|
||||
}
|
||||
}
|
||||
|
||||
void TeaLeafReader::stop()
|
||||
{
|
||||
cleanUpProcess();
|
||||
|
||||
if (m_future) {
|
||||
m_future->reportCanceled();
|
||||
m_future->reportFinished();
|
||||
delete m_future;
|
||||
m_future = nullptr;
|
||||
}
|
||||
}
|
||||
|
||||
bool TeaLeafReader::isParsing() const
|
||||
{
|
||||
return m_cmakeProcess && m_cmakeProcess->state() != QProcess::NotRunning;
|
||||
}
|
||||
|
||||
bool TeaLeafReader::hasData() const
|
||||
{
|
||||
return m_hasData;
|
||||
}
|
||||
|
||||
QList<CMakeBuildTarget> TeaLeafReader::buildTargets() const
|
||||
{
|
||||
return m_buildTargets;
|
||||
}
|
||||
|
||||
CMakeConfig TeaLeafReader::parsedConfiguration() const
|
||||
{
|
||||
CMakeConfig result;
|
||||
Utils::FileName cacheFile = m_parameters.buildDirectory;
|
||||
cacheFile.appendPath(QLatin1String("CMakeCache.txt"));
|
||||
if (!cacheFile.exists())
|
||||
return result;
|
||||
QString errorMessage;
|
||||
m_cmakeCache = parseConfiguration(cacheFile, &errorMessage);
|
||||
if (!errorMessage.isEmpty())
|
||||
emit errorOccured(errorMessage);
|
||||
const Utils::FileName sourceOfBuildDir
|
||||
= Utils::FileName::fromUtf8(CMakeConfigItem::valueOf("CMAKE_HOME_DIRECTORY", m_cmakeCache));
|
||||
const Utils::FileName canonicalSourceOfBuildDir = Utils::FileUtils::canonicalPath(sourceOfBuildDir);
|
||||
const Utils::FileName canonicalSourceDirectory = Utils::FileUtils::canonicalPath(m_parameters.sourceDirectory);
|
||||
if (canonicalSourceOfBuildDir != canonicalSourceDirectory) { // Uses case-insensitive compare where appropriate
|
||||
emit errorOccured(tr("The build directory is not for %1 but for %2")
|
||||
.arg(canonicalSourceOfBuildDir.toUserOutput(),
|
||||
canonicalSourceDirectory.toUserOutput()));
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
CMakeConfig TeaLeafReader::parseConfiguration(const Utils::FileName &cacheFile, QString *errorMessage) const
|
||||
{
|
||||
CMakeConfig result;
|
||||
QFile cache(cacheFile.toString());
|
||||
if (!cache.open(QIODevice::ReadOnly | QIODevice::Text)) {
|
||||
if (errorMessage)
|
||||
*errorMessage = tr("Failed to open %1 for reading.").arg(cacheFile.toUserOutput());
|
||||
return CMakeConfig();
|
||||
}
|
||||
|
||||
QSet<QByteArray> advancedSet;
|
||||
QMap<QByteArray, QByteArray> valuesMap;
|
||||
QByteArray documentation;
|
||||
while (!cache.atEnd()) {
|
||||
const QByteArray line = trimCMakeCacheLine(cache.readLine());
|
||||
|
||||
if (line.isEmpty() || line.startsWith('#'))
|
||||
continue;
|
||||
|
||||
if (line.startsWith("//")) {
|
||||
documentation = line.mid(2);
|
||||
continue;
|
||||
}
|
||||
|
||||
const QByteArrayList pieces = splitCMakeCacheLine(line);
|
||||
if (pieces.isEmpty())
|
||||
continue;
|
||||
|
||||
QTC_ASSERT(pieces.count() == 3, continue);
|
||||
const QByteArray key = pieces.at(0);
|
||||
const QByteArray type = pieces.at(1);
|
||||
const QByteArray value = pieces.at(2);
|
||||
|
||||
if (key.endsWith("-ADVANCED") && value == "1") {
|
||||
advancedSet.insert(key.left(key.count() - 9 /* "-ADVANCED" */));
|
||||
} else if (key.endsWith("-STRINGS") && fromByteArray(type) == CMakeConfigItem::INTERNAL) {
|
||||
valuesMap[key.left(key.count() - 8) /* "-STRINGS" */] = value;
|
||||
} else {
|
||||
CMakeConfigItem::Type t = fromByteArray(type);
|
||||
result << CMakeConfigItem(key, t, documentation, value);
|
||||
}
|
||||
}
|
||||
|
||||
// Set advanced flags:
|
||||
for (int i = 0; i < result.count(); ++i) {
|
||||
CMakeConfigItem &item = result[i];
|
||||
item.isAdvanced = advancedSet.contains(item.key);
|
||||
|
||||
if (valuesMap.contains(item.key)) {
|
||||
item.values = CMakeConfigItem::cmakeSplitValue(QString::fromUtf8(valuesMap[item.key]));
|
||||
} else if (item.key == "CMAKE_BUILD_TYPE") {
|
||||
// WA for known options
|
||||
item.values << "" << "Debug" << "Release" << "MinSizeRel" << "RelWithDebInfo";
|
||||
}
|
||||
}
|
||||
|
||||
Utils::sort(result, CMakeConfigItem::sortOperator());
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
void TeaLeafReader::generateProjectTree(CMakeProjectNode *root)
|
||||
{
|
||||
root->setDisplayName(m_projectName);
|
||||
|
||||
// Delete no longer necessary file watcher:
|
||||
const QSet<Utils::FileName> currentWatched
|
||||
= Utils::transform(m_watchedFiles, [](CMakeFile *cmf) { return cmf->filePath(); });
|
||||
const QSet<Utils::FileName> toWatch = m_cmakeFiles;
|
||||
QSet<Utils::FileName> toDelete = currentWatched;
|
||||
toDelete.subtract(toWatch);
|
||||
m_watchedFiles = Utils::filtered(m_watchedFiles, [&toDelete](Internal::CMakeFile *cmf) {
|
||||
if (toDelete.contains(cmf->filePath())) {
|
||||
delete cmf;
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
});
|
||||
|
||||
// Add new file watchers:
|
||||
QSet<Utils::FileName> toAdd = toWatch;
|
||||
toAdd.subtract(currentWatched);
|
||||
foreach (const Utils::FileName &fn, toAdd) {
|
||||
CMakeFile *cm = new CMakeFile(this, fn);
|
||||
Core::DocumentManager::addDocument(cm);
|
||||
m_watchedFiles.insert(cm);
|
||||
}
|
||||
|
||||
QList<FileNode *> fileNodes = m_files;
|
||||
root->buildTree(fileNodes);
|
||||
m_files.clear(); // Some of the FileNodes in files() were deleted!
|
||||
}
|
||||
|
||||
QSet<Core::Id> TeaLeafReader::updateCodeModel(CppTools::ProjectPartBuilder &ppBuilder)
|
||||
{
|
||||
QSet<Core::Id> languages;
|
||||
ToolChain *tc = ToolChainManager::findToolChain(m_parameters.toolChainId);
|
||||
const Utils::FileName sysroot = m_parameters.sysRoot;
|
||||
|
||||
QHash<QString, QStringList> targetDataCache;
|
||||
foreach (const CMakeBuildTarget &cbt, m_buildTargets) {
|
||||
if (cbt.targetType == UtilityType)
|
||||
continue;
|
||||
|
||||
// CMake shuffles the include paths that it reports via the CodeBlocks generator
|
||||
// So remove the toolchain include paths, so that at least those end up in the correct
|
||||
// place.
|
||||
QStringList cxxflags = getCXXFlagsFor(cbt, targetDataCache);
|
||||
QSet<Utils::FileName> tcIncludes;
|
||||
QStringList includePaths;
|
||||
if (tc) {
|
||||
foreach (const HeaderPath &hp, tc->systemHeaderPaths(cxxflags, sysroot))
|
||||
tcIncludes.insert(Utils::FileName::fromString(hp.path()));
|
||||
foreach (const Utils::FileName &i, cbt.includeFiles) {
|
||||
if (!tcIncludes.contains(i))
|
||||
includePaths.append(i.toString());
|
||||
}
|
||||
} else {
|
||||
includePaths = Utils::transform(cbt.includeFiles, &Utils::FileName::toString);
|
||||
}
|
||||
includePaths += m_parameters.buildDirectory.toString();
|
||||
ppBuilder.setIncludePaths(includePaths);
|
||||
ppBuilder.setCFlags(cxxflags);
|
||||
ppBuilder.setCxxFlags(cxxflags);
|
||||
ppBuilder.setDefines(cbt.defines);
|
||||
ppBuilder.setDisplayName(cbt.title);
|
||||
|
||||
const QSet<Core::Id> partLanguages
|
||||
= QSet<Core::Id>::fromList(ppBuilder.createProjectPartsForFiles(
|
||||
Utils::transform(cbt.files, [](const Utils::FileName &fn) { return fn.toString(); })));
|
||||
|
||||
languages.unite(partLanguages);
|
||||
}
|
||||
return languages;
|
||||
|
||||
}
|
||||
|
||||
void TeaLeafReader::cleanUpProcess()
|
||||
{
|
||||
if (m_cmakeProcess) {
|
||||
m_cmakeProcess->disconnect();
|
||||
Core::Reaper::reap(m_cmakeProcess);
|
||||
m_cmakeProcess = nullptr;
|
||||
}
|
||||
|
||||
// Delete issue parser:
|
||||
if (m_parser)
|
||||
m_parser->flush();
|
||||
delete m_parser;
|
||||
m_parser = nullptr;
|
||||
}
|
||||
|
||||
void TeaLeafReader::extractData()
|
||||
{
|
||||
const Utils::FileName srcDir = m_parameters.sourceDirectory;
|
||||
const Utils::FileName bldDir = m_parameters.buildDirectory;
|
||||
const Utils::FileName topCMake
|
||||
= Utils::FileName::fromString(srcDir.toString() + QLatin1String("/CMakeLists.txt"));
|
||||
|
||||
resetData();
|
||||
|
||||
m_projectName = m_parameters.projectName;
|
||||
m_files.append(new FileNode(topCMake, ProjectFileType, false));
|
||||
// Do not insert topCMake into m_cmakeFiles: The project already watches that!
|
||||
|
||||
// Find cbp file
|
||||
Utils::FileName cbpFile = Utils::FileName::fromString(CMakeManager::findCbpFile(bldDir.toString()));
|
||||
if (cbpFile.isEmpty())
|
||||
return;
|
||||
m_cmakeFiles.insert(cbpFile);
|
||||
|
||||
// Add CMakeCache.txt file:
|
||||
Utils::FileName cacheFile = m_parameters.buildDirectory;
|
||||
cacheFile.appendPath(QLatin1String("CMakeCache.txt"));
|
||||
if (cacheFile.toFileInfo().exists())
|
||||
m_cmakeFiles.insert(cacheFile);
|
||||
|
||||
// setFolderName
|
||||
CMakeCbpParser cbpparser;
|
||||
// Parsing
|
||||
if (!cbpparser.parseCbpFile(m_parameters.pathMapper, cbpFile, srcDir))
|
||||
return;
|
||||
|
||||
m_projectName = cbpparser.projectName();
|
||||
|
||||
m_files = cbpparser.fileList();
|
||||
if (cbpparser.hasCMakeFiles()) {
|
||||
m_files.append(cbpparser.cmakeFileList());
|
||||
foreach (const FileNode *node, cbpparser.cmakeFileList())
|
||||
m_cmakeFiles.insert(node->filePath());
|
||||
}
|
||||
|
||||
// Make sure the top cmakelists.txt file is always listed:
|
||||
if (!Utils::contains(m_files, [topCMake](FileNode *fn) { return fn->filePath() == topCMake; })) {
|
||||
m_files.append(new FileNode(topCMake, ProjectFileType, false));
|
||||
}
|
||||
|
||||
m_buildTargets = cbpparser.buildTargets();
|
||||
}
|
||||
|
||||
void TeaLeafReader::startCMake(const QStringList &configurationArguments)
|
||||
{
|
||||
const Utils::FileName buildDirectory = m_parameters.buildDirectory;
|
||||
QTC_ASSERT(!m_cmakeProcess, return);
|
||||
QTC_ASSERT(!m_parser, return);
|
||||
QTC_ASSERT(!m_future, return);
|
||||
QTC_ASSERT(buildDirectory.exists(), return);
|
||||
|
||||
const QString srcDir = m_parameters.sourceDirectory.toString();
|
||||
|
||||
m_parser = new CMakeParser;
|
||||
QDir source = QDir(srcDir);
|
||||
connect(m_parser, &IOutputParser::addTask, m_parser,
|
||||
[source](const Task &task) {
|
||||
if (task.file.isEmpty() || task.file.toFileInfo().isAbsolute()) {
|
||||
TaskHub::addTask(task);
|
||||
} else {
|
||||
Task t = task;
|
||||
t.file = Utils::FileName::fromString(source.absoluteFilePath(task.file.toString()));
|
||||
TaskHub::addTask(t);
|
||||
}
|
||||
});
|
||||
|
||||
// Always use the sourceDir: If we are triggered because the build directory is getting deleted
|
||||
// then we are racing against CMakeCache.txt also getting deleted.
|
||||
|
||||
m_cmakeProcess = new Utils::QtcProcess;
|
||||
m_cmakeProcess->setWorkingDirectory(buildDirectory.toString());
|
||||
m_cmakeProcess->setEnvironment(m_parameters.environment);
|
||||
|
||||
connect(m_cmakeProcess, &QProcess::readyReadStandardOutput,
|
||||
this, &TeaLeafReader::processCMakeOutput);
|
||||
connect(m_cmakeProcess, &QProcess::readyReadStandardError,
|
||||
this, &TeaLeafReader::processCMakeError);
|
||||
connect(m_cmakeProcess, static_cast<void(QProcess::*)(int, QProcess::ExitStatus)>(&QProcess::finished),
|
||||
this, &TeaLeafReader::cmakeFinished);
|
||||
|
||||
QString args;
|
||||
Utils::QtcProcess::addArg(&args, srcDir);
|
||||
Utils::QtcProcess::addArgs(&args, m_parameters.generatorArguments);
|
||||
Utils::QtcProcess::addArgs(&args, configurationArguments);
|
||||
|
||||
TaskHub::clearTasks(ProjectExplorer::Constants::TASK_CATEGORY_BUILDSYSTEM);
|
||||
|
||||
Core::MessageManager::write(tr("Running \"%1 %2\" in %3.")
|
||||
.arg(m_parameters.cmakeExecutable.toUserOutput())
|
||||
.arg(args)
|
||||
.arg(buildDirectory.toUserOutput()));
|
||||
|
||||
m_future = new QFutureInterface<void>();
|
||||
m_future->setProgressRange(0, 1);
|
||||
Core::ProgressManager::addTask(m_future->future(),
|
||||
tr("Configuring \"%1\"").arg(m_parameters.projectName),
|
||||
"CMake.Configure");
|
||||
|
||||
m_cmakeProcess->setCommand(m_parameters.cmakeExecutable.toString(), args);
|
||||
m_cmakeProcess->start();
|
||||
emit configurationStarted();
|
||||
}
|
||||
|
||||
void TeaLeafReader::cmakeFinished(int code, QProcess::ExitStatus status)
|
||||
{
|
||||
QTC_ASSERT(m_cmakeProcess, return);
|
||||
|
||||
// process rest of the output:
|
||||
processCMakeOutput();
|
||||
processCMakeError();
|
||||
|
||||
m_cmakeProcess->disconnect();
|
||||
cleanUpProcess();
|
||||
|
||||
extractData(); // try even if cmake failed...
|
||||
|
||||
QString msg;
|
||||
if (status != QProcess::NormalExit)
|
||||
msg = tr("*** cmake process crashed!");
|
||||
else if (code != 0)
|
||||
msg = tr("*** cmake process exited with exit code %1.").arg(code);
|
||||
|
||||
if (!msg.isEmpty()) {
|
||||
Core::MessageManager::write(msg);
|
||||
TaskHub::addTask(Task::Error, msg, ProjectExplorer::Constants::TASK_CATEGORY_BUILDSYSTEM);
|
||||
m_future->reportCanceled();
|
||||
} else {
|
||||
m_future->setProgressValue(1);
|
||||
}
|
||||
|
||||
m_future->reportFinished();
|
||||
delete m_future;
|
||||
m_future = nullptr;
|
||||
|
||||
m_hasData = true;
|
||||
emit dataAvailable();
|
||||
}
|
||||
|
||||
void TeaLeafReader::processCMakeOutput()
|
||||
{
|
||||
static QString rest;
|
||||
rest = lineSplit(rest, m_cmakeProcess->readAllStandardOutput(),
|
||||
[this](const QString &s) { Core::MessageManager::write(s); });
|
||||
}
|
||||
|
||||
void TeaLeafReader::processCMakeError()
|
||||
{
|
||||
static QString rest;
|
||||
rest = lineSplit(rest, m_cmakeProcess->readAllStandardError(), [this](const QString &s) {
|
||||
m_parser->stdError(s);
|
||||
Core::MessageManager::write(s);
|
||||
});
|
||||
}
|
||||
|
||||
QStringList TeaLeafReader::getCXXFlagsFor(const CMakeBuildTarget &buildTarget, QHash<QString, QStringList> &cache)
|
||||
{
|
||||
// check cache:
|
||||
auto it = cache.constFind(buildTarget.title);
|
||||
if (it != cache.constEnd())
|
||||
return *it;
|
||||
|
||||
if (extractCXXFlagsFromMake(buildTarget, cache))
|
||||
return cache.value(buildTarget.title);
|
||||
|
||||
if (extractCXXFlagsFromNinja(buildTarget, cache))
|
||||
return cache.value(buildTarget.title);
|
||||
|
||||
cache.insert(buildTarget.title, QStringList());
|
||||
return QStringList();
|
||||
}
|
||||
|
||||
bool TeaLeafReader::extractCXXFlagsFromMake(const CMakeBuildTarget &buildTarget, QHash<QString, QStringList> &cache)
|
||||
{
|
||||
QString makeCommand = buildTarget.makeCommand.toString();
|
||||
int startIndex = makeCommand.indexOf('\"');
|
||||
int endIndex = makeCommand.indexOf('\"', startIndex + 1);
|
||||
if (startIndex != -1 && endIndex != -1) {
|
||||
startIndex += 1;
|
||||
QString makefile = makeCommand.mid(startIndex, endIndex - startIndex);
|
||||
int slashIndex = makefile.lastIndexOf('/');
|
||||
makefile.truncate(slashIndex);
|
||||
makefile.append("/CMakeFiles/" + buildTarget.title + ".dir/flags.make");
|
||||
QFile file(makefile);
|
||||
if (file.exists()) {
|
||||
file.open(QIODevice::ReadOnly | QIODevice::Text);
|
||||
QTextStream stream(&file);
|
||||
while (!stream.atEnd()) {
|
||||
QString line = stream.readLine().trimmed();
|
||||
if (line.startsWith("CXX_FLAGS =")) {
|
||||
// Skip past =
|
||||
cache.insert(buildTarget.title,
|
||||
line.mid(11).trimmed().split(' ', QString::SkipEmptyParts));
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
bool TeaLeafReader::extractCXXFlagsFromNinja(const CMakeBuildTarget &buildTarget, QHash<QString, QStringList> &cache)
|
||||
{
|
||||
Q_UNUSED(buildTarget)
|
||||
if (!cache.isEmpty()) // We fill the cache in one go!
|
||||
return false;
|
||||
|
||||
// Attempt to find build.ninja file and obtain FLAGS (CXX_FLAGS) from there if no suitable flags.make were
|
||||
// found
|
||||
// Get "all" target's working directory
|
||||
QByteArray ninjaFile;
|
||||
QString buildNinjaFile = buildTargets().at(0).workingDirectory.toString();
|
||||
buildNinjaFile += "/build.ninja";
|
||||
QFile buildNinja(buildNinjaFile);
|
||||
if (buildNinja.exists()) {
|
||||
buildNinja.open(QIODevice::ReadOnly | QIODevice::Text);
|
||||
ninjaFile = buildNinja.readAll();
|
||||
buildNinja.close();
|
||||
}
|
||||
|
||||
if (ninjaFile.isEmpty())
|
||||
return false;
|
||||
|
||||
QTextStream stream(ninjaFile);
|
||||
bool cxxFound = false;
|
||||
const QString targetSignature = "# Object build statements for ";
|
||||
QString currentTarget;
|
||||
|
||||
while (!stream.atEnd()) {
|
||||
// 1. Look for a block that refers to the current target
|
||||
// 2. Look for a build rule which invokes CXX_COMPILER
|
||||
// 3. Return the FLAGS definition
|
||||
QString line = stream.readLine().trimmed();
|
||||
if (line.startsWith('#')) {
|
||||
if (line.startsWith(targetSignature)) {
|
||||
int pos = line.lastIndexOf(' ');
|
||||
currentTarget = line.mid(pos + 1);
|
||||
}
|
||||
} else if (!currentTarget.isEmpty() && line.startsWith("build")) {
|
||||
cxxFound = line.indexOf("CXX_COMPILER") != -1;
|
||||
} else if (cxxFound && line.startsWith("FLAGS =")) {
|
||||
// Skip past =
|
||||
cache.insert(currentTarget, line.mid(7).trimmed().split(' ', QString::SkipEmptyParts));
|
||||
}
|
||||
}
|
||||
return !cache.isEmpty();
|
||||
}
|
||||
|
||||
} // namespace Internal
|
||||
} // namespace CMakeProjectManager
|
||||
Reference in New Issue
Block a user