clangeditordocumentprocessor.cpp 10.6 KB
Newer Older
1 2
/****************************************************************************
**
Eike Ziller's avatar
Eike Ziller committed
3 4
** Copyright (C) 2015 The Qt Company Ltd.
** Contact: http://www.qt.io/licensing
5 6 7 8 9 10 11
**
** 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
Eike Ziller's avatar
Eike Ziller committed
12 13
** a written agreement between you and The Qt Company.  For licensing terms and
** conditions see http://www.qt.io/terms-conditions.  For further information
Tobias Hunger's avatar
Tobias Hunger committed
14
** use the contact form at http://www.qt.io/contact-us.
15 16 17
**
** GNU Lesser General Public License Usage
** Alternatively, this file may be used under the terms of the GNU Lesser
Tobias Hunger's avatar
Tobias Hunger committed
18 19 20 21 22 23
** General Public License version 2.1 or version 3 as published by the Free
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
** LICENSE.LGPLv3 included in the packaging of this file.  Please review the
** following information to ensure the GNU Lesser General Public License
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
24
**
Eike Ziller's avatar
Eike Ziller committed
25 26
** In addition, as a special exception, The Qt Company gives you certain additional
** rights.  These rights are described in The Qt Company LGPL Exception
27 28 29 30 31 32
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
****************************************************************************/

#include "clangeditordocumentprocessor.h"

33 34
#include "clangfixitoperation.h"
#include "clangfixitoperationsextractor.h"
35 36
#include "clangmodelmanagersupport.h"
#include "clangutils.h"
37 38 39 40
#include "cppcreatemarkers.h"
#include "diagnostic.h"
#include "pchinfo.h"

Marco Bubke's avatar
Marco Bubke committed
41 42 43
#include <diagnosticcontainer.h>
#include <sourcelocationcontainer.h>

44 45 46
#include <cpptools/cpptoolsplugin.h>
#include <cpptools/cppworkingcopy.h>

47
#include <texteditor/convenience.h>
48
#include <texteditor/fontsettings.h>
Marco Bubke's avatar
Marco Bubke committed
49
#include <texteditor/texteditor.h>
50 51
#include <texteditor/texteditorconstants.h>
#include <texteditor/texteditorsettings.h>
Marco Bubke's avatar
Marco Bubke committed
52

53 54 55 56 57
#include <cplusplus/CppDocument.h>

#include <utils/qtcassert.h>
#include <utils/QtConcurrentTools>

Marco Bubke's avatar
Marco Bubke committed
58 59
#include <QTextBlock>

60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76
namespace {

typedef CPlusPlus::Document::DiagnosticMessage CppToolsDiagnostic;

QList<TextEditor::BlockRange> toTextEditorBlocks(
        const QList<ClangCodeModel::SemanticMarker::Range> &ranges)
{
    QList<TextEditor::BlockRange> result;
    result.reserve(ranges.size());
    foreach (const ClangCodeModel::SemanticMarker::Range &range, ranges)
        result.append(TextEditor::BlockRange(range.first, range.last));
    return result;
}

} // anonymous namespace

namespace ClangCodeModel {
77
namespace Internal {
78

79 80 81
ClangEditorDocumentProcessor::ClangEditorDocumentProcessor(
        ModelManagerSupportClang *modelManagerSupport,
        TextEditor::TextDocument *document)
82
    : BaseEditorDocumentProcessor(document)
83
    , m_diagnosticManager(document)
84
    , m_modelManagerSupport(modelManagerSupport)
85
    , m_parser(new ClangEditorDocumentParser(document->filePath().toString()))
86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101
    , m_parserRevision(0)
    , m_semanticHighlighter(document)
    , m_builtinProcessor(document, /*enableSemanticHighlighter=*/ false)
{
    // Forwarding the semantic info from the builtin processor enables us to provide all
    // editor (widget) related features that are not yet implemented by the clang plugin.
    connect(&m_builtinProcessor, &CppTools::BuiltinEditorDocumentProcessor::cppDocumentUpdated,
            this, &ClangEditorDocumentProcessor::cppDocumentUpdated);
    connect(&m_builtinProcessor, &CppTools::BuiltinEditorDocumentProcessor::semanticInfoUpdated,
            this, &ClangEditorDocumentProcessor::semanticInfoUpdated);

    m_semanticHighlighter.setHighlightingRunner(
        [this]() -> QFuture<TextEditor::HighlightingResult> {
            const int firstLine = 1;
            const int lastLine = baseTextDocument()->document()->blockCount();

102
            CreateMarkers *createMarkers = CreateMarkers::create(m_parser->semanticMarker(),
103
                                                                 baseTextDocument()->filePath().toString(),
104 105 106 107 108 109 110 111 112
                                                                 firstLine, lastLine);
            return createMarkers->start();
        });
}

ClangEditorDocumentProcessor::~ClangEditorDocumentProcessor()
{
    m_parserWatcher.cancel();
    m_parserWatcher.waitForFinished();
113

114 115
    if (m_projectPart) {
        QTC_ASSERT(m_modelManagerSupport, return);
116
        m_modelManagerSupport->ipcCommunicator().unregisterTranslationUnitsForEditor(
117 118
            {ClangBackEnd::FileContainer(filePath(), m_projectPart->id())});
    }
119 120 121 122
}

void ClangEditorDocumentProcessor::run()
{
Marco Bubke's avatar
Marco Bubke committed
123 124
    requestDiagnostics();

125 126 127 128 129 130 131 132 133
    // Run clang parser
    disconnect(&m_parserWatcher, &QFutureWatcher<void>::finished,
               this, &ClangEditorDocumentProcessor::onParserFinished);
    m_parserWatcher.cancel();
    m_parserWatcher.setFuture(QFuture<void>());

    m_parserRevision = revision();
    connect(&m_parserWatcher, &QFutureWatcher<void>::finished,
            this, &ClangEditorDocumentProcessor::onParserFinished);
134 135 136
    const QFuture<void> future = QtConcurrent::run(&runParser,
                                                   parser(),
                                                   ClangEditorDocumentParser::InMemoryInfo(true));
137 138 139 140 141 142
    m_parserWatcher.setFuture(future);

    // Run builtin processor
    m_builtinProcessor.run();
}

143
void ClangEditorDocumentProcessor::recalculateSemanticInfoDetached(bool force)
144
{
145
    m_builtinProcessor.recalculateSemanticInfoDetached(force);
146 147
}

148 149 150 151 152 153
void ClangEditorDocumentProcessor::semanticRehighlight()
{
    m_semanticHighlighter.updateFormatMapFromFontSettings();
    m_semanticHighlighter.run();
}

154 155 156 157 158
CppTools::SemanticInfo ClangEditorDocumentProcessor::recalculateSemanticInfo()
{
    return m_builtinProcessor.recalculateSemanticInfo();
}

159
CppTools::BaseEditorDocumentParser::Ptr ClangEditorDocumentProcessor::parser()
160
{
161
    return m_parser;
162 163
}

164 165 166 167 168
CPlusPlus::Snapshot ClangEditorDocumentProcessor::snapshot()
{
   return m_builtinProcessor.snapshot();
}

169 170 171 172 173
bool ClangEditorDocumentProcessor::isParserRunning() const
{
    return m_parserWatcher.isRunning();
}

174 175 176 177 178
bool ClangEditorDocumentProcessor::hasProjectPart() const
{
    return m_projectPart;
}

179 180 181 182 183
CppTools::ProjectPart::Ptr ClangEditorDocumentProcessor::projectPart() const
{
    return m_projectPart;
}

184 185 186 187 188
void ClangEditorDocumentProcessor::clearProjectPart()
{
    m_projectPart.clear();
}

Marco Bubke's avatar
Marco Bubke committed
189 190 191 192
void ClangEditorDocumentProcessor::updateCodeWarnings(const QVector<ClangBackEnd::DiagnosticContainer> &diagnostics,
                                                      uint documentRevision)
{
    if (documentRevision == revision()) {
193 194
        m_diagnosticManager.processNewDiagnostics(diagnostics);
        const auto codeWarnings = m_diagnosticManager.takeExtraSelections();
Marco Bubke's avatar
Marco Bubke committed
195 196 197 198
        emit codeWarningsUpdated(revision(), codeWarnings);
    }
}

199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216
static int currentLine(const TextEditor::AssistInterface &assistInterface)
{
    int line, column;
    TextEditor::Convenience::convertPosition(assistInterface.textDocument(),
                                             assistInterface.position(),
                                             &line,
                                             &column);
    return line;
}

TextEditor::QuickFixOperations ClangEditorDocumentProcessor::extraRefactoringOperations(
        const TextEditor::AssistInterface &assistInterface)
{
    ClangFixItOperationsExtractor extractor(m_diagnosticManager.diagnosticsWithFixIts());

    return extractor.extract(assistInterface.fileName(), currentLine(assistInterface));
}

217 218 219 220 221
ClangBackEnd::FileContainer ClangEditorDocumentProcessor::fileContainer() const
{
    return fileContainer(m_projectPart.data());
}

222 223 224 225 226
ClangEditorDocumentProcessor *ClangEditorDocumentProcessor::get(const QString &filePath)
{
    return qobject_cast<ClangEditorDocumentProcessor *>(BaseEditorDocumentProcessor::get(filePath));
}

227 228 229 230 231 232
static bool isProjectPartLoadedOrIsFallback(CppTools::ProjectPart::Ptr projectPart)
{
    return projectPart
        && (projectPart->id().isEmpty() || ClangCodeModel::Utils::isProjectPartValid(projectPart));
}

233
void ClangEditorDocumentProcessor::updateProjectPartAndTranslationUnitForEditor()
234
{
235
    const CppTools::ProjectPart::Ptr projectPart = m_parser->projectPart();
236

237
    if (isProjectPartLoadedOrIsFallback(projectPart)) {
238 239
        updateTranslationUnitForEditor(projectPart.data());
        requestDiagnostics(projectPart.data());
Marco Bubke's avatar
Marco Bubke committed
240

241 242
        m_projectPart = projectPart;
    }
243 244
}

245 246 247 248 249 250
void ClangEditorDocumentProcessor::onParserFinished()
{
    if (revision() != m_parserRevision)
        return;

    // Emit ifdefed out blocks
251
    const auto ifdefoutBlocks = toTextEditorBlocks(m_parser->ifdefedOutBlocks());
252 253 254 255
    emit ifdefedOutBlocksUpdated(revision(), ifdefoutBlocks);

    // Run semantic highlighter
    m_semanticHighlighter.run();
256

257
    updateProjectPartAndTranslationUnitForEditor();
258 259
}

260
void ClangEditorDocumentProcessor::updateTranslationUnitForEditor(CppTools::ProjectPart *projectPart)
261 262 263 264 265
{
    QTC_ASSERT(m_modelManagerSupport, return);
    IpcCommunicator &ipcCommunicator = m_modelManagerSupport->ipcCommunicator();

    if (m_projectPart) {
266 267 268
        if (projectPart->id() != m_projectPart->id()) {
            ipcCommunicator.unregisterTranslationUnitsForEditor({fileContainer()});
            ipcCommunicator.registerTranslationUnitsForEditor({fileContainer(projectPart)});
269 270
        }
    } else {
271
        ipcCommunicator.registerTranslationUnitsForEditor({{fileContainer(projectPart)}});
Marco Bubke's avatar
Marco Bubke committed
272 273 274
    }
}

275
void ClangEditorDocumentProcessor::requestDiagnostics(CppTools::ProjectPart *projectPart)
Marco Bubke's avatar
Marco Bubke committed
276
{
277
    if (!m_projectPart || projectPart->id() != m_projectPart->id()) {
Marco Bubke's avatar
Marco Bubke committed
278 279
        IpcCommunicator &ipcCommunicator = m_modelManagerSupport->ipcCommunicator();

280
        ipcCommunicator.requestDiagnostics({fileContainer(projectPart)});
Marco Bubke's avatar
Marco Bubke committed
281 282 283 284 285 286 287 288 289
    }
}

void ClangEditorDocumentProcessor::requestDiagnostics()
{
    // Get diagnostics
    if (m_projectPart) {
        auto  &ipcCommunicator = m_modelManagerSupport->ipcCommunicator();
        ipcCommunicator.requestDiagnostics({filePath(),
290 291 292 293
                                            m_projectPart->id(),
                                            baseTextDocument()->plainText(),
                                            true,
                                            revision()});
294
    }
295 296
}

297 298 299 300 301 302 303 304 305
ClangBackEnd::FileContainer
ClangEditorDocumentProcessor::fileContainer(CppTools::ProjectPart *projectPart) const
{
    if (projectPart)
        return {filePath(), projectPart->id(), revision()};

    return {filePath(), Utf8String(), revision()};
}

306
} // namespace Internal
307
} // namespace ClangCodeModel