Cpp Utilities 1.2.3
All Classes Namespaces Files Functions Variables Typedefs Friends Macros Modules Pages
RWSpinLock.hpp
Go to the documentation of this file.
1#ifndef CPP_UTILITIES_MEMORYSAFETY_RWSPINLOCK_HPP
2#define CPP_UTILITIES_MEMORYSAFETY_RWSPINLOCK_HPP
3/*
4 * Copyright 2011-present Facebook, Inc.
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17 */
148#include <algorithm>
149#include <atomic>
150#include <thread>
151#include "../Common.h"
152
154
159namespace Memory {
165 enum : int32_t { READER = 4, UPGRADED = 2, WRITER = 1 };
166
167public:
168 constexpr RWSpinLock() : bits_(0) {}
169
170 RWSpinLock(RWSpinLock const&) = delete;
171 RWSpinLock& operator=(RWSpinLock const&) = delete;
172
174 void lock() {
175 uint_fast32_t count = 0;
176 while (!try_lock()) {
177 if (++count > 1000) {
178 std::this_thread::yield();
179 }
180 }
181 }
182
184 void unlock() {
185 static_assert(READER > WRITER + UPGRADED, "wrong bits!");
186 bits_.fetch_and(~(WRITER | UPGRADED), std::memory_order_release);
187 }
188
190 void lock_shared() {
191 uint_fast32_t count = 0;
192 while (!try_lock_shared()) {
193 if (++count > 1000) {
194 std::this_thread::yield();
195 }
196 }
197 }
198
200 bits_.fetch_add(-READER, std::memory_order_release);
201 }
202
205 bits_.fetch_add(READER, std::memory_order_acquire);
206 unlock();
207 }
208
211 uint_fast32_t count = 0;
212 while (!try_lock_upgrade()) {
213 if (++count > 1000) {
214 std::this_thread::yield();
215 }
216 }
217 }
218
220 bits_.fetch_add(-UPGRADED, std::memory_order_acq_rel);
221 }
222
225 int64_t count = 0;
226 while (!try_unlock_upgrade_and_lock()) {
227 if (++count > 1000) {
228 std::this_thread::yield();
229 }
230 }
231 }
232
235 bits_.fetch_add(READER - UPGRADED, std::memory_order_acq_rel);
236 }
237
240 // need to do it in two steps here -- as the UPGRADED bit might be OR-ed
241 // at the same time when other threads are trying do try_lock_upgrade().
242 bits_.fetch_or(UPGRADED, std::memory_order_acquire);
243 bits_.fetch_add(-WRITER, std::memory_order_release);
244 }
245
247 bool try_lock() {
248 int32_t expect = 0;
249 return bits_.compare_exchange_strong(
250 expect, WRITER, std::memory_order_acq_rel);
251 }
252
263 // fetch_add is considerably (100%) faster than compare_exchange,
264 // so here we are optimizing for the common (lock success) case.
265 int32_t value = bits_.fetch_add(READER, std::memory_order_acquire);
266 if (value & (WRITER | UPGRADED)) {
267 bits_.fetch_add(-READER, std::memory_order_release);
268 return false;
269 }
270 return true;
271 }
272
275 int32_t expect = UPGRADED;
276 return bits_.compare_exchange_strong(
277 expect, WRITER, std::memory_order_acq_rel);
278 }
279
289 int32_t value = bits_.fetch_or(UPGRADED, std::memory_order_acquire);
290
291 return ((value & (UPGRADED | WRITER)) == 0);
292 }
293
295 int32_t bits() const {
296 return bits_.load(std::memory_order_acquire);
297 }
298
299 class ReadHolder;
300 class UpgradedHolder;
301 class WriteHolder;
302
308 public:
309 explicit ReadHolder(RWSpinLock* lock) : lock_(lock) {
310 if (lock_) {
311 lock_->lock_shared();
312 }
313 }
314
315 explicit ReadHolder(RWSpinLock& lock) : lock_(&lock) {
316 lock_->lock_shared();
317 }
318
319 ReadHolder(ReadHolder&& other) noexcept : lock_(other.lock_) {
320 other.lock_ = nullptr;
321 }
322
324 explicit ReadHolder(UpgradedHolder&& upgraded) : lock_(upgraded.lock_) {
325 upgraded.lock_ = nullptr;
326 if (lock_) {
328 }
329 }
330
331 explicit ReadHolder(WriteHolder&& writer) : lock_(writer.lock_) {
332 writer.lock_ = nullptr;
333 if (lock_) {
334 lock_->unlock_and_lock_shared();
335 }
336 }
337
339 using std::swap;
340 swap(lock_, other.lock_);
341 return *this;
342 }
343
344 ReadHolder(const ReadHolder& other) = delete;
345 ReadHolder& operator=(const ReadHolder& other) = delete;
346
348 if (lock_) {
349 lock_->unlock_shared();
350 }
351 }
352
353 void reset(RWSpinLock* lock = nullptr) {
354 if (lock == lock_) {
355 return;
356 }
357 if (lock_) {
358 lock_->unlock_shared();
359 }
360 lock_ = lock;
361 if (lock_) {
362 lock_->lock_shared();
363 }
364 }
365
366 void swap(ReadHolder& other) {
367 std::swap(lock_, other.lock_);
368 }
369
370 private:
371 friend class UpgradedHolder;
372 friend class WriteHolder;
373 RWSpinLock* lock_;
374 };
375
381 public:
382 explicit UpgradedHolder(RWSpinLock* lock) : lock_(lock) {
383 if (lock_) {
384 lock_->lock_upgrade();
385 }
386 }
387
388 explicit UpgradedHolder(RWSpinLock& lock) : lock_(&lock) {
389 lock_->lock_upgrade();
390 }
391
392 explicit UpgradedHolder(WriteHolder&& writer) {
393 lock_ = writer.lock_;
394 writer.lock_ = nullptr;
395 if (lock_) {
397 }
398 }
399
400 UpgradedHolder(UpgradedHolder&& other) noexcept : lock_(other.lock_) {
401 other.lock_ = nullptr;
402 }
403
405 using std::swap;
406 swap(lock_, other.lock_);
407 return *this;
408 }
409
410 UpgradedHolder(const UpgradedHolder& other) = delete;
411 UpgradedHolder& operator=(const UpgradedHolder& other) = delete;
412
414 if (lock_) {
415 lock_->unlock_upgrade();
416 }
417 }
418
419 void reset(RWSpinLock* lock = nullptr) {
420 if (lock == lock_) {
421 return;
422 }
423 if (lock_) {
424 lock_->unlock_upgrade();
425 }
426 lock_ = lock;
427 if (lock_) {
428 lock_->lock_upgrade();
429 }
430 }
431
432 void swap(UpgradedHolder& other) {
433 using std::swap;
434 swap(lock_, other.lock_);
435 }
436
437 private:
438 friend class WriteHolder;
439 friend class ReadHolder;
440 RWSpinLock* lock_;
441 };
442
448 public:
449 explicit WriteHolder(RWSpinLock* lock) : lock_(lock) {
450 if (lock_) {
451 lock_->lock();
452 }
453 }
454
455 explicit WriteHolder(RWSpinLock& lock) : lock_(&lock) {
456 lock_->lock();
457 }
458
460 explicit WriteHolder(UpgradedHolder&& upgraded) {
461 lock_ = upgraded.lock_;
462 upgraded.lock_ = nullptr;
463 if (lock_) {
465 }
466 }
467
468 WriteHolder(WriteHolder&& other) noexcept : lock_(other.lock_) {
469 other.lock_ = nullptr;
470 }
471
473 using std::swap;
474 swap(lock_, other.lock_);
475 return *this;
476 }
477
478 WriteHolder(const WriteHolder& other) = delete;
479 WriteHolder& operator=(const WriteHolder& other) = delete;
480
482 if (lock_) {
483 lock_->unlock();
484 }
485 }
486
487 void reset(RWSpinLock* lock = nullptr) {
488 if (lock == lock_) {
489 return;
490 }
491 if (lock_) {
492 lock_->unlock();
493 }
494 lock_ = lock;
495 if (lock_) {
496 lock_->lock();
497 }
498 }
499
500 void swap(WriteHolder* other) {
501 using std::swap;
502 swap(lock_, other->lock_);
503 }
504
505 private:
506 friend class ReadHolder;
507 friend class UpgradedHolder;
508 RWSpinLock* lock_;
509 };
510
511private:
512 std::atomic<int32_t> bits_;
513};
514} // namespace Memory
518
519#endif // CPP_UTILITIES_MEMORYSAFETY_RWSPINLOCK_HPP
#define UTILITIES_NAMESPACE_END
Define for end namespace declaration, nothing will be generated if UTILITIES_NAMESPACE isn't defined.
Definition: Common.h:92
#define UTILITIES_NAMESPACE_BEGIN
Define for begin namespace declaration, nothing will be generated if UTILITIES_NAMESPACE isn't define...
Definition: Common.h:91
RAII guard for read lock with RWSpinLock::lock_shared() on construction and RWSpinLock::unlock_shared...
Definition: RWSpinLock.hpp:307
~ReadHolder()
Definition: RWSpinLock.hpp:347
void reset(RWSpinLock *lock=nullptr)
Definition: RWSpinLock.hpp:353
ReadHolder(ReadHolder &&other) noexcept
Definition: RWSpinLock.hpp:319
void swap(ReadHolder &other)
Definition: RWSpinLock.hpp:366
ReadHolder(WriteHolder &&writer)
Definition: RWSpinLock.hpp:331
ReadHolder & operator=(ReadHolder &&other)
Definition: RWSpinLock.hpp:338
ReadHolder(RWSpinLock *lock)
Definition: RWSpinLock.hpp:309
ReadHolder(const ReadHolder &other)=delete
ReadHolder(RWSpinLock &lock)
Definition: RWSpinLock.hpp:315
ReadHolder(UpgradedHolder &&upgraded)
down-grade
Definition: RWSpinLock.hpp:324
ReadHolder & operator=(const ReadHolder &other)=delete
RAII guard for upgrade lock with RWSpinLock::lock_upgrade() on construction and RWSpinLock::unlock_up...
Definition: RWSpinLock.hpp:380
void swap(UpgradedHolder &other)
Definition: RWSpinLock.hpp:432
UpgradedHolder & operator=(const UpgradedHolder &other)=delete
~UpgradedHolder()
Definition: RWSpinLock.hpp:413
UpgradedHolder(RWSpinLock &lock)
Definition: RWSpinLock.hpp:388
void reset(RWSpinLock *lock=nullptr)
Definition: RWSpinLock.hpp:419
UpgradedHolder(WriteHolder &&writer)
Definition: RWSpinLock.hpp:392
UpgradedHolder(UpgradedHolder &&other) noexcept
Definition: RWSpinLock.hpp:400
UpgradedHolder(const UpgradedHolder &other)=delete
UpgradedHolder(RWSpinLock *lock)
Definition: RWSpinLock.hpp:382
UpgradedHolder & operator=(UpgradedHolder &&other)
Definition: RWSpinLock.hpp:404
RAII guard for write lock with RWSpinLock::lock() on construction and RWSpinLock::unlock() on destruc...
Definition: RWSpinLock.hpp:447
WriteHolder(const WriteHolder &other)=delete
WriteHolder & operator=(WriteHolder &&other)
Definition: RWSpinLock.hpp:472
void reset(RWSpinLock *lock=nullptr)
Definition: RWSpinLock.hpp:487
void swap(WriteHolder *other)
Definition: RWSpinLock.hpp:500
WriteHolder(RWSpinLock *lock)
Definition: RWSpinLock.hpp:449
WriteHolder & operator=(const WriteHolder &other)=delete
WriteHolder(WriteHolder &&other) noexcept
Definition: RWSpinLock.hpp:468
WriteHolder(RWSpinLock &lock)
Definition: RWSpinLock.hpp:455
WriteHolder(UpgradedHolder &&upgraded)
promoted from an upgrade lock holder
Definition: RWSpinLock.hpp:460
~WriteHolder()
Definition: RWSpinLock.hpp:481
High-performance read-write-spinlock, see RWSpinLock.hpp for details.
Definition: RWSpinLock.hpp:164
void unlock_upgrade()
Definition: RWSpinLock.hpp:219
RWSpinLock & operator=(RWSpinLock const &)=delete
RWSpinLock(RWSpinLock const &)=delete
void lock_upgrade()
UpgradeLockable Concept.
Definition: RWSpinLock.hpp:210
bool try_lock_shared()
Try to get reader permission on the lock. This can fail if we find out someone is a writer or upgrade...
Definition: RWSpinLock.hpp:262
bool try_lock()
Definition: RWSpinLock.hpp:247
constexpr RWSpinLock()
Definition: RWSpinLock.hpp:168
int32_t bits() const
mainly for debugging purposes.
Definition: RWSpinLock.hpp:295
void lock()
Lockable Concept.
Definition: RWSpinLock.hpp:174
void unlock_shared()
Definition: RWSpinLock.hpp:199
void unlock_upgrade_and_lock()
unlock upgrade and try to acquire write lock
Definition: RWSpinLock.hpp:224
void unlock_and_lock_upgrade()
write unlock and upgrade lock atomically
Definition: RWSpinLock.hpp:239
void unlock_and_lock_shared()
Downgrade the lock from writer status to reader status.
Definition: RWSpinLock.hpp:204
bool try_unlock_upgrade_and_lock()
try to unlock upgrade and write lock atomically
Definition: RWSpinLock.hpp:274
bool try_lock_upgrade()
try to acquire an upgradable lock.
Definition: RWSpinLock.hpp:288
void unlock_upgrade_and_lock_shared()
unlock upgrade and read lock atomically
Definition: RWSpinLock.hpp:234
void unlock()
Writer is responsible for clearing up both the UPGRADED and WRITER bits.
Definition: RWSpinLock.hpp:184
void lock_shared()
SharedLockable Concept.
Definition: RWSpinLock.hpp:190
Namespace for all classes, typedefs and functions of memory safety. See Memory Safety for more instru...
Definition: RWSpinLock.hpp:159