/**************************************************************************** ** ** 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. ** ****************************************************************************/ #include "projectupdater.h" #include "pchmanagerclient.h" #include #include #include #include #include #include #include #include namespace ClangPchManager { class HeaderAndSources { public: void reserve(std::size_t size) { headers.reserve(size); sources.reserve(size); } ClangBackEnd::FilePathIds headers; ClangBackEnd::FilePathIds sources; }; ProjectUpdater::ProjectUpdater(ClangBackEnd::ProjectManagementServerInterface &server, ClangBackEnd::FilePathCachingInterface &filePathCache) : m_server(server), m_filePathCache(filePathCache) { } void ProjectUpdater::updateProjectParts(const std::vector &projectParts, ClangBackEnd::V2::FileContainers &&generatedFiles) { m_excludedPaths = createExcludedPaths(generatedFiles); ClangBackEnd::UpdatePchProjectPartsMessage message{toProjectPartContainers(projectParts), std::move(generatedFiles)}; m_server.updatePchProjectParts(std::move(message)); } void ProjectUpdater::removeProjectParts(const QStringList &projectPartIds) { ClangBackEnd::RemovePchProjectPartsMessage message{Utils::SmallStringVector(projectPartIds)}; m_server.removePchProjectParts(std::move(message)); } void ProjectUpdater::setExcludedPaths(Utils::PathStringVector &&excludedPaths) { m_excludedPaths = excludedPaths; } void ProjectUpdater::addToHeaderAndSources(HeaderAndSources &headerAndSources, const CppTools::ProjectFile &projectFile) const { using ClangBackEnd::FilePathView; Utils::PathString path = projectFile.path; bool exclude = std::binary_search(m_excludedPaths.begin(), m_excludedPaths.end(), path); if (!exclude) { ClangBackEnd::FilePathId filePathId = m_filePathCache.filePathId(FilePathView(path)); if (projectFile.isSource()) headerAndSources.sources.push_back(filePathId); else if (projectFile.isHeader()) headerAndSources.headers.push_back(filePathId); } } HeaderAndSources ProjectUpdater::headerAndSourcesFromProjectPart( CppTools::ProjectPart *projectPart) const { HeaderAndSources headerAndSources; headerAndSources.reserve(std::size_t(projectPart->files.size()) * 3 / 2); for (const CppTools::ProjectFile &projectFile : projectPart->files) addToHeaderAndSources(headerAndSources, projectFile); return headerAndSources; } QStringList ProjectUpdater::compilerArguments(CppTools::ProjectPart *projectPart) { using CppTools::CompilerOptionsBuilder; CompilerOptionsBuilder builder(*projectPart, CLANG_VERSION, CLANG_RESOURCE_DIR); return builder.build(CppTools::ProjectFile::CXXHeader, CompilerOptionsBuilder::PchUsage::None); } ClangBackEnd::V2::ProjectPartContainer ProjectUpdater::toProjectPartContainer( CppTools::ProjectPart *projectPart) const { QStringList arguments = compilerArguments(projectPart); HeaderAndSources headerAndSources = headerAndSourcesFromProjectPart(projectPart); return ClangBackEnd::V2::ProjectPartContainer(projectPart->displayName, Utils::SmallStringVector(arguments), std::move(headerAndSources.headers), std::move(headerAndSources.sources)); } std::vector ProjectUpdater::toProjectPartContainers( std::vector projectParts) const { using namespace std::placeholders; std::vector projectPartContainers; projectPartContainers.reserve(projectParts.size()); std::transform(projectParts.begin(), projectParts.end(), std::back_inserter(projectPartContainers), std::bind(&ProjectUpdater::toProjectPartContainer, this, _1)); return projectPartContainers; } Utils::PathStringVector ProjectUpdater::createExcludedPaths( const ClangBackEnd::V2::FileContainers &generatedFiles) { Utils::PathStringVector excludedPaths; excludedPaths.reserve(generatedFiles.size()); auto convertToPath = [] (const ClangBackEnd::V2::FileContainer &fileContainer) { return fileContainer.filePath().path(); }; std::transform(generatedFiles.begin(), generatedFiles.end(), std::back_inserter(excludedPaths), convertToPath); std::sort(excludedPaths.begin(), excludedPaths.end()); return excludedPaths; } } // namespace ClangPchManager