2016-08-04 15:26:53 +02: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.
|
|
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "refactoringserver.h"
|
|
|
|
|
|
|
|
|
|
#include "symbolfinder.h"
|
2016-11-15 15:38:12 +01:00
|
|
|
#include "clangquery.h"
|
2016-08-04 15:26:53 +02:00
|
|
|
|
|
|
|
|
#include <refactoringclientinterface.h>
|
|
|
|
|
#include <requestsourcelocationforrenamingmessage.h>
|
2016-11-15 15:38:12 +01:00
|
|
|
#include <requestsourcerangesanddiagnosticsforquerymessage.h>
|
2016-08-04 15:26:53 +02:00
|
|
|
#include <sourcelocationsforrenamingmessage.h>
|
2016-11-15 15:38:12 +01:00
|
|
|
#include <sourcerangesanddiagnosticsforquerymessage.h>
|
2016-08-04 15:26:53 +02:00
|
|
|
|
|
|
|
|
#include <QCoreApplication>
|
|
|
|
|
|
2016-11-15 15:38:12 +01:00
|
|
|
#include <algorithm>
|
|
|
|
|
#include <chrono>
|
|
|
|
|
#include <future>
|
2016-11-23 13:13:38 +01:00
|
|
|
#include <atomic>
|
2016-11-15 15:38:12 +01:00
|
|
|
|
2016-08-04 15:26:53 +02:00
|
|
|
namespace ClangBackEnd {
|
|
|
|
|
|
|
|
|
|
RefactoringServer::RefactoringServer()
|
|
|
|
|
{
|
2016-11-23 13:13:38 +01:00
|
|
|
pollEventLoop = [] () { QCoreApplication::processEvents(); };
|
2016-08-04 15:26:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void RefactoringServer::end()
|
|
|
|
|
{
|
|
|
|
|
QCoreApplication::exit();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void RefactoringServer::requestSourceLocationsForRenamingMessage(RequestSourceLocationsForRenamingMessage &&message)
|
|
|
|
|
{
|
|
|
|
|
SymbolFinder symbolFinder(message.line(), message.column());
|
|
|
|
|
|
|
|
|
|
symbolFinder.addFile(message.filePath().directory(),
|
|
|
|
|
message.filePath().name(),
|
|
|
|
|
message.unsavedContent(),
|
|
|
|
|
message.commandLine());
|
|
|
|
|
|
|
|
|
|
symbolFinder.findSymbol();
|
|
|
|
|
|
2016-11-15 15:38:12 +01:00
|
|
|
client()->sourceLocationsForRenamingMessage({symbolFinder.takeSymbolName(),
|
|
|
|
|
symbolFinder.takeSourceLocations(),
|
|
|
|
|
message.textDocumentRevision()});
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-23 13:13:38 +01:00
|
|
|
void RefactoringServer::requestSourceRangesAndDiagnosticsForQueryMessage(
|
|
|
|
|
RequestSourceRangesAndDiagnosticsForQueryMessage &&message)
|
|
|
|
|
{
|
2016-11-30 15:29:36 +01:00
|
|
|
gatherSourceRangesAndDiagnosticsForQueryMessage(message.takeSources(),
|
|
|
|
|
message.takeUnsavedContent(),
|
|
|
|
|
message.takeQuery());
|
2016-11-23 13:13:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void RefactoringServer::cancel()
|
|
|
|
|
{
|
|
|
|
|
cancelWork = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool RefactoringServer::isCancelingJobs() const
|
|
|
|
|
{
|
|
|
|
|
return cancelWork;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void RefactoringServer::supersedePollEventLoop(std::function<void ()> &&pollEventLoop)
|
|
|
|
|
{
|
|
|
|
|
this->pollEventLoop = std::move(pollEventLoop);
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-15 15:38:12 +01:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
|
|
SourceRangesAndDiagnosticsForQueryMessage createSourceRangesAndDiagnosticsForQueryMessage(
|
2016-11-30 15:29:36 +01:00
|
|
|
V2::FileContainer &&source,
|
|
|
|
|
std::vector<V2::FileContainer> &&unsaved,
|
2016-11-23 13:13:38 +01:00
|
|
|
Utils::SmallString &&query,
|
|
|
|
|
const std::atomic_bool &cancelWork) {
|
2016-11-15 15:38:12 +01:00
|
|
|
ClangQuery clangQuery(std::move(query));
|
|
|
|
|
|
2016-11-23 13:13:38 +01:00
|
|
|
if (!cancelWork) {
|
2016-11-30 15:29:36 +01:00
|
|
|
clangQuery.addFile(source.filePath().directory(),
|
|
|
|
|
source.filePath().name(),
|
|
|
|
|
source.takeUnsavedFileContent(),
|
|
|
|
|
source.takeCommandLineArguments());
|
|
|
|
|
|
|
|
|
|
clangQuery.addUnsavedFiles(std::move(unsaved));
|
2016-11-15 15:38:12 +01:00
|
|
|
|
2016-11-23 13:13:38 +01:00
|
|
|
clangQuery.findLocations();
|
|
|
|
|
}
|
2016-11-15 15:38:12 +01:00
|
|
|
|
|
|
|
|
return {clangQuery.takeSourceRanges(), clangQuery.takeDiagnosticContainers()};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void RefactoringServer::gatherSourceRangesAndDiagnosticsForQueryMessage(
|
2016-11-30 15:29:36 +01:00
|
|
|
std::vector<V2::FileContainer> &&sources,
|
|
|
|
|
std::vector<V2::FileContainer> &&unsaved,
|
2016-11-15 15:38:12 +01:00
|
|
|
Utils::SmallString &&query)
|
|
|
|
|
{
|
|
|
|
|
std::vector<Future> futures;
|
|
|
|
|
|
2016-12-07 17:21:44 +01:00
|
|
|
#ifdef _WIN32
|
|
|
|
|
std::size_t freeProcessors = 1;
|
|
|
|
|
#else
|
2016-11-15 15:38:12 +01:00
|
|
|
std::size_t freeProcessors = std::thread::hardware_concurrency();
|
2016-12-07 17:21:44 +01:00
|
|
|
#endif
|
2016-11-15 15:38:12 +01:00
|
|
|
|
2016-11-30 15:29:36 +01:00
|
|
|
while (!sources.empty() || !futures.empty()) {
|
2016-11-15 15:38:12 +01:00
|
|
|
--freeProcessors;
|
|
|
|
|
|
2016-11-30 15:29:36 +01:00
|
|
|
if (!sources.empty()) {
|
2016-11-15 15:38:12 +01:00
|
|
|
Future &&future = std::async(std::launch::async,
|
|
|
|
|
createSourceRangesAndDiagnosticsForQueryMessage,
|
2016-11-30 15:29:36 +01:00
|
|
|
std::move(sources.back()),
|
|
|
|
|
Utils::clone(unsaved),
|
|
|
|
|
query.clone(),
|
|
|
|
|
std::ref(cancelWork));
|
|
|
|
|
sources.pop_back();
|
2016-11-15 15:38:12 +01:00
|
|
|
|
|
|
|
|
futures.emplace_back(std::move(future));
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-30 15:29:36 +01:00
|
|
|
if (freeProcessors == 0 || sources.empty())
|
2016-11-15 15:38:12 +01:00
|
|
|
freeProcessors += waitForNewSourceRangesAndDiagnosticsForQueryMessage(futures);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::size_t RefactoringServer::waitForNewSourceRangesAndDiagnosticsForQueryMessage(std::vector<Future> &futures)
|
|
|
|
|
{
|
|
|
|
|
while (true) {
|
2016-11-23 13:13:38 +01:00
|
|
|
pollEventLoop();
|
|
|
|
|
|
2016-11-15 15:38:12 +01:00
|
|
|
std::vector<Future> readyFutures;
|
|
|
|
|
readyFutures.reserve(futures.size());
|
|
|
|
|
|
|
|
|
|
auto beginReady = std::partition(futures.begin(),
|
|
|
|
|
futures.end(),
|
|
|
|
|
[] (const Future &future) {
|
|
|
|
|
return future.wait_for(std::chrono::duration<int>::zero()) != std::future_status::ready;
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
std::move(beginReady, futures.end(), std::back_inserter(readyFutures));
|
|
|
|
|
futures.erase(beginReady, futures.end());
|
|
|
|
|
|
|
|
|
|
for (Future &readyFuture : readyFutures)
|
2016-12-08 11:46:31 +01:00
|
|
|
client()->sourceRangesAndDiagnosticsForQueryMessage(readyFuture.get());
|
2016-11-15 15:38:12 +01:00
|
|
|
|
|
|
|
|
if (readyFutures.empty())
|
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(20));
|
|
|
|
|
else
|
|
|
|
|
return readyFutures.size();
|
|
|
|
|
}
|
2016-08-04 15:26:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace ClangBackEnd
|