2019-05-01 15:13:48 +02:00
|
|
|
//
|
2022-12-31 22:28:08 +01:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2023
|
2019-05-01 15:13:48 +02: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>
|
2019-05-22 20:17:24 +02:00
|
|
|
#include <memory>
|
|
|
|
|
2019-05-01 15:13:48 +02:00
|
|
|
namespace td {
|
2019-05-22 20:17:24 +02:00
|
|
|
|
2019-05-01 15:13:48 +02:00
|
|
|
namespace detail {
|
|
|
|
struct RawCancellationToken {
|
2021-05-31 20:51:48 +02:00
|
|
|
std::atomic<bool> is_canceled_{false};
|
2019-05-01 15:13:48 +02:00
|
|
|
};
|
|
|
|
} // namespace detail
|
|
|
|
|
|
|
|
class CancellationToken {
|
|
|
|
public:
|
2022-10-09 18:30:18 +02:00
|
|
|
explicit operator bool() const noexcept {
|
2021-05-31 20:51:48 +02:00
|
|
|
// empty CancellationToken is never canceled
|
2020-06-24 13:47:36 +02:00
|
|
|
if (!token_) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-05-31 20:51:48 +02:00
|
|
|
return token_->is_canceled_.load(std::memory_order_acquire);
|
2019-05-01 15:13:48 +02:00
|
|
|
}
|
2020-06-24 13:47:36 +02:00
|
|
|
CancellationToken() = default;
|
2019-05-01 15:13:48 +02:00
|
|
|
explicit CancellationToken(std::shared_ptr<detail::RawCancellationToken> token) : token_(std::move(token)) {
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::shared_ptr<detail::RawCancellationToken> token_;
|
|
|
|
};
|
|
|
|
|
|
|
|
class CancellationTokenSource {
|
|
|
|
public:
|
|
|
|
CancellationTokenSource() = default;
|
2021-10-18 13:36:15 +02:00
|
|
|
CancellationTokenSource(CancellationTokenSource &&other) noexcept : token_(std::move(other.token_)) {
|
2019-05-01 15:13:48 +02:00
|
|
|
}
|
2021-10-18 13:36:15 +02:00
|
|
|
CancellationTokenSource &operator=(CancellationTokenSource &&other) noexcept {
|
2019-05-01 15:13:48 +02:00
|
|
|
cancel();
|
|
|
|
token_ = std::move(other.token_);
|
|
|
|
return *this;
|
|
|
|
}
|
2019-05-22 20:17:24 +02:00
|
|
|
CancellationTokenSource(const CancellationTokenSource &other) = delete;
|
|
|
|
CancellationTokenSource &operator=(const CancellationTokenSource &other) = delete;
|
2019-05-01 15:13:48 +02:00
|
|
|
~CancellationTokenSource() {
|
|
|
|
cancel();
|
|
|
|
}
|
|
|
|
|
|
|
|
CancellationToken get_cancellation_token() {
|
|
|
|
if (!token_) {
|
|
|
|
token_ = std::make_shared<detail::RawCancellationToken>();
|
|
|
|
}
|
|
|
|
return CancellationToken(token_);
|
|
|
|
}
|
|
|
|
void cancel() {
|
|
|
|
if (!token_) {
|
|
|
|
return;
|
|
|
|
}
|
2021-05-31 20:51:48 +02:00
|
|
|
token_->is_canceled_.store(true, std::memory_order_release);
|
2019-05-01 15:13:48 +02:00
|
|
|
token_.reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::shared_ptr<detail::RawCancellationToken> token_;
|
|
|
|
};
|
2019-05-22 20:17:24 +02:00
|
|
|
|
2019-05-01 15:13:48 +02:00
|
|
|
} // namespace td
|