2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** Copyright (C) 2012 Digia Plc and/or its subsidiary(-ies).
|
|
|
|
** Contact: http://www.qt-project.org/legal
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2008-12-02 12:01:29 +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.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +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-17 16:01:08 +01:00
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
2008-12-02 15:08:31 +01:00
|
|
|
|
2008-12-08 12:59:33 +01:00
|
|
|
#include <cplusplus/pp.h>
|
2009-06-04 11:32:01 +02:00
|
|
|
#include <cplusplus/Overview.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2012-11-23 11:47:39 +01:00
|
|
|
#include "builtinindexingsupport.h"
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "cppmodelmanager.h"
|
2012-02-21 10:00:32 +01:00
|
|
|
#include "cppcompletionassist.h"
|
2012-02-20 12:39:08 +01:00
|
|
|
#include "cpphighlightingsupport.h"
|
|
|
|
#include "cpphighlightingsupportinternal.h"
|
2012-10-16 16:02:40 +02:00
|
|
|
#include "cppindexingsupport.h"
|
2010-12-03 13:49:35 +01:00
|
|
|
#include "abstracteditorsupport.h"
|
2012-09-07 10:51:50 +02:00
|
|
|
#include "cpptoolsconstants.h"
|
|
|
|
#include "cpptoolseditorsupport.h"
|
|
|
|
#include "cppfindreferences.h"
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-03-06 11:51:45 +01:00
|
|
|
#include <functional>
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QtConcurrentRun>
|
2012-09-07 10:51:50 +02:00
|
|
|
#include <QFutureSynchronizer>
|
|
|
|
#include <utils/runextensions.h>
|
|
|
|
#include <texteditor/itexteditor.h>
|
|
|
|
#include <texteditor/basetexteditor.h>
|
|
|
|
#include <projectexplorer/project.h>
|
|
|
|
#include <projectexplorer/projectexplorer.h>
|
|
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
|
|
|
#include <projectexplorer/session.h>
|
|
|
|
#include <coreplugin/icore.h>
|
|
|
|
#include <coreplugin/mimedatabase.h>
|
|
|
|
#include <coreplugin/editormanager/editormanager.h>
|
|
|
|
#include <coreplugin/progressmanager/progressmanager.h>
|
|
|
|
#include <extensionsystem/pluginmanager.h>
|
2009-10-05 13:43:05 +02:00
|
|
|
|
2012-08-23 15:53:58 +02:00
|
|
|
#include <utils/hostosinfo.h>
|
2008-12-09 15:25:01 +01:00
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <TranslationUnit.h>
|
|
|
|
#include <AST.h>
|
|
|
|
#include <Scope.h>
|
|
|
|
#include <Literals.h>
|
|
|
|
#include <Symbols.h>
|
|
|
|
#include <Names.h>
|
|
|
|
#include <NameVisitor.h>
|
|
|
|
#include <TypeVisitor.h>
|
2009-02-18 16:01:28 +01:00
|
|
|
#include <ASTVisitor.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <Lexer.h>
|
|
|
|
#include <Token.h>
|
2010-03-18 15:21:07 +01:00
|
|
|
#include <Parser.h>
|
|
|
|
#include <Control.h>
|
2010-12-03 10:13:15 +01:00
|
|
|
#include <CoreTypes.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QCoreApplication>
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QMutexLocker>
|
|
|
|
#include <QTime>
|
|
|
|
#include <QTimer>
|
2009-02-24 11:04:52 +01:00
|
|
|
#include <QtConcurrentMap>
|
2010-04-26 14:02:09 +02:00
|
|
|
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QTextBlock>
|
2010-04-26 14:02:09 +02:00
|
|
|
|
2009-02-18 16:01:28 +01:00
|
|
|
#include <iostream>
|
|
|
|
#include <sstream>
|
2008-12-09 15:25:01 +01:00
|
|
|
|
2012-02-16 15:09:56 +01:00
|
|
|
namespace CPlusPlus {
|
2012-11-23 16:29:00 +01:00
|
|
|
|
2012-02-16 15:09:56 +01:00
|
|
|
uint qHash(const CppModelManagerInterface::ProjectPart &p)
|
|
|
|
{
|
2012-11-23 16:29:00 +01:00
|
|
|
uint h = qHash(p.defines) ^ p.language ^ p.qtVersion;
|
2012-02-16 15:09:56 +01:00
|
|
|
|
|
|
|
foreach (const QString &i, p.includePaths)
|
|
|
|
h ^= qHash(i);
|
|
|
|
|
|
|
|
foreach (const QString &f, p.frameworkPaths)
|
|
|
|
h ^= qHash(f);
|
|
|
|
|
|
|
|
return h;
|
|
|
|
}
|
2012-11-23 16:29:00 +01:00
|
|
|
|
2012-02-16 15:09:56 +01:00
|
|
|
bool operator==(const CppModelManagerInterface::ProjectPart &p1,
|
|
|
|
const CppModelManagerInterface::ProjectPart &p2)
|
|
|
|
{
|
|
|
|
if (p1.defines != p2.defines)
|
|
|
|
return false;
|
|
|
|
if (p1.language != p2.language)
|
|
|
|
return false;
|
2012-11-23 16:29:00 +01:00
|
|
|
if (p1.qtVersion!= p2.qtVersion)
|
2012-02-16 15:09:56 +01:00
|
|
|
return false;
|
|
|
|
if (p1.includePaths != p2.includePaths)
|
|
|
|
return false;
|
|
|
|
return p1.frameworkPaths == p2.frameworkPaths;
|
|
|
|
}
|
2012-11-23 16:29:00 +01:00
|
|
|
|
2012-02-16 15:09:56 +01:00
|
|
|
} // namespace CPlusPlus
|
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
using namespace CppTools;
|
|
|
|
using namespace CppTools::Internal;
|
2008-12-02 12:01:29 +01:00
|
|
|
using namespace CPlusPlus;
|
|
|
|
|
2009-02-18 16:01:28 +01:00
|
|
|
#if defined(QTCREATOR_WITH_DUMP_AST) && defined(Q_CC_GNU)
|
|
|
|
|
|
|
|
#include <cxxabi.h>
|
|
|
|
|
|
|
|
class DumpAST: protected ASTVisitor
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
int depth;
|
|
|
|
|
|
|
|
DumpAST(Control *control)
|
|
|
|
: ASTVisitor(control), depth(0)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
void operator()(AST *ast)
|
|
|
|
{ accept(ast); }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual bool preVisit(AST *ast)
|
|
|
|
{
|
|
|
|
std::ostringstream s;
|
|
|
|
PrettyPrinter pp(control(), s);
|
|
|
|
pp(ast);
|
|
|
|
QString code = QString::fromStdString(s.str());
|
|
|
|
code.replace('\n', ' ');
|
|
|
|
code.replace(QRegExp("\\s+"), " ");
|
|
|
|
|
|
|
|
const char *name = abi::__cxa_demangle(typeid(*ast).name(), 0, 0, 0) + 11;
|
|
|
|
|
|
|
|
QByteArray ind(depth, ' ');
|
|
|
|
ind += name;
|
|
|
|
|
|
|
|
printf("%-40s %s\n", ind.constData(), qPrintable(code));
|
|
|
|
++depth;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void postVisit(AST *)
|
|
|
|
{ --depth; }
|
|
|
|
};
|
|
|
|
|
2009-03-05 09:46:54 +01:00
|
|
|
#endif // QTCREATOR_WITH_DUMP_AST
|
2009-02-18 16:01:28 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
static const char pp_configuration[] =
|
|
|
|
"# 1 \"<configuration>\"\n"
|
|
|
|
"#define __cplusplus 1\n"
|
|
|
|
"#define __extension__\n"
|
|
|
|
"#define __context__\n"
|
|
|
|
"#define __range__\n"
|
|
|
|
"#define restrict\n"
|
|
|
|
"#define __restrict\n"
|
2009-07-17 10:54:08 +02:00
|
|
|
"#define __restrict__\n"
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-02-23 09:53:26 +01:00
|
|
|
"#define __complex__\n"
|
|
|
|
"#define __imag__\n"
|
|
|
|
"#define __real__\n"
|
|
|
|
|
2009-06-18 17:48:55 +02:00
|
|
|
"#define __builtin_va_arg(a,b) ((b)0)\n"
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
// ### add macros for win32
|
|
|
|
"#define __cdecl\n"
|
2009-12-10 16:20:34 +01:00
|
|
|
"#define __stdcall\n"
|
2008-12-02 12:01:29 +01:00
|
|
|
"#define QT_WA(x) x\n"
|
|
|
|
"#define CALLBACK\n"
|
|
|
|
"#define STDMETHODCALLTYPE\n"
|
|
|
|
"#define __RPC_FAR\n"
|
|
|
|
"#define __declspec(a)\n"
|
2011-09-06 16:45:47 +02:00
|
|
|
"#define STDMETHOD(method) virtual HRESULT STDMETHODCALLTYPE method\n"
|
|
|
|
"#define __try try\n"
|
|
|
|
"#define __except catch\n"
|
2011-09-08 11:30:48 +02:00
|
|
|
"#define __finally\n"
|
|
|
|
"#define __inline inline\n"
|
|
|
|
"#define __forceinline inline\n";
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2012-03-30 10:06:02 +02:00
|
|
|
CppPreprocessor::CppPreprocessor(QPointer<CppModelManager> modelManager, bool dumpFileNameWhileParsing)
|
2009-02-09 12:33:43 +01:00
|
|
|
: snapshot(modelManager->snapshot()),
|
|
|
|
m_modelManager(modelManager),
|
2012-03-30 10:06:02 +02:00
|
|
|
m_dumpFileNameWhileParsing(dumpFileNameWhileParsing),
|
2009-07-14 14:23:12 +02:00
|
|
|
preprocess(this, &env),
|
|
|
|
m_revision(0)
|
2011-08-23 14:05:47 +02:00
|
|
|
{
|
|
|
|
preprocess.setKeepComments(true);
|
|
|
|
}
|
2009-03-11 12:00:07 +01:00
|
|
|
|
|
|
|
CppPreprocessor::~CppPreprocessor()
|
2008-12-08 11:08:48 +01:00
|
|
|
{ }
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-07-14 14:23:12 +02:00
|
|
|
void CppPreprocessor::setRevision(unsigned revision)
|
|
|
|
{ m_revision = revision; }
|
|
|
|
|
2010-12-03 13:49:35 +01:00
|
|
|
void CppPreprocessor::setWorkingCopy(const CppModelManagerInterface::WorkingCopy &workingCopy)
|
2008-12-08 11:08:48 +01:00
|
|
|
{ m_workingCopy = workingCopy; }
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
void CppPreprocessor::setIncludePaths(const QStringList &includePaths)
|
2010-03-15 18:09:29 +01:00
|
|
|
{
|
|
|
|
m_includePaths.clear();
|
|
|
|
|
|
|
|
for (int i = 0; i < includePaths.size(); ++i) {
|
2010-03-17 14:49:42 +01:00
|
|
|
const QString &path = includePaths.at(i);
|
2010-03-15 18:09:29 +01:00
|
|
|
|
2012-08-23 15:53:58 +02:00
|
|
|
if (Utils::HostOsInfo::isMacHost()) {
|
|
|
|
if (i + 1 < includePaths.size() && path.endsWith(QLatin1String(".framework/Headers"))) {
|
|
|
|
const QFileInfo pathInfo(path);
|
|
|
|
const QFileInfo frameworkFileInfo(pathInfo.path());
|
|
|
|
const QString frameworkName = frameworkFileInfo.baseName();
|
|
|
|
|
|
|
|
const QFileInfo nextIncludePath = includePaths.at(i + 1);
|
|
|
|
if (nextIncludePath.fileName() == frameworkName) {
|
|
|
|
// We got a QtXXX.framework/Headers followed by $QTDIR/include/QtXXX.
|
|
|
|
// In this case we prefer to include files from $QTDIR/include/QtXXX.
|
|
|
|
continue;
|
|
|
|
}
|
2010-03-15 18:09:29 +01:00
|
|
|
}
|
2012-08-23 15:53:58 +02:00
|
|
|
m_includePaths.append(path);
|
|
|
|
} else {
|
|
|
|
m_includePaths.append(path);
|
2010-03-15 18:09:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-12-08 11:08:48 +01:00
|
|
|
|
|
|
|
void CppPreprocessor::setFrameworkPaths(const QStringList &frameworkPaths)
|
2010-04-13 11:14:34 +02:00
|
|
|
{
|
|
|
|
m_frameworkPaths.clear();
|
|
|
|
|
|
|
|
foreach (const QString &frameworkPath, frameworkPaths) {
|
|
|
|
addFrameworkPath(frameworkPath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the given framework path, and expand private frameworks.
|
|
|
|
//
|
|
|
|
// Example:
|
|
|
|
// <framework-path>/ApplicationServices.framework
|
|
|
|
// has private frameworks in:
|
|
|
|
// <framework-path>/ApplicationServices.framework/Frameworks
|
|
|
|
// if the "Frameworks" folder exists inside the top level framework.
|
|
|
|
void CppPreprocessor::addFrameworkPath(const QString &frameworkPath)
|
|
|
|
{
|
|
|
|
// The algorithm below is a bit too eager, but that's because we're not getting
|
|
|
|
// in the frameworks we're linking against. If we would have that, then we could
|
|
|
|
// add only those private frameworks.
|
|
|
|
if (!m_frameworkPaths.contains(frameworkPath)) {
|
|
|
|
m_frameworkPaths.append(frameworkPath);
|
|
|
|
}
|
|
|
|
|
|
|
|
const QDir frameworkDir(frameworkPath);
|
|
|
|
const QStringList filter = QStringList() << QLatin1String("*.framework");
|
|
|
|
foreach (const QFileInfo &framework, frameworkDir.entryInfoList(filter)) {
|
|
|
|
if (!framework.isDir())
|
|
|
|
continue;
|
|
|
|
const QFileInfo privateFrameworks(framework.absoluteFilePath(), QLatin1String("Frameworks"));
|
|
|
|
if (privateFrameworks.exists() && privateFrameworks.isDir()) {
|
|
|
|
addFrameworkPath(privateFrameworks.absoluteFilePath());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
void CppPreprocessor::setProjectFiles(const QStringList &files)
|
|
|
|
{ m_projectFiles = files; }
|
|
|
|
|
2009-02-23 15:57:37 +01:00
|
|
|
void CppPreprocessor::setTodo(const QStringList &files)
|
|
|
|
{ m_todo = QSet<QString>::fromList(files); }
|
|
|
|
|
2010-02-03 13:21:08 +10:00
|
|
|
namespace {
|
2009-03-06 11:51:45 +01:00
|
|
|
class Process: public std::unary_function<Document::Ptr, void>
|
2009-02-24 11:04:52 +01:00
|
|
|
{
|
|
|
|
QPointer<CppModelManager> _modelManager;
|
2009-06-02 16:50:43 +02:00
|
|
|
Snapshot _snapshot;
|
|
|
|
Document::Ptr _doc;
|
2010-05-10 10:19:37 +02:00
|
|
|
Document::CheckMode _mode;
|
2009-02-24 11:04:52 +01:00
|
|
|
|
|
|
|
public:
|
2009-06-02 16:50:43 +02:00
|
|
|
Process(QPointer<CppModelManager> modelManager,
|
2010-05-10 10:19:37 +02:00
|
|
|
Document::Ptr doc,
|
|
|
|
const Snapshot &snapshot,
|
2009-12-15 15:26:40 +01:00
|
|
|
const CppModelManager::WorkingCopy &workingCopy)
|
2009-06-02 16:50:43 +02:00
|
|
|
: _modelManager(modelManager),
|
|
|
|
_snapshot(snapshot),
|
2010-05-10 10:19:37 +02:00
|
|
|
_doc(doc),
|
|
|
|
_mode(Document::FastCheck)
|
2009-06-02 16:50:43 +02:00
|
|
|
{
|
2009-06-26 09:11:14 +02:00
|
|
|
|
2010-05-10 10:19:37 +02:00
|
|
|
if (workingCopy.contains(_doc->fileName()))
|
|
|
|
_mode = Document::FullCheck;
|
|
|
|
}
|
2009-06-26 09:11:14 +02:00
|
|
|
|
2010-05-10 10:19:37 +02:00
|
|
|
void operator()()
|
|
|
|
{
|
|
|
|
_doc->check(_mode);
|
2010-09-02 12:50:37 +02:00
|
|
|
|
2009-02-24 11:04:52 +01:00
|
|
|
if (_modelManager)
|
2012-10-15 16:38:56 +02:00
|
|
|
_modelManager->emitDocumentUpdated(_doc);
|
2011-08-23 12:02:29 +02:00
|
|
|
|
|
|
|
_doc->releaseSourceAndAST();
|
2009-02-24 11:04:52 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
} // end of anonymous namespace
|
|
|
|
|
|
|
|
void CppPreprocessor::run(const QString &fileName)
|
|
|
|
{
|
|
|
|
QString absoluteFilePath = fileName;
|
C++: Core changes in preprocessing
Summary of most relevant items:
- Preprocessor output format change. No more gen true/false. Instead
a more intuitive and natural expansion (like from a real compiler) is
performed directly corresponding to the macro invocation. Notice that
information about the generated tokens is not lost, because it's now
embedded in the expansion section header (in terms of lines and columns
as explained in the code). In addition the location on where the macro
expansion happens is also documented for future use.
- Fix line control directives and associated token line numbers.
This was not detected in tests cases because some of them were
actually wrong: Within expansions the line information was being
considered as originally computed in the macro definition, while
the desired and expected for Creator's reporting mechanism (just
like regular compilers) is the line from the expanded version
of the tokens.
- Do not allow for eager expansion. This was previously being done
inside define directives. However, it's not allowed and might
lead to incorrect results, since the argument substitution should
only happen upon the macro invocation (and following nested ones).
At least GCC and clang are consistent with that. See test case
tst_Preprocessor:dont_eagerly_expand for a detailed explanation.
- Revive the 'expanded' token flag. This is used to mark every token
that originates from a macro expansion. Notice, however, that
expanded tokens are not necessarily generated tokens (although
every generated token is a expanded token). Expanded tokens that
are not generated are those which are still considered by our
code model features, since they are visible on the editor. The
translation unit is smart enough to calculate line/column position
for such tokens based on the information from the expansion section
header.
- How expansions are tracked has also changed. Now, we simply add
two surrounding marker tokens to each "top-level" expansion
sequence. There is an enumeration that control expansion states.
Also, no "previous" token is kept around.
- Preprocessor client methods suffered a change in signature so
they now receive the line number of the action in question as
a paramater. Previously such line could be retrieved by the client
implementation by accessing the environment line. However, this
is not reliable because we try to avoid synchronization of the
output/environment lines in order to avoid unnecessary output,
while expanding macros or handling preprocessor directives.
- Although macros are not expanded during define directives (as
mentioned above) the preprocessor client is now "notified"
when it sees a macro. This is to allow usage tracking.
- Other small stuff.
This is all in one patch because the fixes are a consequence
of the change in preprocessing control.
Change-Id: I8f4c6e6366f37756ec65d0a93b79f72a3ac4ed50
Reviewed-by: Roberto Raggi <roberto.raggi@nokia.com>
2012-06-20 15:22:02 +02:00
|
|
|
sourceNeeded(0, absoluteFilePath, IncludeGlobal);
|
2009-02-24 11:04:52 +01:00
|
|
|
}
|
2008-12-08 11:08:48 +01:00
|
|
|
|
2009-02-23 12:49:56 +01:00
|
|
|
void CppPreprocessor::resetEnvironment()
|
2009-03-09 12:51:13 +01:00
|
|
|
{
|
|
|
|
env.reset();
|
|
|
|
m_processed.clear();
|
|
|
|
}
|
2009-02-23 12:49:56 +01:00
|
|
|
|
2009-12-15 15:52:55 +01:00
|
|
|
bool CppPreprocessor::includeFile(const QString &absoluteFilePath, QString *result, unsigned *revision)
|
2008-12-08 11:08:48 +01:00
|
|
|
{
|
2009-10-26 15:31:43 +01:00
|
|
|
if (absoluteFilePath.isEmpty() || m_included.contains(absoluteFilePath))
|
2008-12-08 11:08:48 +01:00
|
|
|
return true;
|
|
|
|
|
|
|
|
if (m_workingCopy.contains(absoluteFilePath)) {
|
|
|
|
m_included.insert(absoluteFilePath);
|
2009-12-15 15:52:55 +01:00
|
|
|
const QPair<QString, unsigned> r = m_workingCopy.get(absoluteFilePath);
|
|
|
|
*result = r.first;
|
|
|
|
*revision = r.second;
|
2008-12-08 11:08:48 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
QFileInfo fileInfo(absoluteFilePath);
|
|
|
|
if (! fileInfo.isFile())
|
2008-12-02 12:01:29 +01:00
|
|
|
return false;
|
2008-12-08 11:08:48 +01:00
|
|
|
|
|
|
|
QFile file(absoluteFilePath);
|
2011-09-06 11:28:47 +02:00
|
|
|
if (file.open(QFile::ReadOnly | QFile::Text)) {
|
2008-12-08 11:08:48 +01:00
|
|
|
m_included.insert(absoluteFilePath);
|
2012-09-29 20:35:54 +02:00
|
|
|
QTextCodec *defaultCodec = Core::EditorManager::instance()->defaultTextCodec();
|
2008-12-08 11:08:48 +01:00
|
|
|
QTextStream stream(&file);
|
2012-09-29 20:35:54 +02:00
|
|
|
stream.setCodec(defaultCodec);
|
|
|
|
if (result)
|
|
|
|
*result = stream.readAll();
|
2008-12-08 11:08:48 +01:00
|
|
|
file.close();
|
|
|
|
return true;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-12-15 15:52:55 +01:00
|
|
|
QString CppPreprocessor::tryIncludeFile(QString &fileName, IncludeType type, unsigned *revision)
|
2010-05-10 09:54:30 +02:00
|
|
|
{
|
|
|
|
if (type == IncludeGlobal) {
|
|
|
|
const QString fn = m_fileNameCache.value(fileName);
|
|
|
|
|
|
|
|
if (! fn.isEmpty()) {
|
|
|
|
fileName = fn;
|
|
|
|
|
|
|
|
if (revision)
|
|
|
|
*revision = 0;
|
|
|
|
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const QString originalFileName = fileName;
|
|
|
|
const QString contents = tryIncludeFile_helper(fileName, type, revision);
|
|
|
|
if (type == IncludeGlobal)
|
|
|
|
m_fileNameCache.insert(originalFileName, fileName);
|
|
|
|
return contents;
|
|
|
|
}
|
|
|
|
|
2010-10-03 09:23:48 +02:00
|
|
|
static inline void appendDirSeparatorIfNeeded(QString &path)
|
|
|
|
{
|
|
|
|
if (!path.endsWith(QLatin1Char('/'), Qt::CaseInsensitive))
|
|
|
|
path += QLatin1Char('/');
|
|
|
|
}
|
|
|
|
|
2010-05-10 09:54:30 +02:00
|
|
|
QString CppPreprocessor::tryIncludeFile_helper(QString &fileName, IncludeType type, unsigned *revision)
|
2008-12-08 11:08:48 +01:00
|
|
|
{
|
|
|
|
QFileInfo fileInfo(fileName);
|
2012-11-28 09:52:19 +01:00
|
|
|
if (fileName == Preprocessor::configurationFileName || fileInfo.isAbsolute()) {
|
2009-06-15 15:38:20 +02:00
|
|
|
QString contents;
|
2009-12-15 15:52:55 +01:00
|
|
|
includeFile(fileName, &contents, revision);
|
2008-12-08 11:08:48 +01:00
|
|
|
return contents;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type == IncludeLocal && m_currentDoc) {
|
|
|
|
QFileInfo currentFileInfo(m_currentDoc->fileName());
|
|
|
|
QString path = currentFileInfo.absolutePath();
|
2010-10-03 09:23:48 +02:00
|
|
|
appendDirSeparatorIfNeeded(path);
|
2008-12-08 11:08:48 +01:00
|
|
|
path += fileName;
|
|
|
|
path = QDir::cleanPath(path);
|
2009-06-15 15:38:20 +02:00
|
|
|
QString contents;
|
2009-12-15 15:52:55 +01:00
|
|
|
if (includeFile(path, &contents, revision)) {
|
2008-12-08 11:08:48 +01:00
|
|
|
fileName = path;
|
2008-12-02 12:01:29 +01:00
|
|
|
return contents;
|
|
|
|
}
|
2008-12-08 11:08:48 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-11-13 15:19:15 +01:00
|
|
|
foreach (const QString &includePath, m_includePaths) {
|
|
|
|
QString path = includePath;
|
2010-10-03 09:23:48 +02:00
|
|
|
appendDirSeparatorIfNeeded(path);
|
2008-12-08 11:08:48 +01:00
|
|
|
path += fileName;
|
|
|
|
path = QDir::cleanPath(path);
|
2009-06-15 15:38:20 +02:00
|
|
|
QString contents;
|
2009-12-15 15:52:55 +01:00
|
|
|
if (includeFile(path, &contents, revision)) {
|
2008-12-08 11:08:48 +01:00
|
|
|
fileName = path;
|
|
|
|
return contents;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
2008-12-08 11:08:48 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
// look in the system include paths
|
|
|
|
foreach (const QString &includePath, m_systemIncludePaths) {
|
|
|
|
QString path = includePath;
|
2010-10-03 09:23:48 +02:00
|
|
|
appendDirSeparatorIfNeeded(path);
|
2008-12-08 11:08:48 +01:00
|
|
|
path += fileName;
|
|
|
|
path = QDir::cleanPath(path);
|
2009-06-15 15:38:20 +02:00
|
|
|
QString contents;
|
2009-12-15 15:52:55 +01:00
|
|
|
if (includeFile(path, &contents, revision)) {
|
2008-12-08 11:08:48 +01:00
|
|
|
fileName = path;
|
|
|
|
return contents;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
2008-12-08 11:08:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int index = fileName.indexOf(QLatin1Char('/'));
|
|
|
|
if (index != -1) {
|
|
|
|
QString frameworkName = fileName.left(index);
|
|
|
|
QString name = fileName.mid(index + 1);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
foreach (const QString &frameworkPath, m_frameworkPaths) {
|
|
|
|
QString path = frameworkPath;
|
2010-10-03 09:23:48 +02:00
|
|
|
appendDirSeparatorIfNeeded(path);
|
2008-12-08 11:08:48 +01:00
|
|
|
path += frameworkName;
|
|
|
|
path += QLatin1String(".framework/Headers/");
|
|
|
|
path += name;
|
2009-06-02 14:56:03 +02:00
|
|
|
path = QDir::cleanPath(path);
|
2009-06-15 15:38:20 +02:00
|
|
|
QString contents;
|
2009-12-15 15:52:55 +01:00
|
|
|
if (includeFile(path, &contents, revision)) {
|
2008-12-02 12:01:29 +01:00
|
|
|
fileName = path;
|
|
|
|
return contents;
|
|
|
|
}
|
|
|
|
}
|
2008-12-08 11:08:48 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
QString path = fileName;
|
|
|
|
if (path.at(0) != QLatin1Char('/'))
|
|
|
|
path.prepend(QLatin1Char('/'));
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
foreach (const QString &projectFile, m_projectFiles) {
|
|
|
|
if (projectFile.endsWith(path)) {
|
|
|
|
fileName = projectFile;
|
2009-06-15 15:38:20 +02:00
|
|
|
QString contents;
|
2009-12-15 15:52:55 +01:00
|
|
|
includeFile(fileName, &contents, revision);
|
2008-12-08 11:08:48 +01:00
|
|
|
return contents;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
//qDebug() << "**** file" << fileName << "not found!";
|
2009-06-15 15:38:20 +02:00
|
|
|
return QString();
|
2008-12-08 11:08:48 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 12:59:33 +01:00
|
|
|
void CppPreprocessor::macroAdded(const Macro ¯o)
|
2008-12-08 11:08:48 +01:00
|
|
|
{
|
|
|
|
if (! m_currentDoc)
|
|
|
|
return;
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 12:59:33 +01:00
|
|
|
m_currentDoc->appendMacro(macro);
|
2008-12-08 11:08:48 +01:00
|
|
|
}
|
2008-12-04 12:05:04 +01:00
|
|
|
|
2012-10-11 16:16:01 +02:00
|
|
|
static inline const Macro revision(const CppModelManagerInterface::WorkingCopy &s, const Macro ¯o)
|
|
|
|
{
|
|
|
|
Macro newMacro(macro);
|
|
|
|
newMacro.setFileRevision(s.get(macro.fileName()).second);
|
|
|
|
return newMacro;
|
|
|
|
}
|
|
|
|
|
C++: Core changes in preprocessing
Summary of most relevant items:
- Preprocessor output format change. No more gen true/false. Instead
a more intuitive and natural expansion (like from a real compiler) is
performed directly corresponding to the macro invocation. Notice that
information about the generated tokens is not lost, because it's now
embedded in the expansion section header (in terms of lines and columns
as explained in the code). In addition the location on where the macro
expansion happens is also documented for future use.
- Fix line control directives and associated token line numbers.
This was not detected in tests cases because some of them were
actually wrong: Within expansions the line information was being
considered as originally computed in the macro definition, while
the desired and expected for Creator's reporting mechanism (just
like regular compilers) is the line from the expanded version
of the tokens.
- Do not allow for eager expansion. This was previously being done
inside define directives. However, it's not allowed and might
lead to incorrect results, since the argument substitution should
only happen upon the macro invocation (and following nested ones).
At least GCC and clang are consistent with that. See test case
tst_Preprocessor:dont_eagerly_expand for a detailed explanation.
- Revive the 'expanded' token flag. This is used to mark every token
that originates from a macro expansion. Notice, however, that
expanded tokens are not necessarily generated tokens (although
every generated token is a expanded token). Expanded tokens that
are not generated are those which are still considered by our
code model features, since they are visible on the editor. The
translation unit is smart enough to calculate line/column position
for such tokens based on the information from the expansion section
header.
- How expansions are tracked has also changed. Now, we simply add
two surrounding marker tokens to each "top-level" expansion
sequence. There is an enumeration that control expansion states.
Also, no "previous" token is kept around.
- Preprocessor client methods suffered a change in signature so
they now receive the line number of the action in question as
a paramater. Previously such line could be retrieved by the client
implementation by accessing the environment line. However, this
is not reliable because we try to avoid synchronization of the
output/environment lines in order to avoid unnecessary output,
while expanding macros or handling preprocessor directives.
- Although macros are not expanded during define directives (as
mentioned above) the preprocessor client is now "notified"
when it sees a macro. This is to allow usage tracking.
- Other small stuff.
This is all in one patch because the fixes are a consequence
of the change in preprocessing control.
Change-Id: I8f4c6e6366f37756ec65d0a93b79f72a3ac4ed50
Reviewed-by: Roberto Raggi <roberto.raggi@nokia.com>
2012-06-20 15:22:02 +02:00
|
|
|
void CppPreprocessor::passedMacroDefinitionCheck(unsigned offset, unsigned line, const Macro ¯o)
|
2009-09-25 16:00:14 +02:00
|
|
|
{
|
|
|
|
if (! m_currentDoc)
|
|
|
|
return;
|
|
|
|
|
2012-10-11 16:16:01 +02:00
|
|
|
m_currentDoc->addMacroUse(revision(m_workingCopy, macro), offset, macro.name().length(), line,
|
2012-03-26 15:18:01 +02:00
|
|
|
QVector<MacroArgumentReference>());
|
2009-09-25 16:00:14 +02:00
|
|
|
}
|
|
|
|
|
2012-04-18 17:35:28 +02:00
|
|
|
void CppPreprocessor::failedMacroDefinitionCheck(unsigned offset, const ByteArrayRef &name)
|
2009-09-25 16:00:14 +02:00
|
|
|
{
|
|
|
|
if (! m_currentDoc)
|
|
|
|
return;
|
|
|
|
|
2012-04-18 17:28:33 +02:00
|
|
|
m_currentDoc->addUndefinedMacroUse(QByteArray(name.start(), name.size()), offset);
|
2009-09-25 16:00:14 +02:00
|
|
|
}
|
|
|
|
|
C++: Core changes in preprocessing
Summary of most relevant items:
- Preprocessor output format change. No more gen true/false. Instead
a more intuitive and natural expansion (like from a real compiler) is
performed directly corresponding to the macro invocation. Notice that
information about the generated tokens is not lost, because it's now
embedded in the expansion section header (in terms of lines and columns
as explained in the code). In addition the location on where the macro
expansion happens is also documented for future use.
- Fix line control directives and associated token line numbers.
This was not detected in tests cases because some of them were
actually wrong: Within expansions the line information was being
considered as originally computed in the macro definition, while
the desired and expected for Creator's reporting mechanism (just
like regular compilers) is the line from the expanded version
of the tokens.
- Do not allow for eager expansion. This was previously being done
inside define directives. However, it's not allowed and might
lead to incorrect results, since the argument substitution should
only happen upon the macro invocation (and following nested ones).
At least GCC and clang are consistent with that. See test case
tst_Preprocessor:dont_eagerly_expand for a detailed explanation.
- Revive the 'expanded' token flag. This is used to mark every token
that originates from a macro expansion. Notice, however, that
expanded tokens are not necessarily generated tokens (although
every generated token is a expanded token). Expanded tokens that
are not generated are those which are still considered by our
code model features, since they are visible on the editor. The
translation unit is smart enough to calculate line/column position
for such tokens based on the information from the expansion section
header.
- How expansions are tracked has also changed. Now, we simply add
two surrounding marker tokens to each "top-level" expansion
sequence. There is an enumeration that control expansion states.
Also, no "previous" token is kept around.
- Preprocessor client methods suffered a change in signature so
they now receive the line number of the action in question as
a paramater. Previously such line could be retrieved by the client
implementation by accessing the environment line. However, this
is not reliable because we try to avoid synchronization of the
output/environment lines in order to avoid unnecessary output,
while expanding macros or handling preprocessor directives.
- Although macros are not expanded during define directives (as
mentioned above) the preprocessor client is now "notified"
when it sees a macro. This is to allow usage tracking.
- Other small stuff.
This is all in one patch because the fixes are a consequence
of the change in preprocessing control.
Change-Id: I8f4c6e6366f37756ec65d0a93b79f72a3ac4ed50
Reviewed-by: Roberto Raggi <roberto.raggi@nokia.com>
2012-06-20 15:22:02 +02:00
|
|
|
void CppPreprocessor::notifyMacroReference(unsigned offset, unsigned line, const Macro ¯o)
|
|
|
|
{
|
|
|
|
if (! m_currentDoc)
|
|
|
|
return;
|
|
|
|
|
2012-10-11 16:16:01 +02:00
|
|
|
m_currentDoc->addMacroUse(revision(m_workingCopy, macro), offset, macro.name().length(), line,
|
C++: Core changes in preprocessing
Summary of most relevant items:
- Preprocessor output format change. No more gen true/false. Instead
a more intuitive and natural expansion (like from a real compiler) is
performed directly corresponding to the macro invocation. Notice that
information about the generated tokens is not lost, because it's now
embedded in the expansion section header (in terms of lines and columns
as explained in the code). In addition the location on where the macro
expansion happens is also documented for future use.
- Fix line control directives and associated token line numbers.
This was not detected in tests cases because some of them were
actually wrong: Within expansions the line information was being
considered as originally computed in the macro definition, while
the desired and expected for Creator's reporting mechanism (just
like regular compilers) is the line from the expanded version
of the tokens.
- Do not allow for eager expansion. This was previously being done
inside define directives. However, it's not allowed and might
lead to incorrect results, since the argument substitution should
only happen upon the macro invocation (and following nested ones).
At least GCC and clang are consistent with that. See test case
tst_Preprocessor:dont_eagerly_expand for a detailed explanation.
- Revive the 'expanded' token flag. This is used to mark every token
that originates from a macro expansion. Notice, however, that
expanded tokens are not necessarily generated tokens (although
every generated token is a expanded token). Expanded tokens that
are not generated are those which are still considered by our
code model features, since they are visible on the editor. The
translation unit is smart enough to calculate line/column position
for such tokens based on the information from the expansion section
header.
- How expansions are tracked has also changed. Now, we simply add
two surrounding marker tokens to each "top-level" expansion
sequence. There is an enumeration that control expansion states.
Also, no "previous" token is kept around.
- Preprocessor client methods suffered a change in signature so
they now receive the line number of the action in question as
a paramater. Previously such line could be retrieved by the client
implementation by accessing the environment line. However, this
is not reliable because we try to avoid synchronization of the
output/environment lines in order to avoid unnecessary output,
while expanding macros or handling preprocessor directives.
- Although macros are not expanded during define directives (as
mentioned above) the preprocessor client is now "notified"
when it sees a macro. This is to allow usage tracking.
- Other small stuff.
This is all in one patch because the fixes are a consequence
of the change in preprocessing control.
Change-Id: I8f4c6e6366f37756ec65d0a93b79f72a3ac4ed50
Reviewed-by: Roberto Raggi <roberto.raggi@nokia.com>
2012-06-20 15:22:02 +02:00
|
|
|
QVector<MacroArgumentReference>());
|
|
|
|
}
|
|
|
|
|
|
|
|
void CppPreprocessor::startExpandingMacro(unsigned offset, unsigned line,
|
2008-12-09 15:23:47 +01:00
|
|
|
const Macro ¯o,
|
2009-03-04 11:47:30 +01:00
|
|
|
const QVector<MacroArgumentReference> &actuals)
|
2008-12-08 11:08:48 +01:00
|
|
|
{
|
|
|
|
if (! m_currentDoc)
|
|
|
|
return;
|
2008-12-04 12:05:04 +01:00
|
|
|
|
2012-10-11 16:16:01 +02:00
|
|
|
m_currentDoc->addMacroUse(revision(m_workingCopy, macro), offset, macro.name().length(), line, actuals);
|
2008-12-08 11:08:48 +01:00
|
|
|
}
|
2008-12-04 12:05:04 +01:00
|
|
|
|
2008-12-08 12:17:39 +01:00
|
|
|
void CppPreprocessor::stopExpandingMacro(unsigned, const Macro &)
|
2008-12-08 11:08:48 +01:00
|
|
|
{
|
|
|
|
if (! m_currentDoc)
|
|
|
|
return;
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
//qDebug() << "stop expanding:" << macro.name;
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
void CppPreprocessor::mergeEnvironment(Document::Ptr doc)
|
|
|
|
{
|
|
|
|
if (! doc)
|
|
|
|
return;
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
const QString fn = doc->fileName();
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-03-09 12:51:13 +01:00
|
|
|
if (m_processed.contains(fn))
|
2008-12-08 11:08:48 +01:00
|
|
|
return;
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-03-09 12:51:13 +01:00
|
|
|
m_processed.insert(fn);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-02-23 12:49:56 +01:00
|
|
|
foreach (const Document::Include &incl, doc->includes()) {
|
2009-02-09 12:33:43 +01:00
|
|
|
QString includedFile = incl.fileName();
|
|
|
|
|
2009-12-07 10:54:27 +01:00
|
|
|
if (Document::Ptr includedDoc = snapshot.document(includedFile))
|
2009-03-09 12:51:13 +01:00
|
|
|
mergeEnvironment(includedDoc);
|
2009-02-09 12:33:43 +01:00
|
|
|
else
|
|
|
|
run(includedFile);
|
2008-12-08 12:59:33 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-02-23 12:49:56 +01:00
|
|
|
env.addMacros(doc->definedMacros());
|
2008-12-08 11:08:48 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
void CppPreprocessor::startSkippingBlocks(unsigned offset)
|
|
|
|
{
|
|
|
|
//qDebug() << "start skipping blocks:" << offset;
|
|
|
|
if (m_currentDoc)
|
|
|
|
m_currentDoc->startSkippingBlocks(offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CppPreprocessor::stopSkippingBlocks(unsigned offset)
|
|
|
|
{
|
|
|
|
//qDebug() << "stop skipping blocks:" << offset;
|
|
|
|
if (m_currentDoc)
|
|
|
|
m_currentDoc->stopSkippingBlocks(offset);
|
|
|
|
}
|
|
|
|
|
C++: Core changes in preprocessing
Summary of most relevant items:
- Preprocessor output format change. No more gen true/false. Instead
a more intuitive and natural expansion (like from a real compiler) is
performed directly corresponding to the macro invocation. Notice that
information about the generated tokens is not lost, because it's now
embedded in the expansion section header (in terms of lines and columns
as explained in the code). In addition the location on where the macro
expansion happens is also documented for future use.
- Fix line control directives and associated token line numbers.
This was not detected in tests cases because some of them were
actually wrong: Within expansions the line information was being
considered as originally computed in the macro definition, while
the desired and expected for Creator's reporting mechanism (just
like regular compilers) is the line from the expanded version
of the tokens.
- Do not allow for eager expansion. This was previously being done
inside define directives. However, it's not allowed and might
lead to incorrect results, since the argument substitution should
only happen upon the macro invocation (and following nested ones).
At least GCC and clang are consistent with that. See test case
tst_Preprocessor:dont_eagerly_expand for a detailed explanation.
- Revive the 'expanded' token flag. This is used to mark every token
that originates from a macro expansion. Notice, however, that
expanded tokens are not necessarily generated tokens (although
every generated token is a expanded token). Expanded tokens that
are not generated are those which are still considered by our
code model features, since they are visible on the editor. The
translation unit is smart enough to calculate line/column position
for such tokens based on the information from the expansion section
header.
- How expansions are tracked has also changed. Now, we simply add
two surrounding marker tokens to each "top-level" expansion
sequence. There is an enumeration that control expansion states.
Also, no "previous" token is kept around.
- Preprocessor client methods suffered a change in signature so
they now receive the line number of the action in question as
a paramater. Previously such line could be retrieved by the client
implementation by accessing the environment line. However, this
is not reliable because we try to avoid synchronization of the
output/environment lines in order to avoid unnecessary output,
while expanding macros or handling preprocessor directives.
- Although macros are not expanded during define directives (as
mentioned above) the preprocessor client is now "notified"
when it sees a macro. This is to allow usage tracking.
- Other small stuff.
This is all in one patch because the fixes are a consequence
of the change in preprocessing control.
Change-Id: I8f4c6e6366f37756ec65d0a93b79f72a3ac4ed50
Reviewed-by: Roberto Raggi <roberto.raggi@nokia.com>
2012-06-20 15:22:02 +02:00
|
|
|
void CppPreprocessor::sourceNeeded(unsigned line, QString &fileName, IncludeType type)
|
2008-12-08 11:08:48 +01:00
|
|
|
{
|
|
|
|
if (fileName.isEmpty())
|
|
|
|
return;
|
|
|
|
|
2009-12-15 15:52:55 +01:00
|
|
|
unsigned editorRevision = 0;
|
|
|
|
QString contents = tryIncludeFile(fileName, type, &editorRevision);
|
2009-11-09 13:53:28 +01:00
|
|
|
fileName = QDir::cleanPath(fileName);
|
2008-12-08 11:08:48 +01:00
|
|
|
if (m_currentDoc) {
|
2008-12-10 17:21:01 +01:00
|
|
|
m_currentDoc->addIncludeFile(fileName, line);
|
2009-02-23 12:49:56 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
if (contents.isEmpty() && ! QFileInfo(fileName).isAbsolute()) {
|
2009-04-17 21:11:52 +02:00
|
|
|
QString msg = QCoreApplication::translate(
|
|
|
|
"CppPreprocessor", "%1: No such file or directory").arg(fileName);
|
2009-02-23 12:49:56 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
Document::DiagnosticMessage d(Document::DiagnosticMessage::Warning,
|
|
|
|
m_currentDoc->fileName(),
|
C++: Core changes in preprocessing
Summary of most relevant items:
- Preprocessor output format change. No more gen true/false. Instead
a more intuitive and natural expansion (like from a real compiler) is
performed directly corresponding to the macro invocation. Notice that
information about the generated tokens is not lost, because it's now
embedded in the expansion section header (in terms of lines and columns
as explained in the code). In addition the location on where the macro
expansion happens is also documented for future use.
- Fix line control directives and associated token line numbers.
This was not detected in tests cases because some of them were
actually wrong: Within expansions the line information was being
considered as originally computed in the macro definition, while
the desired and expected for Creator's reporting mechanism (just
like regular compilers) is the line from the expanded version
of the tokens.
- Do not allow for eager expansion. This was previously being done
inside define directives. However, it's not allowed and might
lead to incorrect results, since the argument substitution should
only happen upon the macro invocation (and following nested ones).
At least GCC and clang are consistent with that. See test case
tst_Preprocessor:dont_eagerly_expand for a detailed explanation.
- Revive the 'expanded' token flag. This is used to mark every token
that originates from a macro expansion. Notice, however, that
expanded tokens are not necessarily generated tokens (although
every generated token is a expanded token). Expanded tokens that
are not generated are those which are still considered by our
code model features, since they are visible on the editor. The
translation unit is smart enough to calculate line/column position
for such tokens based on the information from the expansion section
header.
- How expansions are tracked has also changed. Now, we simply add
two surrounding marker tokens to each "top-level" expansion
sequence. There is an enumeration that control expansion states.
Also, no "previous" token is kept around.
- Preprocessor client methods suffered a change in signature so
they now receive the line number of the action in question as
a paramater. Previously such line could be retrieved by the client
implementation by accessing the environment line. However, this
is not reliable because we try to avoid synchronization of the
output/environment lines in order to avoid unnecessary output,
while expanding macros or handling preprocessor directives.
- Although macros are not expanded during define directives (as
mentioned above) the preprocessor client is now "notified"
when it sees a macro. This is to allow usage tracking.
- Other small stuff.
This is all in one patch because the fixes are a consequence
of the change in preprocessing control.
Change-Id: I8f4c6e6366f37756ec65d0a93b79f72a3ac4ed50
Reviewed-by: Roberto Raggi <roberto.raggi@nokia.com>
2012-06-20 15:22:02 +02:00
|
|
|
line, /*column = */ 0,
|
2008-12-08 11:08:48 +01:00
|
|
|
msg);
|
2009-02-23 12:49:56 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
m_currentDoc->addDiagnosticMessage(d);
|
2009-02-23 12:49:56 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
//qWarning() << "file not found:" << fileName << m_currentDoc->fileName() << env.current_line;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
2008-12-08 11:08:48 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2012-03-30 10:06:02 +02:00
|
|
|
if (m_dumpFileNameWhileParsing) {
|
|
|
|
qDebug() << "Parsing file:" << fileName
|
2012-03-26 15:18:01 +02:00
|
|
|
// << "contents:" << contents.size()
|
2012-03-30 10:06:02 +02:00
|
|
|
;
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-12-07 10:54:27 +01:00
|
|
|
Document::Ptr doc = snapshot.document(fileName);
|
2009-02-23 12:49:56 +01:00
|
|
|
if (doc) {
|
|
|
|
mergeEnvironment(doc);
|
|
|
|
return;
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-02-23 12:49:56 +01:00
|
|
|
doc = Document::create(fileName);
|
2009-07-14 14:23:12 +02:00
|
|
|
doc->setRevision(m_revision);
|
2009-12-15 15:52:55 +01:00
|
|
|
doc->setEditorRevision(editorRevision);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-11-05 12:34:02 +01:00
|
|
|
QFileInfo info(fileName);
|
|
|
|
if (info.exists())
|
|
|
|
doc->setLastModified(info.lastModified());
|
|
|
|
|
2009-02-23 12:49:56 +01:00
|
|
|
Document::Ptr previousDoc = switchDocument(doc);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2012-04-19 10:35:49 +02:00
|
|
|
const QByteArray preprocessedCode = preprocess.run(fileName, contents);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2012-03-26 15:18:01 +02:00
|
|
|
// { QByteArray b(preprocessedCode); b.replace("\n", "<<<\n"); qDebug("Preprocessed code for \"%s\": [[%s]]", fileName.toUtf8().constData(), b.constData()); }
|
|
|
|
|
2012-01-12 17:53:56 +01:00
|
|
|
doc->setUtf8Source(preprocessedCode);
|
2011-08-23 12:02:29 +02:00
|
|
|
doc->keepSourceAndAST();
|
2009-02-24 11:04:52 +01:00
|
|
|
doc->tokenize();
|
2009-02-23 12:49:56 +01:00
|
|
|
|
2009-12-07 10:54:27 +01:00
|
|
|
snapshot.insert(doc);
|
2009-03-06 11:51:45 +01:00
|
|
|
m_todo.remove(fileName);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2010-05-10 10:19:37 +02:00
|
|
|
Process process(m_modelManager, doc, snapshot, m_workingCopy);
|
2009-03-17 14:39:52 +01:00
|
|
|
|
2010-05-10 10:19:37 +02:00
|
|
|
process();
|
2009-02-23 12:49:56 +01:00
|
|
|
|
|
|
|
(void) switchDocument(previousDoc);
|
2008-12-08 11:08:48 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 11:08:48 +01:00
|
|
|
Document::Ptr CppPreprocessor::switchDocument(Document::Ptr doc)
|
|
|
|
{
|
|
|
|
Document::Ptr previousDoc = m_currentDoc;
|
|
|
|
m_currentDoc = doc;
|
|
|
|
return previousDoc;
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2010-12-03 13:49:35 +01:00
|
|
|
void CppModelManager::updateModifiedSourceFiles()
|
2009-11-05 12:34:02 +01:00
|
|
|
{
|
|
|
|
const Snapshot snapshot = this->snapshot();
|
|
|
|
QStringList sourceFiles;
|
|
|
|
|
|
|
|
foreach (const Document::Ptr doc, snapshot) {
|
|
|
|
const QDateTime lastModified = doc->lastModified();
|
|
|
|
|
|
|
|
if (! lastModified.isNull()) {
|
|
|
|
QFileInfo fileInfo(doc->fileName());
|
|
|
|
|
|
|
|
if (fileInfo.exists() && fileInfo.lastModified() != lastModified)
|
|
|
|
sourceFiles.append(doc->fileName());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
updateSourceFiles(sourceFiles);
|
|
|
|
}
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
/*!
|
|
|
|
\class CppTools::CppModelManager
|
|
|
|
\brief The CppModelManager keeps track of one CppCodeModel instance
|
|
|
|
for each project and all related CppCodeModelPart instances.
|
|
|
|
|
|
|
|
It also takes care of updating the code models when C++ files are
|
2009-05-08 13:01:32 +02:00
|
|
|
modified within Qt Creator.
|
2008-12-02 12:01:29 +01:00
|
|
|
*/
|
|
|
|
|
2012-10-15 16:38:56 +02:00
|
|
|
QMutex CppModelManager::m_modelManagerMutex;
|
|
|
|
CppModelManager *CppModelManager::m_modelManagerInstance = 0;
|
|
|
|
|
|
|
|
CppModelManager *CppModelManager::instance()
|
|
|
|
{
|
|
|
|
if (m_modelManagerInstance)
|
|
|
|
return m_modelManagerInstance;
|
|
|
|
QMutexLocker locker(&m_modelManagerMutex);
|
|
|
|
if (!m_modelManagerInstance) {
|
|
|
|
m_modelManagerInstance = new CppModelManager;
|
|
|
|
}
|
|
|
|
return m_modelManagerInstance;
|
|
|
|
}
|
|
|
|
|
2009-01-20 11:52:04 +01:00
|
|
|
CppModelManager::CppModelManager(QObject *parent)
|
2009-11-10 18:02:42 +01:00
|
|
|
: CppModelManagerInterface(parent)
|
2012-11-23 11:47:39 +01:00
|
|
|
, m_completionAssistProvider(0)
|
|
|
|
, m_highlightingFactory(0)
|
|
|
|
, m_indexingSupporter(0)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2009-08-07 13:02:36 +02:00
|
|
|
m_findReferences = new CppFindReferences(this);
|
2009-12-07 12:03:31 +01:00
|
|
|
m_indexerEnabled = qgetenv("QTCREATOR_NO_CODE_INDEXER").isNull();
|
2009-03-11 12:00:07 +01:00
|
|
|
|
2008-12-04 17:07:43 +01:00
|
|
|
m_dirty = true;
|
|
|
|
|
2009-03-04 11:47:30 +01:00
|
|
|
ProjectExplorer::ProjectExplorerPlugin *pe =
|
2009-03-04 09:38:01 +01:00
|
|
|
ProjectExplorer::ProjectExplorerPlugin::instance();
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-03-04 09:38:01 +01:00
|
|
|
QTC_ASSERT(pe, return);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-03-04 09:38:01 +01:00
|
|
|
ProjectExplorer::SessionManager *session = pe->session();
|
2009-02-10 22:56:04 +01:00
|
|
|
m_updateEditorSelectionsTimer = new QTimer(this);
|
|
|
|
m_updateEditorSelectionsTimer->setInterval(500);
|
|
|
|
m_updateEditorSelectionsTimer->setSingleShot(true);
|
|
|
|
connect(m_updateEditorSelectionsTimer, SIGNAL(timeout()),
|
|
|
|
this, SLOT(updateEditorSelections()));
|
|
|
|
|
2008-12-04 17:07:43 +01:00
|
|
|
connect(session, SIGNAL(projectAdded(ProjectExplorer::Project*)),
|
|
|
|
this, SLOT(onProjectAdded(ProjectExplorer::Project*)));
|
|
|
|
|
2012-03-05 22:30:59 +01:00
|
|
|
connect(session, SIGNAL(aboutToRemoveProject(ProjectExplorer::Project*)),
|
|
|
|
this, SLOT(onAboutToRemoveProject(ProjectExplorer::Project*)));
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2012-02-10 12:46:33 +01:00
|
|
|
connect(session, SIGNAL(aboutToUnloadSession(QString)),
|
2009-08-13 17:10:53 +02:00
|
|
|
this, SLOT(onAboutToUnloadSession()));
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
qRegisterMetaType<CPlusPlus::Document::Ptr>("CPlusPlus::Document::Ptr");
|
|
|
|
|
|
|
|
// thread connections
|
|
|
|
connect(this, SIGNAL(documentUpdated(CPlusPlus::Document::Ptr)),
|
|
|
|
this, SLOT(onDocumentUpdated(CPlusPlus::Document::Ptr)));
|
2011-08-31 09:58:40 +02:00
|
|
|
connect(this, SIGNAL(extraDiagnosticsUpdated(QString)),
|
|
|
|
this, SLOT(onExtraDiagnosticsUpdated(QString)));
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
// Listen for editor closed and opened events so that we can keep track of changing files
|
2012-03-05 22:30:59 +01:00
|
|
|
connect(Core::ICore::editorManager(), SIGNAL(editorOpened(Core::IEditor*)),
|
|
|
|
this, SLOT(editorOpened(Core::IEditor*)));
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2012-03-05 22:30:59 +01:00
|
|
|
connect(Core::ICore::editorManager(), SIGNAL(editorAboutToClose(Core::IEditor*)),
|
|
|
|
this, SLOT(editorAboutToClose(Core::IEditor*)));
|
2012-02-20 12:39:08 +01:00
|
|
|
|
2012-02-21 10:00:32 +01:00
|
|
|
m_completionFallback = new InternalCompletionAssistProvider;
|
|
|
|
m_completionAssistProvider = m_completionFallback;
|
2012-06-18 11:34:15 +02:00
|
|
|
ExtensionSystem::PluginManager::addObject(m_completionAssistProvider);
|
2012-02-20 12:39:08 +01:00
|
|
|
m_highlightingFallback = new CppHighlightingSupportInternalFactory;
|
|
|
|
m_highlightingFactory = m_highlightingFallback;
|
2012-11-28 09:52:19 +01:00
|
|
|
m_internalIndexingSupport = new BuiltinIndexingSupport;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
CppModelManager::~CppModelManager()
|
2012-02-20 12:39:08 +01:00
|
|
|
{
|
2012-06-18 11:34:15 +02:00
|
|
|
ExtensionSystem::PluginManager::removeObject(m_completionAssistProvider);
|
2012-02-20 12:39:08 +01:00
|
|
|
delete m_completionFallback;
|
|
|
|
delete m_highlightingFallback;
|
2012-10-16 16:02:40 +02:00
|
|
|
delete m_internalIndexingSupport;
|
2012-02-20 12:39:08 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-12 10:07:58 +01:00
|
|
|
Snapshot CppModelManager::snapshot() const
|
2009-03-04 09:38:01 +01:00
|
|
|
{
|
2009-06-02 15:27:13 +02:00
|
|
|
QMutexLocker locker(&protectSnapshot);
|
2009-03-04 09:38:01 +01:00
|
|
|
return m_snapshot;
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 10:44:56 +01:00
|
|
|
void CppModelManager::ensureUpdated()
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&mutex);
|
|
|
|
if (! m_dirty)
|
|
|
|
return;
|
|
|
|
|
2008-12-08 14:48:51 +01:00
|
|
|
m_projectFiles = internalProjectFiles();
|
|
|
|
m_includePaths = internalIncludePaths();
|
|
|
|
m_frameworkPaths = internalFrameworkPaths();
|
|
|
|
m_definedMacros = internalDefinedMacros();
|
2008-12-08 10:44:56 +01:00
|
|
|
m_dirty = false;
|
|
|
|
}
|
|
|
|
|
2008-12-08 14:48:51 +01:00
|
|
|
QStringList CppModelManager::internalProjectFiles() const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
QStringList files;
|
|
|
|
QMapIterator<ProjectExplorer::Project *, ProjectInfo> it(m_projects);
|
|
|
|
while (it.hasNext()) {
|
|
|
|
it.next();
|
|
|
|
ProjectInfo pinfo = it.value();
|
2012-11-23 16:29:00 +01:00
|
|
|
foreach (const ProjectPart::Ptr &part, pinfo.projectParts()) {
|
|
|
|
files += part->headerFiles;
|
2012-02-16 15:09:56 +01:00
|
|
|
files += part->sourceFiles;
|
2012-11-23 16:29:00 +01:00
|
|
|
files += part->objcSourceFiles;
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
2008-12-08 14:48:51 +01:00
|
|
|
files.removeDuplicates();
|
2008-12-02 12:01:29 +01:00
|
|
|
return files;
|
|
|
|
}
|
|
|
|
|
2008-12-08 14:48:51 +01:00
|
|
|
QStringList CppModelManager::internalIncludePaths() const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
QStringList includePaths;
|
|
|
|
QMapIterator<ProjectExplorer::Project *, ProjectInfo> it(m_projects);
|
|
|
|
while (it.hasNext()) {
|
|
|
|
it.next();
|
|
|
|
ProjectInfo pinfo = it.value();
|
2012-02-16 15:09:56 +01:00
|
|
|
foreach (const ProjectPart::Ptr &part, pinfo.projectParts())
|
|
|
|
includePaths += part->includePaths;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
2008-12-08 14:48:51 +01:00
|
|
|
includePaths.removeDuplicates();
|
2008-12-02 12:01:29 +01:00
|
|
|
return includePaths;
|
|
|
|
}
|
|
|
|
|
2008-12-08 14:48:51 +01:00
|
|
|
QStringList CppModelManager::internalFrameworkPaths() const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
QStringList frameworkPaths;
|
|
|
|
QMapIterator<ProjectExplorer::Project *, ProjectInfo> it(m_projects);
|
|
|
|
while (it.hasNext()) {
|
|
|
|
it.next();
|
|
|
|
ProjectInfo pinfo = it.value();
|
2012-02-16 15:09:56 +01:00
|
|
|
foreach (const ProjectPart::Ptr &part, pinfo.projectParts())
|
|
|
|
frameworkPaths += part->frameworkPaths;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
2008-12-08 14:48:51 +01:00
|
|
|
frameworkPaths.removeDuplicates();
|
2008-12-02 12:01:29 +01:00
|
|
|
return frameworkPaths;
|
|
|
|
}
|
|
|
|
|
2008-12-08 14:48:51 +01:00
|
|
|
QByteArray CppModelManager::internalDefinedMacros() const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
QByteArray macros;
|
2012-03-27 15:29:12 +02:00
|
|
|
QSet<QByteArray> alreadyIn;
|
2008-12-02 12:01:29 +01:00
|
|
|
QMapIterator<ProjectExplorer::Project *, ProjectInfo> it(m_projects);
|
|
|
|
while (it.hasNext()) {
|
|
|
|
it.next();
|
|
|
|
ProjectInfo pinfo = it.value();
|
2012-03-27 15:29:12 +02:00
|
|
|
foreach (const ProjectPart::Ptr &part, pinfo.projectParts()) {
|
|
|
|
const QList<QByteArray> defs = part->defines.split('\n');
|
|
|
|
foreach (const QByteArray &def, defs) {
|
|
|
|
if (!alreadyIn.contains(def)) {
|
|
|
|
macros += def;
|
|
|
|
macros.append('\n');
|
|
|
|
alreadyIn.insert(def);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
return macros;
|
|
|
|
}
|
|
|
|
|
2012-11-06 10:48:16 +01:00
|
|
|
/// This method will aquire the mutex!
|
|
|
|
void CppModelManager::dumpModelManagerConfiguration()
|
|
|
|
{
|
|
|
|
// Tons of debug output...
|
|
|
|
qDebug()<<"========= CppModelManager::dumpModelManagerConfiguration ======";
|
|
|
|
foreach (const ProjectInfo &pinfo, m_projects.values()) {
|
|
|
|
qDebug()<<" for project:"<< pinfo.project().data()->document()->fileName();
|
|
|
|
foreach (const ProjectPart::Ptr &part, pinfo.projectParts()) {
|
|
|
|
qDebug() << "=== part ===";
|
2012-11-23 16:29:00 +01:00
|
|
|
const char* lang;
|
|
|
|
switch (part->language) {
|
|
|
|
case ProjectPart::CXX: lang = "C++"; break;
|
|
|
|
case ProjectPart::CXX11: lang = "C++11"; break;
|
|
|
|
case ProjectPart::C89: lang = "C89"; break;
|
|
|
|
case ProjectPart::C99: lang = "C99"; break;
|
|
|
|
default: lang = "INVALID";
|
|
|
|
}
|
|
|
|
|
|
|
|
qDebug() << "language:" << lang;
|
2012-11-06 10:48:16 +01:00
|
|
|
qDebug() << "Qt version:" << part->qtVersion;
|
|
|
|
qDebug() << "precompiled header:" << part->precompiledHeaders;
|
|
|
|
qDebug() << "defines:" << part->defines;
|
|
|
|
qDebug() << "includes:" << part->includePaths;
|
|
|
|
qDebug() << "frameworkPaths:" << part->frameworkPaths;
|
2012-11-23 16:29:00 +01:00
|
|
|
qDebug() << "headers:" << part->headerFiles;
|
2012-11-06 10:48:16 +01:00
|
|
|
qDebug() << "sources:" << part->sourceFiles;
|
2012-11-23 16:29:00 +01:00
|
|
|
qDebug() << "objc sources:" << part->objcSourceFiles;
|
2012-11-06 10:48:16 +01:00
|
|
|
qDebug() << "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ensureUpdated();
|
|
|
|
qDebug() << "=== Merged include paths ===";
|
|
|
|
foreach (const QString &inc, m_includePaths)
|
|
|
|
qDebug() << inc;
|
|
|
|
qDebug() << "=== Merged framework paths ===";
|
|
|
|
foreach (const QString &inc, m_frameworkPaths)
|
|
|
|
qDebug() << inc;
|
|
|
|
qDebug() << "=== Merged defined macros ===";
|
|
|
|
qDebug() << m_definedMacros;
|
|
|
|
qDebug()<<"========= End of dump ======";
|
|
|
|
}
|
|
|
|
|
2009-05-12 13:45:24 +02:00
|
|
|
void CppModelManager::addEditorSupport(AbstractEditorSupport *editorSupport)
|
|
|
|
{
|
|
|
|
m_addtionalEditorSupport.insert(editorSupport);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CppModelManager::removeEditorSupport(AbstractEditorSupport *editorSupport)
|
|
|
|
{
|
|
|
|
m_addtionalEditorSupport.remove(editorSupport);
|
|
|
|
}
|
|
|
|
|
2010-05-31 12:09:28 +02:00
|
|
|
QList<int> CppModelManager::references(CPlusPlus::Symbol *symbol, const LookupContext &context)
|
2009-09-30 13:25:40 +02:00
|
|
|
{
|
2010-05-31 12:09:28 +02:00
|
|
|
return m_findReferences->references(symbol, context);
|
2009-09-30 13:25:40 +02:00
|
|
|
}
|
|
|
|
|
2010-05-31 12:09:28 +02:00
|
|
|
void CppModelManager::findUsages(CPlusPlus::Symbol *symbol, const CPlusPlus::LookupContext &context)
|
2009-08-07 13:02:36 +02:00
|
|
|
{
|
2009-09-24 16:51:40 +02:00
|
|
|
if (symbol->identifier())
|
2010-05-31 12:09:28 +02:00
|
|
|
m_findReferences->findUsages(symbol, context);
|
2009-10-05 15:17:25 +02:00
|
|
|
}
|
|
|
|
|
2010-07-20 11:02:37 +02:00
|
|
|
void CppModelManager::renameUsages(CPlusPlus::Symbol *symbol, const CPlusPlus::LookupContext &context,
|
|
|
|
const QString &replacement)
|
2009-10-05 15:17:25 +02:00
|
|
|
{
|
|
|
|
if (symbol->identifier())
|
2010-07-20 11:02:37 +02:00
|
|
|
m_findReferences->renameUsages(symbol, context, replacement);
|
2009-08-07 13:02:36 +02:00
|
|
|
}
|
|
|
|
|
2009-12-21 14:54:10 +01:00
|
|
|
void CppModelManager::findMacroUsages(const CPlusPlus::Macro ¯o)
|
|
|
|
{
|
|
|
|
m_findReferences->findMacroUses(macro);
|
|
|
|
}
|
|
|
|
|
2012-03-17 13:26:27 +01:00
|
|
|
void CppModelManager::renameMacroUsages(const CPlusPlus::Macro ¯o, const QString &replacement)
|
|
|
|
{
|
|
|
|
m_findReferences->renameMacroUses(macro, replacement);
|
|
|
|
}
|
|
|
|
|
2009-12-15 15:16:46 +01:00
|
|
|
CppModelManager::WorkingCopy CppModelManager::buildWorkingCopyList()
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2009-12-15 15:16:46 +01:00
|
|
|
WorkingCopy workingCopy;
|
2008-12-02 12:01:29 +01:00
|
|
|
QMapIterator<TextEditor::ITextEditor *, CppEditorSupport *> it(m_editorSupport);
|
|
|
|
while (it.hasNext()) {
|
|
|
|
it.next();
|
|
|
|
TextEditor::ITextEditor *textEditor = it.key();
|
|
|
|
CppEditorSupport *editorSupport = it.value();
|
2012-02-14 16:43:51 +01:00
|
|
|
QString fileName = textEditor->document()->fileName();
|
2009-12-15 15:52:55 +01:00
|
|
|
workingCopy.insert(fileName, editorSupport->contents(), editorSupport->editorRevision());
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
2009-05-12 13:45:24 +02:00
|
|
|
QSetIterator<AbstractEditorSupport *> jt(m_addtionalEditorSupport);
|
|
|
|
while (jt.hasNext()) {
|
|
|
|
AbstractEditorSupport *es = jt.next();
|
2012-11-21 22:36:47 +02:00
|
|
|
workingCopy.insert(es->fileName(), QString::fromUtf8(es->contents()));
|
2009-05-12 13:45:24 +02:00
|
|
|
}
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
// add the project configuration file
|
|
|
|
QByteArray conf(pp_configuration);
|
|
|
|
conf += definedMacros();
|
2012-11-28 09:52:19 +01:00
|
|
|
workingCopy.insert(configurationFileName(), QString::fromUtf8(conf));
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
return workingCopy;
|
|
|
|
}
|
|
|
|
|
2009-12-15 15:16:46 +01:00
|
|
|
CppModelManager::WorkingCopy CppModelManager::workingCopy() const
|
2009-10-12 10:38:00 +02:00
|
|
|
{
|
|
|
|
return const_cast<CppModelManager *>(this)->buildWorkingCopyList();
|
|
|
|
}
|
|
|
|
|
2010-03-02 12:51:47 +01:00
|
|
|
QFuture<void> CppModelManager::updateSourceFiles(const QStringList &sourceFiles)
|
2012-10-16 16:02:40 +02:00
|
|
|
{
|
|
|
|
if (sourceFiles.isEmpty() || !m_indexerEnabled)
|
|
|
|
return QFuture<void>();
|
|
|
|
|
2012-11-23 11:47:39 +01:00
|
|
|
if (m_indexingSupporter)
|
|
|
|
m_indexingSupporter->refreshSourceFiles(sourceFiles);
|
2012-10-16 16:02:40 +02:00
|
|
|
return m_internalIndexingSupport->refreshSourceFiles(sourceFiles);
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-08 10:44:56 +01:00
|
|
|
QList<CppModelManager::ProjectInfo> CppModelManager::projectInfos() const
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&mutex);
|
|
|
|
|
|
|
|
return m_projects.values();
|
|
|
|
}
|
|
|
|
|
|
|
|
CppModelManager::ProjectInfo CppModelManager::projectInfo(ProjectExplorer::Project *project) const
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&mutex);
|
|
|
|
|
|
|
|
return m_projects.value(project, ProjectInfo(project));
|
|
|
|
}
|
|
|
|
|
|
|
|
void CppModelManager::updateProjectInfo(const ProjectInfo &pinfo)
|
|
|
|
{
|
2012-11-06 10:48:16 +01:00
|
|
|
{ // only hold the mutex for a limited scope, so the dumping afterwards can aquire it without deadlocking.
|
|
|
|
QMutexLocker locker(&mutex);
|
2012-02-16 15:09:56 +01:00
|
|
|
|
2012-11-06 10:48:16 +01:00
|
|
|
if (! pinfo.isValid())
|
|
|
|
return;
|
2008-12-08 10:44:56 +01:00
|
|
|
|
2012-11-06 10:48:16 +01:00
|
|
|
ProjectExplorer::Project *project = pinfo.project().data();
|
|
|
|
m_projects.insert(project, pinfo);
|
|
|
|
m_dirty = true;
|
2008-12-08 10:44:56 +01:00
|
|
|
|
2012-11-06 10:48:16 +01:00
|
|
|
m_srcToProjectPart.clear();
|
2012-02-16 15:09:56 +01:00
|
|
|
|
2012-11-23 16:29:00 +01:00
|
|
|
foreach (const ProjectInfo &projectInfo, m_projects.values()) {
|
|
|
|
foreach (const ProjectPart::Ptr &projectPart, projectInfo.projectParts()) {
|
2012-11-06 10:48:16 +01:00
|
|
|
foreach (const QString &sourceFile, projectPart->sourceFiles)
|
|
|
|
m_srcToProjectPart[sourceFile].append(projectPart);
|
2012-11-23 16:29:00 +01:00
|
|
|
foreach (const QString &objcSourceFile, projectPart->objcSourceFiles)
|
|
|
|
m_srcToProjectPart[objcSourceFile].append(projectPart);
|
|
|
|
foreach (const QString &headerFile, projectPart->headerFiles)
|
|
|
|
m_srcToProjectPart[headerFile].append(projectPart);
|
|
|
|
}
|
|
|
|
}
|
2012-11-06 10:48:16 +01:00
|
|
|
}
|
2012-02-16 15:09:56 +01:00
|
|
|
|
2012-11-06 10:48:16 +01:00
|
|
|
if (!qgetenv("QTCREATOR_DUMP_PROJECT_INFO").isEmpty())
|
|
|
|
dumpModelManagerConfiguration();
|
2012-02-16 15:09:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QList<CppModelManager::ProjectPart::Ptr> CppModelManager::projectPart(const QString &fileName) const
|
|
|
|
{
|
|
|
|
QList<CppModelManager::ProjectPart::Ptr> parts = m_srcToProjectPart.value(fileName);
|
|
|
|
if (!parts.isEmpty())
|
|
|
|
return parts;
|
|
|
|
|
|
|
|
DependencyTable table;
|
|
|
|
table.build(snapshot());
|
|
|
|
QStringList deps = table.filesDependingOn(fileName);
|
|
|
|
foreach (const QString &dep, deps) {
|
|
|
|
parts = m_srcToProjectPart.value(dep);
|
|
|
|
if (!parts.isEmpty())
|
|
|
|
return parts;
|
|
|
|
}
|
|
|
|
|
|
|
|
return parts;
|
2008-12-08 10:44:56 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
/*!
|
|
|
|
\fn void CppModelManager::editorOpened(Core::IEditor *editor)
|
|
|
|
\brief If a C++ editor is opened, the model manager listens to content changes
|
|
|
|
in order to update the CppCodeModel accordingly. It also updates the
|
|
|
|
CppCodeModel for the first time with this editor.
|
|
|
|
|
|
|
|
\sa void CppModelManager::editorContentsChanged()
|
|
|
|
*/
|
|
|
|
void CppModelManager::editorOpened(Core::IEditor *editor)
|
|
|
|
{
|
|
|
|
if (isCppEditor(editor)) {
|
|
|
|
TextEditor::ITextEditor *textEditor = qobject_cast<TextEditor::ITextEditor *>(editor);
|
2008-12-09 15:25:01 +01:00
|
|
|
QTC_ASSERT(textEditor, return);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
CppEditorSupport *editorSupport = new CppEditorSupport(this);
|
|
|
|
editorSupport->setTextEditor(textEditor);
|
|
|
|
m_editorSupport[textEditor] = editorSupport;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CppModelManager::editorAboutToClose(Core::IEditor *editor)
|
|
|
|
{
|
|
|
|
if (isCppEditor(editor)) {
|
|
|
|
TextEditor::ITextEditor *textEditor = qobject_cast<TextEditor::ITextEditor *>(editor);
|
2008-12-09 15:25:01 +01:00
|
|
|
QTC_ASSERT(textEditor, return);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
CppEditorSupport *editorSupport = m_editorSupport.value(textEditor);
|
|
|
|
m_editorSupport.remove(textEditor);
|
|
|
|
delete editorSupport;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CppModelManager::isCppEditor(Core::IEditor *editor) const
|
|
|
|
{
|
2010-06-25 17:37:59 +02:00
|
|
|
return editor->context().contains(ProjectExplorer::Constants::LANG_CXX);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CppModelManager::emitDocumentUpdated(Document::Ptr doc)
|
2010-06-25 17:37:59 +02:00
|
|
|
{
|
|
|
|
emit documentUpdated(doc);
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
void CppModelManager::onDocumentUpdated(Document::Ptr doc)
|
|
|
|
{
|
|
|
|
const QString fileName = doc->fileName();
|
2009-06-02 15:27:13 +02:00
|
|
|
|
2009-07-14 14:23:12 +02:00
|
|
|
bool outdated = false;
|
|
|
|
|
2009-06-02 15:27:13 +02:00
|
|
|
protectSnapshot.lock();
|
2009-07-14 14:23:12 +02:00
|
|
|
|
2009-12-07 10:54:27 +01:00
|
|
|
Document::Ptr previous = m_snapshot.document(fileName);
|
2009-07-14 14:23:12 +02:00
|
|
|
|
|
|
|
if (previous && (doc->revision() != 0 && doc->revision() < previous->revision()))
|
|
|
|
outdated = true;
|
|
|
|
else
|
|
|
|
m_snapshot.insert(doc);
|
|
|
|
|
2009-06-02 15:27:13 +02:00
|
|
|
protectSnapshot.unlock();
|
|
|
|
|
2009-07-14 14:23:12 +02:00
|
|
|
if (outdated)
|
|
|
|
return;
|
|
|
|
|
2011-08-31 09:58:40 +02:00
|
|
|
updateEditor(doc);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CppModelManager::onExtraDiagnosticsUpdated(const QString &fileName)
|
|
|
|
{
|
|
|
|
protectSnapshot.lock();
|
|
|
|
Document::Ptr doc = m_snapshot.document(fileName);
|
|
|
|
protectSnapshot.unlock();
|
|
|
|
if (doc)
|
|
|
|
updateEditor(doc);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CppModelManager::updateEditor(Document::Ptr doc)
|
|
|
|
{
|
|
|
|
const QString fileName = doc->fileName();
|
|
|
|
|
2012-01-24 15:36:40 +01:00
|
|
|
QList<Core::IEditor *> openedEditors = Core::ICore::editorManager()->openedEditors();
|
2008-12-02 12:01:29 +01:00
|
|
|
foreach (Core::IEditor *editor, openedEditors) {
|
2012-02-14 16:43:51 +01:00
|
|
|
if (editor->document()->fileName() == fileName) {
|
2008-12-02 12:01:29 +01:00
|
|
|
TextEditor::ITextEditor *textEditor = qobject_cast<TextEditor::ITextEditor *>(editor);
|
|
|
|
if (! textEditor)
|
|
|
|
continue;
|
|
|
|
|
2011-02-21 16:02:26 +01:00
|
|
|
TextEditor::BaseTextEditorWidget *ed = qobject_cast<TextEditor::BaseTextEditorWidget *>(textEditor->widget());
|
2008-12-02 12:01:29 +01:00
|
|
|
if (! ed)
|
|
|
|
continue;
|
|
|
|
|
2011-02-21 16:02:26 +01:00
|
|
|
QList<TextEditor::BaseTextEditorWidget::BlockRange> blockRanges;
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-02-23 12:49:56 +01:00
|
|
|
foreach (const Document::Block &block, doc->skippedBlocks()) {
|
2011-02-21 16:02:26 +01:00
|
|
|
blockRanges.append(TextEditor::BaseTextEditorWidget::BlockRange(block.begin(), block.end()));
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// set up the format for the errors
|
|
|
|
QTextCharFormat errorFormat;
|
|
|
|
errorFormat.setUnderlineStyle(QTextCharFormat::WaveUnderline);
|
|
|
|
errorFormat.setUnderlineColor(Qt::red);
|
|
|
|
|
|
|
|
// set up the format for the warnings.
|
|
|
|
QTextCharFormat warningFormat;
|
|
|
|
warningFormat.setUnderlineStyle(QTextCharFormat::WaveUnderline);
|
|
|
|
warningFormat.setUnderlineColor(Qt::darkYellow);
|
|
|
|
|
2009-02-10 22:56:04 +01:00
|
|
|
QList<Editor> todo;
|
2012-03-15 10:52:55 +01:00
|
|
|
foreach (const Editor &e, m_todo) {
|
2009-03-02 16:30:25 +01:00
|
|
|
if (e.textEditor != textEditor)
|
2009-02-10 22:56:04 +01:00
|
|
|
todo.append(e);
|
|
|
|
}
|
|
|
|
Editor e;
|
2012-03-15 10:52:55 +01:00
|
|
|
|
|
|
|
if (m_highlightingFactory->hightlighterHandlesDiagnostics()) {
|
|
|
|
e.updateSelections = false;
|
|
|
|
} else {
|
|
|
|
QSet<int> lines;
|
|
|
|
QList<Document::DiagnosticMessage> messages = doc->diagnosticMessages();
|
|
|
|
messages += extraDiagnostics(doc->fileName());
|
|
|
|
foreach (const Document::DiagnosticMessage &m, messages) {
|
|
|
|
if (m.fileName() != fileName)
|
|
|
|
continue;
|
|
|
|
else if (lines.contains(m.line()))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
lines.insert(m.line());
|
|
|
|
|
|
|
|
QTextEdit::ExtraSelection sel;
|
|
|
|
if (m.isWarning())
|
|
|
|
sel.format = warningFormat;
|
|
|
|
else
|
|
|
|
sel.format = errorFormat;
|
|
|
|
|
|
|
|
QTextCursor c(ed->document()->findBlockByNumber(m.line() - 1));
|
|
|
|
const QString text = c.block().text();
|
2010-10-18 17:45:49 +02:00
|
|
|
if (m.length() > 0 && m.column() + m.length() < (unsigned)text.size()) {
|
|
|
|
int column = m.column() > 0 ? m.column() - 1 : 0;
|
|
|
|
c.setPosition(c.position() + column);
|
|
|
|
c.movePosition(QTextCursor::NextCharacter, QTextCursor::KeepAnchor, m.length());
|
|
|
|
} else {
|
|
|
|
for (int i = 0; i < text.size(); ++i) {
|
|
|
|
if (! text.at(i).isSpace()) {
|
|
|
|
c.setPosition(c.position() + i);
|
|
|
|
break;
|
|
|
|
}
|
2012-03-15 10:52:55 +01:00
|
|
|
}
|
2010-10-18 17:45:49 +02:00
|
|
|
c.movePosition(QTextCursor::EndOfBlock, QTextCursor::KeepAnchor);
|
2012-03-15 10:52:55 +01:00
|
|
|
}
|
|
|
|
sel.cursor = c;
|
|
|
|
sel.format.setToolTip(m.text());
|
|
|
|
e.selections.append(sel);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-07-10 13:57:21 +02:00
|
|
|
e.revision = ed->document()->revision();
|
2009-03-02 16:30:25 +01:00
|
|
|
e.textEditor = textEditor;
|
2009-02-23 17:49:03 +01:00
|
|
|
e.ifdefedOutBlocks = blockRanges;
|
2009-02-10 22:56:04 +01:00
|
|
|
todo.append(e);
|
|
|
|
m_todo = todo;
|
|
|
|
postEditorUpdate();
|
2008-12-02 12:01:29 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-10 22:56:04 +01:00
|
|
|
void CppModelManager::postEditorUpdate()
|
|
|
|
{
|
|
|
|
m_updateEditorSelectionsTimer->start(500);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CppModelManager::updateEditorSelections()
|
|
|
|
{
|
2009-02-23 12:49:56 +01:00
|
|
|
foreach (const Editor &ed, m_todo) {
|
2009-03-02 16:30:25 +01:00
|
|
|
if (! ed.textEditor)
|
2009-02-10 22:56:04 +01:00
|
|
|
continue;
|
|
|
|
|
2009-03-02 16:30:25 +01:00
|
|
|
TextEditor::ITextEditor *textEditor = ed.textEditor;
|
2011-02-21 16:02:26 +01:00
|
|
|
TextEditor::BaseTextEditorWidget *editor = qobject_cast<TextEditor::BaseTextEditorWidget *>(textEditor->widget());
|
2009-07-10 13:57:21 +02:00
|
|
|
|
2009-03-02 16:30:25 +01:00
|
|
|
if (! editor)
|
|
|
|
continue;
|
2009-07-10 13:57:21 +02:00
|
|
|
else if (editor->document()->revision() != ed.revision)
|
|
|
|
continue; // outdated
|
2009-03-02 16:30:25 +01:00
|
|
|
|
2012-03-15 10:52:55 +01:00
|
|
|
if (ed.updateSelections)
|
|
|
|
editor->setExtraSelections(TextEditor::BaseTextEditorWidget::CodeWarningsSelection,
|
|
|
|
ed.selections);
|
2009-02-23 17:49:03 +01:00
|
|
|
|
2009-03-02 16:30:25 +01:00
|
|
|
editor->setIfdefedOutBlocks(ed.ifdefedOutBlocks);
|
2009-02-10 22:56:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
m_todo.clear();
|
2009-03-02 16:30:25 +01:00
|
|
|
|
2009-02-10 22:56:04 +01:00
|
|
|
}
|
|
|
|
|
2008-12-04 17:07:43 +01:00
|
|
|
void CppModelManager::onProjectAdded(ProjectExplorer::Project *)
|
|
|
|
{
|
2008-12-08 10:44:56 +01:00
|
|
|
QMutexLocker locker(&mutex);
|
2008-12-04 17:07:43 +01:00
|
|
|
m_dirty = true;
|
|
|
|
}
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
void CppModelManager::onAboutToRemoveProject(ProjectExplorer::Project *project)
|
|
|
|
{
|
2008-12-08 10:44:56 +01:00
|
|
|
do {
|
|
|
|
QMutexLocker locker(&mutex);
|
|
|
|
m_dirty = true;
|
|
|
|
m_projects.remove(project);
|
|
|
|
} while (0);
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
GC();
|
|
|
|
}
|
|
|
|
|
2009-08-13 17:10:53 +02:00
|
|
|
void CppModelManager::onAboutToUnloadSession()
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2012-01-24 15:36:40 +01:00
|
|
|
if (Core::ProgressManager *pm = Core::ICore::progressManager()) {
|
2012-11-21 22:36:47 +02:00
|
|
|
pm->cancelTasks(QLatin1String(CppTools::Constants::TASK_INDEX));
|
2008-12-04 17:07:43 +01:00
|
|
|
}
|
2008-12-08 10:44:56 +01:00
|
|
|
do {
|
|
|
|
QMutexLocker locker(&mutex);
|
|
|
|
m_projects.clear();
|
|
|
|
m_dirty = true;
|
|
|
|
} while (0);
|
|
|
|
|
|
|
|
GC();
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CppModelManager::GC()
|
|
|
|
{
|
2009-06-02 15:27:13 +02:00
|
|
|
protectSnapshot.lock();
|
2009-12-07 10:54:27 +01:00
|
|
|
Snapshot currentSnapshot = m_snapshot;
|
2009-06-02 15:27:13 +02:00
|
|
|
protectSnapshot.unlock();
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
QSet<QString> processed;
|
2008-12-04 17:07:43 +01:00
|
|
|
QStringList todo = projectFiles();
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
while (! todo.isEmpty()) {
|
|
|
|
QString fn = todo.last();
|
|
|
|
todo.removeLast();
|
|
|
|
|
|
|
|
if (processed.contains(fn))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
processed.insert(fn);
|
|
|
|
|
2009-12-07 10:54:27 +01:00
|
|
|
if (Document::Ptr doc = currentSnapshot.document(fn)) {
|
2008-12-02 12:01:29 +01:00
|
|
|
todo += doc->includedFiles();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList removedFiles;
|
2009-12-07 10:54:27 +01:00
|
|
|
|
|
|
|
Snapshot newSnapshot;
|
|
|
|
for (Snapshot::const_iterator it = currentSnapshot.begin(); it != currentSnapshot.end(); ++it) {
|
|
|
|
const QString fileName = it.key();
|
|
|
|
|
|
|
|
if (processed.contains(fileName))
|
|
|
|
newSnapshot.insert(it.value());
|
|
|
|
else
|
|
|
|
removedFiles.append(fileName);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
emit aboutToRemoveFiles(removedFiles);
|
2009-06-02 15:27:13 +02:00
|
|
|
|
|
|
|
protectSnapshot.lock();
|
2009-12-07 10:54:27 +01:00
|
|
|
m_snapshot = newSnapshot;
|
2009-06-02 15:27:13 +02:00
|
|
|
protectSnapshot.unlock();
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
2010-12-03 13:49:35 +01:00
|
|
|
|
2011-07-05 10:46:46 +02:00
|
|
|
void CppModelManager::finishedRefreshingSourceFiles(const QStringList &files)
|
|
|
|
{
|
|
|
|
emit sourceFilesRefreshed(files);
|
|
|
|
}
|
|
|
|
|
2012-02-07 15:09:08 +01:00
|
|
|
CppCompletionSupport *CppModelManager::completionSupport(Core::IEditor *editor) const
|
|
|
|
{
|
2012-02-20 12:39:08 +01:00
|
|
|
if (TextEditor::ITextEditor *textEditor = qobject_cast<TextEditor::ITextEditor *>(editor))
|
2012-02-21 10:00:32 +01:00
|
|
|
return m_completionAssistProvider->completionSupport(textEditor);
|
2012-02-07 15:09:08 +01:00
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-02-21 10:00:32 +01:00
|
|
|
void CppModelManager::setCppCompletionAssistProvider(CppCompletionAssistProvider *completionAssistProvider)
|
2012-02-20 12:39:08 +01:00
|
|
|
{
|
2012-06-18 11:34:15 +02:00
|
|
|
ExtensionSystem::PluginManager::removeObject(m_completionAssistProvider);
|
2012-02-21 10:00:32 +01:00
|
|
|
if (completionAssistProvider)
|
|
|
|
m_completionAssistProvider = completionAssistProvider;
|
2012-02-20 12:39:08 +01:00
|
|
|
else
|
2012-02-21 10:00:32 +01:00
|
|
|
m_completionAssistProvider = m_completionFallback;
|
2012-06-18 11:34:15 +02:00
|
|
|
ExtensionSystem::PluginManager::addObject(m_completionAssistProvider);
|
2012-02-20 12:39:08 +01:00
|
|
|
}
|
|
|
|
|
2012-02-07 15:09:08 +01:00
|
|
|
CppHighlightingSupport *CppModelManager::highlightingSupport(Core::IEditor *editor) const
|
|
|
|
{
|
2012-02-20 12:39:08 +01:00
|
|
|
if (TextEditor::ITextEditor *textEditor = qobject_cast<TextEditor::ITextEditor *>(editor))
|
|
|
|
return m_highlightingFactory->highlightingSupport(textEditor);
|
2012-02-07 15:09:08 +01:00
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
2011-08-24 10:55:48 +02:00
|
|
|
|
2012-02-20 12:39:08 +01:00
|
|
|
void CppModelManager::setHighlightingSupportFactory(CppHighlightingSupportFactory *highlightingFactory)
|
|
|
|
{
|
|
|
|
if (highlightingFactory)
|
|
|
|
m_highlightingFactory = highlightingFactory;
|
|
|
|
else
|
|
|
|
m_highlightingFactory = m_highlightingFallback;
|
|
|
|
}
|
|
|
|
|
2012-11-23 11:47:39 +01:00
|
|
|
void CppModelManager::setIndexingSupport(CppIndexingSupport *indexingSupport)
|
2012-10-16 16:02:40 +02:00
|
|
|
{
|
|
|
|
if (indexingSupport)
|
2012-11-23 11:47:39 +01:00
|
|
|
m_indexingSupporter = indexingSupport;
|
|
|
|
}
|
|
|
|
|
|
|
|
CppIndexingSupport *CppModelManager::indexingSupport()
|
|
|
|
{
|
|
|
|
return m_indexingSupporter ? m_indexingSupporter : m_internalIndexingSupport;
|
2012-10-16 16:02:40 +02:00
|
|
|
}
|
|
|
|
|
2011-08-24 10:55:48 +02:00
|
|
|
void CppModelManager::setExtraDiagnostics(const QString &fileName, int kind,
|
|
|
|
const QList<Document::DiagnosticMessage> &diagnostics)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&protectExtraDiagnostics);
|
2011-08-31 09:58:40 +02:00
|
|
|
if (m_extraDiagnostics[fileName][kind] == diagnostics)
|
|
|
|
return;
|
2011-08-24 10:55:48 +02:00
|
|
|
m_extraDiagnostics[fileName].insert(kind, diagnostics);
|
|
|
|
}
|
2011-08-31 09:58:40 +02:00
|
|
|
emit extraDiagnosticsUpdated(fileName);
|
2011-08-24 10:55:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
QList<Document::DiagnosticMessage> CppModelManager::extraDiagnostics(const QString &fileName, int kind) const
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&protectExtraDiagnostics);
|
|
|
|
if (kind == -1) {
|
|
|
|
QList<Document::DiagnosticMessage> messages;
|
|
|
|
foreach (const QList<Document::DiagnosticMessage> &list, m_extraDiagnostics.value(fileName))
|
|
|
|
messages += list;
|
|
|
|
return messages;
|
|
|
|
}
|
|
|
|
return m_extraDiagnostics.value(fileName).value(kind);
|
|
|
|
}
|