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

Side by Side 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, 9 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 unified diff | Download patch
OLDNEW
1 /* 1 /*
2 * Copyright (C) 2011 Apple Inc. All rights reserved. 2 * Copyright (C) 2011 Apple Inc. All rights reserved.
3 * 3 *
4 * Redistribution and use in source and binary forms, with or without 4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions 5 * modification, are permitted provided that the following conditions
6 * are met: 6 * are met:
7 * 1. Redistributions of source code must retain the above copyright 7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer. 8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright 9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the 10 * notice, this list of conditions and the following disclaimer in the
(...skipping 23 matching lines...) Expand all
34 #include "wtf/RefPtr.h" 34 #include "wtf/RefPtr.h"
35 #include "wtf/ThreadSafeRefCounted.h" 35 #include "wtf/ThreadSafeRefCounted.h"
36 #include "wtf/WeakPtr.h" 36 #include "wtf/WeakPtr.h"
37 #include <tuple> 37 #include <tuple>
38 #include <utility> 38 #include <utility>
39 39
40 namespace WTF { 40 namespace WTF {
41 41
42 // Functional.h provides a very simple way to bind a function pointer and argume nts together into a function object 42 // Functional.h provides a very simple way to bind a function pointer and argume nts together into a function object
43 // that can be stored, copied and invoked, similar to how boost::bind and std::b ind in C++11. 43 // that can be stored, copied and invoked, similar to how boost::bind and std::b ind in C++11.
44
45 // Thread Safety:
44 // 46 //
45 // Use threadSafeBind() or createCrossThreadTask() if the function/task is 47 // WTF::bind() and SameThreadClosure should be used for same-thread closures
46 // called on a (potentially) different thread from the current thread. 48 // only, i.e. the closures must be created, executed and destructed on
49 // the same thread.
50 // Use threadSafeBind() and CrossThreadClosure if the function/task is called
51 // or destructed on a (potentially) different thread from the current thread.
47 52
48 // WTF::bind() and move semantics 53 // WTF::bind() and move semantics
49 // ============================== 54 // ==============================
50 // 55 //
51 // For unbound parameters (arguments supplied later on the bound functor directl y), there are two ways to pass movable 56 // For unbound parameters (arguments supplied later on the bound functor directl y), there are two ways to pass movable
52 // arguments: 57 // arguments:
53 // 58 //
54 // 1) Pass by rvalue reference. 59 // 1) Pass by rvalue reference.
55 // 60 //
56 // void yourFunction(Argument&& argument) { ... } 61 // void yourFunction(Argument&& argument) { ... }
(...skipping 127 matching lines...) Expand 10 before | Expand all | Expand 10 after
184 static typename RetainPtr<T>::PtrType unwrap(const StorageType& value) { ret urn value.get(); } 189 static typename RetainPtr<T>::PtrType unwrap(const StorageType& value) { ret urn value.get(); }
185 }; 190 };
186 191
187 template<> struct ParamStorageTraits<void*> { 192 template<> struct ParamStorageTraits<void*> {
188 typedef void* StorageType; 193 typedef void* StorageType;
189 194
190 static StorageType wrap(void* value) { return value; } 195 static StorageType wrap(void* value) { return value; }
191 static void* unwrap(const StorageType& value) { return value; } 196 static void* unwrap(const StorageType& value) { return value; }
192 }; 197 };
193 198
194 template<typename> 199 enum FunctionThreadAffinity {
200 CrossThreadAffinity,
201 SameThreadAffinity
202 };
203
204 template<typename, FunctionThreadAffinity threadAffinity = SameThreadAffinity>
195 class Function; 205 class Function;
196 206
197 template<typename R, typename... Args> 207 template<FunctionThreadAffinity threadAffinity, typename R, typename... Args>
198 class Function<R(Args...)> { 208 class Function<R(Args...), threadAffinity> {
199 USING_FAST_MALLOC(Function); 209 USING_FAST_MALLOC(Function);
200 WTF_MAKE_NONCOPYABLE(Function); 210 WTF_MAKE_NONCOPYABLE(Function);
201 public: 211 public:
202 virtual ~Function() { } 212 virtual ~Function() { }
203 virtual R operator()(Args... args) = 0; 213 virtual R operator()(Args... args) = 0;
204 protected: 214 protected:
205 Function() = default; 215 Function() = default;
216 void checkThread() { }
206 }; 217 };
207 218
208 template <typename BoundParametersTuple, typename FunctionWrapper, typename... U nboundParameters> 219 #if ENABLE(ASSERT)
220 template<typename R, typename... Args>
221 class Function<R(Args...), SameThreadAffinity> {
222 USING_FAST_MALLOC(Function);
223 WTF_MAKE_NONCOPYABLE(Function);
224 public:
225 virtual ~Function()
226 {
227 checkThread();
228 }
229 virtual R operator()(Args... args) = 0;
230 protected:
231 Function()
232 : m_createdThread(currentThread())
233 {
234 }
235
236 void NEVER_INLINE checkThread()
237 {
238 // Function with SameThreadAffinity, including SameThreadClosure
239 // created by WTF::bind() or blink::createSameThreadTask(),
240 // must be called and destructed on the thread where it is created.
241 // If it is intended to be used cross-thread, use
242 // blink::threadSafeBind() or blink::createCrossThreadTask() instead.
243 RELEASE_ASSERT(m_createdThread == currentThread());
244 }
245
246 private:
247 const ThreadIdentifier m_createdThread;
248 };
249 #endif
250
251 template <FunctionThreadAffinity threadAffinity, typename BoundParametersTuple, typename FunctionWrapper, typename... UnboundParameters>
209 class PartBoundFunctionImpl; 252 class PartBoundFunctionImpl;
210 253
211 template <typename... BoundParameters, typename FunctionWrapper, typename... Unb oundParameters> 254 template <FunctionThreadAffinity threadAffinity, typename... BoundParameters, ty pename FunctionWrapper, typename... UnboundParameters>
212 class PartBoundFunctionImpl<std::tuple<BoundParameters...>, FunctionWrapper, Unb oundParameters...> final : public Function<typename FunctionWrapper::ResultType( UnboundParameters...)> { 255 class PartBoundFunctionImpl<threadAffinity, std::tuple<BoundParameters...>, Func tionWrapper, UnboundParameters...> final : public Function<typename FunctionWrap per::ResultType(UnboundParameters...), threadAffinity> {
213 public: 256 public:
214 // We would like to use StorageTraits<UnboundParameters>... with StorageTrai ts defined as below in order to obtain 257 // We would like to use StorageTraits<UnboundParameters>... with StorageTrai ts defined as below in order to obtain
215 // storage traits of UnboundParameters, but unfortunately MSVC can't handle template using declarations correctly. 258 // storage traits of UnboundParameters, but unfortunately MSVC can't handle template using declarations correctly.
216 // So, sadly, we have write down the full type signature in all places where storage traits are needed. 259 // So, sadly, we have write down the full type signature in all places where storage traits are needed.
217 // 260 //
218 // template <typename T> 261 // template <typename T>
219 // using StorageTraits = ParamStorageTraits<typename std::decay<T>::type>; 262 // using StorageTraits = ParamStorageTraits<typename std::decay<T>::type>;
220 263
221 // Note that BoundParameters can be const T&, T&& or a mix of these. 264 // Note that BoundParameters can be const T&, T&& or a mix of these.
222 explicit PartBoundFunctionImpl(FunctionWrapper functionWrapper, BoundParamet ers... bound) 265 explicit PartBoundFunctionImpl(FunctionWrapper functionWrapper, BoundParamet ers... bound)
223 : m_functionWrapper(functionWrapper) 266 : m_functionWrapper(functionWrapper)
224 , m_bound(ParamStorageTraits<typename std::decay<BoundParameters>::type> ::wrap(std::forward<BoundParameters>(bound))...) 267 , m_bound(ParamStorageTraits<typename std::decay<BoundParameters>::type> ::wrap(std::forward<BoundParameters>(bound))...)
225 { 268 {
226 } 269 }
227 270
228 typename FunctionWrapper::ResultType operator()(UnboundParameters... unbound ) override 271 typename FunctionWrapper::ResultType operator()(UnboundParameters... unbound ) override
229 { 272 {
230 // What we really want to do is to call m_functionWrapper(m_bound..., un bound...), but to do that we need to 273 // What we really want to do is to call m_functionWrapper(m_bound..., un bound...), but to do that we need to
231 // pass a list of indices to a worker function template. 274 // pass a list of indices to a worker function template.
232 return callInternal(base::MakeIndexSequence<sizeof...(BoundParameters)>( ), std::forward<UnboundParameters>(unbound)...); 275 return callInternal(base::MakeIndexSequence<sizeof...(BoundParameters)>( ), std::forward<UnboundParameters>(unbound)...);
233 } 276 }
234 277
235 private: 278 private:
236 template <std::size_t... boundIndices, typename... IncomingUnboundParameters > 279 template <std::size_t... boundIndices, typename... IncomingUnboundParameters >
237 typename FunctionWrapper::ResultType callInternal(const base::IndexSeque nce<boundIndices...>&, IncomingUnboundParameters&&... unbound) 280 typename FunctionWrapper::ResultType callInternal(const base::IndexSeque nce<boundIndices...>&, IncomingUnboundParameters&&... unbound)
238 { 281 {
282 this->checkThread();
239 // Get each element in m_bound, unwrap them, and call the function with the desired arguments. 283 // Get each element in m_bound, unwrap them, and call the function with the desired arguments.
240 return m_functionWrapper(ParamStorageTraits<typename std::decay<BoundPar ameters>::type>::unwrap(std::get<boundIndices>(m_bound))..., std::forward<Incomi ngUnboundParameters>(unbound)...); 284 return m_functionWrapper(ParamStorageTraits<typename std::decay<BoundPar ameters>::type>::unwrap(std::get<boundIndices>(m_bound))..., std::forward<Incomi ngUnboundParameters>(unbound)...);
241 } 285 }
242 286
243 FunctionWrapper m_functionWrapper; 287 FunctionWrapper m_functionWrapper;
244 std::tuple<typename ParamStorageTraits<typename std::decay<BoundParameters>: :type>::StorageType...> m_bound; 288 std::tuple<typename ParamStorageTraits<typename std::decay<BoundParameters>: :type>::StorageType...> m_bound;
245 }; 289 };
246 290
247 template <typename... UnboundParameters, typename FunctionType, typename... Boun dParameters> 291 template <FunctionThreadAffinity threadAffinity, typename... UnboundParameters, typename FunctionType, typename... BoundParameters>
248 PassOwnPtr<Function<typename FunctionWrapper<FunctionType>::ResultType(UnboundPa rameters...)>> bind(FunctionType function, BoundParameters&&... boundParameters) 292 PassOwnPtr<Function<typename FunctionWrapper<FunctionType>::ResultType(UnboundPa rameters...), threadAffinity>> bindInternal(FunctionType function, BoundParamete rs&&... boundParameters)
249 { 293 {
250 // Bound parameters' types are wrapped with std::tuple so we can pass two te mplate parameter packs (bound 294 // Bound parameters' types are wrapped with std::tuple so we can pass two te mplate parameter packs (bound
251 // parameters and unbound) to PartBoundFunctionImpl. Note that a tuple of th is type isn't actually created; 295 // parameters and unbound) to PartBoundFunctionImpl. Note that a tuple of th is type isn't actually created;
252 // std::tuple<> is just for carrying the bound parameters' types. Any other class template taking a type parameter 296 // std::tuple<> is just for carrying the bound parameters' types. Any other class template taking a type parameter
253 // pack can be used instead of std::tuple. std::tuple is used just because i t's most convenient for this purpose. 297 // pack can be used instead of std::tuple. std::tuple is used just because i t's most convenient for this purpose.
254 using BoundFunctionType = PartBoundFunctionImpl<std::tuple<BoundParameters&& ...>, FunctionWrapper<FunctionType>, UnboundParameters...>; 298 using BoundFunctionType = PartBoundFunctionImpl<threadAffinity, std::tuple<B oundParameters&&...>, FunctionWrapper<FunctionType>, UnboundParameters...>;
255 return adoptPtr(new BoundFunctionType(FunctionWrapper<FunctionType>(function ), std::forward<BoundParameters>(boundParameters)...)); 299 return adoptPtr(new BoundFunctionType(FunctionWrapper<FunctionType>(function ), std::forward<BoundParameters>(boundParameters)...));
256 } 300 }
257 301
258 typedef Function<void()> Closure; 302 template <typename... UnboundParameters, typename FunctionType, typename... Boun dParameters>
303 PassOwnPtr<Function<typename FunctionWrapper<FunctionType>::ResultType(UnboundPa rameters...), SameThreadAffinity>> bind(FunctionType function, BoundParameters&& ... boundParameters)
304 {
305 return bindInternal<SameThreadAffinity, UnboundParameters...>(function, std: :forward<BoundParameters>(boundParameters)...);
306 }
307
308 typedef Function<void(), SameThreadAffinity> SameThreadClosure;
309 typedef Function<void(), CrossThreadAffinity> CrossThreadClosure;
259 310
260 } // namespace WTF 311 } // namespace WTF
261 312
262 using WTF::Function; 313 using WTF::Function;
263 using WTF::bind; 314 using WTF::bind;
264 using WTF::Closure; 315 using WTF::SameThreadClosure;
316 using WTF::CrossThreadClosure;
265 317
266 #endif // WTF_Functional_h 318 #endif // WTF_Functional_h
OLDNEW
« 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