-
Notifications
You must be signed in to change notification settings - Fork 1k
/
Copy pathtask_scheduler_observer.h
116 lines (93 loc) · 4.47 KB
/
task_scheduler_observer.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
/*
Copyright (c) 2005-2021 Intel Corporation
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
#ifndef __TBB_task_scheduler_observer_H
#define __TBB_task_scheduler_observer_H
#include "detail/_namespace_injection.h"
#include "task_arena.h"
#include <atomic>
namespace tbb {
namespace detail {
namespace d1 {
class task_scheduler_observer;
}
namespace r1 {
class observer_proxy;
class observer_list;
//! Enable or disable observation
/** For local observers the method can be used only when the current thread
has the task scheduler initialized or is attached to an arena.
Repeated calls with the same state are no-ops. **/
void __TBB_EXPORTED_FUNC observe(d1::task_scheduler_observer&, bool state = true);
}
namespace d1 {
class task_scheduler_observer {
friend class r1::observer_proxy;
friend class r1::observer_list;
friend void r1::observe(d1::task_scheduler_observer&, bool);
//! Pointer to the proxy holding this observer.
/** Observers are proxied by the scheduler to maintain persistent lists of them. **/
std::atomic<r1::observer_proxy*> my_proxy{ nullptr };
//! Counter preventing the observer from being destroyed while in use by the scheduler.
/** Valid only when observation is on. **/
std::atomic<intptr_t> my_busy_count{ 0 };
//! Contains task_arena pointer
task_arena* my_task_arena{ nullptr };
public:
//! Returns true if observation is enabled, false otherwise.
bool is_observing() const { return my_proxy.load(std::memory_order_relaxed) != nullptr; }
//! Entry notification
/** Invoked from inside observe(true) call and whenever a worker enters the arena
this observer is associated with. If a thread is already in the arena when
the observer is activated, the entry notification is called before it
executes the first stolen task. **/
virtual void on_scheduler_entry( bool /*is_worker*/ ) {}
//! Exit notification
/** Invoked from inside observe(false) call and whenever a worker leaves the
arena this observer is associated with. **/
virtual void on_scheduler_exit( bool /*is_worker*/ ) {}
//! Construct local or global observer in inactive state (observation disabled).
/** For a local observer entry/exit notifications are invoked whenever a worker
thread joins/leaves the arena of the observer's owner thread. If a thread is
already in the arena when the observer is activated, the entry notification is
called before it executes the first stolen task. **/
explicit task_scheduler_observer() = default;
//! Construct local observer for a given arena in inactive state (observation disabled).
/** entry/exit notifications are invoked whenever a thread joins/leaves arena.
If a thread is already in the arena when the observer is activated, the entry notification
is called before it executes the first stolen task. **/
explicit task_scheduler_observer(task_arena& a) : my_task_arena(&a) {}
/** Destructor protects instance of the observer from concurrent notification.
It is recommended to disable observation before destructor of a derived class starts,
otherwise it can lead to concurrent notification callback on partly destroyed object **/
virtual ~task_scheduler_observer() {
if (my_proxy.load(std::memory_order_relaxed)) {
observe(false);
}
}
//! Enable or disable observation
/** Warning: concurrent invocations of this method are not safe.
Repeated calls with the same state are no-ops. **/
void observe(bool state = true) {
if( state && !my_proxy.load(std::memory_order_relaxed) ) {
__TBB_ASSERT( my_busy_count.load(std::memory_order_relaxed) == 0, "Inconsistent state of task_scheduler_observer instance");
}
r1::observe(*this, state);
}
};
} // namespace d1
} // namespace detail
inline namespace v1 {
using detail::d1::task_scheduler_observer;
}
} // namespace tbb
#endif /* __TBB_task_scheduler_observer_H */