2018-01-31 16:51:16 +08:00
|
|
|
//===-- TUSchedulerTests.cpp ------------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
[clangd] Pass Context implicitly using TLS.
Summary:
Instead of passing Context explicitly around, we now have a thread-local
Context object `Context::current()` which is an implicit argument to
every function.
Most manipulation of this should use the WithContextValue helper, which
augments the current Context to add a single KV pair, and restores the
old context on destruction.
Advantages are:
- less boilerplate in functions that just propagate contexts
- reading most code doesn't require understanding context at all, and
using context as values in fewer places still
- fewer options to pass the "wrong" context when it changes within a
scope (e.g. when using Span)
- contexts pass through interfaces we can't modify, such as VFS
- propagating contexts across threads was slightly tricky (e.g.
copy vs move, no move-init in lambdas), and is now encapsulated in
the threadpool
Disadvantages are all the usual TLS stuff - hidden magic, and
potential for higher memory usage on threads that don't use the
context. (In practice, it's just one pointer)
Reviewers: ilya-biryukov
Subscribers: klimek, jkorous-apple, ioeric, cfe-commits
Differential Revision: https://reviews.llvm.org/D42517
llvm-svn: 323872
2018-01-31 21:40:48 +08:00
|
|
|
#include "Context.h"
|
2018-01-31 16:51:16 +08:00
|
|
|
#include "TUScheduler.h"
|
|
|
|
#include "TestFS.h"
|
|
|
|
#include "gmock/gmock.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include <utility>
|
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace clangd {
|
|
|
|
|
|
|
|
using ::testing::Pair;
|
|
|
|
using ::testing::Pointee;
|
|
|
|
|
2018-03-12 23:28:22 +08:00
|
|
|
void ignoreUpdate(llvm::Optional<std::vector<Diag>>) {}
|
2018-01-31 16:51:16 +08:00
|
|
|
void ignoreError(llvm::Error Err) {
|
|
|
|
handleAllErrors(std::move(Err), [](const llvm::ErrorInfoBase &) {});
|
|
|
|
}
|
|
|
|
|
|
|
|
class TUSchedulerTests : public ::testing::Test {
|
|
|
|
protected:
|
|
|
|
ParseInputs getInputs(PathRef File, std::string Contents) {
|
|
|
|
return ParseInputs{*CDB.getCompileCommand(File), buildTestFS(Files),
|
|
|
|
std::move(Contents)};
|
|
|
|
}
|
|
|
|
|
2018-02-16 17:41:43 +08:00
|
|
|
llvm::StringMap<std::string> Files;
|
2018-01-31 16:51:16 +08:00
|
|
|
|
|
|
|
private:
|
|
|
|
MockCompilationDatabase CDB;
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(TUSchedulerTests, MissingFiles) {
|
|
|
|
TUScheduler S(getDefaultAsyncThreadsCount(),
|
|
|
|
/*StorePreamblesInMemory=*/true,
|
2018-03-02 16:56:37 +08:00
|
|
|
/*ASTParsedCallback=*/nullptr,
|
|
|
|
/*UpdateDebounce=*/std::chrono::steady_clock::duration::zero());
|
2018-01-31 16:51:16 +08:00
|
|
|
|
2018-02-16 17:41:43 +08:00
|
|
|
auto Added = testPath("added.cpp");
|
|
|
|
Files[Added] = "";
|
2018-01-31 16:51:16 +08:00
|
|
|
|
2018-02-16 17:41:43 +08:00
|
|
|
auto Missing = testPath("missing.cpp");
|
|
|
|
Files[Missing] = "";
|
2018-01-31 16:51:16 +08:00
|
|
|
|
2018-02-22 21:11:12 +08:00
|
|
|
S.update(Added, getInputs(Added, ""), WantDiagnostics::No, ignoreUpdate);
|
2018-01-31 16:51:16 +08:00
|
|
|
|
|
|
|
// Assert each operation for missing file is an error (even if it's available
|
|
|
|
// in VFS).
|
2018-02-19 17:56:28 +08:00
|
|
|
S.runWithAST("", Missing, [&](llvm::Expected<InputsAndAST> AST) {
|
2018-01-31 16:51:16 +08:00
|
|
|
ASSERT_FALSE(bool(AST));
|
|
|
|
ignoreError(AST.takeError());
|
|
|
|
});
|
2018-02-19 17:56:28 +08:00
|
|
|
S.runWithPreamble("", Missing,
|
|
|
|
[&](llvm::Expected<InputsAndPreamble> Preamble) {
|
|
|
|
ASSERT_FALSE(bool(Preamble));
|
|
|
|
ignoreError(Preamble.takeError());
|
|
|
|
});
|
2018-02-08 15:37:35 +08:00
|
|
|
// remove() shouldn't crash on missing files.
|
|
|
|
S.remove(Missing);
|
2018-01-31 16:51:16 +08:00
|
|
|
|
|
|
|
// Assert there aren't any errors for added file.
|
2018-02-19 17:56:28 +08:00
|
|
|
S.runWithAST("", Added, [&](llvm::Expected<InputsAndAST> AST) {
|
|
|
|
EXPECT_TRUE(bool(AST));
|
|
|
|
});
|
|
|
|
S.runWithPreamble("", Added, [&](llvm::Expected<InputsAndPreamble> Preamble) {
|
2018-01-31 16:51:16 +08:00
|
|
|
EXPECT_TRUE(bool(Preamble));
|
|
|
|
});
|
2018-02-08 15:37:35 +08:00
|
|
|
S.remove(Added);
|
2018-01-31 16:51:16 +08:00
|
|
|
|
|
|
|
// Assert that all operations fail after removing the file.
|
2018-02-19 17:56:28 +08:00
|
|
|
S.runWithAST("", Added, [&](llvm::Expected<InputsAndAST> AST) {
|
2018-01-31 16:51:16 +08:00
|
|
|
ASSERT_FALSE(bool(AST));
|
|
|
|
ignoreError(AST.takeError());
|
|
|
|
});
|
2018-02-19 17:56:28 +08:00
|
|
|
S.runWithPreamble("", Added, [&](llvm::Expected<InputsAndPreamble> Preamble) {
|
2018-01-31 16:51:16 +08:00
|
|
|
ASSERT_FALSE(bool(Preamble));
|
|
|
|
ignoreError(Preamble.takeError());
|
|
|
|
});
|
2018-02-08 15:37:35 +08:00
|
|
|
// remove() shouldn't crash on missing files.
|
|
|
|
S.remove(Added);
|
2018-01-31 16:51:16 +08:00
|
|
|
}
|
|
|
|
|
2018-02-22 21:11:12 +08:00
|
|
|
TEST_F(TUSchedulerTests, WantDiagnostics) {
|
|
|
|
std::atomic<int> CallbackCount(0);
|
|
|
|
{
|
2018-02-22 23:33:33 +08:00
|
|
|
// To avoid a racy test, don't allow tasks to actualy run on the worker
|
|
|
|
// thread until we've scheduled them all.
|
|
|
|
Notification Ready;
|
2018-03-02 16:56:37 +08:00
|
|
|
TUScheduler S(
|
|
|
|
getDefaultAsyncThreadsCount(),
|
|
|
|
/*StorePreamblesInMemory=*/true,
|
|
|
|
/*ASTParsedCallback=*/nullptr,
|
|
|
|
/*UpdateDebounce=*/std::chrono::steady_clock::duration::zero());
|
2018-02-22 21:11:12 +08:00
|
|
|
auto Path = testPath("foo.cpp");
|
|
|
|
S.update(Path, getInputs(Path, ""), WantDiagnostics::Yes,
|
2018-03-12 23:28:22 +08:00
|
|
|
[&](std::vector<Diag>) { Ready.wait(); });
|
2018-02-22 21:11:12 +08:00
|
|
|
|
|
|
|
S.update(Path, getInputs(Path, "request diags"), WantDiagnostics::Yes,
|
2018-03-12 23:28:22 +08:00
|
|
|
[&](std::vector<Diag> Diags) { ++CallbackCount; });
|
2018-02-22 21:11:12 +08:00
|
|
|
S.update(Path, getInputs(Path, "auto (clobbered)"), WantDiagnostics::Auto,
|
2018-03-12 23:28:22 +08:00
|
|
|
[&](std::vector<Diag> Diags) {
|
2018-02-22 21:11:12 +08:00
|
|
|
ADD_FAILURE() << "auto should have been cancelled by auto";
|
|
|
|
});
|
|
|
|
S.update(Path, getInputs(Path, "request no diags"), WantDiagnostics::No,
|
2018-03-12 23:28:22 +08:00
|
|
|
[&](std::vector<Diag> Diags) {
|
2018-02-22 21:11:12 +08:00
|
|
|
ADD_FAILURE() << "no diags should not be called back";
|
|
|
|
});
|
|
|
|
S.update(Path, getInputs(Path, "auto (produces)"), WantDiagnostics::Auto,
|
2018-03-12 23:28:22 +08:00
|
|
|
[&](std::vector<Diag> Diags) { ++CallbackCount; });
|
2018-02-22 21:11:12 +08:00
|
|
|
Ready.notify();
|
|
|
|
}
|
|
|
|
EXPECT_EQ(2, CallbackCount);
|
|
|
|
}
|
|
|
|
|
2018-03-02 16:56:37 +08:00
|
|
|
TEST_F(TUSchedulerTests, Debounce) {
|
|
|
|
std::atomic<int> CallbackCount(0);
|
|
|
|
{
|
|
|
|
TUScheduler S(getDefaultAsyncThreadsCount(),
|
|
|
|
/*StorePreamblesInMemory=*/true,
|
|
|
|
/*ASTParsedCallback=*/nullptr,
|
2018-03-03 02:23:41 +08:00
|
|
|
/*UpdateDebounce=*/std::chrono::seconds(1));
|
|
|
|
// FIXME: we could probably use timeouts lower than 1 second here.
|
2018-03-02 16:56:37 +08:00
|
|
|
auto Path = testPath("foo.cpp");
|
|
|
|
S.update(Path, getInputs(Path, "auto (debounced)"), WantDiagnostics::Auto,
|
2018-03-12 23:28:22 +08:00
|
|
|
[&](std::vector<Diag> Diags) {
|
2018-03-02 16:56:37 +08:00
|
|
|
ADD_FAILURE() << "auto should have been debounced and canceled";
|
|
|
|
});
|
2018-03-03 02:23:41 +08:00
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(200));
|
2018-03-02 16:56:37 +08:00
|
|
|
S.update(Path, getInputs(Path, "auto (timed out)"), WantDiagnostics::Auto,
|
2018-03-12 23:28:22 +08:00
|
|
|
[&](std::vector<Diag> Diags) { ++CallbackCount; });
|
2018-03-03 02:23:41 +08:00
|
|
|
std::this_thread::sleep_for(std::chrono::seconds(2));
|
2018-03-02 16:56:37 +08:00
|
|
|
S.update(Path, getInputs(Path, "auto (shut down)"), WantDiagnostics::Auto,
|
2018-03-12 23:28:22 +08:00
|
|
|
[&](std::vector<Diag> Diags) { ++CallbackCount; });
|
2018-03-02 16:56:37 +08:00
|
|
|
}
|
|
|
|
EXPECT_EQ(2, CallbackCount);
|
|
|
|
}
|
|
|
|
|
2018-01-31 16:51:16 +08:00
|
|
|
TEST_F(TUSchedulerTests, ManyUpdates) {
|
|
|
|
const int FilesCount = 3;
|
|
|
|
const int UpdatesPerFile = 10;
|
|
|
|
|
|
|
|
std::mutex Mut;
|
|
|
|
int TotalASTReads = 0;
|
|
|
|
int TotalPreambleReads = 0;
|
|
|
|
int TotalUpdates = 0;
|
|
|
|
|
|
|
|
// Run TUScheduler and collect some stats.
|
|
|
|
{
|
|
|
|
TUScheduler S(getDefaultAsyncThreadsCount(),
|
|
|
|
/*StorePreamblesInMemory=*/true,
|
2018-03-02 16:56:37 +08:00
|
|
|
/*ASTParsedCallback=*/nullptr,
|
|
|
|
/*UpdateDebounce=*/std::chrono::milliseconds(50));
|
2018-01-31 16:51:16 +08:00
|
|
|
|
|
|
|
std::vector<std::string> Files;
|
|
|
|
for (int I = 0; I < FilesCount; ++I) {
|
2018-02-16 17:41:43 +08:00
|
|
|
std::string Name = "foo" + std::to_string(I) + ".cpp";
|
|
|
|
Files.push_back(testPath(Name));
|
|
|
|
this->Files[Files.back()] = "";
|
2018-01-31 16:51:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
llvm::StringRef Contents1 = R"cpp(int a;)cpp";
|
|
|
|
llvm::StringRef Contents2 = R"cpp(int main() { return 1; })cpp";
|
|
|
|
llvm::StringRef Contents3 =
|
|
|
|
R"cpp(int a; int b; int sum() { return a + b; })cpp";
|
|
|
|
|
|
|
|
llvm::StringRef AllContents[] = {Contents1, Contents2, Contents3};
|
|
|
|
const int AllContentsSize = 3;
|
|
|
|
|
[clangd] Pass Context implicitly using TLS.
Summary:
Instead of passing Context explicitly around, we now have a thread-local
Context object `Context::current()` which is an implicit argument to
every function.
Most manipulation of this should use the WithContextValue helper, which
augments the current Context to add a single KV pair, and restores the
old context on destruction.
Advantages are:
- less boilerplate in functions that just propagate contexts
- reading most code doesn't require understanding context at all, and
using context as values in fewer places still
- fewer options to pass the "wrong" context when it changes within a
scope (e.g. when using Span)
- contexts pass through interfaces we can't modify, such as VFS
- propagating contexts across threads was slightly tricky (e.g.
copy vs move, no move-init in lambdas), and is now encapsulated in
the threadpool
Disadvantages are all the usual TLS stuff - hidden magic, and
potential for higher memory usage on threads that don't use the
context. (In practice, it's just one pointer)
Reviewers: ilya-biryukov
Subscribers: klimek, jkorous-apple, ioeric, cfe-commits
Differential Revision: https://reviews.llvm.org/D42517
llvm-svn: 323872
2018-01-31 21:40:48 +08:00
|
|
|
// Scheduler may run tasks asynchronously, but should propagate the context.
|
|
|
|
// We stash a nonce in the context, and verify it in the task.
|
|
|
|
static Key<int> NonceKey;
|
|
|
|
int Nonce = 0;
|
|
|
|
|
2018-01-31 16:51:16 +08:00
|
|
|
for (int FileI = 0; FileI < FilesCount; ++FileI) {
|
|
|
|
for (int UpdateI = 0; UpdateI < UpdatesPerFile; ++UpdateI) {
|
|
|
|
auto Contents = AllContents[(FileI + UpdateI) % AllContentsSize];
|
|
|
|
|
|
|
|
auto File = Files[FileI];
|
|
|
|
auto Inputs = getInputs(File, Contents.str());
|
[clangd] Pass Context implicitly using TLS.
Summary:
Instead of passing Context explicitly around, we now have a thread-local
Context object `Context::current()` which is an implicit argument to
every function.
Most manipulation of this should use the WithContextValue helper, which
augments the current Context to add a single KV pair, and restores the
old context on destruction.
Advantages are:
- less boilerplate in functions that just propagate contexts
- reading most code doesn't require understanding context at all, and
using context as values in fewer places still
- fewer options to pass the "wrong" context when it changes within a
scope (e.g. when using Span)
- contexts pass through interfaces we can't modify, such as VFS
- propagating contexts across threads was slightly tricky (e.g.
copy vs move, no move-init in lambdas), and is now encapsulated in
the threadpool
Disadvantages are all the usual TLS stuff - hidden magic, and
potential for higher memory usage on threads that don't use the
context. (In practice, it's just one pointer)
Reviewers: ilya-biryukov
Subscribers: klimek, jkorous-apple, ioeric, cfe-commits
Differential Revision: https://reviews.llvm.org/D42517
llvm-svn: 323872
2018-01-31 21:40:48 +08:00
|
|
|
|
|
|
|
{
|
|
|
|
WithContextValue WithNonce(NonceKey, ++Nonce);
|
2018-02-22 21:11:12 +08:00
|
|
|
S.update(File, Inputs, WantDiagnostics::Auto,
|
2018-03-12 23:28:22 +08:00
|
|
|
[Nonce, &Mut,
|
|
|
|
&TotalUpdates](llvm::Optional<std::vector<Diag>> Diags) {
|
[clangd] Pass Context implicitly using TLS.
Summary:
Instead of passing Context explicitly around, we now have a thread-local
Context object `Context::current()` which is an implicit argument to
every function.
Most manipulation of this should use the WithContextValue helper, which
augments the current Context to add a single KV pair, and restores the
old context on destruction.
Advantages are:
- less boilerplate in functions that just propagate contexts
- reading most code doesn't require understanding context at all, and
using context as values in fewer places still
- fewer options to pass the "wrong" context when it changes within a
scope (e.g. when using Span)
- contexts pass through interfaces we can't modify, such as VFS
- propagating contexts across threads was slightly tricky (e.g.
copy vs move, no move-init in lambdas), and is now encapsulated in
the threadpool
Disadvantages are all the usual TLS stuff - hidden magic, and
potential for higher memory usage on threads that don't use the
context. (In practice, it's just one pointer)
Reviewers: ilya-biryukov
Subscribers: klimek, jkorous-apple, ioeric, cfe-commits
Differential Revision: https://reviews.llvm.org/D42517
llvm-svn: 323872
2018-01-31 21:40:48 +08:00
|
|
|
EXPECT_THAT(Context::current().get(NonceKey),
|
|
|
|
Pointee(Nonce));
|
|
|
|
|
|
|
|
std::lock_guard<std::mutex> Lock(Mut);
|
|
|
|
++TotalUpdates;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
WithContextValue WithNonce(NonceKey, ++Nonce);
|
2018-02-19 17:56:28 +08:00
|
|
|
S.runWithAST("CheckAST", File,
|
|
|
|
[Inputs, Nonce, &Mut,
|
|
|
|
&TotalASTReads](llvm::Expected<InputsAndAST> AST) {
|
|
|
|
EXPECT_THAT(Context::current().get(NonceKey),
|
|
|
|
Pointee(Nonce));
|
|
|
|
|
|
|
|
ASSERT_TRUE((bool)AST);
|
|
|
|
EXPECT_EQ(AST->Inputs.FS, Inputs.FS);
|
|
|
|
EXPECT_EQ(AST->Inputs.Contents, Inputs.Contents);
|
|
|
|
|
|
|
|
std::lock_guard<std::mutex> Lock(Mut);
|
|
|
|
++TotalASTReads;
|
|
|
|
});
|
[clangd] Pass Context implicitly using TLS.
Summary:
Instead of passing Context explicitly around, we now have a thread-local
Context object `Context::current()` which is an implicit argument to
every function.
Most manipulation of this should use the WithContextValue helper, which
augments the current Context to add a single KV pair, and restores the
old context on destruction.
Advantages are:
- less boilerplate in functions that just propagate contexts
- reading most code doesn't require understanding context at all, and
using context as values in fewer places still
- fewer options to pass the "wrong" context when it changes within a
scope (e.g. when using Span)
- contexts pass through interfaces we can't modify, such as VFS
- propagating contexts across threads was slightly tricky (e.g.
copy vs move, no move-init in lambdas), and is now encapsulated in
the threadpool
Disadvantages are all the usual TLS stuff - hidden magic, and
potential for higher memory usage on threads that don't use the
context. (In practice, it's just one pointer)
Reviewers: ilya-biryukov
Subscribers: klimek, jkorous-apple, ioeric, cfe-commits
Differential Revision: https://reviews.llvm.org/D42517
llvm-svn: 323872
2018-01-31 21:40:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
WithContextValue WithNonce(NonceKey, ++Nonce);
|
|
|
|
S.runWithPreamble(
|
2018-02-19 17:56:28 +08:00
|
|
|
"CheckPreamble", File,
|
|
|
|
[Inputs, Nonce, &Mut, &TotalPreambleReads](
|
|
|
|
llvm::Expected<InputsAndPreamble> Preamble) {
|
[clangd] Pass Context implicitly using TLS.
Summary:
Instead of passing Context explicitly around, we now have a thread-local
Context object `Context::current()` which is an implicit argument to
every function.
Most manipulation of this should use the WithContextValue helper, which
augments the current Context to add a single KV pair, and restores the
old context on destruction.
Advantages are:
- less boilerplate in functions that just propagate contexts
- reading most code doesn't require understanding context at all, and
using context as values in fewer places still
- fewer options to pass the "wrong" context when it changes within a
scope (e.g. when using Span)
- contexts pass through interfaces we can't modify, such as VFS
- propagating contexts across threads was slightly tricky (e.g.
copy vs move, no move-init in lambdas), and is now encapsulated in
the threadpool
Disadvantages are all the usual TLS stuff - hidden magic, and
potential for higher memory usage on threads that don't use the
context. (In practice, it's just one pointer)
Reviewers: ilya-biryukov
Subscribers: klimek, jkorous-apple, ioeric, cfe-commits
Differential Revision: https://reviews.llvm.org/D42517
llvm-svn: 323872
2018-01-31 21:40:48 +08:00
|
|
|
EXPECT_THAT(Context::current().get(NonceKey), Pointee(Nonce));
|
|
|
|
|
|
|
|
ASSERT_TRUE((bool)Preamble);
|
2018-03-15 01:46:52 +08:00
|
|
|
EXPECT_EQ(Preamble->Contents, Inputs.Contents);
|
[clangd] Pass Context implicitly using TLS.
Summary:
Instead of passing Context explicitly around, we now have a thread-local
Context object `Context::current()` which is an implicit argument to
every function.
Most manipulation of this should use the WithContextValue helper, which
augments the current Context to add a single KV pair, and restores the
old context on destruction.
Advantages are:
- less boilerplate in functions that just propagate contexts
- reading most code doesn't require understanding context at all, and
using context as values in fewer places still
- fewer options to pass the "wrong" context when it changes within a
scope (e.g. when using Span)
- contexts pass through interfaces we can't modify, such as VFS
- propagating contexts across threads was slightly tricky (e.g.
copy vs move, no move-init in lambdas), and is now encapsulated in
the threadpool
Disadvantages are all the usual TLS stuff - hidden magic, and
potential for higher memory usage on threads that don't use the
context. (In practice, it's just one pointer)
Reviewers: ilya-biryukov
Subscribers: klimek, jkorous-apple, ioeric, cfe-commits
Differential Revision: https://reviews.llvm.org/D42517
llvm-svn: 323872
2018-01-31 21:40:48 +08:00
|
|
|
|
|
|
|
std::lock_guard<std::mutex> Lock(Mut);
|
|
|
|
++TotalPreambleReads;
|
|
|
|
});
|
|
|
|
}
|
2018-01-31 16:51:16 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} // TUScheduler destructor waits for all operations to finish.
|
|
|
|
|
|
|
|
std::lock_guard<std::mutex> Lock(Mut);
|
|
|
|
EXPECT_EQ(TotalUpdates, FilesCount * UpdatesPerFile);
|
|
|
|
EXPECT_EQ(TotalASTReads, FilesCount * UpdatesPerFile);
|
|
|
|
EXPECT_EQ(TotalPreambleReads, FilesCount * UpdatesPerFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace clangd
|
|
|
|
} // namespace clang
|