2013-12-10 14:37:32 +01:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
2013-12-10 14:37:32 +01:00
|
|
|
**
|
|
|
|
|
** 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
|
2016-01-15 14:57:40 +01:00
|
|
|
** 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.
|
2013-12-10 14:37:32 +01:00
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** 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.
|
2013-12-10 14:37:32 +01:00
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "clangutils.h"
|
|
|
|
|
|
2015-10-13 15:56:41 +02:00
|
|
|
#include "clangeditordocumentprocessor.h"
|
2016-11-21 13:30:26 +01:00
|
|
|
#include "clangmodelmanagersupport.h"
|
2015-10-13 15:56:41 +02:00
|
|
|
|
2018-02-09 13:51:39 +01:00
|
|
|
#include <clangsupport/tokeninfocontainer.h>
|
|
|
|
|
|
2013-12-10 14:37:32 +01:00
|
|
|
#include <coreplugin/icore.h>
|
|
|
|
|
#include <coreplugin/idocument.h>
|
2015-05-08 15:48:17 +02:00
|
|
|
#include <cpptools/baseeditordocumentparser.h>
|
2017-09-27 10:32:52 +02:00
|
|
|
#include <cpptools/compileroptionsbuilder.h>
|
2019-03-13 10:22:16 +01:00
|
|
|
#include <cpptools/cppcodemodelsettings.h>
|
2015-12-15 11:59:48 +01:00
|
|
|
#include <cpptools/cppmodelmanager.h>
|
2019-03-13 10:22:16 +01:00
|
|
|
#include <cpptools/cpptoolsreuse.h>
|
2016-11-22 15:17:42 +01:00
|
|
|
#include <cpptools/editordocumenthandle.h>
|
2016-01-12 12:51:33 +01:00
|
|
|
#include <cpptools/projectpart.h>
|
2018-07-10 15:53:51 +02:00
|
|
|
#include <projectexplorer/buildconfiguration.h>
|
2019-03-13 10:22:16 +01:00
|
|
|
#include <projectexplorer/kitinformation.h>
|
2015-11-17 16:25:02 +01:00
|
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
2018-07-10 15:53:51 +02:00
|
|
|
#include <projectexplorer/target.h>
|
2018-02-09 13:51:39 +01:00
|
|
|
|
|
|
|
|
#include <utils/algorithm.h>
|
2018-07-10 15:53:51 +02:00
|
|
|
#include <utils/fileutils.h>
|
2015-05-08 15:48:17 +02:00
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
|
2014-01-21 14:57:00 +01:00
|
|
|
#include <QDir>
|
2013-12-10 14:37:32 +01:00
|
|
|
#include <QFile>
|
2018-07-10 15:53:51 +02:00
|
|
|
#include <QJsonArray>
|
|
|
|
|
#include <QJsonDocument>
|
|
|
|
|
#include <QJsonObject>
|
2015-11-27 18:21:46 +01:00
|
|
|
#include <QStringList>
|
2018-02-01 13:27:50 +01:00
|
|
|
#include <QTextBlock>
|
2013-12-10 14:37:32 +01:00
|
|
|
|
|
|
|
|
using namespace Core;
|
|
|
|
|
using namespace CppTools;
|
2020-02-18 18:25:26 +01:00
|
|
|
using namespace ProjectExplorer;
|
2020-07-06 15:49:35 +02:00
|
|
|
using namespace Utils;
|
2013-12-10 14:37:32 +01:00
|
|
|
|
|
|
|
|
namespace ClangCodeModel {
|
2020-07-06 15:49:35 +02:00
|
|
|
namespace Internal {
|
2013-12-10 14:37:32 +01:00
|
|
|
|
2017-09-27 10:32:52 +02:00
|
|
|
class LibClangOptionsBuilder final : public CompilerOptionsBuilder
|
2013-12-10 14:37:32 +01:00
|
|
|
{
|
2015-06-26 10:50:06 +02:00
|
|
|
public:
|
2019-01-31 10:16:28 +01:00
|
|
|
LibClangOptionsBuilder(const ProjectPart &projectPart,
|
|
|
|
|
UseBuildSystemWarnings useBuildSystemWarnings)
|
2018-08-23 15:45:33 +02:00
|
|
|
: CompilerOptionsBuilder(projectPart,
|
|
|
|
|
UseSystemHeader::No,
|
2018-11-30 12:15:07 +01:00
|
|
|
UseTweakedHeaderPaths::Yes,
|
2018-11-30 11:02:49 +01:00
|
|
|
UseLanguageDefines::No,
|
2019-01-31 10:16:28 +01:00
|
|
|
useBuildSystemWarnings,
|
2018-08-23 15:45:33 +02:00
|
|
|
QString(CLANG_VERSION),
|
2020-08-24 10:46:58 +02:00
|
|
|
QString(CLANG_INCLUDE_DIR))
|
2015-06-26 10:50:06 +02:00
|
|
|
{
|
2015-07-27 10:26:09 +02:00
|
|
|
}
|
|
|
|
|
|
2019-01-11 15:50:39 +01:00
|
|
|
void addProjectMacros() final
|
2017-11-20 13:30:36 +01:00
|
|
|
{
|
|
|
|
|
addMacros({ProjectExplorer::Macro("Q_CREATOR_RUN", "1")});
|
2019-01-11 15:50:39 +01:00
|
|
|
CompilerOptionsBuilder::addProjectMacros();
|
2017-11-20 13:30:36 +01:00
|
|
|
}
|
|
|
|
|
|
2017-09-11 14:08:00 +02:00
|
|
|
void addExtraOptions() final
|
2015-11-17 16:25:02 +01:00
|
|
|
{
|
2017-09-11 14:08:00 +02:00
|
|
|
addDummyUiHeaderOnDiskIncludePath();
|
2019-01-11 15:32:12 +01:00
|
|
|
add("-fmessage-length=0", /*gccOnlyOption=*/true);
|
|
|
|
|
add("-fdiagnostics-show-note-include-stack", /*gccOnlyOption=*/true);
|
|
|
|
|
add("-fretain-comments-from-system-headers", /*gccOnlyOption=*/true);
|
2017-09-27 10:32:52 +02:00
|
|
|
add("-fmacro-backtrace-limit=0");
|
|
|
|
|
add("-ferror-limit=1000");
|
2015-06-25 13:46:22 +02:00
|
|
|
}
|
|
|
|
|
|
2017-09-11 14:08:00 +02:00
|
|
|
private:
|
2016-11-21 13:30:26 +01:00
|
|
|
void addDummyUiHeaderOnDiskIncludePath()
|
|
|
|
|
{
|
2018-10-18 09:21:35 +02:00
|
|
|
const QString path = ClangModelManagerSupport::instance()->dummyUiHeaderOnDiskDirPath();
|
2019-01-11 15:32:12 +01:00
|
|
|
if (!path.isEmpty())
|
|
|
|
|
add({"-I", QDir::toNativeSeparators(path)});
|
2016-11-21 13:30:26 +01:00
|
|
|
}
|
2015-06-26 10:50:06 +02:00
|
|
|
};
|
2015-06-25 13:46:22 +02:00
|
|
|
|
2019-01-31 10:16:28 +01:00
|
|
|
QStringList createClangOptions(const ProjectPart &projectPart,
|
|
|
|
|
UseBuildSystemWarnings useBuildSystemWarnings,
|
|
|
|
|
ProjectFile::Kind fileKind)
|
2015-06-26 10:50:06 +02:00
|
|
|
{
|
2019-01-31 10:16:28 +01:00
|
|
|
return LibClangOptionsBuilder(projectPart, useBuildSystemWarnings)
|
|
|
|
|
.build(fileKind, UsePrecompiledHeaders::No);
|
2013-12-10 14:37:32 +01:00
|
|
|
}
|
|
|
|
|
|
2015-05-08 15:48:17 +02:00
|
|
|
ProjectPart::Ptr projectPartForFile(const QString &filePath)
|
|
|
|
|
{
|
2015-09-01 17:34:07 +02:00
|
|
|
if (const auto parser = CppTools::BaseEditorDocumentParser::get(filePath))
|
2017-01-17 15:27:31 +01:00
|
|
|
return parser->projectPartInfo().projectPart;
|
2015-05-08 15:48:17 +02:00
|
|
|
return ProjectPart::Ptr();
|
|
|
|
|
}
|
|
|
|
|
|
2015-10-13 15:56:41 +02:00
|
|
|
ProjectPart::Ptr projectPartForFileBasedOnProcessor(const QString &filePath)
|
|
|
|
|
{
|
|
|
|
|
if (const auto processor = ClangEditorDocumentProcessor::get(filePath))
|
|
|
|
|
return processor->projectPart();
|
|
|
|
|
return ProjectPart::Ptr();
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-24 14:25:53 +02:00
|
|
|
bool isProjectPartLoaded(const ProjectPart::Ptr projectPart)
|
2015-05-08 15:48:17 +02:00
|
|
|
{
|
|
|
|
|
if (projectPart)
|
2020-09-18 12:39:32 +02:00
|
|
|
return !CppModelManager::instance()->projectPartForId(projectPart->id()).isNull();
|
2015-05-08 15:48:17 +02:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-14 17:30:17 +02:00
|
|
|
QString projectPartIdForFile(const QString &filePath)
|
2015-05-08 15:48:17 +02:00
|
|
|
{
|
|
|
|
|
const ProjectPart::Ptr projectPart = projectPartForFile(filePath);
|
|
|
|
|
|
2015-09-24 14:25:53 +02:00
|
|
|
if (isProjectPartLoaded(projectPart))
|
2015-07-14 17:30:17 +02:00
|
|
|
return projectPart->id(); // OK, Project Part is still loaded
|
2015-05-08 15:48:17 +02:00
|
|
|
return QString();
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-22 15:17:42 +01:00
|
|
|
CppEditorDocumentHandle *cppDocument(const QString &filePath)
|
|
|
|
|
{
|
|
|
|
|
return CppTools::CppModelManager::instance()->cppEditorDocument(filePath);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void setLastSentDocumentRevision(const QString &filePath, uint revision)
|
|
|
|
|
{
|
|
|
|
|
if (CppEditorDocumentHandle *document = cppDocument(filePath))
|
|
|
|
|
document->sendTracker().setLastSentRevision(int(revision));
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-01 13:27:50 +01:00
|
|
|
int clangColumn(const QTextBlock &line, int cppEditorColumn)
|
2018-01-26 17:04:38 +01:00
|
|
|
{
|
|
|
|
|
// (1) cppEditorColumn is the actual column shown by CppEditor.
|
|
|
|
|
// (2) The return value is the column in Clang which is the utf8 byte offset from the beginning
|
|
|
|
|
// of the line.
|
|
|
|
|
// Here we convert column from (1) to (2).
|
2018-09-25 16:19:41 +02:00
|
|
|
// '- 1' and '+ 1' are because of 1-based columns
|
|
|
|
|
return line.text().left(cppEditorColumn - 1).toUtf8().size() + 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int cppEditorColumn(const QTextBlock &line, int clangColumn)
|
|
|
|
|
{
|
|
|
|
|
// (1) clangColumn is the column in Clang which is the utf8 byte offset from the beginning
|
|
|
|
|
// of the line.
|
|
|
|
|
// (2) The return value is the actual column shown by CppEditor.
|
|
|
|
|
// Here we convert column from (1) to (2).
|
|
|
|
|
// '- 1' and '+ 1' are because of 1-based columns
|
|
|
|
|
return QString::fromUtf8(line.text().toUtf8().left(clangColumn - 1)).size() + 1;
|
2018-01-26 17:04:38 +01:00
|
|
|
}
|
|
|
|
|
|
2020-07-06 15:49:35 +02:00
|
|
|
CodeModelIcon::Type iconTypeForToken(const ClangBackEnd::TokenInfoContainer &token)
|
2018-02-09 13:51:39 +01:00
|
|
|
{
|
2018-04-04 18:25:23 +02:00
|
|
|
const ClangBackEnd::ExtraInfo &extraInfo = token.extraInfo;
|
2018-02-09 13:51:39 +01:00
|
|
|
if (extraInfo.signal)
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::Signal;
|
2018-02-09 13:51:39 +01:00
|
|
|
|
|
|
|
|
ClangBackEnd::AccessSpecifier access = extraInfo.accessSpecifier;
|
|
|
|
|
if (extraInfo.slot) {
|
|
|
|
|
switch (access) {
|
|
|
|
|
case ClangBackEnd::AccessSpecifier::Public:
|
|
|
|
|
case ClangBackEnd::AccessSpecifier::Invalid:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::SlotPublic;
|
2018-02-09 13:51:39 +01:00
|
|
|
case ClangBackEnd::AccessSpecifier::Protected:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::SlotProtected;
|
2018-02-09 13:51:39 +01:00
|
|
|
case ClangBackEnd::AccessSpecifier::Private:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::SlotPrivate;
|
2018-02-09 13:51:39 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-04 18:25:23 +02:00
|
|
|
ClangBackEnd::HighlightingType mainType = token.types.mainHighlightingType;
|
2018-02-09 13:51:39 +01:00
|
|
|
|
|
|
|
|
if (mainType == ClangBackEnd::HighlightingType::QtProperty)
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::Property;
|
2018-02-09 13:51:39 +01:00
|
|
|
|
|
|
|
|
if (mainType == ClangBackEnd::HighlightingType::PreprocessorExpansion
|
|
|
|
|
|| mainType == ClangBackEnd::HighlightingType::PreprocessorDefinition) {
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::Macro;
|
2018-02-09 13:51:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (mainType == ClangBackEnd::HighlightingType::Enumeration)
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::Enumerator;
|
2018-02-09 13:51:39 +01:00
|
|
|
|
2018-04-06 14:07:35 +02:00
|
|
|
if (mainType == ClangBackEnd::HighlightingType::Type
|
|
|
|
|
|| mainType == ClangBackEnd::HighlightingType::Keyword) {
|
2018-04-04 18:25:23 +02:00
|
|
|
const ClangBackEnd::MixinHighlightingTypes &types = token.types.mixinHighlightingTypes;
|
2018-02-09 13:51:39 +01:00
|
|
|
if (types.contains(ClangBackEnd::HighlightingType::Enum))
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::Enum;
|
2018-02-09 13:51:39 +01:00
|
|
|
if (types.contains(ClangBackEnd::HighlightingType::Struct))
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::Struct;
|
2018-02-09 13:51:39 +01:00
|
|
|
if (types.contains(ClangBackEnd::HighlightingType::Namespace))
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::Namespace;
|
2018-04-06 14:07:35 +02:00
|
|
|
if (types.contains(ClangBackEnd::HighlightingType::Class))
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::Class;
|
2018-04-06 14:07:35 +02:00
|
|
|
if (mainType == ClangBackEnd::HighlightingType::Keyword)
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::Keyword;
|
|
|
|
|
return CodeModelIcon::Class;
|
2018-02-09 13:51:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ClangBackEnd::StorageClass storageClass = extraInfo.storageClass;
|
|
|
|
|
if (mainType == ClangBackEnd::HighlightingType::VirtualFunction
|
2016-11-05 15:29:10 +01:00
|
|
|
|| mainType == ClangBackEnd::HighlightingType::Function
|
2018-08-22 17:25:12 +02:00
|
|
|
|| token.types.mixinHighlightingTypes.contains(
|
|
|
|
|
ClangBackEnd::HighlightingType::Operator)) {
|
2018-02-09 13:51:39 +01:00
|
|
|
if (storageClass != ClangBackEnd::StorageClass::Static) {
|
|
|
|
|
switch (access) {
|
|
|
|
|
case ClangBackEnd::AccessSpecifier::Public:
|
|
|
|
|
case ClangBackEnd::AccessSpecifier::Invalid:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::FuncPublic;
|
2018-02-09 13:51:39 +01:00
|
|
|
case ClangBackEnd::AccessSpecifier::Protected:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::FuncProtected;
|
2018-02-09 13:51:39 +01:00
|
|
|
case ClangBackEnd::AccessSpecifier::Private:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::FuncPrivate;
|
2018-02-09 13:51:39 +01:00
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
switch (access) {
|
|
|
|
|
case ClangBackEnd::AccessSpecifier::Public:
|
|
|
|
|
case ClangBackEnd::AccessSpecifier::Invalid:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::FuncPublicStatic;
|
2018-02-09 13:51:39 +01:00
|
|
|
case ClangBackEnd::AccessSpecifier::Protected:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::FuncProtectedStatic;
|
2018-02-09 13:51:39 +01:00
|
|
|
case ClangBackEnd::AccessSpecifier::Private:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::FuncPrivateStatic;
|
2018-02-09 13:51:39 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (mainType == ClangBackEnd::HighlightingType::GlobalVariable
|
|
|
|
|
|| mainType == ClangBackEnd::HighlightingType::Field) {
|
|
|
|
|
if (storageClass != ClangBackEnd::StorageClass::Static) {
|
|
|
|
|
switch (access) {
|
|
|
|
|
case ClangBackEnd::AccessSpecifier::Public:
|
|
|
|
|
case ClangBackEnd::AccessSpecifier::Invalid:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::VarPublic;
|
2018-02-09 13:51:39 +01:00
|
|
|
case ClangBackEnd::AccessSpecifier::Protected:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::VarProtected;
|
2018-02-09 13:51:39 +01:00
|
|
|
case ClangBackEnd::AccessSpecifier::Private:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::VarPrivate;
|
2018-02-09 13:51:39 +01:00
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
switch (access) {
|
|
|
|
|
case ClangBackEnd::AccessSpecifier::Public:
|
|
|
|
|
case ClangBackEnd::AccessSpecifier::Invalid:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::VarPublicStatic;
|
2018-02-09 13:51:39 +01:00
|
|
|
case ClangBackEnd::AccessSpecifier::Protected:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::VarProtectedStatic;
|
2018-02-09 13:51:39 +01:00
|
|
|
case ClangBackEnd::AccessSpecifier::Private:
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::VarPrivateStatic;
|
2018-02-09 13:51:39 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-06 15:49:35 +02:00
|
|
|
return CodeModelIcon::Unknown;
|
2018-02-09 13:51:39 +01:00
|
|
|
}
|
|
|
|
|
|
2018-05-09 15:12:01 +02:00
|
|
|
QString diagnosticCategoryPrefixRemoved(const QString &text)
|
|
|
|
|
{
|
|
|
|
|
QString theText = text;
|
|
|
|
|
|
|
|
|
|
// Prefixes are taken from $LLVM_SOURCE_DIR/tools/clang/lib/Frontend/TextDiagnostic.cpp,
|
|
|
|
|
// function TextDiagnostic::printDiagnosticLevel (llvm-3.6.2).
|
|
|
|
|
static const QStringList categoryPrefixes = {
|
|
|
|
|
QStringLiteral("note"),
|
|
|
|
|
QStringLiteral("remark"),
|
|
|
|
|
QStringLiteral("warning"),
|
|
|
|
|
QStringLiteral("error"),
|
|
|
|
|
QStringLiteral("fatal error")
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
for (const QString &prefix : categoryPrefixes) {
|
|
|
|
|
const QString fullPrefix = prefix + QStringLiteral(": ");
|
|
|
|
|
if (theText.startsWith(fullPrefix)) {
|
|
|
|
|
theText.remove(0, fullPrefix.length());
|
|
|
|
|
return theText;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return text;
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-06 15:49:35 +02:00
|
|
|
static FilePath compilerPath(const CppTools::ProjectPart &projectPart)
|
2018-07-10 15:53:51 +02:00
|
|
|
{
|
2020-02-18 18:25:26 +01:00
|
|
|
Target *target = projectPart.project->activeTarget();
|
2018-07-10 15:53:51 +02:00
|
|
|
if (!target)
|
2020-07-06 15:49:35 +02:00
|
|
|
return FilePath();
|
2018-07-10 15:53:51 +02:00
|
|
|
|
2020-02-18 18:25:26 +01:00
|
|
|
ToolChain *toolchain = ToolChainKitAspect::cxxToolChain(target->kit());
|
2019-03-13 10:22:16 +01:00
|
|
|
|
|
|
|
|
return toolchain->compilerCommand();
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-06 15:49:35 +02:00
|
|
|
static FilePath buildDirectory(const ProjectExplorer::Project &project)
|
2018-07-10 15:53:51 +02:00
|
|
|
{
|
2019-10-25 19:48:14 +02:00
|
|
|
if (auto *target = project.activeTarget()) {
|
|
|
|
|
if (auto *bc = target->activeBuildConfiguration())
|
|
|
|
|
return bc->buildDirectory();
|
|
|
|
|
}
|
|
|
|
|
return {};
|
2018-07-10 15:53:51 +02:00
|
|
|
}
|
|
|
|
|
|
2019-03-13 10:22:16 +01:00
|
|
|
static QStringList projectPartArguments(const ProjectPart &projectPart)
|
2018-07-10 15:53:51 +02:00
|
|
|
{
|
2019-03-13 10:22:16 +01:00
|
|
|
QStringList args;
|
|
|
|
|
args << compilerPath(projectPart).toString();
|
|
|
|
|
args << "-c";
|
|
|
|
|
if (projectPart.toolchainType != ProjectExplorer::Constants::MSVC_TOOLCHAIN_TYPEID) {
|
|
|
|
|
args << "--target=" + projectPart.toolChainTargetTriple;
|
|
|
|
|
args << (projectPart.toolChainWordWidth == ProjectPart::WordWidth64Bit
|
|
|
|
|
? QLatin1String("-m64")
|
|
|
|
|
: QLatin1String("-m32"));
|
|
|
|
|
}
|
|
|
|
|
args << projectPart.compilerFlags;
|
|
|
|
|
for (const ProjectExplorer::HeaderPath &headerPath : projectPart.headerPaths) {
|
|
|
|
|
if (headerPath.type == ProjectExplorer::HeaderPathType::User) {
|
|
|
|
|
args << "-I" + headerPath.path;
|
|
|
|
|
} else if (headerPath.type == ProjectExplorer::HeaderPathType::System) {
|
|
|
|
|
args << (projectPart.toolchainType == ProjectExplorer::Constants::MSVC_TOOLCHAIN_TYPEID
|
|
|
|
|
? "-I"
|
|
|
|
|
: "-isystem")
|
|
|
|
|
+ headerPath.path;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
for (const ProjectExplorer::Macro ¯o : projectPart.projectMacros) {
|
|
|
|
|
args.append(QString::fromUtf8(
|
|
|
|
|
macro.toKeyValue(macro.type == ProjectExplorer::MacroType::Define ? "-D" : "-U")));
|
|
|
|
|
}
|
2018-07-10 15:53:51 +02:00
|
|
|
|
2019-03-13 10:22:16 +01:00
|
|
|
return args;
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-06 15:49:35 +02:00
|
|
|
static QJsonObject createFileObject(const FilePath &buildDir,
|
2019-03-13 10:22:16 +01:00
|
|
|
const QStringList &arguments,
|
|
|
|
|
const ProjectPart &projectPart,
|
|
|
|
|
const ProjectFile &projFile)
|
|
|
|
|
{
|
2018-07-10 15:53:51 +02:00
|
|
|
QJsonObject fileObject;
|
|
|
|
|
fileObject["file"] = projFile.path;
|
2019-03-13 10:22:16 +01:00
|
|
|
QJsonArray args = QJsonArray::fromStringList(arguments);
|
|
|
|
|
|
|
|
|
|
const ProjectFile::Kind kind = ProjectFile::classify(projFile.path);
|
|
|
|
|
if (projectPart.toolchainType == ProjectExplorer::Constants::MSVC_TOOLCHAIN_TYPEID
|
|
|
|
|
|| projectPart.toolchainType == ProjectExplorer::Constants::CLANG_CL_TOOLCHAIN_TYPEID) {
|
|
|
|
|
if (ProjectFile::isC(kind))
|
|
|
|
|
args.append("/TC");
|
|
|
|
|
else if (ProjectFile::isCxx(kind))
|
|
|
|
|
args.append("/TP");
|
|
|
|
|
} else {
|
|
|
|
|
QStringList langOption
|
|
|
|
|
= createLanguageOptionGcc(kind,
|
|
|
|
|
projectPart.languageExtensions
|
2020-07-06 15:49:35 +02:00
|
|
|
& LanguageExtension::ObjectiveC);
|
2019-03-13 10:22:16 +01:00
|
|
|
for (const QString &langOptionPart : langOption)
|
|
|
|
|
args.append(langOptionPart);
|
|
|
|
|
}
|
2018-07-10 15:53:51 +02:00
|
|
|
args.append(QDir::toNativeSeparators(projFile.path));
|
|
|
|
|
fileObject["arguments"] = args;
|
|
|
|
|
fileObject["directory"] = buildDir.toString();
|
|
|
|
|
return fileObject;
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-13 17:58:06 +01:00
|
|
|
GenerateCompilationDbResult generateCompilationDB(CppTools::ProjectInfo projectInfo)
|
2018-07-10 15:53:51 +02:00
|
|
|
{
|
2020-07-06 15:49:35 +02:00
|
|
|
const FilePath buildDir = buildDirectory(*projectInfo.project());
|
2019-03-13 17:58:06 +01:00
|
|
|
QTC_ASSERT(!buildDir.isEmpty(), return GenerateCompilationDbResult(QString(),
|
|
|
|
|
QCoreApplication::translate("ClangUtils", "Could not retrieve build directory.")));
|
2018-07-10 15:53:51 +02:00
|
|
|
|
2019-03-13 14:31:50 +01:00
|
|
|
QDir dir(buildDir.toString());
|
|
|
|
|
if (!dir.exists())
|
|
|
|
|
dir.mkpath(dir.path());
|
|
|
|
|
QFile compileCommandsFile(buildDir.toString() + "/compile_commands.json");
|
2018-12-03 08:26:48 +01:00
|
|
|
const bool fileOpened = compileCommandsFile.open(QIODevice::WriteOnly | QIODevice::Truncate);
|
2019-03-13 17:58:06 +01:00
|
|
|
if (!fileOpened) {
|
|
|
|
|
return GenerateCompilationDbResult(QString(),
|
|
|
|
|
QCoreApplication::translate("ClangUtils", "Could not create \"%1\": %2")
|
|
|
|
|
.arg(compileCommandsFile.fileName(), compileCommandsFile.errorString()));
|
|
|
|
|
}
|
2018-09-20 00:42:28 +03:00
|
|
|
compileCommandsFile.write("[");
|
2018-07-10 15:53:51 +02:00
|
|
|
|
|
|
|
|
for (ProjectPart::Ptr projectPart : projectInfo.projectParts()) {
|
2019-03-13 10:22:16 +01:00
|
|
|
const QStringList args = projectPartArguments(*projectPart);
|
2018-09-20 00:42:28 +03:00
|
|
|
for (const ProjectFile &projFile : projectPart->files) {
|
2019-03-13 10:22:16 +01:00
|
|
|
const QJsonObject json = createFileObject(buildDir, args, *projectPart, projFile);
|
2018-09-20 00:42:28 +03:00
|
|
|
if (compileCommandsFile.size() > 1)
|
|
|
|
|
compileCommandsFile.write(",");
|
|
|
|
|
compileCommandsFile.write('\n' + QJsonDocument(json).toJson().trimmed());
|
|
|
|
|
}
|
2018-07-10 15:53:51 +02:00
|
|
|
}
|
|
|
|
|
|
2018-09-20 00:42:28 +03:00
|
|
|
compileCommandsFile.write("\n]");
|
2018-07-10 15:53:51 +02:00
|
|
|
compileCommandsFile.close();
|
2019-03-13 17:58:06 +01:00
|
|
|
return GenerateCompilationDbResult(compileCommandsFile.fileName(), QString());
|
2018-07-10 15:53:51 +02:00
|
|
|
}
|
|
|
|
|
|
2019-01-29 16:03:38 +01:00
|
|
|
QString currentCppEditorDocumentFilePath()
|
|
|
|
|
{
|
|
|
|
|
QString filePath;
|
|
|
|
|
|
|
|
|
|
const auto currentEditor = Core::EditorManager::currentEditor();
|
|
|
|
|
if (currentEditor && CppTools::CppModelManager::isCppEditor(currentEditor)) {
|
|
|
|
|
const auto currentDocument = currentEditor->document();
|
|
|
|
|
if (currentDocument)
|
|
|
|
|
filePath = currentDocument->filePath().toString();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return filePath;
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-28 12:40:03 +01:00
|
|
|
DiagnosticTextInfo::DiagnosticTextInfo(const QString &text)
|
|
|
|
|
: m_text(text)
|
|
|
|
|
, m_squareBracketStartIndex(text.lastIndexOf('['))
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
QString DiagnosticTextInfo::textWithoutOption() const
|
|
|
|
|
{
|
|
|
|
|
if (m_squareBracketStartIndex == -1)
|
|
|
|
|
return m_text;
|
|
|
|
|
|
|
|
|
|
return m_text.mid(0, m_squareBracketStartIndex - 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString DiagnosticTextInfo::option() const
|
|
|
|
|
{
|
|
|
|
|
if (m_squareBracketStartIndex == -1)
|
|
|
|
|
return QString();
|
|
|
|
|
|
|
|
|
|
const int index = m_squareBracketStartIndex + 1;
|
|
|
|
|
return m_text.mid(index, m_text.count() - index - 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString DiagnosticTextInfo::category() const
|
|
|
|
|
{
|
|
|
|
|
if (m_squareBracketStartIndex == -1)
|
|
|
|
|
return QString();
|
|
|
|
|
|
|
|
|
|
const int index = m_squareBracketStartIndex + 1;
|
|
|
|
|
if (isClazyOption(m_text.mid(index)))
|
|
|
|
|
return QCoreApplication::translate("ClangDiagnosticWidget", "Clazy Issue");
|
|
|
|
|
else
|
|
|
|
|
return QCoreApplication::translate("ClangDiagnosticWidget", "Clang-Tidy Issue");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool DiagnosticTextInfo::isClazyOption(const QString &option)
|
|
|
|
|
{
|
|
|
|
|
return option.startsWith("-Wclazy");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString DiagnosticTextInfo::clazyCheckName(const QString &option)
|
|
|
|
|
{
|
|
|
|
|
if (option.startsWith("-Wclazy"))
|
|
|
|
|
return option.mid(8); // Chop "-Wclazy-"
|
|
|
|
|
return option;
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-06 15:49:35 +02:00
|
|
|
} // namespace Internal
|
2013-12-10 14:37:32 +01:00
|
|
|
} // namespace Clang
|