2017-12-12 19:16:45 +08:00
|
|
|
//===--- Context.h - Mechanism for passing implicit data --------*- C++-*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Context for storing and retrieving implicit data. Useful for passing implicit
|
|
|
|
// parameters on a per-request basis.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLVM_CLANG_TOOLS_EXTRA_CLANGD_CONTEXT_H_
|
|
|
|
#define LLVM_CLANG_TOOLS_EXTRA_CLANGD_CONTEXT_H_
|
|
|
|
|
|
|
|
#include "llvm/ADT/STLExtras.h"
|
[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 "llvm/Support/Compiler.h"
|
2017-12-12 19:16:45 +08:00
|
|
|
#include <memory>
|
|
|
|
#include <type_traits>
|
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace clangd {
|
|
|
|
|
[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
|
|
|
/// Values in a Context are indexed by typed keys.
|
|
|
|
/// Key<T> serves two purposes:
|
|
|
|
/// - it provides a lookup key for the context (each Key is unique),
|
|
|
|
/// - it makes lookup type-safe: a Key<T> can only map to a T (or nothing).
|
|
|
|
///
|
|
|
|
/// Example:
|
|
|
|
/// Key<int> RequestID;
|
|
|
|
/// Key<int> Version;
|
|
|
|
///
|
|
|
|
/// Context Ctx = Context::empty().derive(RequestID, 10).derive(Version, 3);
|
|
|
|
/// assert(*Ctx.get(RequestID) == 10);
|
|
|
|
/// assert(*Ctx.get(Version) == 3);
|
|
|
|
///
|
|
|
|
/// Keys are typically used across multiple functions, so most of the time you
|
|
|
|
/// would want to make them static class members or global variables.
|
2017-12-12 19:16:45 +08:00
|
|
|
template <class Type> class Key {
|
|
|
|
public:
|
|
|
|
static_assert(!std::is_reference<Type>::value,
|
|
|
|
"Reference arguments to Key<> are not allowed");
|
|
|
|
|
2018-06-15 20:39:21 +08:00
|
|
|
constexpr Key() = default;
|
2017-12-12 19:16:45 +08:00
|
|
|
|
|
|
|
Key(Key const &) = delete;
|
|
|
|
Key &operator=(Key const &) = delete;
|
|
|
|
Key(Key &&) = delete;
|
|
|
|
Key &operator=(Key &&) = delete;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// A context is an immutable container for per-request data that must be
|
|
|
|
/// propagated through layers that don't care about it. An example is a request
|
|
|
|
/// ID that we may want to use when logging.
|
|
|
|
///
|
|
|
|
/// Conceptually, a context is a heterogeneous map<Key<T>, T>. Each key has
|
|
|
|
/// an associated value type, which allows the map to be typesafe.
|
|
|
|
///
|
[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
|
|
|
/// There is an "ambient" context for each thread, Context::current().
|
|
|
|
/// Most functions should read from this, and use WithContextValue or
|
|
|
|
/// WithContext to extend or replace the context within a block scope.
|
|
|
|
/// Only code dealing with threads and extension points should need to use
|
|
|
|
/// other Context objects.
|
|
|
|
///
|
2017-12-12 19:16:45 +08:00
|
|
|
/// You can't add data to an existing context, instead you create a new
|
|
|
|
/// immutable context derived from it with extra data added. When you retrieve
|
|
|
|
/// data, the context will walk up the parent chain until the key is found.
|
|
|
|
class Context {
|
|
|
|
public:
|
[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
|
|
|
/// Returns an empty root context that contains no data.
|
2017-12-12 19:16:45 +08:00
|
|
|
static Context empty();
|
[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
|
|
|
/// Returns the context for the current thread, creating it if needed.
|
|
|
|
static const Context ¤t();
|
|
|
|
// Sets the current() context to Replacement, and returns the old context.
|
|
|
|
// Prefer to use WithContext or WithContextValue to do this safely.
|
|
|
|
static Context swapCurrent(Context Replacement);
|
2017-12-12 19:16:45 +08:00
|
|
|
|
|
|
|
private:
|
|
|
|
struct Data;
|
|
|
|
Context(std::shared_ptr<const Data> DataPtr);
|
|
|
|
|
|
|
|
public:
|
2017-12-13 21:43:47 +08:00
|
|
|
/// Same as Context::empty(), please use Context::empty() instead.
|
|
|
|
/// Constructor is defined to workaround a bug in MSVC's version of STL.
|
|
|
|
/// (arguments of std::future<> must be default-construcitble in MSVC).
|
|
|
|
Context() = default;
|
|
|
|
|
[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
|
|
|
/// Copy operations for this class are deleted, use an explicit clone() method
|
|
|
|
/// when you need a copy of the context instead.
|
2017-12-12 19:16:45 +08:00
|
|
|
Context(Context const &) = delete;
|
|
|
|
Context &operator=(const Context &) = delete;
|
|
|
|
|
|
|
|
Context(Context &&) = default;
|
|
|
|
Context &operator=(Context &&) = default;
|
|
|
|
|
|
|
|
/// Get data stored for a typed \p Key. If values are not found
|
|
|
|
/// \returns Pointer to the data associated with \p Key. If no data is
|
|
|
|
/// specified for \p Key, return null.
|
|
|
|
template <class Type> const Type *get(const Key<Type> &Key) const {
|
|
|
|
for (const Data *DataPtr = this->DataPtr.get(); DataPtr != nullptr;
|
|
|
|
DataPtr = DataPtr->Parent.get()) {
|
|
|
|
if (DataPtr->KeyPtr == &Key)
|
|
|
|
return static_cast<const Type *>(DataPtr->Value->getValuePtr());
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A helper to get a reference to a \p Key that must exist in the map.
|
|
|
|
/// Must not be called for keys that are not in the map.
|
|
|
|
template <class Type> const Type &getExisting(const Key<Type> &Key) const {
|
|
|
|
auto Val = get(Key);
|
|
|
|
assert(Val && "Key does not exist");
|
|
|
|
return *Val;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Derives a child context
|
[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
|
|
|
/// It is safe to move or destroy a parent context after calling derive().
|
|
|
|
/// The child will keep its parent alive, and its data remains accessible.
|
2017-12-12 19:16:45 +08:00
|
|
|
template <class Type>
|
|
|
|
Context derive(const Key<Type> &Key,
|
|
|
|
typename std::decay<Type>::type Value) const & {
|
|
|
|
return Context(std::make_shared<Data>(Data{
|
|
|
|
/*Parent=*/DataPtr, &Key,
|
|
|
|
llvm::make_unique<TypedAnyStorage<typename std::decay<Type>::type>>(
|
|
|
|
std::move(Value))}));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class Type>
|
|
|
|
Context
|
|
|
|
derive(const Key<Type> &Key,
|
|
|
|
typename std::decay<Type>::type Value) && /* takes ownership */ {
|
|
|
|
return Context(std::make_shared<Data>(Data{
|
|
|
|
/*Parent=*/std::move(DataPtr), &Key,
|
|
|
|
llvm::make_unique<TypedAnyStorage<typename std::decay<Type>::type>>(
|
|
|
|
std::move(Value))}));
|
|
|
|
}
|
|
|
|
|
2018-01-26 17:00:30 +08:00
|
|
|
/// Derives a child context, using an anonymous key.
|
|
|
|
/// Intended for objects stored only for their destructor's side-effect.
|
|
|
|
template <class Type> Context derive(Type &&Value) const & {
|
|
|
|
static Key<typename std::decay<Type>::type> Private;
|
|
|
|
return derive(Private, std::forward<Type>(Value));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class Type> Context derive(Type &&Value) && {
|
|
|
|
static Key<typename std::decay<Type>::type> Private;
|
2018-04-23 23:27:42 +08:00
|
|
|
return std::move(*this).derive(Private, std::forward<Type>(Value));
|
2018-01-26 17:00:30 +08:00
|
|
|
}
|
|
|
|
|
2017-12-12 19:16:45 +08:00
|
|
|
/// Clone this context object.
|
|
|
|
Context clone() const;
|
|
|
|
|
|
|
|
private:
|
|
|
|
class AnyStorage {
|
|
|
|
public:
|
|
|
|
virtual ~AnyStorage() = default;
|
|
|
|
virtual void *getValuePtr() = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class T> class TypedAnyStorage : public Context::AnyStorage {
|
|
|
|
static_assert(std::is_same<typename std::decay<T>::type, T>::value,
|
|
|
|
"Argument to TypedAnyStorage must be decayed");
|
|
|
|
|
|
|
|
public:
|
|
|
|
TypedAnyStorage(T &&Value) : Value(std::move(Value)) {}
|
|
|
|
|
|
|
|
void *getValuePtr() override { return &Value; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
T Value;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Data {
|
|
|
|
// We need to make sure Parent outlives the Value, so the order of members
|
|
|
|
// is important. We do that to allow classes stored in Context's child
|
|
|
|
// layers to store references to the data in the parent layers.
|
|
|
|
std::shared_ptr<const Data> Parent;
|
|
|
|
const void *KeyPtr;
|
|
|
|
std::unique_ptr<AnyStorage> Value;
|
|
|
|
};
|
|
|
|
|
|
|
|
std::shared_ptr<const Data> DataPtr;
|
[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
|
|
|
};
|
|
|
|
|
|
|
|
/// WithContext replaces Context::current() with a provided scope.
|
|
|
|
/// When the WithContext is destroyed, the original scope is restored.
|
|
|
|
/// For extending the current context with new value, prefer WithContextValue.
|
|
|
|
class LLVM_NODISCARD WithContext {
|
|
|
|
public:
|
|
|
|
WithContext(Context C) : Restore(Context::swapCurrent(std::move(C))) {}
|
|
|
|
~WithContext() { Context::swapCurrent(std::move(Restore)); }
|
|
|
|
WithContext(const WithContext &) = delete;
|
|
|
|
WithContext &operator=(const WithContext &) = delete;
|
|
|
|
WithContext(WithContext &&) = delete;
|
|
|
|
WithContext &operator=(WithContext &&) = delete;
|
|
|
|
|
|
|
|
private:
|
|
|
|
Context Restore;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// WithContextValue extends Context::current() with a single value.
|
|
|
|
/// When the WithContextValue is destroyed, the original scope is restored.
|
|
|
|
class LLVM_NODISCARD WithContextValue {
|
|
|
|
public:
|
|
|
|
template <typename T>
|
|
|
|
WithContextValue(const Key<T> &K, typename std::decay<T>::type V)
|
|
|
|
: Restore(Context::current().derive(K, std::move(V))) {}
|
|
|
|
|
|
|
|
// Anonymous values can be used for the destructor side-effect.
|
|
|
|
template <typename T>
|
|
|
|
WithContextValue(T &&V)
|
|
|
|
: Restore(Context::current().derive(std::forward<T>(V))) {}
|
|
|
|
|
|
|
|
private:
|
|
|
|
WithContext Restore;
|
|
|
|
};
|
2017-12-12 19:16:45 +08:00
|
|
|
|
|
|
|
} // namespace clangd
|
|
|
|
} // namespace clang
|
|
|
|
|
|
|
|
#endif // LLVM_CLANG_TOOLS_EXTRA_CLANGD_CONTEXT_H_
|