1 | /* |
2 | Copyright (c) 2005-2019 Intel Corporation |
3 | |
4 | Licensed under the Apache License, Version 2.0 (the "License"); |
5 | you may not use this file except in compliance with the License. |
6 | You may obtain a copy of the License at |
7 | |
8 | http://www.apache.org/licenses/LICENSE-2.0 |
9 | |
10 | Unless required by applicable law or agreed to in writing, software |
11 | distributed under the License is distributed on an "AS IS" BASIS, |
12 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
13 | See the License for the specific language governing permissions and |
14 | limitations under the License. |
15 | */ |
16 | |
17 | #include "spin_rw_mutex_v2.h" |
18 | #include "tbb/tbb_machine.h" |
19 | #include "../tbb/itt_notify.h" |
20 | #include "tbb/atomic.h" |
21 | |
22 | namespace tbb { |
23 | |
24 | using namespace internal; |
25 | |
26 | static inline bool CAS(volatile uintptr_t &addr, uintptr_t newv, uintptr_t oldv) { |
27 | return as_atomic(addr).compare_and_swap(newv, oldv) == oldv; |
28 | } |
29 | |
30 | //! Signal that write lock is released |
31 | void spin_rw_mutex::internal_itt_releasing(spin_rw_mutex *mutex) { |
32 | __TBB_ASSERT_EX(mutex, NULL); // To prevent compiler warnings |
33 | ITT_NOTIFY(sync_releasing, mutex); |
34 | } |
35 | |
36 | //! Acquire write (exclusive) lock on the given mutex. |
37 | bool spin_rw_mutex::internal_acquire_writer(spin_rw_mutex *mutex) |
38 | { |
39 | ITT_NOTIFY(sync_prepare, mutex); |
40 | for( atomic_backoff backoff;;backoff.pause() ) { |
41 | state_t s = mutex->state; |
42 | if( !(s & BUSY) ) { // no readers, no writers |
43 | if( CAS(mutex->state, WRITER, s) ) |
44 | break; // successfully stored writer flag |
45 | backoff.reset(); // we could be very close to complete op. |
46 | } else if( !(s & WRITER_PENDING) ) { // no pending writers |
47 | __TBB_AtomicOR(&mutex->state, WRITER_PENDING); |
48 | } |
49 | } |
50 | ITT_NOTIFY(sync_acquired, mutex); |
51 | __TBB_ASSERT( (mutex->state & BUSY)==WRITER, "invalid state of a write lock" ); |
52 | return false; |
53 | } |
54 | |
55 | //! Release write lock on the given mutex |
56 | void spin_rw_mutex::internal_release_writer(spin_rw_mutex *mutex) { |
57 | __TBB_ASSERT( (mutex->state & BUSY)==WRITER, "invalid state of a write lock" ); |
58 | ITT_NOTIFY(sync_releasing, mutex); |
59 | mutex->state = 0; |
60 | } |
61 | |
62 | //! Acquire read (shared) lock on the given mutex. |
63 | void spin_rw_mutex::internal_acquire_reader(spin_rw_mutex *mutex) { |
64 | ITT_NOTIFY(sync_prepare, mutex); |
65 | for( atomic_backoff backoff;;backoff.pause() ) { |
66 | state_t s = mutex->state; |
67 | if( !(s & (WRITER|WRITER_PENDING)) ) { // no writer or write requests |
68 | if( CAS(mutex->state, s+ONE_READER, s) ) |
69 | break; // successfully stored increased number of readers |
70 | backoff.reset(); // we could be very close to complete op. |
71 | } |
72 | } |
73 | ITT_NOTIFY(sync_acquired, mutex); |
74 | __TBB_ASSERT( mutex->state & READERS, "invalid state of a read lock: no readers" ); |
75 | __TBB_ASSERT( !(mutex->state & WRITER), "invalid state of a read lock: active writer" ); |
76 | } |
77 | |
78 | //! Upgrade reader to become a writer. |
79 | /** Returns whether the upgrade happened without releasing and re-acquiring the lock */ |
80 | bool spin_rw_mutex::internal_upgrade(spin_rw_mutex *mutex) { |
81 | state_t s = mutex->state; |
82 | __TBB_ASSERT( s & READERS, "invalid state before upgrade: no readers " ); |
83 | __TBB_ASSERT( !(s & WRITER), "invalid state before upgrade: active writer " ); |
84 | // check and set writer-pending flag |
85 | // required conditions: either no pending writers, or we are the only reader |
86 | // (with multiple readers and pending writer, another upgrade could have been requested) |
87 | while( (s & READERS)==ONE_READER || !(s & WRITER_PENDING) ) { |
88 | if( CAS(mutex->state, s | WRITER_PENDING, s) ) |
89 | { |
90 | ITT_NOTIFY(sync_prepare, mutex); |
91 | atomic_backoff backoff; |
92 | while( (mutex->state & READERS) != ONE_READER ) backoff.pause(); |
93 | __TBB_ASSERT(mutex->state == (ONE_READER | WRITER_PENDING),"invalid state when upgrading to writer" ); |
94 | // both new readers and writers are blocked at this time |
95 | mutex->state = WRITER; |
96 | ITT_NOTIFY(sync_acquired, mutex); |
97 | __TBB_ASSERT( (mutex->state & BUSY) == WRITER, "invalid state after upgrade" ); |
98 | return true; // successfully upgraded |
99 | } else { |
100 | s = mutex->state; // re-read |
101 | } |
102 | } |
103 | // slow reacquire |
104 | internal_release_reader(mutex); |
105 | return internal_acquire_writer(mutex); // always returns false |
106 | } |
107 | |
108 | //! Downgrade writer to a reader |
109 | void spin_rw_mutex::internal_downgrade(spin_rw_mutex *mutex) { |
110 | __TBB_ASSERT( (mutex->state & BUSY) == WRITER, "invalid state before downgrade" ); |
111 | ITT_NOTIFY(sync_releasing, mutex); |
112 | mutex->state = ONE_READER; |
113 | __TBB_ASSERT( mutex->state & READERS, "invalid state after downgrade: no readers" ); |
114 | __TBB_ASSERT( !(mutex->state & WRITER), "invalid state after downgrade: active writer" ); |
115 | } |
116 | |
117 | //! Release read lock on the given mutex |
118 | void spin_rw_mutex::internal_release_reader(spin_rw_mutex *mutex) |
119 | { |
120 | __TBB_ASSERT( mutex->state & READERS, "invalid state of a read lock: no readers" ); |
121 | __TBB_ASSERT( !(mutex->state & WRITER), "invalid state of a read lock: active writer" ); |
122 | ITT_NOTIFY(sync_releasing, mutex); // release reader |
123 | __TBB_FetchAndAddWrelease((volatile void *)&(mutex->state),-(intptr_t)ONE_READER); |
124 | } |
125 | |
126 | //! Try to acquire write lock on the given mutex |
127 | bool spin_rw_mutex::internal_try_acquire_writer( spin_rw_mutex * mutex ) |
128 | { |
129 | // for a writer: only possible to acquire if no active readers or writers |
130 | state_t s = mutex->state; // on IA-64 architecture, this volatile load has acquire semantic |
131 | if( !(s & BUSY) ) // no readers, no writers; mask is 1..1101 |
132 | if( CAS(mutex->state, WRITER, s) ) { |
133 | ITT_NOTIFY(sync_acquired, mutex); |
134 | return true; // successfully stored writer flag |
135 | } |
136 | return false; |
137 | } |
138 | |
139 | //! Try to acquire read lock on the given mutex |
140 | bool spin_rw_mutex::internal_try_acquire_reader( spin_rw_mutex * mutex ) |
141 | { |
142 | // for a reader: acquire if no active or waiting writers |
143 | state_t s = mutex->state; // on IA-64 architecture, a load of volatile variable has acquire semantic |
144 | while( !(s & (WRITER|WRITER_PENDING)) ) // no writers |
145 | if( CAS(mutex->state, s+ONE_READER, s) ) { |
146 | ITT_NOTIFY(sync_acquired, mutex); |
147 | return true; // successfully stored increased number of readers |
148 | } |
149 | return false; |
150 | } |
151 | |
152 | } // namespace tbb |
153 | |