diff options
author | Erik Verbruggen <erik.verbruggen@digia.com> | 2013-12-10 14:37:32 +0100 |
---|---|---|
committer | hjk <hjk121@nokiamail.com> | 2013-12-20 17:05:09 +0100 |
commit | 5beb74fd9d11b31b360e0a336e269b81cbca1f5a (patch) | |
tree | 6b1bad7c9798e29c694e216f1196e0f47b77ceff /src/plugins/clangcodemodel/dependencygraph.cpp | |
parent | 93b7528431857d67aa0ffdc60f835d987aa7b101 (diff) | |
download | qt-creator-5beb74fd9d11b31b360e0a336e269b81cbca1f5a.tar.gz |
Add experimental clang code-model plug-in.
Previously known as the wip/clang branch.
Contributors (in alphabetical order):
- Christian Kamm <christian.d.kamm@nokia.com>
- Erik Verbruggen <erik.verbruggen@digia.com>
- Leandro Melo <leandro.melo@nokia.com>
- Peter Kuemmel <syntheticpp@gmx.net>
- Sergey Shambir <sergey.shambir.auto@gmail.com>
Change-Id: I4c3ff600a19b6732641c1d5ef28236bf2cc17737
Reviewed-by: hjk <hjk121@nokiamail.com>
Diffstat (limited to 'src/plugins/clangcodemodel/dependencygraph.cpp')
-rw-r--r-- | src/plugins/clangcodemodel/dependencygraph.cpp | 208 |
1 files changed, 208 insertions, 0 deletions
diff --git a/src/plugins/clangcodemodel/dependencygraph.cpp b/src/plugins/clangcodemodel/dependencygraph.cpp new file mode 100644 index 0000000000..f0aadd6692 --- /dev/null +++ b/src/plugins/clangcodemodel/dependencygraph.cpp @@ -0,0 +1,208 @@ +/**************************************************************************** +** +** 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 "dependencygraph.h" + +#include <QtCore/QtConcurrentRun> + +using namespace ClangCodeModel; +using namespace Internal; + +DependencyGraph::DependencyGraph() +{ + m_includeTracker.setResolutionMode(IncludeTracker::EveryMatchResolution); +} + +DependencyGraph::~DependencyGraph() +{ + discard(); +} + +void DependencyGraph::cancel() +{ + if (m_computeWatcher.isRunning()) { + m_computeWatcher.cancel(); + m_computeWatcher.waitForFinished(); + } +} + +void DependencyGraph::addFile(const QString &fileName, const QStringList &compilationOptions) +{ + cancel(); + + m_files.append(qMakePair(fileName, compilationOptions)); +} + +QFuture<void> DependencyGraph::compute() +{ + QFuture<void> future = QtConcurrent::run(this, &DependencyGraph::computeCore); + m_computeWatcher.setFuture(future); + return future; +} + +void DependencyGraph::computeCore() +{ + for (int i = 0; i < m_files.size(); ++i) { + if (m_computeWatcher.isCanceled()) + break; + + const QPair<QString, QStringList> &p = m_files.at(i); + const QString ¤tFile = p.first; + const QStringList &options = p.second; + const QPair<bool, NodeRefSetIt> &v = findVertex(currentFile); + if (!v.first) + processIncludes(insertVertex(currentFile), options); + } + + emit dependencyGraphAvailable(); +} + +void DependencyGraph::processIncludes(NodeRefSetIt currentIt, + const QStringList &compilationOptions) +{ + const QString ¤tFile = currentIt.key(); + const QStringList &includes = m_includeTracker.directIncludes(currentFile, compilationOptions); + foreach (const QString &include, includes) { + if (m_computeWatcher.isCanceled()) + return; + + QPair<bool, NodeRefSetIt> v = findVertex(include); + if (!v.first) { + v.second = insertVertex(include); + processIncludes(v.second, compilationOptions); + } + insertEdge(currentIt, v.second); + } +} + +namespace { + +struct SimpleVisitor +{ + bool acceptFile(const QString &fileName) + { + m_allFiles.append(fileName); + return false; + } + + QStringList m_allFiles; +}; + +} + +QStringList DependencyGraph::collectDependencies(const QString &referenceFile, + DependencyRole role) const +{ + SimpleVisitor visitor; + collectDependencies(referenceFile, role, &visitor); + + return visitor.m_allFiles; +} + +bool DependencyGraph::hasDependency(const QString &referenceFile, DependencyRole role) const +{ + QPair<bool, NodeRefSetIt> v = findVertex(referenceFile); + if (!v.first) + return false; + + NodeListIt nodeIt = v.second.value(); + + if (role == FilesDirectlyIncludedBy || role == FilesIncludedBy) + return nodeIt->m_out != 0; + + return nodeIt->m_in != 0; +} + +void DependencyGraph::discard() +{ + cancel(); + + for (NodeListIt it = m_nodes.begin(); it != m_nodes.end(); ++it) { + deleteAdjacencies(it->m_out); + deleteAdjacencies(it->m_in); + } + m_nodes.clear(); + m_nodesRefs.clear(); + m_files.clear(); +} + + +DependencyGraph::Node::Node(const QString &fileName) + : m_fileName(fileName) + , m_out(0) + , m_in(0) +{} + +DependencyGraph::AdjacencyNode::AdjacencyNode(NodeListIt it) + : m_next(0) + , m_nodeIt(it) +{} + +QPair<bool, DependencyGraph::NodeRefSetIt> DependencyGraph::findVertex(const QString &s) const +{ + bool found = false; + NodeRefSetIt it = const_cast<NodeRefSet &>(m_nodesRefs).find(s); + if (it != m_nodesRefs.end()) + found = true; + return qMakePair(found, it); +} + +DependencyGraph::NodeRefSetIt DependencyGraph::insertVertex(const QString &s) +{ + Q_ASSERT(m_nodesRefs.find(s) == m_nodesRefs.end()); + + m_nodes.append(Node(s)); + return m_nodesRefs.insert(s, m_nodes.end() - 1); +} + +void DependencyGraph::insertEdge(DependencyGraph::NodeRefSetIt fromIt, + DependencyGraph::NodeRefSetIt toIt) +{ + NodeListIt nodeFromIt = fromIt.value(); + NodeListIt nodeToIt = toIt.value(); + + createAdjacency(&nodeFromIt->m_out, new AdjacencyNode(nodeToIt)); + createAdjacency(&nodeToIt->m_in, new AdjacencyNode(nodeFromIt)); +} + +void DependencyGraph::deleteAdjacencies(AdjacencyNode *node) +{ + while (node) { + AdjacencyNode *next = node->m_next; + delete node; + node = next; + } +} + +void DependencyGraph::createAdjacency(AdjacencyNode **node, AdjacencyNode *newNode) +{ + if (*node) + newNode->m_next = *node; + *node = newNode; +} |