task_scheduler_observer.h 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. /*
  2. Copyright 2005-2013 Intel Corporation. All Rights Reserved.
  3. This file is part of Threading Building Blocks.
  4. Threading Building Blocks is free software; you can redistribute it
  5. and/or modify it under the terms of the GNU General Public License
  6. version 2 as published by the Free Software Foundation.
  7. Threading Building Blocks is distributed in the hope that it will be
  8. useful, but WITHOUT ANY WARRANTY; without even the implied warranty
  9. of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with Threading Building Blocks; if not, write to the Free Software
  13. Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
  14. As a special exception, you may use this file as part of a free software
  15. library without restriction. Specifically, if other files instantiate
  16. templates or use macros or inline functions from this file, or you compile
  17. this file and link it with other files to produce an executable, this
  18. file does not by itself cause the resulting executable to be covered by
  19. the GNU General Public License. This exception does not however
  20. invalidate any other reasons why the executable file might be covered by
  21. the GNU General Public License.
  22. */
  23. #ifndef __TBB_task_scheduler_observer_H
  24. #define __TBB_task_scheduler_observer_H
  25. #include "atomic.h"
  26. #if __TBB_TASK_ARENA
  27. #include "task_arena.h"
  28. #endif //__TBB_TASK_ARENA
  29. #if __TBB_SCHEDULER_OBSERVER
  30. namespace tbb {
  31. namespace interface6 {
  32. class task_scheduler_observer;
  33. }
  34. namespace internal {
  35. class observer_proxy;
  36. class observer_list;
  37. class task_scheduler_observer_v3 {
  38. friend class observer_proxy;
  39. friend class observer_list;
  40. friend class interface6::task_scheduler_observer;
  41. //! Pointer to the proxy holding this observer.
  42. /** Observers are proxied by the scheduler to maintain persistent lists of them. **/
  43. observer_proxy* my_proxy;
  44. //! Counter preventing the observer from being destroyed while in use by the scheduler.
  45. /** Valid only when observation is on. **/
  46. atomic<intptr_t> my_busy_count;
  47. public:
  48. //! Enable or disable observation
  49. /** For local observers the method can be used only when the current thread
  50. has the task scheduler initialized or is attached to an arena.
  51. Repeated calls with the same state are no-ops. **/
  52. void __TBB_EXPORTED_METHOD observe( bool state=true );
  53. //! Returns true if observation is enabled, false otherwise.
  54. bool is_observing() const {return my_proxy!=NULL;}
  55. //! Construct observer with observation disabled.
  56. task_scheduler_observer_v3() : my_proxy(NULL) { my_busy_count.store<relaxed>(0); }
  57. //! Entry notification
  58. /** Invoked from inside observe(true) call and whenever a worker enters the arena
  59. this observer is associated with. If a thread is already in the arena when
  60. the observer is activated, the entry notification is called before it
  61. executes the first stolen task.
  62. Obsolete semantics. For global observers it is called by a thread before
  63. the first steal since observation became enabled. **/
  64. virtual void on_scheduler_entry( bool /*is_worker*/ ) {}
  65. //! Exit notification
  66. /** Invoked from inside observe(false) call and whenever a worker leaves the
  67. arena this observer is associated with.
  68. Obsolete semantics. For global observers it is called by a thread before
  69. the first steal since observation became enabled. **/
  70. virtual void on_scheduler_exit( bool /*is_worker*/ ) {}
  71. //! Destructor automatically switches observation off if it is enabled.
  72. virtual ~task_scheduler_observer_v3() { if(my_proxy) observe(false);}
  73. };
  74. } // namespace internal
  75. #if TBB_PREVIEW_LOCAL_OBSERVER
  76. namespace interface6 {
  77. class task_scheduler_observer : public internal::task_scheduler_observer_v3 {
  78. friend class internal::task_scheduler_observer_v3;
  79. friend class internal::observer_proxy;
  80. friend class internal::observer_list;
  81. /** Negative numbers with the largest absolute value to minimize probability
  82. of coincidence in case of a bug in busy count usage. **/
  83. // TODO: take more high bits for version number
  84. static const intptr_t v6_trait = (intptr_t)((~(uintptr_t)0 >> 1) + 1);
  85. //! contains task_arena pointer or tag indicating local or global semantics of the observer
  86. intptr_t my_context_tag;
  87. enum { global_tag = 0, implicit_tag = 1 };
  88. public:
  89. //! Construct local or global observer in inactive state (observation disabled).
  90. /** For a local observer entry/exit notifications are invoked whenever a worker
  91. thread joins/leaves the arena of the observer's owner thread. If a thread is
  92. already in the arena when the observer is activated, the entry notification is
  93. called before it executes the first stolen task. **/
  94. /** TODO: Obsolete.
  95. Global observer semantics is obsolete as it violates master thread isolation
  96. guarantees and is not composable. Thus the current default behavior of the
  97. constructor is obsolete too and will be changed in one of the future versions
  98. of the library. **/
  99. task_scheduler_observer( bool local = false ) {
  100. my_busy_count.store<relaxed>(v6_trait);
  101. my_context_tag = local? implicit_tag : global_tag;
  102. }
  103. #if __TBB_TASK_ARENA
  104. //! Construct local observer for a given arena in inactive state (observation disabled).
  105. /** entry/exit notifications are invoked whenever a thread joins/leaves arena.
  106. If a thread is already in the arena when the observer is activated, the entry notification
  107. is called before it executes the first stolen task. **/
  108. task_scheduler_observer( task_arena & a) {
  109. my_busy_count.store<relaxed>(v6_trait);
  110. my_context_tag = (intptr_t)&a;
  111. }
  112. #endif //__TBB_TASK_ARENA
  113. //! The callback can be invoked in a worker thread before it leaves an arena.
  114. /** If it returns false, the thread remains in the arena. Will not be called for masters
  115. or if the worker leaves arena due to rebalancing or priority changes, etc.
  116. NOTE: The preview library must be linked for this method to take effect **/
  117. virtual bool on_scheduler_leaving() { return true; }
  118. //! Destructor additionally protects concurrent on_scheduler_leaving notification
  119. // It is recommended to disable observation before destructor of a derived class starts,
  120. // otherwise it can lead to concurrent notification callback on partly destroyed object
  121. virtual ~task_scheduler_observer() { if(my_proxy) observe(false);}
  122. };
  123. } //namespace interface6
  124. using interface6::task_scheduler_observer;
  125. #else /*TBB_PREVIEW_LOCAL_OBSERVER*/
  126. typedef tbb::internal::task_scheduler_observer_v3 task_scheduler_observer;
  127. #endif /*TBB_PREVIEW_LOCAL_OBSERVER*/
  128. } // namespace tbb
  129. #endif /* __TBB_SCHEDULER_OBSERVER */
  130. #endif /* __TBB_task_scheduler_observer_H */