2017-01-30 11:24:46 +01:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** Copyright (C) 2016 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.
|
|
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
2017-09-29 13:46:53 +02:00
|
|
|
#include "sourcelocationsutils.h"
|
2017-09-21 11:43:59 +02:00
|
|
|
#include <filepathcachinginterface.h>
|
|
|
|
|
#include <filepathid.h>
|
|
|
|
|
|
|
|
|
|
#include <utils/smallstringvector.h>
|
2017-01-30 11:24:46 +01:00
|
|
|
|
|
|
|
|
#include <clang/Basic/SourceManager.h>
|
|
|
|
|
#include <clang/Lex/MacroInfo.h>
|
|
|
|
|
#include <clang/Lex/HeaderSearch.h>
|
|
|
|
|
#include <clang/Lex/PPCallbacks.h>
|
|
|
|
|
#include <clang/Lex/Preprocessor.h>
|
|
|
|
|
|
2017-02-20 17:14:17 +01:00
|
|
|
#include <QFile>
|
|
|
|
|
#include <QDir>
|
|
|
|
|
#include <QTemporaryDir>
|
|
|
|
|
|
2017-01-30 11:24:46 +01:00
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
|
|
namespace ClangBackEnd {
|
|
|
|
|
|
|
|
|
|
class CollectIncludesPreprocessorCallbacks final : public clang::PPCallbacks
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
CollectIncludesPreprocessorCallbacks(clang::HeaderSearch &headerSearch,
|
2017-09-21 11:43:59 +02:00
|
|
|
FilePathIds &includeIds,
|
2018-03-28 16:00:10 +02:00
|
|
|
FilePathIds &topIncludeIds,
|
2017-09-21 11:43:59 +02:00
|
|
|
FilePathCachingInterface &filePathCache,
|
|
|
|
|
const std::vector<uint> &excludedIncludeUID,
|
2017-09-29 13:46:53 +02:00
|
|
|
std::vector<uint> &alreadyIncludedFileUIDs,
|
|
|
|
|
clang::SourceManager &sourceManager)
|
2017-01-30 11:24:46 +01:00
|
|
|
: m_headerSearch(headerSearch),
|
|
|
|
|
m_includeIds(includeIds),
|
2018-03-28 16:00:10 +02:00
|
|
|
m_topIncludeIds(topIncludeIds),
|
2017-01-30 11:24:46 +01:00
|
|
|
m_filePathCache(filePathCache),
|
|
|
|
|
m_excludedIncludeUID(excludedIncludeUID),
|
2017-09-29 13:46:53 +02:00
|
|
|
m_alreadyIncludedFileUIDs(alreadyIncludedFileUIDs),
|
|
|
|
|
m_sourceManager(sourceManager)
|
2017-01-30 11:24:46 +01:00
|
|
|
{}
|
|
|
|
|
|
2017-09-29 13:46:53 +02:00
|
|
|
void InclusionDirective(clang::SourceLocation hashLocation,
|
2017-01-30 11:24:46 +01:00
|
|
|
const clang::Token &/*includeToken*/,
|
2017-02-20 18:26:18 +01:00
|
|
|
llvm::StringRef /*fileName*/,
|
2017-01-30 11:24:46 +01:00
|
|
|
bool /*isAngled*/,
|
|
|
|
|
clang::CharSourceRange /*fileNameRange*/,
|
|
|
|
|
const clang::FileEntry *file,
|
2017-02-20 18:26:18 +01:00
|
|
|
llvm::StringRef /*searchPath*/,
|
2017-01-30 11:24:46 +01:00
|
|
|
llvm::StringRef /*relativePath*/,
|
2017-02-01 14:10:55 +01:00
|
|
|
const clang::Module * /*imported*/) override
|
2017-01-30 11:24:46 +01:00
|
|
|
{
|
2017-02-20 17:14:17 +01:00
|
|
|
if (!m_skipInclude && file) {
|
2017-01-31 14:21:05 +01:00
|
|
|
auto fileUID = file->getUID();
|
2018-03-28 16:00:10 +02:00
|
|
|
auto sourceFileUID = m_sourceManager.getFileEntryForID(m_sourceManager.getFileID(hashLocation))->getUID();
|
|
|
|
|
if (isNotInExcludedIncludeUID(fileUID)) {
|
2017-01-31 14:21:05 +01:00
|
|
|
auto notAlreadyIncluded = isNotAlreadyIncluded(fileUID);
|
|
|
|
|
if (notAlreadyIncluded.first) {
|
|
|
|
|
m_alreadyIncludedFileUIDs.insert(notAlreadyIncluded.second, fileUID);
|
2017-11-16 17:48:53 +01:00
|
|
|
FilePath filePath = filePathFromFile(file);
|
|
|
|
|
if (!filePath.empty()) {
|
2017-09-21 11:43:59 +02:00
|
|
|
FilePathId includeId = m_filePathCache.filePathId(filePath);
|
2017-02-16 18:25:19 +01:00
|
|
|
m_includeIds.emplace_back(includeId);
|
2018-03-28 16:00:10 +02:00
|
|
|
if (isInExcludedIncludeUID(sourceFileUID))
|
|
|
|
|
m_topIncludeIds.emplace_back(includeId);
|
2017-02-16 18:25:19 +01:00
|
|
|
}
|
2017-01-31 14:21:05 +01:00
|
|
|
}
|
2017-01-30 11:24:46 +01:00
|
|
|
}
|
|
|
|
|
}
|
2017-02-20 17:14:17 +01:00
|
|
|
|
|
|
|
|
m_skipInclude = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool FileNotFound(clang::StringRef fileNameRef, clang::SmallVectorImpl<char> &recoveryPath) override
|
|
|
|
|
{
|
|
|
|
|
QTemporaryDir temporaryDirectory;
|
|
|
|
|
temporaryDirectory.setAutoRemove(false);
|
|
|
|
|
const QByteArray temporaryDirUtf8 = temporaryDirectory.path().toUtf8();
|
|
|
|
|
|
|
|
|
|
const QString fileName = QString::fromUtf8(fileNameRef.data(), int(fileNameRef.size()));
|
|
|
|
|
QString filePath = temporaryDirectory.path() + '/' + fileName;
|
|
|
|
|
|
|
|
|
|
ensureDirectory(temporaryDirectory.path(), fileName);
|
|
|
|
|
createFakeFile(filePath);
|
|
|
|
|
|
|
|
|
|
recoveryPath.append(temporaryDirUtf8.cbegin(), temporaryDirUtf8.cend());
|
|
|
|
|
|
|
|
|
|
m_skipInclude = true;
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ensureDirectory(const QString &directory, const QString &fileName)
|
|
|
|
|
{
|
|
|
|
|
QStringList directoryEntries = fileName.split('/');
|
|
|
|
|
directoryEntries.pop_back();
|
|
|
|
|
|
|
|
|
|
if (!directoryEntries.isEmpty())
|
|
|
|
|
QDir(directory).mkpath(directoryEntries.join('/'));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void createFakeFile(const QString &filePath)
|
|
|
|
|
{
|
|
|
|
|
QFile fakeFile;
|
|
|
|
|
fakeFile.setFileName(filePath);
|
|
|
|
|
|
|
|
|
|
fakeFile.open(QIODevice::ReadWrite);
|
|
|
|
|
fakeFile.close();
|
2017-01-30 11:24:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool isNotInExcludedIncludeUID(uint uid) const
|
|
|
|
|
{
|
2018-03-28 16:00:10 +02:00
|
|
|
return !isInExcludedIncludeUID(uid);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool isInExcludedIncludeUID(uint uid) const
|
|
|
|
|
{
|
|
|
|
|
return std::binary_search(m_excludedIncludeUID.begin(),
|
|
|
|
|
m_excludedIncludeUID.end(),
|
|
|
|
|
uid);
|
2017-01-30 11:24:46 +01:00
|
|
|
}
|
|
|
|
|
|
2017-09-21 11:43:59 +02:00
|
|
|
std::pair<bool, std::vector<uint>::iterator> isNotAlreadyIncluded(uint uid) const
|
2017-01-30 11:24:46 +01:00
|
|
|
{
|
|
|
|
|
auto range = std::equal_range(m_alreadyIncludedFileUIDs.begin(),
|
|
|
|
|
m_alreadyIncludedFileUIDs.end(),
|
|
|
|
|
uid);
|
|
|
|
|
|
|
|
|
|
return {range.first == range.second, range.first};
|
|
|
|
|
}
|
|
|
|
|
|
2017-11-16 17:48:53 +01:00
|
|
|
static FilePath filePathFromFile(const clang::FileEntry *file)
|
2017-02-16 18:25:19 +01:00
|
|
|
{
|
2017-11-16 17:48:53 +01:00
|
|
|
return FilePath::fromNativeFilePath(absolutePath(file->getName()));
|
2017-02-16 18:25:19 +01:00
|
|
|
}
|
|
|
|
|
|
2017-01-30 11:24:46 +01:00
|
|
|
private:
|
|
|
|
|
clang::HeaderSearch &m_headerSearch;
|
2017-09-21 11:43:59 +02:00
|
|
|
FilePathIds &m_includeIds;
|
2018-03-28 16:00:10 +02:00
|
|
|
FilePathIds &m_topIncludeIds;
|
2017-09-21 11:43:59 +02:00
|
|
|
FilePathCachingInterface &m_filePathCache;
|
|
|
|
|
const std::vector<uint> &m_excludedIncludeUID;
|
|
|
|
|
std::vector<uint> &m_alreadyIncludedFileUIDs;
|
2017-09-29 13:46:53 +02:00
|
|
|
clang::SourceManager &m_sourceManager;
|
2017-02-20 17:14:17 +01:00
|
|
|
bool m_skipInclude = false;
|
2017-01-30 11:24:46 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
} // namespace ClangBackEnd
|