2018-12-31 22:04:05 +03:00
|
|
|
//
|
2018-01-02 16:42:31 +03:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018
|
2018-12-31 22:04:05 +03:00
|
|
|
//
|
|
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
//
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <atomic>
|
|
|
|
|
|
|
|
#include "td/utils/port/thread.h"
|
|
|
|
|
|
|
|
namespace td {
|
|
|
|
|
|
|
|
class SpinLock {
|
|
|
|
struct Unlock {
|
|
|
|
void operator()(SpinLock *ptr) {
|
|
|
|
ptr->unlock();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class InfBackoff {
|
|
|
|
int cnt = 0;
|
|
|
|
|
|
|
|
public:
|
|
|
|
bool next() {
|
|
|
|
cnt++;
|
|
|
|
if (cnt < 50) {
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
td::this_thread::yield();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
public:
|
|
|
|
using Lock = std::unique_ptr<SpinLock, Unlock>;
|
|
|
|
|
|
|
|
Lock lock() {
|
|
|
|
InfBackoff backoff;
|
|
|
|
while (!try_lock()) {
|
|
|
|
backoff.next();
|
|
|
|
}
|
|
|
|
return Lock(this);
|
|
|
|
}
|
|
|
|
bool try_lock() {
|
|
|
|
return !flag_.test_and_set(std::memory_order_acquire);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::atomic_flag flag_ = ATOMIC_FLAG_INIT;
|
|
|
|
void unlock() {
|
|
|
|
flag_.clear(std::memory_order_release);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace td
|