Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 // This file contains utility functions and classes that help the | 5 // This file contains utility functions and classes that help the |
| 6 // implementation, and management of the Callback objects. | 6 // implementation, and management of the Callback objects. |
| 7 | 7 |
| 8 #ifndef BASE_CALLBACK_INTERNAL_H_ | 8 #ifndef BASE_CALLBACK_INTERNAL_H_ |
| 9 #define BASE_CALLBACK_INTERNAL_H_ | 9 #define BASE_CALLBACK_INTERNAL_H_ |
| 10 | 10 |
| 11 #include "base/atomic_ref_count.h" | |
| 12 #include "base/base_export.h" | 11 #include "base/base_export.h" |
| 13 #include "base/callback_forward.h" | 12 #include "base/callback_forward.h" |
| 14 #include "base/macros.h" | 13 #include "base/macros.h" |
| 15 #include "base/memory/ref_counted.h" | 14 #include "base/memory/ref_counted.h" |
| 16 | 15 |
| 17 namespace base { | 16 namespace base { |
| 17 | |
| 18 struct FakeBindState; | |
| 19 | |
| 18 namespace internal { | 20 namespace internal { |
| 21 | |
| 19 template <CopyMode copy_mode> | 22 template <CopyMode copy_mode> |
| 20 class CallbackBase; | 23 class CallbackBase; |
| 21 | 24 |
| 25 class BindStateBase; | |
| 26 | |
| 27 template <typename Functor, typename... BoundArgs> | |
| 28 struct BindState; | |
| 29 | |
| 30 struct BindStateBaseRefCountTraits { | |
| 31 static void Destruct(const BindStateBase*); | |
| 32 }; | |
| 33 | |
| 22 // BindStateBase is used to provide an opaque handle that the Callback | 34 // BindStateBase is used to provide an opaque handle that the Callback |
| 23 // class can use to represent a function object with bound arguments. It | 35 // class can use to represent a function object with bound arguments. It |
| 24 // behaves as an existential type that is used by a corresponding | 36 // behaves as an existential type that is used by a corresponding |
| 25 // DoInvoke function to perform the function execution. This allows | 37 // DoInvoke function to perform the function execution. This allows |
| 26 // us to shield the Callback class from the types of the bound argument via | 38 // us to shield the Callback class from the types of the bound argument via |
| 27 // "type erasure." | 39 // "type erasure." |
| 28 // At the base level, the only task is to add reference counting data. Don't use | 40 // At the base level, the only task is to add reference counting data. Don't use |
| 29 // RefCountedThreadSafe since it requires the destructor to be a virtual method. | 41 // RefCountedThreadSafe since it requires the destructor to be a virtual method. |
| 30 // Creating a vtable for every BindState template instantiation results in a lot | 42 // Creating a vtable for every BindState template instantiation results in a lot |
| 31 // of bloat. Its only task is to call the destructor which can be done with a | 43 // of bloat. Its only task is to call the destructor which can be done with a |
| 32 // function pointer. | 44 // function pointer. |
| 33 class BASE_EXPORT BindStateBase { | 45 class BASE_EXPORT BindStateBase |
| 46 : public RefCountedThreadSafe<BindStateBase, BindStateBaseRefCountTraits> { | |
| 34 public: | 47 public: |
| 35 using InvokeFuncStorage = void(*)(); | 48 using InvokeFuncStorage = void(*)(); |
| 36 | 49 |
| 37 protected: | 50 private: |
| 38 BindStateBase(InvokeFuncStorage polymorphic_invoke, | 51 BindStateBase(InvokeFuncStorage polymorphic_invoke, |
| 39 void (*destructor)(const BindStateBase*)); | 52 void (*destructor)(const BindStateBase*)); |
| 40 BindStateBase(InvokeFuncStorage polymorphic_invoke, | 53 BindStateBase(InvokeFuncStorage polymorphic_invoke, |
| 41 void (*destructor)(const BindStateBase*), | 54 void (*destructor)(const BindStateBase*), |
| 42 bool (*is_cancelled)(const BindStateBase*)); | 55 bool (*is_cancelled)(const BindStateBase*)); |
| 56 | |
| 43 ~BindStateBase() = default; | 57 ~BindStateBase() = default; |
| 44 | 58 |
| 45 private: | 59 friend struct BindStateBaseRefCountTraits; |
| 46 friend class scoped_refptr<BindStateBase>; | 60 friend class RefCountedThreadSafe<BindStateBase, BindStateBaseRefCountTraits>; |
| 61 | |
| 47 template <CopyMode copy_mode> | 62 template <CopyMode copy_mode> |
| 48 friend class CallbackBase; | 63 friend class CallbackBase; |
| 49 | 64 |
| 65 // Whitelist subclasses that access the destructor of BindStateBase. | |
|
dcheng
2017/03/22 00:26:17
Hmm, wouldn't it be easier just to keep the destru
tzik
2017/03/22 01:50:05
That hits our clang plugin check. It requires the
dcheng
2017/03/27 07:15:26
Hmm... so this seems pretty dangerous (though it's
tzik
2017/03/29 08:29:47
Do we have a pitfall where we may easily fall into
dcheng
2017/03/29 17:38:00
Right, this is the thing I'm concerned about. I do
tzik
2017/03/29 18:25:26
CallbackBase holds it as a scoped_refptr of BindSt
dcheng
2017/03/30 04:11:02
Ah... right. This is a bit subtle, but I guess it'
| |
| 66 template <typename Functor, typename... BoundArgs> | |
| 67 friend struct BindState; | |
| 68 friend struct ::base::FakeBindState; | |
| 69 | |
| 50 bool IsCancelled() const { | 70 bool IsCancelled() const { |
| 51 return is_cancelled_(this); | 71 return is_cancelled_(this); |
| 52 } | 72 } |
| 53 | 73 |
| 54 void AddRef() const; | |
| 55 void Release() const; | |
| 56 | |
| 57 // In C++, it is safe to cast function pointers to function pointers of | 74 // In C++, it is safe to cast function pointers to function pointers of |
| 58 // another type. It is not okay to use void*. We create a InvokeFuncStorage | 75 // another type. It is not okay to use void*. We create a InvokeFuncStorage |
| 59 // that that can store our function pointer, and then cast it back to | 76 // that that can store our function pointer, and then cast it back to |
| 60 // the original type on usage. | 77 // the original type on usage. |
| 61 InvokeFuncStorage polymorphic_invoke_; | 78 InvokeFuncStorage polymorphic_invoke_; |
| 62 | 79 |
| 63 mutable AtomicRefCount ref_count_; | |
| 64 | |
| 65 // Pointer to a function that will properly destroy |this|. | 80 // Pointer to a function that will properly destroy |this|. |
| 66 void (*destructor_)(const BindStateBase*); | 81 void (*destructor_)(const BindStateBase*); |
| 67 bool (*is_cancelled_)(const BindStateBase*); | 82 bool (*is_cancelled_)(const BindStateBase*); |
| 68 | 83 |
| 69 DISALLOW_COPY_AND_ASSIGN(BindStateBase); | 84 DISALLOW_COPY_AND_ASSIGN(BindStateBase); |
| 70 }; | 85 }; |
| 71 | 86 |
| 72 // Holds the Callback methods that don't require specialization to reduce | 87 // Holds the Callback methods that don't require specialization to reduce |
| 73 // template bloat. | 88 // template bloat. |
| 74 // CallbackBase<MoveOnly> is a direct base class of MoveOnly callbacks, and | 89 // CallbackBase<MoveOnly> is a direct base class of MoveOnly callbacks, and |
| 75 // CallbackBase<Copyable> uses CallbackBase<MoveOnly> for its implementation. | 90 // CallbackBase<Copyable> uses CallbackBase<MoveOnly> for its implementation. |
| 76 template <> | 91 template <> |
| 77 class BASE_EXPORT CallbackBase<CopyMode::MoveOnly> { | 92 class BASE_EXPORT CallbackBase<CopyMode::MoveOnly> { |
| 78 public: | 93 public: |
| 79 CallbackBase(CallbackBase&& c); | 94 CallbackBase(CallbackBase&& c); |
| 80 CallbackBase& operator=(CallbackBase&& c); | 95 CallbackBase& operator=(CallbackBase&& c); |
| 81 | 96 |
| 82 explicit CallbackBase(const CallbackBase<CopyMode::Copyable>& c); | 97 explicit CallbackBase(const CallbackBase<CopyMode::Copyable>& c); |
| 83 CallbackBase& operator=(const CallbackBase<CopyMode::Copyable>& c); | 98 CallbackBase& operator=(const CallbackBase<CopyMode::Copyable>& c); |
| 84 | 99 |
| 85 explicit CallbackBase(CallbackBase<CopyMode::Copyable>&& c); | 100 explicit CallbackBase(CallbackBase<CopyMode::Copyable>&& c); |
| 86 CallbackBase& operator=(CallbackBase<CopyMode::Copyable>&& c); | 101 CallbackBase& operator=(CallbackBase<CopyMode::Copyable>&& c); |
| 87 | 102 |
| 88 // Returns true if Callback is null (doesn't refer to anything). | 103 // Returns true if Callback is null (doesn't refer to anything). |
| 89 bool is_null() const { return bind_state_.get() == NULL; } | 104 bool is_null() const { return !bind_state_; } |
| 90 explicit operator bool() const { return !is_null(); } | 105 explicit operator bool() const { return !is_null(); } |
| 91 | 106 |
| 92 // Returns true if the callback invocation will be nop due to an cancellation. | 107 // Returns true if the callback invocation will be nop due to an cancellation. |
| 93 // It's invalid to call this on uninitialized callback. | 108 // It's invalid to call this on uninitialized callback. |
| 94 bool IsCancelled() const; | 109 bool IsCancelled() const; |
| 95 | 110 |
| 96 // Returns the Callback into an uninitialized state. | 111 // Returns the Callback into an uninitialized state. |
| 97 void Reset(); | 112 void Reset(); |
| 98 | 113 |
| 99 protected: | 114 protected: |
| (...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 133 ~CallbackBase() {} | 148 ~CallbackBase() {} |
| 134 }; | 149 }; |
| 135 | 150 |
| 136 extern template class CallbackBase<CopyMode::MoveOnly>; | 151 extern template class CallbackBase<CopyMode::MoveOnly>; |
| 137 extern template class CallbackBase<CopyMode::Copyable>; | 152 extern template class CallbackBase<CopyMode::Copyable>; |
| 138 | 153 |
| 139 } // namespace internal | 154 } // namespace internal |
| 140 } // namespace base | 155 } // namespace base |
| 141 | 156 |
| 142 #endif // BASE_CALLBACK_INTERNAL_H_ | 157 #endif // BASE_CALLBACK_INTERNAL_H_ |
| OLD | NEW |