101 lines
		
	
	
	
		
			1.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			101 lines
		
	
	
	
		
			1.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| This file is part of Telegram Desktop,
 | |
| the official desktop application for the Telegram messaging service.
 | |
| 
 | |
| For license and copyright information please follow this link:
 | |
| https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | |
| */
 | |
| #pragma once
 | |
| 
 | |
| #include "base/algorithm.h"
 | |
| 
 | |
| #include <atomic>
 | |
| 
 | |
| namespace base {
 | |
| 
 | |
| class binary_guard {
 | |
| public:
 | |
| 	binary_guard() = default;
 | |
| 	binary_guard(binary_guard &&other);
 | |
| 	binary_guard &operator=(binary_guard &&other);
 | |
| 	~binary_guard();
 | |
| 
 | |
| 	binary_guard &operator=(std::nullptr_t);
 | |
| 
 | |
| 	bool alive() const;
 | |
| 	binary_guard make_guard();
 | |
| 
 | |
| 	explicit operator bool() const;
 | |
| 
 | |
| private:
 | |
| 	void destroy();
 | |
| 
 | |
| 	std::atomic<bool> *_bothAlive = nullptr;
 | |
| 
 | |
| };
 | |
| 
 | |
| inline binary_guard::binary_guard(binary_guard &&other)
 | |
| : _bothAlive(base::take(other._bothAlive)) {
 | |
| }
 | |
| 
 | |
| inline binary_guard &binary_guard::operator=(binary_guard &&other) {
 | |
| 	if (this != &other) {
 | |
| 		destroy();
 | |
| 		_bothAlive = base::take(other._bothAlive);
 | |
| 	}
 | |
| 	return *this;
 | |
| }
 | |
| 
 | |
| inline binary_guard::~binary_guard() {
 | |
| 	destroy();
 | |
| }
 | |
| 
 | |
| inline binary_guard &binary_guard::operator=(std::nullptr_t) {
 | |
| 	destroy();
 | |
| 	return *this;
 | |
| }
 | |
| 
 | |
| inline binary_guard::operator bool() const {
 | |
| 	return alive();
 | |
| }
 | |
| 
 | |
| inline bool binary_guard::alive() const {
 | |
| 	return _bothAlive && _bothAlive->load();
 | |
| }
 | |
| 
 | |
| inline void binary_guard::destroy() {
 | |
| 	if (const auto both = base::take(_bothAlive)) {
 | |
| 		auto old = true;
 | |
| 		if (!both->compare_exchange_strong(old, false)) {
 | |
| 			delete both;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| inline binary_guard binary_guard::make_guard() {
 | |
| 	destroy();
 | |
| 
 | |
| 	auto result = binary_guard();
 | |
| 	_bothAlive = result._bothAlive = new std::atomic<bool>(true);
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| } // namespace base
 | |
| 
 | |
| namespace crl {
 | |
| 
 | |
| template <typename T, typename Enable>
 | |
| struct guard_traits;
 | |
| 
 | |
| template <>
 | |
| struct guard_traits<base::binary_guard, void> {
 | |
| 	static base::binary_guard create(base::binary_guard value) {
 | |
| 		return value;
 | |
| 	}
 | |
| 	static bool check(const base::binary_guard &guard) {
 | |
| 		return guard.alive();
 | |
| 	}
 | |
| 
 | |
| };
 | |
| 
 | |
| } // namespace crl
 | 
