/**************************************************************************** ** ** 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 #include "stringcache.h" #include #include #include #include #include #include #include namespace ClangBackEnd { class CollectIncludesPreprocessorCallbacks final : public clang::PPCallbacks { public: CollectIncludesPreprocessorCallbacks(clang::HeaderSearch &headerSearch, std::vector &includeIds, StringCache &filePathCache, const std::vector &excludedIncludeUID, std::vector &alreadyIncludedFileUIDs) : m_headerSearch(headerSearch), m_includeIds(includeIds), m_filePathCache(filePathCache), m_excludedIncludeUID(excludedIncludeUID), m_alreadyIncludedFileUIDs(alreadyIncludedFileUIDs) {} void InclusionDirective(clang::SourceLocation /*hashLocation*/, const clang::Token &/*includeToken*/, llvm::StringRef fileName, bool /*isAngled*/, clang::CharSourceRange /*fileNameRange*/, const clang::FileEntry *file, llvm::StringRef /*searchPath*/, llvm::StringRef /*relativePath*/, const clang::Module * /*imported*/) override { if (file) { auto fileUID = file->getUID(); flagIncludeAlreadyRead(file); if (isNotInExcludedIncludeUID(fileUID)) { auto notAlreadyIncluded = isNotAlreadyIncluded(fileUID); if (notAlreadyIncluded.first) { m_alreadyIncludedFileUIDs.insert(notAlreadyIncluded.second, fileUID); uint includeId = m_filePathCache.stringId({fileName.data(), fileName.size()}); m_includeIds.emplace_back(includeId); } } } } bool isNotInExcludedIncludeUID(uint uid) const { return !std::binary_search(m_excludedIncludeUID.begin(), m_excludedIncludeUID.end(), uid); } std::pair::iterator> isNotAlreadyIncluded(uint uid) const { auto range = std::equal_range(m_alreadyIncludedFileUIDs.begin(), m_alreadyIncludedFileUIDs.end(), uid); return {range.first == range.second, range.first}; } void flagIncludeAlreadyRead(const clang::FileEntry *file) { auto &headerFileInfo = m_headerSearch.getFileInfo(file); headerFileInfo.isImport = true; ++headerFileInfo.NumIncludes; } private: clang::HeaderSearch &m_headerSearch; std::vector &m_includeIds; StringCache &m_filePathCache; const std::vector &m_excludedIncludeUID; std::vector &m_alreadyIncludedFileUIDs; }; } // namespace ClangBackEnd