forked from qt-creator/qt-creator
Clang: extra clangbackend job to collect full token info
Limit document annotations job to only highlighting data collection and move more expensive calls into separate job that runs after it. Change-Id: Ie792a3f741ac45c81033dd5b3a20ed061604f927 Reviewed-by: Nikolai Kosjar <nikolai.kosjar@qt.io>
This commit is contained in:
153
src/tools/clangbackend/source/fulltokeninfo.cpp
Normal file
153
src/tools/clangbackend/source/fulltokeninfo.cpp
Normal file
@@ -0,0 +1,153 @@
|
||||
/****************************************************************************
|
||||
**
|
||||
** Copyright (C) 2018 The Qt Company Ltd.
|
||||
** Contact: https://www.qt.io/licensing/
|
||||
**
|
||||
** 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 The Qt Company. For licensing terms
|
||||
** and conditions see https://www.qt.io/terms-conditions. For further
|
||||
** information use the contact form at https://www.qt.io/contact-us.
|
||||
**
|
||||
** GNU General Public License Usage
|
||||
** Alternatively, this file may be used under the terms of the GNU
|
||||
** General Public License version 3 as published by the Free Software
|
||||
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
||||
** included in the packaging of this file. Please review the following
|
||||
** information to ensure the GNU General Public License requirements will
|
||||
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
||||
**
|
||||
****************************************************************************/
|
||||
|
||||
#include "clangstring.h"
|
||||
#include "cursor.h"
|
||||
#include "fulltokeninfo.h"
|
||||
|
||||
namespace ClangBackEnd {
|
||||
|
||||
FullTokenInfo::FullTokenInfo(const CXCursor &cxCursor,
|
||||
CXToken *cxToken,
|
||||
CXTranslationUnit cxTranslationUnit,
|
||||
std::vector<CXSourceRange> ¤tOutputArgumentRanges)
|
||||
: TokenInfo(cxCursor, cxToken, cxTranslationUnit, currentOutputArgumentRanges)
|
||||
{
|
||||
}
|
||||
|
||||
FullTokenInfo::operator TokenInfoContainer() const
|
||||
{
|
||||
return TokenInfoContainer(line(), column(), length(), types(), m_extraInfo);
|
||||
}
|
||||
|
||||
static Utf8String fullyQualifiedType(const Cursor &cursor)
|
||||
{
|
||||
Utf8String typeSpelling = cursor.type().canonical().utf8Spelling();
|
||||
if (typeSpelling.isEmpty()) {
|
||||
// Only if it's the namespaces level.
|
||||
typeSpelling = cursor.unifiedSymbolResolution();
|
||||
typeSpelling.replace(Utf8StringLiteral("c:@N@"), Utf8StringLiteral(""));
|
||||
typeSpelling.replace(Utf8StringLiteral("@N@"), Utf8StringLiteral("::"));
|
||||
}
|
||||
return typeSpelling;
|
||||
}
|
||||
|
||||
void FullTokenInfo::updateTypeSpelling(const Cursor &cursor, bool functionLike)
|
||||
{
|
||||
m_extraInfo.typeSpelling = fullyQualifiedType(cursor);
|
||||
m_extraInfo.semanticParentTypeSpelling = fullyQualifiedType(cursor.semanticParent());
|
||||
if (!functionLike)
|
||||
return;
|
||||
Type type = cursor.type().canonical();
|
||||
m_extraInfo.resultTypeSpelling = type.resultType().utf8Spelling();
|
||||
bool hasSpaceAfterReturnType = false;
|
||||
if (m_extraInfo.resultTypeSpelling.byteSize() < m_extraInfo.typeSpelling.byteSize()) {
|
||||
const char *data = m_extraInfo.typeSpelling.constData();
|
||||
hasSpaceAfterReturnType = (data[m_extraInfo.resultTypeSpelling.byteSize()] == ' ');
|
||||
}
|
||||
m_extraInfo.typeSpelling
|
||||
= m_extraInfo.typeSpelling.mid(m_extraInfo.resultTypeSpelling.byteSize()
|
||||
+ (hasSpaceAfterReturnType ? 1 : 0));
|
||||
}
|
||||
|
||||
void FullTokenInfo::identifierKind(const Cursor &cursor, Recursion recursion)
|
||||
{
|
||||
updateTypeSpelling(cursor);
|
||||
|
||||
TokenInfo::identifierKind(cursor, recursion);
|
||||
|
||||
m_extraInfo.identifier = (cursor.kind() != CXCursor_PreprocessingDirective);
|
||||
}
|
||||
|
||||
void FullTokenInfo::referencedTypeKind(const Cursor &cursor)
|
||||
{
|
||||
updateTypeSpelling(cursor.referenced());
|
||||
|
||||
TokenInfo::referencedTypeKind(cursor);
|
||||
}
|
||||
|
||||
void FullTokenInfo::functionKind(const Cursor &cursor, Recursion recursion)
|
||||
{
|
||||
updateTypeSpelling(cursor, true);
|
||||
|
||||
TokenInfo::functionKind(cursor, recursion);
|
||||
|
||||
m_extraInfo.accessSpecifier = cursor.accessSpecifier();
|
||||
m_extraInfo.storageClass = cursor.storageClass();
|
||||
|
||||
bool isSignal = false;
|
||||
bool isSlot = false;
|
||||
m_originalCursor.visit([&isSignal, &isSlot](CXCursor cursor, CXCursor) {
|
||||
Cursor cur(cursor);
|
||||
ClangString spelling = cur.spelling();
|
||||
if (spelling == "qt_signal")
|
||||
isSignal = true;
|
||||
else if (spelling == "qt_slot")
|
||||
isSlot = true;
|
||||
return CXChildVisit_Break;
|
||||
});
|
||||
m_extraInfo.signal = isSignal;
|
||||
m_extraInfo.slot = isSlot;
|
||||
}
|
||||
|
||||
void FullTokenInfo::variableKind(const Cursor &cursor)
|
||||
{
|
||||
TokenInfo::variableKind(cursor);
|
||||
|
||||
m_extraInfo.storageClass = cursor.storageClass();
|
||||
}
|
||||
|
||||
void FullTokenInfo::fieldKind(const Cursor &cursor)
|
||||
{
|
||||
TokenInfo::fieldKind(cursor);
|
||||
|
||||
m_extraInfo.accessSpecifier = cursor.accessSpecifier();
|
||||
m_extraInfo.storageClass = cursor.storageClass();
|
||||
}
|
||||
|
||||
void FullTokenInfo::memberReferenceKind(const Cursor &cursor)
|
||||
{
|
||||
TokenInfo::memberReferenceKind(cursor);
|
||||
if (cursor.isDynamicCall()) {
|
||||
m_extraInfo.storageClass = cursor.storageClass();
|
||||
m_extraInfo.accessSpecifier = cursor.accessSpecifier();
|
||||
}
|
||||
}
|
||||
|
||||
void FullTokenInfo::evaluate()
|
||||
{
|
||||
TokenInfo::evaluate();
|
||||
|
||||
m_extraInfo.token = ClangString(clang_getTokenSpelling(m_cxTranslationUnit, *m_cxToken));
|
||||
|
||||
auto cxTokenKind = clang_getTokenKind(*m_cxToken);
|
||||
if (cxTokenKind == CXToken_Identifier) {
|
||||
m_extraInfo.declaration = m_originalCursor.isDeclaration();
|
||||
m_extraInfo.definition = m_originalCursor.isDefinition();
|
||||
}
|
||||
m_extraInfo.includeDirectivePath = (m_originalCursor.kind() == CXCursor_InclusionDirective);
|
||||
}
|
||||
|
||||
} // namespace ClangBackEnd
|
||||
Reference in New Issue
Block a user