2018-12-31 22:04:05 +03:00
|
|
|
//
|
2019-01-01 01:02:34 +03:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2019
|
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 "td/utils/port/thread.h"
|
|
|
|
|
2018-02-03 15:58:18 +03:00
|
|
|
#include <atomic>
|
2018-09-27 04:19:03 +03:00
|
|
|
#include <memory>
|
2018-02-03 15:58:18 +03:00
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
namespace td {
|
|
|
|
|
|
|
|
class SpinLock {
|
|
|
|
struct Unlock {
|
|
|
|
void operator()(SpinLock *ptr) {
|
|
|
|
ptr->unlock();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class InfBackoff {
|
|
|
|
int cnt = 0;
|
|
|
|
|
|
|
|
public:
|
|
|
|
bool next() {
|
|
|
|
cnt++;
|
|
|
|
if (cnt < 50) {
|
2019-07-06 13:29:15 +02:00
|
|
|
//TODO pause
|
2018-12-31 22:04:05 +03:00
|
|
|
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
|