summaryrefslogtreecommitdiff
path: root/src/plugins/cpptools/builtinindexingsupport.cpp
diff options
context:
space:
mode:
authorNikolai Kosjar <nikolai.kosjar@digia.com>2014-08-05 11:34:52 +0200
committerNikolai Kosjar <nikolai.kosjar@digia.com>2014-08-07 11:04:53 +0200
commit0f3032f8403de34692edd5c12ed28b2ddfec777f (patch)
treec788d313386608baf2f0bb0c0a20450e626b5f68 /src/plugins/cpptools/builtinindexingsupport.cpp
parent8bf4a5b089a26474099f576e285026b7dcc47a7b (diff)
downloadqt-creator-0f3032f8403de34692edd5c12ed28b2ddfec777f.tar.gz
CppTools: Tests: Add "Find Errors Indexing" mode
Adding QTC_FIND_ERRORS_INDEXING=1 to the run environment will Creator force to do some "extended indexing" in order to find bugs: 1) The project files are parsed as if they would be opened in an editor. That is, parsing happens as precisely as possible, based on the ProjectPart information. 2) Symbols are looked up by invoking CheckSymbols, the backend of the semantic highlighter. 3) A "Task List File" (*.tasks) will be written with all diagnostic messages. This special indexing mode is meant for testing purposes. E.g. it can be run nightly on some bigger projects to find regressions. Note that Qt Creator will quit exactly after the first time some source files are indexed. E.g. that will happen if you open a file manually or if you open a new unconfigured project. Therefore it's required to configure projects as needed before invoking in this indexing mode. Change-Id: If25b83e67d24df9e28e107cb062f21cbf3b4c643 Reviewed-by: Erik Verbruggen <erik.verbruggen@digia.com>
Diffstat (limited to 'src/plugins/cpptools/builtinindexingsupport.cpp')
-rw-r--r--src/plugins/cpptools/builtinindexingsupport.cpp162
1 files changed, 148 insertions, 14 deletions
diff --git a/src/plugins/cpptools/builtinindexingsupport.cpp b/src/plugins/cpptools/builtinindexingsupport.cpp
index ad0d1f4211..60874352c9 100644
--- a/src/plugins/cpptools/builtinindexingsupport.cpp
+++ b/src/plugins/cpptools/builtinindexingsupport.cpp
@@ -1,23 +1,29 @@
#include "builtinindexingsupport.h"
+#include "cppchecksymbols.h"
#include "cppmodelmanager.h"
+#include "cppprojectfile.h"
+#include "cppsnapshotupdater.h"
#include "cppsourceprocessor.h"
-#include "searchsymbols.h"
#include "cpptoolsconstants.h"
#include "cpptoolsplugin.h"
-#include "cppprojectfile.h"
+#include "searchsymbols.h"
#include <coreplugin/icore.h>
#include <coreplugin/progressmanager/progressmanager.h>
+#include <cplusplus/LookupContext.h>
+#include <utils/qtcassert.h>
#include <utils/runextensions.h>
#include <QCoreApplication>
+#include <QElapsedTimer>
using namespace CppTools;
using namespace CppTools::Internal;
static const bool DumpFileNameWhileParsing = qgetenv("QTC_DUMP_FILENAME_WHILE_PARSING") == "1";
+static const bool FindErrorsIndexing = qgetenv("QTC_FIND_ERRORS_INDEXING") == "1";
namespace {
@@ -36,11 +42,125 @@ public:
QStringList sourceFiles;
};
-static void parse(QFutureInterface<void> &future, const ParseParams params)
+class WriteTaskFileForDiagnostics
{
- if (params.sourceFiles.isEmpty())
- return;
+ Q_DISABLE_COPY(WriteTaskFileForDiagnostics)
+
+public:
+ WriteTaskFileForDiagnostics()
+ : m_processedDiagnostics(0)
+ {
+ const QString fileName = QDir::tempPath()
+ + QLatin1String("/qtc_findErrorsIndexing.diagnostics.")
+ + QDateTime::currentDateTime().toString(QLatin1String("yyMMdd_HHmm"))
+ + QLatin1String(".tasks");
+
+ m_file.setFileName(fileName);
+ Q_ASSERT(m_file.open(QIODevice::WriteOnly | QIODevice::Text));
+ m_out.setDevice(&m_file);
+
+ qDebug("FindErrorsIndexing: Task file for diagnostics is \"%s\".",
+ qPrintable(m_file.fileName()));
+ }
+
+ ~WriteTaskFileForDiagnostics()
+ {
+ qDebug("FindErrorsIndexing: %d diagnostic messages written to \"%s\".",
+ m_processedDiagnostics, qPrintable(m_file.fileName()));
+ }
+
+ int processedDiagnostics() const { return m_processedDiagnostics; }
+
+ void process(const CPlusPlus::Document::Ptr document)
+ {
+ using namespace CPlusPlus;
+ const QString fileName = document->fileName();
+
+ foreach (const Document::DiagnosticMessage &message, document->diagnosticMessages()) {
+ ++m_processedDiagnostics;
+
+ QString type;
+ switch (message.level()) {
+ case Document::DiagnosticMessage::Warning:
+ type = QLatin1String("warn"); break;
+ case Document::DiagnosticMessage::Error:
+ case Document::DiagnosticMessage::Fatal:
+ type = QLatin1String("err"); break;
+ default:
+ break;
+ }
+
+ // format: file\tline\ttype\tdescription
+ m_out << fileName << "\t"
+ << message.line() << "\t"
+ << type << "\t"
+ << message.text() << "\n";
+ }
+ }
+
+private:
+ QFile m_file;
+ QTextStream m_out;
+ int m_processedDiagnostics;
+};
+
+void classifyFiles(const QStringList &files, QStringList *headers, QStringList *sources)
+{
+ foreach (const QString &file, files) {
+ if (ProjectFile::isSource(ProjectFile::classify(file)))
+ sources->append(file);
+ else
+ headers->append(file);
+ }
+}
+
+void indexFindErrors(QFutureInterface<void> &future, const ParseParams params)
+{
+ QStringList files = params.sourceFiles;
+ files.sort();
+ QStringList sources, headers;
+ classifyFiles(files, &headers, &sources);
+ files = sources + headers;
+
+ WriteTaskFileForDiagnostics taskFileWriter;
+ QElapsedTimer timer;
+ timer.start();
+
+ for (int i = 0, end = files.size(); i < end ; ++i) {
+ if (future.isPaused())
+ future.waitForResume();
+ if (future.isCanceled())
+ break;
+
+ const QString file = files.at(i);
+ qDebug("FindErrorsIndexing: \"%s\"", qPrintable(file));
+
+ // Parse the file as precisely as possible
+ SnapshotUpdater updater(file);
+ updater.setReleaseSourceAndAST(false);
+ updater.update(params.workingCopy);
+ CPlusPlus::Document::Ptr document = updater.document();
+ QTC_ASSERT(document, return);
+ // Write diagnostic messages
+ taskFileWriter.process(document);
+
+ // Look up symbols
+ CPlusPlus::LookupContext context(document, updater.snapshot());
+ CheckSymbols::go(document, context, QList<CheckSymbols::Result>()).waitForFinished();
+
+ document->releaseSourceAndAST();
+
+ future.setProgressValue(files.size() - (files.size() - (i + 1)));
+ }
+
+ const QTime format = QTime(0, 0, 0, 0).addMSecs(timer.elapsed() + 500);
+ const QString time = format.toString(QLatin1String("hh:mm:ss"));
+ qDebug("FindErrorsIndexing: Finished after %s.", qPrintable(time));
+}
+
+void index(QFutureInterface<void> &future, const ParseParams params)
+{
QScopedPointer<CppSourceProcessor> sourceProcessor(CppModelManager::createSourceProcessor());
sourceProcessor->setDumpFileNameWhileParsing(params.dumpFileNameWhileParsing);
sourceProcessor->setRevision(params.revision);
@@ -48,16 +168,13 @@ static void parse(QFutureInterface<void> &future, const ParseParams params)
sourceProcessor->setWorkingCopy(params.workingCopy);
QStringList files = params.sourceFiles;
+
QStringList sources;
QStringList headers;
+ classifyFiles(files, &headers, &sources);
- foreach (const QString &file, files) {
+ foreach (const QString &file, files)
sourceProcessor->removeFromCache(file);
- if (ProjectFile::isSource(ProjectFile::classify(file)))
- sources.append(file);
- else
- headers.append(file);
- }
const int sourceCount = sources.size();
files = sources;
@@ -65,8 +182,6 @@ static void parse(QFutureInterface<void> &future, const ParseParams params)
sourceProcessor->setTodo(files);
- future.setProgressRange(0, files.size());
-
const QString conf = CppModelManagerInterface::configurationFileName();
bool processingHeaders = false;
@@ -102,9 +217,23 @@ static void parse(QFutureInterface<void> &future, const ParseParams params)
if (isSourceFile)
sourceProcessor->resetEnvironment();
}
+}
+
+void parse(QFutureInterface<void> &future, const ParseParams params)
+{
+ const QStringList files = params.sourceFiles;
+ if (files.isEmpty())
+ return;
+
+ future.setProgressRange(0, files.size());
+
+ if (FindErrorsIndexing)
+ indexFindErrors(future, params);
+ else
+ index(future, params);
future.setProgressValue(files.size());
- cmm->finishedRefreshingSourceFiles(files);
+ CppModelManager::instance()->finishedRefreshingSourceFiles(files);
}
class BuiltinSymbolSearcher: public SymbolSearcher
@@ -236,3 +365,8 @@ SymbolSearcher *BuiltinIndexingSupport::createSymbolSearcher(SymbolSearcher::Par
{
return new BuiltinSymbolSearcher(CppModelManager::instance()->snapshot(), parameters, fileNames);
}
+
+bool BuiltinIndexingSupport::isFindErrorsIndexingActive()
+{
+ return FindErrorsIndexing;
+}