/**************************************************************************** ** ** Copyright (C) 2017 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 "clangquerygatherer.h" #include "clangquery.h" namespace ClangBackEnd { ClangQueryGatherer::ClangQueryGatherer(FilePathCachingInterface *filePathCache, std::vector &&sources, std::vector &&unsaved, Utils::SmallString &&query) : m_filePathCache(filePathCache), m_sourceRangeFilter(sources.size()), m_sources(std::move(sources)), m_unsaved(std::move(unsaved)), m_query(std::move(query)) { } SourceRangesForQueryMessage ClangQueryGatherer::createSourceRangesForSource( FilePathCachingInterface *filePathCache, V2::FileContainer &&source, const std::vector &unsaved, Utils::SmallString &&query) { ClangQuery clangQuery(*filePathCache, std::move(query)); clangQuery.addFile(std::move(source.filePath), std::move(source.unsavedFileContent), std::move(source.commandLineArguments)); clangQuery.addUnsavedFiles(unsaved); clangQuery.findLocations(); return {clangQuery.takeSourceRanges()}; } bool ClangQueryGatherer::canCreateSourceRanges() const { return !m_sources.empty(); } SourceRangesForQueryMessage ClangQueryGatherer::createNextSourceRanges() { auto message = createSourceRangesForSource(m_filePathCache, std::move(m_sources.back()), m_unsaved, m_query.clone()); m_sources.pop_back(); return message; } ClangQueryGatherer::Future ClangQueryGatherer::startCreateNextSourceRangesMessage() { Future future = std::async(std::launch::async, createSourceRangesForSource, m_filePathCache, std::move(m_sources.back()), m_unsaved, m_query.clone()); m_sources.pop_back(); return future; } void ClangQueryGatherer::startCreateNextSourceRangesMessages() { std::vector futures; while (!m_sources.empty() && m_sourceFutures.size() < m_processingSlotCount) m_sourceFutures.push_back(startCreateNextSourceRangesMessage()); } void ClangQueryGatherer::waitForFinished() { for (Future &future : m_sourceFutures) future.wait(); } bool ClangQueryGatherer::isFinished() const { return m_sources.empty() && m_sourceFutures.empty(); } const std::vector &ClangQueryGatherer::sources() const { return m_sources; } const std::vector &ClangQueryGatherer::sourceFutures() const { return m_sourceFutures; } std::vector ClangQueryGatherer::allCurrentProcessedMessages() { std::vector messages; for (Future &future : m_sourceFutures) messages.push_back(m_sourceRangeFilter.removeDuplicates(future.get())); return messages; } std::vector ClangQueryGatherer::finishedMessages() { std::vector messages; for (auto &&future : finishedFutures()) messages.push_back(m_sourceRangeFilter.removeDuplicates(future.get())); return messages; } void ClangQueryGatherer::setProcessingSlotCount(uint count) { m_processingSlotCount = count; } std::vector ClangQueryGatherer::finishedFutures() { std::vector finishedFutures; finishedFutures.reserve(m_sourceFutures.size()); auto beginReady = std::partition(m_sourceFutures.begin(), m_sourceFutures.end(), [] (const Future &future) { return future.wait_for(std::chrono::duration::zero()) != std::future_status::ready; }); std::move(beginReady, m_sourceFutures.end(), std::back_inserter(finishedFutures)); m_sourceFutures.erase(beginReady, m_sourceFutures.end()); return finishedFutures; } } // namespace ClangBackEnd