Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(552)

Unified Diff: third_party/WebKit/Source/wtf/Functional.h

Issue 1549143002: Add thread affinity and ASSERT() for same-thread restriction to WTF::Function (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@TRV_ThreadSafeBindByVariadicTemplate
Patch Set: Rebase. Created 4 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « third_party/WebKit/Source/wtf/Forward.h ('k') | third_party/WebKit/public/platform/WebTaskRunner.h » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: third_party/WebKit/Source/wtf/Functional.h
diff --git a/third_party/WebKit/Source/wtf/Functional.h b/third_party/WebKit/Source/wtf/Functional.h
index d274422347d94e6a684054784cea0e864812c219..b56188dfebf113dbef9af4262e79945a7dd1481b 100644
--- a/third_party/WebKit/Source/wtf/Functional.h
+++ b/third_party/WebKit/Source/wtf/Functional.h
@@ -41,9 +41,14 @@ namespace WTF {
// Functional.h provides a very simple way to bind a function pointer and arguments together into a function object
// that can be stored, copied and invoked, similar to how boost::bind and std::bind in C++11.
+
+// Thread Safety:
//
-// Use threadSafeBind() or createCrossThreadTask() if the function/task is
-// called on a (potentially) different thread from the current thread.
+// WTF::bind() and SameThreadClosure should be used for same-thread closures
+// only, i.e. the closures must be created, executed and destructed on
+// the same thread.
+// Use threadSafeBind() and CrossThreadClosure if the function/task is called
+// or destructed on a (potentially) different thread from the current thread.
// WTF::bind() and move semantics
// ==============================
@@ -191,11 +196,16 @@ template<> struct ParamStorageTraits<void*> {
static void* unwrap(const StorageType& value) { return value; }
};
-template<typename>
+enum FunctionThreadAffinity {
+ CrossThreadAffinity,
+ SameThreadAffinity
+};
+
+template<typename, FunctionThreadAffinity threadAffinity = SameThreadAffinity>
class Function;
-template<typename R, typename... Args>
-class Function<R(Args...)> {
+template<FunctionThreadAffinity threadAffinity, typename R, typename... Args>
+class Function<R(Args...), threadAffinity> {
USING_FAST_MALLOC(Function);
WTF_MAKE_NONCOPYABLE(Function);
public:
@@ -203,13 +213,46 @@ public:
virtual R operator()(Args... args) = 0;
protected:
Function() = default;
+ void checkThread() { }
+};
+
+#if ENABLE(ASSERT)
+template<typename R, typename... Args>
+class Function<R(Args...), SameThreadAffinity> {
+ USING_FAST_MALLOC(Function);
+ WTF_MAKE_NONCOPYABLE(Function);
+public:
+ virtual ~Function()
+ {
+ checkThread();
+ }
+ virtual R operator()(Args... args) = 0;
+protected:
+ Function()
+ : m_createdThread(currentThread())
+ {
+ }
+
+ void NEVER_INLINE checkThread()
+ {
+ // Function with SameThreadAffinity, including SameThreadClosure
+ // created by WTF::bind() or blink::createSameThreadTask(),
+ // must be called and destructed on the thread where it is created.
+ // If it is intended to be used cross-thread, use
+ // blink::threadSafeBind() or blink::createCrossThreadTask() instead.
+ RELEASE_ASSERT(m_createdThread == currentThread());
+ }
+
+private:
+ const ThreadIdentifier m_createdThread;
};
+#endif
-template <typename BoundParametersTuple, typename FunctionWrapper, typename... UnboundParameters>
+template <FunctionThreadAffinity threadAffinity, typename BoundParametersTuple, typename FunctionWrapper, typename... UnboundParameters>
class PartBoundFunctionImpl;
-template <typename... BoundParameters, typename FunctionWrapper, typename... UnboundParameters>
-class PartBoundFunctionImpl<std::tuple<BoundParameters...>, FunctionWrapper, UnboundParameters...> final : public Function<typename FunctionWrapper::ResultType(UnboundParameters...)> {
+template <FunctionThreadAffinity threadAffinity, typename... BoundParameters, typename FunctionWrapper, typename... UnboundParameters>
+class PartBoundFunctionImpl<threadAffinity, std::tuple<BoundParameters...>, FunctionWrapper, UnboundParameters...> final : public Function<typename FunctionWrapper::ResultType(UnboundParameters...), threadAffinity> {
public:
// We would like to use StorageTraits<UnboundParameters>... with StorageTraits defined as below in order to obtain
// storage traits of UnboundParameters, but unfortunately MSVC can't handle template using declarations correctly.
@@ -236,6 +279,7 @@ private:
template <std::size_t... boundIndices, typename... IncomingUnboundParameters>
typename FunctionWrapper::ResultType callInternal(const base::IndexSequence<boundIndices...>&, IncomingUnboundParameters&&... unbound)
{
+ this->checkThread();
// Get each element in m_bound, unwrap them, and call the function with the desired arguments.
return m_functionWrapper(ParamStorageTraits<typename std::decay<BoundParameters>::type>::unwrap(std::get<boundIndices>(m_bound))..., std::forward<IncomingUnboundParameters>(unbound)...);
}
@@ -244,23 +288,31 @@ private:
std::tuple<typename ParamStorageTraits<typename std::decay<BoundParameters>::type>::StorageType...> m_bound;
};
-template <typename... UnboundParameters, typename FunctionType, typename... BoundParameters>
-PassOwnPtr<Function<typename FunctionWrapper<FunctionType>::ResultType(UnboundParameters...)>> bind(FunctionType function, BoundParameters&&... boundParameters)
+template <FunctionThreadAffinity threadAffinity, typename... UnboundParameters, typename FunctionType, typename... BoundParameters>
+PassOwnPtr<Function<typename FunctionWrapper<FunctionType>::ResultType(UnboundParameters...), threadAffinity>> bindInternal(FunctionType function, BoundParameters&&... boundParameters)
{
// Bound parameters' types are wrapped with std::tuple so we can pass two template parameter packs (bound
// parameters and unbound) to PartBoundFunctionImpl. Note that a tuple of this type isn't actually created;
// std::tuple<> is just for carrying the bound parameters' types. Any other class template taking a type parameter
// pack can be used instead of std::tuple. std::tuple is used just because it's most convenient for this purpose.
- using BoundFunctionType = PartBoundFunctionImpl<std::tuple<BoundParameters&&...>, FunctionWrapper<FunctionType>, UnboundParameters...>;
+ using BoundFunctionType = PartBoundFunctionImpl<threadAffinity, std::tuple<BoundParameters&&...>, FunctionWrapper<FunctionType>, UnboundParameters...>;
return adoptPtr(new BoundFunctionType(FunctionWrapper<FunctionType>(function), std::forward<BoundParameters>(boundParameters)...));
}
-typedef Function<void()> Closure;
+template <typename... UnboundParameters, typename FunctionType, typename... BoundParameters>
+PassOwnPtr<Function<typename FunctionWrapper<FunctionType>::ResultType(UnboundParameters...), SameThreadAffinity>> bind(FunctionType function, BoundParameters&&... boundParameters)
+{
+ return bindInternal<SameThreadAffinity, UnboundParameters...>(function, std::forward<BoundParameters>(boundParameters)...);
+}
+
+typedef Function<void(), SameThreadAffinity> SameThreadClosure;
+typedef Function<void(), CrossThreadAffinity> CrossThreadClosure;
} // namespace WTF
using WTF::Function;
using WTF::bind;
-using WTF::Closure;
+using WTF::SameThreadClosure;
+using WTF::CrossThreadClosure;
#endif // WTF_Functional_h
« no previous file with comments | « third_party/WebKit/Source/wtf/Forward.h ('k') | third_party/WebKit/public/platform/WebTaskRunner.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698