|  | /* | 
|  | *  Copyright 2019 The WebRTC project authors. All Rights Reserved. | 
|  | * | 
|  | *  Use of this source code is governed by a BSD-style license | 
|  | *  that can be found in the LICENSE file in the root of the source | 
|  | *  tree. An additional intellectual property rights grant can be found | 
|  | *  in the file PATENTS.  All contributing project authors may | 
|  | *  be found in the AUTHORS file in the root of the source tree. | 
|  | */ | 
|  | #ifndef API_SEQUENCE_CHECKER_H_ | 
|  | #define API_SEQUENCE_CHECKER_H_ | 
|  |  | 
|  | #include "api/task_queue/task_queue_base.h" | 
|  | #include "rtc_base/checks.h" | 
|  | #include "rtc_base/synchronization/sequence_checker_internal.h" | 
|  | #include "rtc_base/thread_annotations.h" | 
|  |  | 
|  | namespace webrtc { | 
|  |  | 
|  | // SequenceChecker is a helper class used to help verify that some methods | 
|  | // of a class are called on the same task queue or thread. A | 
|  | // SequenceChecker is bound to a a task queue if the object is | 
|  | // created on a task queue, or a thread otherwise. | 
|  | // | 
|  | // | 
|  | // Example: | 
|  | // class MyClass { | 
|  | //  public: | 
|  | //   void Foo() { | 
|  | //     RTC_DCHECK_RUN_ON(&sequence_checker_); | 
|  | //     ... (do stuff) ... | 
|  | //   } | 
|  | // | 
|  | //  private: | 
|  | //   SequenceChecker sequence_checker_; | 
|  | // } | 
|  | // | 
|  | // In Release mode, IsCurrent will always return true. | 
|  | class RTC_LOCKABLE SequenceChecker | 
|  | #if RTC_DCHECK_IS_ON | 
|  | : public webrtc_sequence_checker_internal::SequenceCheckerImpl { | 
|  | using Impl = webrtc_sequence_checker_internal::SequenceCheckerImpl; | 
|  | #else | 
|  | : public webrtc_sequence_checker_internal::SequenceCheckerDoNothing { | 
|  | using Impl = webrtc_sequence_checker_internal::SequenceCheckerDoNothing; | 
|  | #endif | 
|  | public: | 
|  | enum InitialState : bool { kDetached = false, kAttached = true }; | 
|  |  | 
|  | // TODO(tommi): We could maybe join these two ctors and have fewer factory | 
|  | // functions. At the moment they're separate to minimize code changes when | 
|  | // we added the second ctor as well as avoiding to have unnecessary code at | 
|  | // the SequenceChecker which much only run for the SequenceCheckerImpl | 
|  | // implementation. | 
|  | // In theory we could have something like: | 
|  | // | 
|  | //  SequenceChecker(InitialState initial_state = kAttached, | 
|  | //                  TaskQueueBase* attached_queue = TaskQueueBase::Current()); | 
|  | // | 
|  | // But the problem with that is having the call to `Current()` exist for | 
|  | // `SequenceCheckerDoNothing`. | 
|  | explicit SequenceChecker(InitialState initial_state = kAttached) | 
|  | : Impl(initial_state) {} | 
|  | explicit SequenceChecker(TaskQueueBase* attached_queue) | 
|  | : Impl(attached_queue) {} | 
|  |  | 
|  | // Returns true if sequence checker is attached to the current sequence. | 
|  | bool IsCurrent() const { return Impl::IsCurrent(); } | 
|  | // Detaches checker from sequence to which it is attached. Next attempt | 
|  | // to do a check with this checker will result in attaching this checker | 
|  | // to the sequence on which check was performed. | 
|  | void Detach() { Impl::Detach(); } | 
|  | }; | 
|  |  | 
|  | }  // namespace webrtc | 
|  |  | 
|  | // RTC_RUN_ON/RTC_GUARDED_BY/RTC_DCHECK_RUN_ON macros allows to annotate | 
|  | // variables are accessed from same thread/task queue. | 
|  | // Using tools designed to check mutexes, it checks at compile time everywhere | 
|  | // variable is access, there is a run-time dcheck thread/task queue is correct. | 
|  | // | 
|  | // class SequenceCheckerExample { | 
|  | //  public: | 
|  | //   int CalledFromPacer() RTC_RUN_ON(pacer_sequence_checker_) { | 
|  | //     return var2_; | 
|  | //   } | 
|  | // | 
|  | //   void CallMeFromPacer() { | 
|  | //     RTC_DCHECK_RUN_ON(&pacer_sequence_checker_) | 
|  | //        << "Should be called from pacer"; | 
|  | //     CalledFromPacer(); | 
|  | //   } | 
|  | // | 
|  | //  private: | 
|  | //   int pacer_var_ RTC_GUARDED_BY(pacer_sequence_checker_); | 
|  | //   SequenceChecker pacer_sequence_checker_; | 
|  | // }; | 
|  | // | 
|  | // class TaskQueueExample { | 
|  | //  public: | 
|  | //   class Encoder { | 
|  | //    public: | 
|  | //     rtc::TaskQueueBase& Queue() { return encoder_queue_; } | 
|  | //     void Encode() { | 
|  | //       RTC_DCHECK_RUN_ON(&encoder_queue_); | 
|  | //       DoSomething(var_); | 
|  | //     } | 
|  | // | 
|  | //    private: | 
|  | //     rtc::TaskQueueBase& encoder_queue_; | 
|  | //     Frame var_ RTC_GUARDED_BY(encoder_queue_); | 
|  | //   }; | 
|  | // | 
|  | //   void Encode() { | 
|  | //     // Will fail at runtime when DCHECK is enabled: | 
|  | //     // encoder_->Encode(); | 
|  | //     // Will work: | 
|  | //     rtc::scoped_refptr<Encoder> encoder = encoder_; | 
|  | //     encoder_->Queue().PostTask([encoder] { encoder->Encode(); }); | 
|  | //   } | 
|  | // | 
|  | //  private: | 
|  | //   rtc::scoped_refptr<Encoder> encoder_; | 
|  | // } | 
|  |  | 
|  | // Document if a function expected to be called from same thread/task queue. | 
|  | #define RTC_RUN_ON(x) \ | 
|  | RTC_THREAD_ANNOTATION_ATTRIBUTE__(exclusive_locks_required(x)) | 
|  |  | 
|  | // Checks current code is running on the desired sequence. | 
|  | // | 
|  | // First statement validates it is running on the sequence `x`. | 
|  | // Second statement annotates for the thread safety analyzer the check was done. | 
|  | // Such annotation has to be attached to a function, and that function has to be | 
|  | // called. Thus current implementation creates a noop lambda and calls it. | 
|  | #define RTC_DCHECK_RUN_ON(x)                                               \ | 
|  | RTC_DCHECK((x)->IsCurrent())                                             \ | 
|  | << webrtc::webrtc_sequence_checker_internal::ExpectationToString(x); \ | 
|  | []() RTC_ASSERT_EXCLUSIVE_LOCK(x) {}() | 
|  |  | 
|  | #endif  // API_SEQUENCE_CHECKER_H_ |