/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of Qt Creator. ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and 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. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ****************************************************************************/ #include "cppmodelmanager.h" #include "builtinindexingsupport.h" #include "cppcompletionassist.h" #include "cpphighlightingsupport.h" #include "cpphighlightingsupportinternal.h" #include "cppindexingsupport.h" #include "abstracteditorsupport.h" #include "cpptoolseditorsupport.h" #include "cppfindreferences.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace CppTools { uint qHash(const ProjectPart &p) { uint h = qHash(p.defines) ^ p.cVersion ^ p.cxxVersion ^ p.cxxExtensions ^ p.qtVersion; foreach (const QString &i, p.includePaths) h ^= qHash(i); foreach (const QString &f, p.frameworkPaths) h ^= qHash(f); return h; } bool operator==(const ProjectPart &p1, const ProjectPart &p2) { if (p1.defines != p2.defines) return false; if (p1.cVersion != p2.cVersion) return false; if (p1.cxxVersion != p2.cxxVersion) return false; if (p1.cxxExtensions != p2.cxxExtensions) return false; if (p1.qtVersion!= p2.qtVersion) return false; if (p1.includePaths != p2.includePaths) return false; return p1.frameworkPaths == p2.frameworkPaths; } } // namespace CppTools using namespace CppTools; using namespace CppTools::Internal; using namespace CPlusPlus; #if defined(QTCREATOR_WITH_DUMP_AST) && defined(Q_CC_GNU) #include 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; } }; #endif // QTCREATOR_WITH_DUMP_AST static const char pp_configuration[] = "# 1 \"\"\n" "#define __cplusplus 1\n" "#define __extension__\n" "#define __context__\n" "#define __range__\n" "#define restrict\n" "#define __restrict\n" "#define __restrict__\n" "#define __complex__\n" "#define __imag__\n" "#define __real__\n" "#define __builtin_va_arg(a,b) ((b)0)\n" // ### add macros for win32 "#define __cdecl\n" "#define __stdcall\n" "#define QT_WA(x) x\n" "#define CALLBACK\n" "#define STDMETHODCALLTYPE\n" "#define __RPC_FAR\n" "#define __declspec(a)\n" "#define STDMETHOD(method) virtual HRESULT STDMETHODCALLTYPE method\n" "#define __try try\n" "#define __except catch\n" "#define __finally\n" "#define __inline inline\n" "#define __forceinline inline\n"; CppPreprocessor::CppPreprocessor(QPointer modelManager, bool dumpFileNameWhileParsing) : m_snapshot(modelManager->snapshot()), m_modelManager(modelManager), m_dumpFileNameWhileParsing(dumpFileNameWhileParsing), m_preprocess(this, &m_env), m_revision(0) { m_preprocess.setKeepComments(true); } CppPreprocessor::~CppPreprocessor() { } void CppPreprocessor::setRevision(unsigned revision) { m_revision = revision; } void CppPreprocessor::setWorkingCopy(const CppModelManagerInterface::WorkingCopy &workingCopy) { m_workingCopy = workingCopy; } void CppPreprocessor::setIncludePaths(const QStringList &includePaths) { m_includePaths.clear(); for (int i = 0; i < includePaths.size(); ++i) { const QString &path = includePaths.at(i); 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; } } } m_includePaths.append(cleanPath(path)); } } void CppPreprocessor::setFrameworkPaths(const QStringList &frameworkPaths) { m_frameworkPaths.clear(); foreach (const QString &frameworkPath, frameworkPaths) { addFrameworkPath(frameworkPath); } } // Add the given framework path, and expand private frameworks. // // Example: // /ApplicationServices.framework // has private frameworks in: // /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. QString cleanFrameworkPath = cleanPath(frameworkPath); if (!m_frameworkPaths.contains(cleanFrameworkPath)) m_frameworkPaths.append(cleanFrameworkPath); const QDir frameworkDir(cleanFrameworkPath); 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()); } } void CppPreprocessor::setTodo(const QStringList &files) { m_todo = QSet::fromList(files); } namespace { class Process: public std::unary_function { QPointer _modelManager; Document::Ptr _doc; Document::CheckMode _mode; public: Process(QPointer modelManager, Document::Ptr doc, const CppModelManager::WorkingCopy &workingCopy) : _modelManager(modelManager), _doc(doc), _mode(Document::FastCheck) { if (workingCopy.contains(_doc->fileName())) _mode = Document::FullCheck; } void operator()() { _doc->check(_mode); if (_modelManager) _modelManager->emitDocumentUpdated(_doc); _doc->releaseSourceAndAST(); } }; } // end of anonymous namespace void CppPreprocessor::run(const QString &fileName) { sourceNeeded(0, fileName, IncludeGlobal); } void CppPreprocessor::removeFromCache(const QString &fileName) { m_snapshot.remove(fileName); } void CppPreprocessor::resetEnvironment() { m_env.reset(); m_processed.clear(); } void CppPreprocessor::getFileContents(const QString &absoluteFilePath, QString *contents, unsigned *revision) const { if (absoluteFilePath.isEmpty()) return; if (m_workingCopy.contains(absoluteFilePath)) { QPair entry = m_workingCopy.get(absoluteFilePath); if (contents) *contents = entry.first; if (revision) *revision = entry.second; return; } QFile file(absoluteFilePath); if (file.open(QFile::ReadOnly | QFile::Text)) { QTextCodec *defaultCodec = Core::EditorManager::instance()->defaultTextCodec(); QTextStream stream(&file); stream.setCodec(defaultCodec); if (contents) *contents = stream.readAll(); if (revision) *revision = 0; file.close(); } } bool CppPreprocessor::checkFile(const QString &absoluteFilePath) const { if (absoluteFilePath.isEmpty() || m_included.contains(absoluteFilePath)) return true; QFileInfo fileInfo(absoluteFilePath); return fileInfo.isFile() && fileInfo.isReadable(); } /// Resolve the given file name to its absolute path w.r.t. the include type. QString CppPreprocessor::resolveFile(const QString &fileName, IncludeType type) { if (type == IncludeGlobal) { QString fn = m_fileNameCache.value(fileName); if (! fn.isEmpty()) return fn; fn = resolveFile_helper(fileName, type); m_fileNameCache.insert(fileName, fn); return fn; } // IncludeLocal, IncludeNext return resolveFile_helper(fileName, type); } QString CppPreprocessor::cleanPath(const QString &path) { QString result = QDir::cleanPath(path); const QChar slash(QLatin1Char('/')); if (!result.endsWith(slash)) result.append(slash); return result; } QString CppPreprocessor::resolveFile_helper(const QString &fileName, IncludeType type) { QFileInfo fileInfo(fileName); if (fileName == Preprocessor::configurationFileName || fileInfo.isAbsolute()) return fileName; if (type == IncludeLocal && m_currentDoc) { QFileInfo currentFileInfo(m_currentDoc->fileName()); QString path = cleanPath(currentFileInfo.absolutePath()) + fileName; if (checkFile(path)) return path; } foreach (const QString &includePath, m_includePaths) { QString path = includePath + fileName; if (checkFile(path)) return path; } int index = fileName.indexOf(QLatin1Char('/')); if (index != -1) { QString frameworkName = fileName.left(index); QString name = frameworkName + QLatin1String(".framework/Headers/") + fileName.mid(index + 1); foreach (const QString &frameworkPath, m_frameworkPaths) { QString path = frameworkPath + name; if (checkFile(path)) return path; } } //qDebug() << "**** file" << fileName << "not found!"; return QString(); } void CppPreprocessor::macroAdded(const Macro ¯o) { if (! m_currentDoc) return; m_currentDoc->appendMacro(macro); } static inline const Macro revision(const CppModelManagerInterface::WorkingCopy &s, const Macro ¯o) { Macro newMacro(macro); newMacro.setFileRevision(s.get(macro.fileName()).second); return newMacro; } void CppPreprocessor::passedMacroDefinitionCheck(unsigned offset, unsigned line, const Macro ¯o) { if (! m_currentDoc) return; m_currentDoc->addMacroUse(revision(m_workingCopy, macro), offset, macro.name().length(), line, QVector()); } void CppPreprocessor::failedMacroDefinitionCheck(unsigned offset, const ByteArrayRef &name) { if (! m_currentDoc) return; m_currentDoc->addUndefinedMacroUse(QByteArray(name.start(), name.size()), offset); } void CppPreprocessor::notifyMacroReference(unsigned offset, unsigned line, const Macro ¯o) { if (! m_currentDoc) return; m_currentDoc->addMacroUse(revision(m_workingCopy, macro), offset, macro.name().length(), line, QVector()); } void CppPreprocessor::startExpandingMacro(unsigned offset, unsigned line, const Macro ¯o, const QVector &actuals) { if (! m_currentDoc) return; m_currentDoc->addMacroUse(revision(m_workingCopy, macro), offset, macro.name().length(), line, actuals); } void CppPreprocessor::stopExpandingMacro(unsigned, const Macro &) { if (! m_currentDoc) return; //qDebug() << "stop expanding:" << macro.name; } void CppPreprocessor::markAsIncludeGuard(const QByteArray ¯oName) { if (!m_currentDoc) return; m_currentDoc->setIncludeGuardMacroName(macroName); } void CppPreprocessor::mergeEnvironment(Document::Ptr doc) { if (! doc) return; const QString fn = doc->fileName(); if (m_processed.contains(fn)) return; m_processed.insert(fn); foreach (const Document::Include &incl, doc->includes()) { QString includedFile = incl.fileName(); if (Document::Ptr includedDoc = m_snapshot.document(includedFile)) mergeEnvironment(includedDoc); else run(includedFile); } m_env.addMacros(doc->definedMacros()); } 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); } void CppPreprocessor::sourceNeeded(unsigned line, const QString &fileName, IncludeType type) { if (fileName.isEmpty()) return; QString absoluteFileName = resolveFile(fileName, type); absoluteFileName = QDir::cleanPath(absoluteFileName); if (m_currentDoc && !absoluteFileName.isEmpty()) m_currentDoc->addIncludeFile(absoluteFileName, line); if (m_included.contains(absoluteFileName)) return; // we've already seen this file. m_included.insert(absoluteFileName); unsigned editorRevision = 0; QString contents; getFileContents(absoluteFileName, &contents, &editorRevision); if (m_currentDoc) { if (contents.isEmpty() && ! QFileInfo(absoluteFileName).isAbsolute()) { QString msg = QCoreApplication::translate( "CppPreprocessor", "%1: No such file or directory").arg(fileName); Document::DiagnosticMessage d(Document::DiagnosticMessage::Warning, m_currentDoc->fileName(), line, /*column = */ 0, msg); m_currentDoc->addDiagnosticMessage(d); //qWarning() << "file not found:" << fileName << m_currentDoc->fileName() << env.current_line; return; } } if (m_dumpFileNameWhileParsing) { qDebug() << "Parsing file:" << absoluteFileName // << "contents:" << contents.size() ; } Document::Ptr doc = m_snapshot.document(absoluteFileName); if (doc) { mergeEnvironment(doc); return; } doc = Document::create(absoluteFileName); doc->setRevision(m_revision); doc->setEditorRevision(editorRevision); QFileInfo info(absoluteFileName); if (info.exists()) doc->setLastModified(info.lastModified()); Document::Ptr previousDoc = switchDocument(doc); const QByteArray preprocessedCode = m_preprocess.run(absoluteFileName, contents); // { QByteArray b(preprocessedCode); b.replace("\n", "<<<\n"); qDebug("Preprocessed code for \"%s\": [[%s]]", fileName.toUtf8().constData(), b.constData()); } doc->setUtf8Source(preprocessedCode); doc->keepSourceAndAST(); doc->tokenize(); m_snapshot.insert(doc); m_todo.remove(absoluteFileName); Process process(m_modelManager, doc, m_workingCopy); process(); (void) switchDocument(previousDoc); } Document::Ptr CppPreprocessor::switchDocument(Document::Ptr doc) { Document::Ptr previousDoc = m_currentDoc; m_currentDoc = doc; return previousDoc; } void CppModelManager::updateModifiedSourceFiles() { 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); } /*! \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 modified within Qt Creator. */ 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; } CppModelManager::CppModelManager(QObject *parent) : CppModelManagerInterface(parent) , m_completionAssistProvider(0) , m_highlightingFactory(0) , m_indexingSupporter(0) { m_findReferences = new CppFindReferences(this); m_indexerEnabled = qgetenv("QTCREATOR_NO_CODE_INDEXER").isNull(); m_dirty = true; ProjectExplorer::ProjectExplorerPlugin *pe = ProjectExplorer::ProjectExplorerPlugin::instance(); QTC_ASSERT(pe, return); ProjectExplorer::SessionManager *session = pe->session(); m_updateEditorSelectionsTimer = new QTimer(this); m_updateEditorSelectionsTimer->setInterval(500); m_updateEditorSelectionsTimer->setSingleShot(true); connect(m_updateEditorSelectionsTimer, SIGNAL(timeout()), this, SLOT(updateEditorSelections())); connect(session, SIGNAL(projectAdded(ProjectExplorer::Project*)), this, SLOT(onProjectAdded(ProjectExplorer::Project*))); connect(session, SIGNAL(aboutToRemoveProject(ProjectExplorer::Project*)), this, SLOT(onAboutToRemoveProject(ProjectExplorer::Project*))); connect(session, SIGNAL(aboutToUnloadSession(QString)), this, SLOT(onAboutToUnloadSession())); qRegisterMetaType("CPlusPlus::Document::Ptr"); // thread connections connect(this, SIGNAL(documentUpdated(CPlusPlus::Document::Ptr)), this, SLOT(onDocumentUpdated(CPlusPlus::Document::Ptr))); connect(this, SIGNAL(extraDiagnosticsUpdated(QString)), this, SLOT(onExtraDiagnosticsUpdated(QString))); // Listen for editor closed and opened events so that we can keep track of changing files connect(Core::ICore::editorManager(), SIGNAL(editorOpened(Core::IEditor*)), this, SLOT(editorOpened(Core::IEditor*))); connect(Core::ICore::editorManager(), SIGNAL(editorAboutToClose(Core::IEditor*)), this, SLOT(editorAboutToClose(Core::IEditor*))); m_completionFallback = new InternalCompletionAssistProvider; m_completionAssistProvider = m_completionFallback; ExtensionSystem::PluginManager::addObject(m_completionAssistProvider); m_highlightingFallback = new CppHighlightingSupportInternalFactory; m_highlightingFactory = m_highlightingFallback; m_internalIndexingSupport = new BuiltinIndexingSupport; } CppModelManager::~CppModelManager() { ExtensionSystem::PluginManager::removeObject(m_completionAssistProvider); delete m_completionFallback; delete m_highlightingFallback; delete m_internalIndexingSupport; } Snapshot CppModelManager::snapshot() const { QMutexLocker locker(&m_protectSnapshot); return m_snapshot; } Document::Ptr CppModelManager::document(const QString &fileName) const { QMutexLocker locker(&m_protectSnapshot); return m_snapshot.document(fileName); } /// Replace the document in the snapshot. /// /// \returns true if successful, false if the new document is out-dated. bool CppModelManager::replaceDocument(Document::Ptr newDoc) { QMutexLocker locker(&m_protectSnapshot); Document::Ptr previous = m_snapshot.document(newDoc->fileName()); if (previous && (newDoc->revision() != 0 && newDoc->revision() < previous->revision())) // the new document is outdated return false; m_snapshot.insert(newDoc); return true; } void CppModelManager::ensureUpdated() { QMutexLocker locker(&m_mutex); if (! m_dirty) return; m_projectFiles = internalProjectFiles(); m_includePaths = internalIncludePaths(); m_frameworkPaths = internalFrameworkPaths(); m_definedMacros = internalDefinedMacros(); m_dirty = false; } QStringList CppModelManager::internalProjectFiles() const { QStringList files; QMapIterator it(m_projects); while (it.hasNext()) { it.next(); ProjectInfo pinfo = it.value(); foreach (const ProjectPart::Ptr &part, pinfo.projectParts()) { foreach (const ProjectFile &file, part->files) files += file.path; } } files.removeDuplicates(); return files; } QStringList CppModelManager::internalIncludePaths() const { QStringList includePaths; QMapIterator it(m_projects); while (it.hasNext()) { it.next(); ProjectInfo pinfo = it.value(); foreach (const ProjectPart::Ptr &part, pinfo.projectParts()) foreach (const QString &path, part->includePaths) includePaths.append(CppPreprocessor::cleanPath(path)); } includePaths.removeDuplicates(); return includePaths; } QStringList CppModelManager::internalFrameworkPaths() const { QStringList frameworkPaths; QMapIterator it(m_projects); while (it.hasNext()) { it.next(); ProjectInfo pinfo = it.value(); foreach (const ProjectPart::Ptr &part, pinfo.projectParts()) foreach (const QString &path, part->frameworkPaths) frameworkPaths.append(CppPreprocessor::cleanPath(path)); } frameworkPaths.removeDuplicates(); return frameworkPaths; } QByteArray CppModelManager::internalDefinedMacros() const { QByteArray macros; QSet alreadyIn; QMapIterator it(m_projects); while (it.hasNext()) { it.next(); ProjectInfo pinfo = it.value(); foreach (const ProjectPart::Ptr &part, pinfo.projectParts()) { const QList defs = part->defines.split('\n'); foreach (const QByteArray &def, defs) { if (!alreadyIn.contains(def)) { macros += def; macros.append('\n'); alreadyIn.insert(def); } } } } return macros; } /// This method will aquire the mutex! void CppModelManager::dumpModelManagerConfiguration() { // Tons of debug output... qDebug()<<"========= CppModelManager::dumpModelManagerConfiguration ======"; foreach (const ProjectInfo &pinfo, m_projects) { qDebug()<<" for project:"<< pinfo.project().data()->document()->fileName(); foreach (const ProjectPart::Ptr &part, pinfo.projectParts()) { qDebug() << "=== part ==="; const char* cVersion; switch (part->cVersion) { case ProjectPart::C89: cVersion = "C89"; break; case ProjectPart::C99: cVersion = "C99"; break; case ProjectPart::C11: cVersion = "C11"; break; default: cVersion = "INVALID"; } const char* cxxVersion; switch (part->cxxVersion) { case ProjectPart::CXX98: cxxVersion = "CXX98"; break; case ProjectPart::CXX11: cxxVersion = "CXX11"; break; default: cxxVersion = "INVALID"; } QStringList cxxExtensions; if (part->cxxExtensions & ProjectPart::GnuExtensions) cxxExtensions << QLatin1String("GnuExtensions"); if (part->cxxExtensions & ProjectPart::MicrosoftExtensions) cxxExtensions << QLatin1String("MicrosoftExtensions"); if (part->cxxExtensions & ProjectPart::BorlandExtensions) cxxExtensions << QLatin1String("BorlandExtensions"); if (part->cxxExtensions & ProjectPart::OpenMP) cxxExtensions << QLatin1String("OpenMP"); qDebug() << "cVersion:" << cVersion; qDebug() << "cxxVersion:" << cxxVersion; qDebug() << "cxxExtensions:" << cxxExtensions; qDebug() << "Qt version:" << part->qtVersion; qDebug() << "precompiled header:" << part->precompiledHeaders; qDebug() << "defines:" << part->defines; qDebug() << "includes:" << part->includePaths; qDebug() << "frameworkPaths:" << part->frameworkPaths; qDebug() << "files:" << part->files; 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 ======"; } void CppModelManager::addEditorSupport(AbstractEditorSupport *editorSupport) { m_addtionalEditorSupport.insert(editorSupport); } void CppModelManager::removeEditorSupport(AbstractEditorSupport *editorSupport) { m_addtionalEditorSupport.remove(editorSupport); } QList CppModelManager::references(CPlusPlus::Symbol *symbol, const LookupContext &context) { return m_findReferences->references(symbol, context); } void CppModelManager::findUsages(CPlusPlus::Symbol *symbol, const CPlusPlus::LookupContext &context) { if (symbol->identifier()) m_findReferences->findUsages(symbol, context); } void CppModelManager::renameUsages(CPlusPlus::Symbol *symbol, const CPlusPlus::LookupContext &context, const QString &replacement) { if (symbol->identifier()) m_findReferences->renameUsages(symbol, context, replacement); } void CppModelManager::findMacroUsages(const CPlusPlus::Macro ¯o) { m_findReferences->findMacroUses(macro); } void CppModelManager::renameMacroUsages(const CPlusPlus::Macro ¯o, const QString &replacement) { m_findReferences->renameMacroUses(macro, replacement); } void CppModelManager::replaceSnapshot(const CPlusPlus::Snapshot &newSnapshot) { QMutexLocker snapshotLocker(&m_protectSnapshot); m_snapshot = newSnapshot; } CppModelManager::WorkingCopy CppModelManager::buildWorkingCopyList() { WorkingCopy workingCopy; QMapIterator it(m_editorSupport); while (it.hasNext()) { it.next(); TextEditor::ITextEditor *textEditor = it.key(); CppEditorSupport *editorSupport = it.value(); QString fileName = textEditor->document()->fileName(); workingCopy.insert(fileName, editorSupport->contents(), editorSupport->editorRevision()); } QSetIterator jt(m_addtionalEditorSupport); while (jt.hasNext()) { AbstractEditorSupport *es = jt.next(); workingCopy.insert(es->fileName(), QString::fromUtf8(es->contents())); } // add the project configuration file QByteArray conf(pp_configuration); conf += definedMacros(); workingCopy.insert(configurationFileName(), QString::fromUtf8(conf)); return workingCopy; } CppModelManager::WorkingCopy CppModelManager::workingCopy() const { return const_cast(this)->buildWorkingCopyList(); } QFuture CppModelManager::updateSourceFiles(const QStringList &sourceFiles) { if (sourceFiles.isEmpty() || !m_indexerEnabled) return QFuture(); if (m_indexingSupporter) m_indexingSupporter->refreshSourceFiles(sourceFiles); return m_internalIndexingSupport->refreshSourceFiles(sourceFiles); } QList CppModelManager::projectInfos() const { QMutexLocker locker(&m_mutex); return m_projects.values(); } CppModelManager::ProjectInfo CppModelManager::projectInfo(ProjectExplorer::Project *project) const { QMutexLocker locker(&m_mutex); return m_projects.value(project, ProjectInfo(project)); } void CppModelManager::updateProjectInfo(const ProjectInfo &pinfo) { { // only hold the mutex for a limited scope, so the dumping afterwards can aquire it without deadlocking. QMutexLocker locker(&m_mutex); if (! pinfo.isValid()) return; ProjectExplorer::Project *project = pinfo.project().data(); m_projects.insert(project, pinfo); m_dirty = true; m_srcToProjectPart.clear(); foreach (const ProjectInfo &projectInfo, m_projects) { foreach (const ProjectPart::Ptr &projectPart, projectInfo.projectParts()) { foreach (const ProjectFile &cxxFile, projectPart->files) { m_srcToProjectPart[cxxFile.path].append(projectPart); foreach (const QString &fileName, m_snapshot.allIncludesForDocument(cxxFile.path)) m_snapshot.remove(fileName); m_snapshot.remove(cxxFile.path); } } } m_snapshot.remove(configurationFileName()); } if (!qgetenv("QTCREATOR_DUMP_PROJECT_INFO").isEmpty()) dumpModelManagerConfiguration(); emit projectPartsUpdated(pinfo.project().data()); } QList CppModelManager::projectPart(const QString &fileName) const { QList 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; } /*! \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(editor); QTC_ASSERT(textEditor, return); 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(editor); QTC_ASSERT(textEditor, return); CppEditorSupport *editorSupport = m_editorSupport.value(textEditor); m_editorSupport.remove(textEditor); delete editorSupport; } } bool CppModelManager::isCppEditor(Core::IEditor *editor) const { return editor->context().contains(ProjectExplorer::Constants::LANG_CXX); } void CppModelManager::emitDocumentUpdated(Document::Ptr doc) { emit documentUpdated(doc); } void CppModelManager::onDocumentUpdated(Document::Ptr doc) { if (replaceDocument(doc)) updateEditor(doc); } void CppModelManager::onExtraDiagnosticsUpdated(const QString &fileName) { if (Document::Ptr doc = document(fileName)) updateEditor(doc); } void CppModelManager::updateEditor(Document::Ptr doc) { const QString fileName = doc->fileName(); QList openedEditors = Core::ICore::editorManager()->openedEditors(); foreach (Core::IEditor *editor, openedEditors) { if (editor->document()->fileName() == fileName) { TextEditor::ITextEditor *textEditor = qobject_cast(editor); if (! textEditor) continue; TextEditor::BaseTextEditorWidget *ed = qobject_cast(textEditor->widget()); if (! ed) continue; QList blockRanges; foreach (const Document::Block &block, doc->skippedBlocks()) { blockRanges.append(TextEditor::BaseTextEditorWidget::BlockRange(block.begin(), block.end())); } // 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); QList todo; foreach (const Editor &e, m_todo) { if (e.textEditor != textEditor) todo.append(e); } Editor e; if (m_highlightingFactory->hightlighterHandlesDiagnostics()) { e.updateSelections = false; } else { QSet lines; QList 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(); 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; } } c.movePosition(QTextCursor::EndOfBlock, QTextCursor::KeepAnchor); } sel.cursor = c; sel.format.setToolTip(m.text()); e.selections.append(sel); } } e.revision = ed->document()->revision(); e.textEditor = textEditor; e.ifdefedOutBlocks = blockRanges; todo.append(e); m_todo = todo; postEditorUpdate(); break; } } } void CppModelManager::postEditorUpdate() { m_updateEditorSelectionsTimer->start(500); } void CppModelManager::updateEditorSelections() { foreach (const Editor &ed, m_todo) { if (! ed.textEditor) continue; TextEditor::ITextEditor *textEditor = ed.textEditor; TextEditor::BaseTextEditorWidget *editor = qobject_cast(textEditor->widget()); if (! editor) continue; else if (editor->document()->revision() != ed.revision) continue; // outdated if (ed.updateSelections) editor->setExtraSelections(TextEditor::BaseTextEditorWidget::CodeWarningsSelection, ed.selections); editor->setIfdefedOutBlocks(ed.ifdefedOutBlocks); } m_todo.clear(); } void CppModelManager::onProjectAdded(ProjectExplorer::Project *) { QMutexLocker locker(&m_mutex); m_dirty = true; } void CppModelManager::onAboutToRemoveProject(ProjectExplorer::Project *project) { do { QMutexLocker locker(&m_mutex); m_dirty = true; m_projects.remove(project); } while (0); GC(); } void CppModelManager::onAboutToUnloadSession() { if (Core::ProgressManager *pm = Core::ICore::progressManager()) pm->cancelTasks(QLatin1String(CppTools::Constants::TASK_INDEX)); do { QMutexLocker locker(&m_mutex); m_projects.clear(); m_dirty = true; } while (0); GC(); } void CppModelManager::GC() { Snapshot currentSnapshot = snapshot(); QSet processed; QStringList todo = projectFiles(); while (! todo.isEmpty()) { QString fn = todo.last(); todo.removeLast(); if (processed.contains(fn)) continue; processed.insert(fn); if (Document::Ptr doc = currentSnapshot.document(fn)) todo += doc->includedFiles(); } QStringList removedFiles; 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); } emit aboutToRemoveFiles(removedFiles); replaceSnapshot(newSnapshot); } void CppModelManager::finishedRefreshingSourceFiles(const QStringList &files) { emit sourceFilesRefreshed(files); } CppCompletionSupport *CppModelManager::completionSupport(Core::IEditor *editor) const { if (TextEditor::ITextEditor *textEditor = qobject_cast(editor)) return m_completionAssistProvider->completionSupport(textEditor); else return 0; } void CppModelManager::setCppCompletionAssistProvider(CppCompletionAssistProvider *completionAssistProvider) { ExtensionSystem::PluginManager::removeObject(m_completionAssistProvider); if (completionAssistProvider) m_completionAssistProvider = completionAssistProvider; else m_completionAssistProvider = m_completionFallback; ExtensionSystem::PluginManager::addObject(m_completionAssistProvider); } CppHighlightingSupport *CppModelManager::highlightingSupport(Core::IEditor *editor) const { if (TextEditor::ITextEditor *textEditor = qobject_cast(editor)) return m_highlightingFactory->highlightingSupport(textEditor); else return 0; } void CppModelManager::setHighlightingSupportFactory(CppHighlightingSupportFactory *highlightingFactory) { if (highlightingFactory) m_highlightingFactory = highlightingFactory; else m_highlightingFactory = m_highlightingFallback; } void CppModelManager::setIndexingSupport(CppIndexingSupport *indexingSupport) { if (indexingSupport) m_indexingSupporter = indexingSupport; } CppIndexingSupport *CppModelManager::indexingSupport() { return m_indexingSupporter ? m_indexingSupporter : m_internalIndexingSupport; } void CppModelManager::setExtraDiagnostics(const QString &fileName, int kind, const QList &diagnostics) { { QMutexLocker locker(&m_protectExtraDiagnostics); if (m_extraDiagnostics[fileName][kind] == diagnostics) return; m_extraDiagnostics[fileName].insert(kind, diagnostics); } emit extraDiagnosticsUpdated(fileName); } QList CppModelManager::extraDiagnostics(const QString &fileName, int kind) const { QMutexLocker locker(&m_protectExtraDiagnostics); if (kind == -1) { QList messages; foreach (const QList &list, m_extraDiagnostics.value(fileName)) messages += list; return messages; } return m_extraDiagnostics.value(fileName).value(kind); }