2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2010-12-03 13:49:35 +01:00
|
|
|
**
|
2014-01-07 13:27:11 +01:00
|
|
|
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
|
2012-10-02 09:12:39 +02:00
|
|
|
** Contact: http://www.qt-project.org/legal
|
2010-12-03 13:49:35 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2010-12-03 13:49:35 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** 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 Digia. For licensing terms and
|
|
|
|
|
** conditions see http://qt.digia.com/licensing. For further information
|
|
|
|
|
** use the contact form at http://qt.digia.com/contact-us.
|
2010-12-03 13:49:35 +01:00
|
|
|
**
|
|
|
|
|
** GNU Lesser General Public License Usage
|
2012-10-02 09:12:39 +02:00
|
|
|
** 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, Digia gives you certain additional
|
|
|
|
|
** rights. These rights are described in the Digia Qt LGPL Exception
|
2010-12-03 13:49:35 +01:00
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
2010-12-03 13:49:35 +01:00
|
|
|
|
2013-04-02 11:52:31 +02:00
|
|
|
#include "cppmodelmanagerinterface.h"
|
2013-03-27 18:54:03 +01:00
|
|
|
|
|
|
|
|
#include <cplusplus/pp-engine.h>
|
2010-12-03 13:49:35 +01:00
|
|
|
|
2013-07-11 11:13:07 +02:00
|
|
|
#include <projectexplorer/headerpath.h>
|
|
|
|
|
#include <projectexplorer/toolchain.h>
|
|
|
|
|
|
|
|
|
|
#include <QSet>
|
|
|
|
|
|
|
|
|
|
using namespace CppTools;
|
|
|
|
|
using namespace ProjectExplorer;
|
2012-02-16 15:09:56 +01:00
|
|
|
|
2013-06-21 08:42:27 +02:00
|
|
|
/*!
|
|
|
|
|
\enum CppTools::CppModelManagerInterface::ProgressNotificationMode
|
|
|
|
|
|
|
|
|
|
This enum type specifies whether a progress bar notification should be
|
|
|
|
|
shown if more than one file is requested to update via
|
|
|
|
|
CppModelManagerInterface::updateSourceFiles().
|
|
|
|
|
|
|
|
|
|
\value ForcedProgressNotification
|
|
|
|
|
Notify regardless of the number of files requested for update.
|
|
|
|
|
\value ReservedProgressNotification
|
|
|
|
|
Notify only if more than one file is requested for update.
|
|
|
|
|
*/
|
|
|
|
|
|
2012-12-08 23:38:26 +04:00
|
|
|
/*!
|
2013-04-02 11:28:11 +02:00
|
|
|
\enum CppTools::CppModelManagerInterface::QtVersion
|
2013-07-11 11:13:07 +02:00
|
|
|
|
2012-12-08 23:38:26 +04:00
|
|
|
Allows C++ parser engine to inject headers or change inner settings as
|
|
|
|
|
needed to parse Qt language extensions for concrete major Qt version
|
2013-07-11 11:13:07 +02:00
|
|
|
|
2012-12-08 23:38:26 +04:00
|
|
|
\value UnknownQt
|
|
|
|
|
Parser may choose any policy
|
|
|
|
|
\value NoQt
|
|
|
|
|
Parser must not use special tricks, because it parses non-qt project
|
|
|
|
|
\value Qt4
|
|
|
|
|
Parser may enable tricks for Qt v4.x
|
|
|
|
|
\value Qt5
|
|
|
|
|
Parser may enable tricks for Qt v5.x
|
|
|
|
|
*/
|
|
|
|
|
|
2013-07-10 14:46:08 +02:00
|
|
|
/*!
|
|
|
|
|
\fn virtual QFuture<void> updateProjectInfo(const ProjectInfo &pinfo) = 0;
|
|
|
|
|
\param pinfo Updated ProjectInfo.
|
|
|
|
|
\return A future that reports progress and allows to cancel the reparsing operation.
|
|
|
|
|
|
|
|
|
|
This function is expected to be called by the project managers to update the
|
|
|
|
|
code model with new project information.
|
|
|
|
|
|
|
|
|
|
In particular, the function should be called in case:
|
|
|
|
|
1. A new project is opened/created
|
|
|
|
|
2. The project configuration changed. This includes
|
|
|
|
|
2.1 Changes of defines, includes, framework paths
|
|
|
|
|
2.2 Addition/Removal of project files
|
|
|
|
|
|
|
|
|
|
\sa CppTools::CppModelManagerInterface::updateSourceFiles()
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
|
\fn virtual QFuture<void> updateSourceFiles(const QStringList &sourceFiles, ProgressNotificationMode mode = ReservedProgressNotification) = 0;
|
|
|
|
|
\param sourceFiles List of source file to update. The items are absolute paths.
|
|
|
|
|
\param mode The progress modification mode.
|
|
|
|
|
\return A future that reports progress and allows to cancel the reparsing operation.
|
|
|
|
|
|
|
|
|
|
Trigger an asynchronous reparsing of the given source files.
|
|
|
|
|
|
|
|
|
|
This function is not meant to be called by the project managers.
|
|
|
|
|
|
|
|
|
|
\sa CppTools::CppModelManagerInterface::ProgressNotificationMode
|
|
|
|
|
\sa CppTools::CppModelManagerInterface::updateProjectInfo()
|
|
|
|
|
*/
|
|
|
|
|
|
2013-05-08 18:27:03 +04:00
|
|
|
ProjectPart::ProjectPart()
|
2013-09-30 15:59:49 +02:00
|
|
|
: project(0)
|
|
|
|
|
, cVersion(C89)
|
2013-05-08 18:27:03 +04:00
|
|
|
, cxxVersion(CXX11)
|
|
|
|
|
, cxxExtensions(NoExtensions)
|
|
|
|
|
, qtVersion(UnknownQt)
|
|
|
|
|
, cWarningFlags(ProjectExplorer::ToolChain::WarningsDefault)
|
|
|
|
|
, cxxWarningFlags(ProjectExplorer::ToolChain::WarningsDefault)
|
2013-09-30 15:59:49 +02:00
|
|
|
|
2013-05-08 18:27:03 +04:00
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2013-07-11 11:13:07 +02:00
|
|
|
/*!
|
|
|
|
|
\brief Retrieves info from concrete compiler using it's flags.
|
|
|
|
|
|
|
|
|
|
\param tc Either nullptr or toolchain for project's active target.
|
|
|
|
|
\param cxxflags C++ or Objective-C++ flags.
|
|
|
|
|
\param cflags C or ObjectiveC flags if possible, \a cxxflags otherwise.
|
|
|
|
|
*/
|
2013-04-28 15:40:54 +04:00
|
|
|
void ProjectPart::evaluateToolchain(const ToolChain *tc,
|
|
|
|
|
const QStringList &cxxflags,
|
|
|
|
|
const QStringList &cflags,
|
|
|
|
|
const Utils::FileName &sysRoot)
|
|
|
|
|
{
|
|
|
|
|
if (!tc)
|
|
|
|
|
return;
|
2013-07-11 11:13:07 +02:00
|
|
|
|
2013-04-28 15:40:54 +04:00
|
|
|
ToolChain::CompilerFlags cxx = tc->compilerFlags(cxxflags);
|
|
|
|
|
ToolChain::CompilerFlags c = (cxxflags == cflags)
|
|
|
|
|
? cxx : tc->compilerFlags(cflags);
|
|
|
|
|
|
2013-05-26 18:15:24 +04:00
|
|
|
if (c & ToolChain::StandardC11)
|
2013-04-28 15:40:54 +04:00
|
|
|
cVersion = C11;
|
2013-05-26 18:15:24 +04:00
|
|
|
else if (c & ToolChain::StandardC99)
|
2013-04-28 15:40:54 +04:00
|
|
|
cVersion = C99;
|
|
|
|
|
else
|
|
|
|
|
cVersion = C89;
|
|
|
|
|
|
2013-05-26 18:15:24 +04:00
|
|
|
if (cxx & ToolChain::StandardCxx11)
|
2013-04-28 15:40:54 +04:00
|
|
|
cxxVersion = CXX11;
|
|
|
|
|
else
|
|
|
|
|
cxxVersion = CXX98;
|
|
|
|
|
|
2013-05-26 18:15:24 +04:00
|
|
|
if (cxx & ToolChain::BorlandExtensions)
|
2013-04-28 15:40:54 +04:00
|
|
|
cxxExtensions |= BorlandExtensions;
|
2013-05-26 18:15:24 +04:00
|
|
|
if (cxx & ToolChain::GnuExtensions)
|
2013-04-28 15:40:54 +04:00
|
|
|
cxxExtensions |= GnuExtensions;
|
2013-05-26 18:15:24 +04:00
|
|
|
if (cxx & ToolChain::MicrosoftExtensions)
|
2013-04-28 15:40:54 +04:00
|
|
|
cxxExtensions |= MicrosoftExtensions;
|
2013-05-26 18:15:24 +04:00
|
|
|
if (cxx & ToolChain::OpenMP)
|
2013-08-19 18:20:49 +02:00
|
|
|
cxxExtensions |= OpenMPExtensions;
|
2013-04-28 15:40:54 +04:00
|
|
|
|
2013-05-08 18:27:03 +04:00
|
|
|
cWarningFlags = tc->warningFlags(cflags);
|
|
|
|
|
cxxWarningFlags = tc->warningFlags(cxxflags);
|
|
|
|
|
|
2013-07-11 11:13:07 +02:00
|
|
|
const QList<HeaderPath> headers = tc->systemHeaderPaths(cxxflags, sysRoot);
|
2013-04-28 15:40:54 +04:00
|
|
|
foreach (const HeaderPath &header, headers)
|
|
|
|
|
if (header.kind() == HeaderPath::FrameworkHeaderPath)
|
|
|
|
|
frameworkPaths << header.path();
|
|
|
|
|
else
|
|
|
|
|
includePaths << header.path();
|
|
|
|
|
|
2013-11-27 15:17:51 +01:00
|
|
|
toolchainDefines = tc->predefinedMacros(cxxflags);
|
2013-04-28 15:40:54 +04:00
|
|
|
}
|
2010-12-03 13:49:35 +01:00
|
|
|
|
2012-11-28 09:52:19 +01:00
|
|
|
const QString CppModelManagerInterface::configurationFileName()
|
2013-04-02 11:28:11 +02:00
|
|
|
{ return CPlusPlus::Preprocessor::configurationFileName; }
|
2012-11-28 09:52:19 +01:00
|
|
|
|
2013-10-21 12:59:48 +02:00
|
|
|
const QString CppModelManagerInterface::editorConfigurationFileName()
|
|
|
|
|
{
|
|
|
|
|
return QLatin1String("<per-editor-defines>");
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-03 13:49:35 +01:00
|
|
|
CppModelManagerInterface::CppModelManagerInterface(QObject *parent)
|
2014-01-23 14:28:31 +01:00
|
|
|
: CPlusPlus::CppModelManagerBase(parent)
|
|
|
|
|
{ }
|
2010-12-03 13:49:35 +01:00
|
|
|
|
|
|
|
|
CppModelManagerInterface::~CppModelManagerInterface()
|
2014-01-23 14:28:31 +01:00
|
|
|
{ }
|
2010-12-03 13:49:35 +01:00
|
|
|
|
|
|
|
|
CppModelManagerInterface *CppModelManagerInterface::instance()
|
|
|
|
|
{
|
2014-01-23 14:28:31 +01:00
|
|
|
return qobject_cast<CppModelManagerInterface *>(CPlusPlus::CppModelManagerBase::instance());
|
2010-12-03 13:49:35 +01:00
|
|
|
}
|
|
|
|
|
|
2012-02-16 15:09:56 +01:00
|
|
|
void CppModelManagerInterface::ProjectInfo::clearProjectParts()
|
|
|
|
|
{
|
|
|
|
|
m_projectParts.clear();
|
|
|
|
|
m_includePaths.clear();
|
|
|
|
|
m_frameworkPaths.clear();
|
|
|
|
|
m_sourceFiles.clear();
|
|
|
|
|
m_defines.clear();
|
|
|
|
|
}
|
|
|
|
|
|
2013-07-11 11:13:07 +02:00
|
|
|
void CppModelManagerInterface::ProjectInfo::appendProjectPart(const ProjectPart::Ptr &part)
|
2012-02-16 15:09:56 +01:00
|
|
|
{
|
|
|
|
|
if (!part)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
m_projectParts.append(part);
|
|
|
|
|
|
2013-07-11 11:13:07 +02:00
|
|
|
// Update include paths
|
2012-02-16 15:09:56 +01:00
|
|
|
QSet<QString> incs = QSet<QString>::fromList(m_includePaths);
|
|
|
|
|
foreach (const QString &ins, part->includePaths)
|
|
|
|
|
incs.insert(ins);
|
|
|
|
|
m_includePaths = incs.toList();
|
|
|
|
|
|
2013-07-11 11:13:07 +02:00
|
|
|
// Update framework paths
|
2012-02-16 15:09:56 +01:00
|
|
|
QSet<QString> frms = QSet<QString>::fromList(m_frameworkPaths);
|
|
|
|
|
foreach (const QString &frm, part->frameworkPaths)
|
|
|
|
|
frms.insert(frm);
|
|
|
|
|
m_frameworkPaths = frms.toList();
|
|
|
|
|
|
2013-07-11 11:13:07 +02:00
|
|
|
// Update source files
|
2012-02-16 15:09:56 +01:00
|
|
|
QSet<QString> srcs = QSet<QString>::fromList(m_sourceFiles);
|
2013-03-04 01:30:46 +04:00
|
|
|
foreach (const ProjectFile &file, part->files)
|
|
|
|
|
srcs.insert(file.path);
|
2012-02-16 15:09:56 +01:00
|
|
|
m_sourceFiles = srcs.toList();
|
|
|
|
|
|
2013-07-11 11:13:07 +02:00
|
|
|
// Update defines
|
2012-02-16 15:09:56 +01:00
|
|
|
if (!m_defines.isEmpty())
|
|
|
|
|
m_defines.append('\n');
|
2013-11-27 15:17:51 +01:00
|
|
|
m_defines.append(part->toolchainDefines);
|
|
|
|
|
m_defines.append(part->projectDefines);
|
2014-02-05 16:44:35 +01:00
|
|
|
if (!part->projectConfigFile.isEmpty()) {
|
|
|
|
|
m_defines.append('\n');
|
|
|
|
|
m_defines += readProjectConfigFile(part);
|
|
|
|
|
m_defines.append('\n');
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QByteArray CppModelManagerInterface::readProjectConfigFile(const ProjectPart::Ptr &part)
|
|
|
|
|
{
|
|
|
|
|
QByteArray result;
|
|
|
|
|
|
|
|
|
|
QFile f(part->projectConfigFile);
|
|
|
|
|
if (f.open(QIODevice::ReadOnly)) {
|
|
|
|
|
QTextStream is(&f);
|
|
|
|
|
result = is.readAll().toUtf8();
|
|
|
|
|
f.close();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
2012-02-16 15:09:56 +01:00
|
|
|
}
|
2014-02-05 16:44:35 +01:00
|
|
|
|