-
Notifications
You must be signed in to change notification settings - Fork 5.2k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Optimize and rename base::weak_unique_ptr.
Rename base::enable_weak_from_this -> base::has_weak_ptr. Rename base::weak_unique_ptr -> base::weak_ptr. Rename base::make_weak_unique -> base::make_weak. Rename base/weak_unique_ptr.h -> base/weak_ptr.h
- Loading branch information
1 parent
0bf854b
commit 2432845
Showing
26 changed files
with
388 additions
and
230 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,302 @@ | ||
/* | ||
This file is part of Telegram Desktop, | ||
the official desktop version of Telegram messaging app, see https://telegram.org | ||
Telegram Desktop is free software: you can redistribute it and/or modify | ||
it under the terms of the GNU General Public License as published by | ||
the Free Software Foundation, either version 3 of the License, or | ||
(at your option) any later version. | ||
It is distributed in the hope that it will be useful, | ||
but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
GNU General Public License for more details. | ||
In addition, as a special exception, the copyright holders give permission | ||
to link the code of portions of this program with the OpenSSL library. | ||
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE | ||
Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org | ||
*/ | ||
#pragma once | ||
|
||
#include <atomic> | ||
|
||
namespace base { | ||
namespace details { | ||
|
||
struct alive_tracker { | ||
std::atomic<int> counter = 1; | ||
std::atomic<bool> dead = false; | ||
}; | ||
|
||
inline alive_tracker *check_and_increment(alive_tracker *tracker) noexcept { | ||
if (tracker) { | ||
++tracker->counter; | ||
} | ||
return tracker; | ||
} | ||
|
||
inline void decrement(alive_tracker *tracker) noexcept { | ||
if (tracker->counter.fetch_sub(1) == 0) { | ||
delete tracker; | ||
} | ||
} | ||
|
||
} // namespace details | ||
|
||
class has_weak_ptr; | ||
|
||
template <typename T> | ||
class weak_ptr; | ||
|
||
class has_weak_ptr { | ||
public: | ||
has_weak_ptr() = default; | ||
has_weak_ptr(const has_weak_ptr &other) noexcept { | ||
} | ||
has_weak_ptr(has_weak_ptr &&other) noexcept { | ||
} | ||
has_weak_ptr &operator=(const has_weak_ptr &other) noexcept { | ||
return *this; | ||
} | ||
has_weak_ptr &operator=(has_weak_ptr &&other) noexcept { | ||
return *this; | ||
} | ||
|
||
~has_weak_ptr() { | ||
if (auto alive = _alive.load()) { | ||
alive->dead.store(true); | ||
details::decrement(alive); | ||
} | ||
} | ||
|
||
private: | ||
template <typename Child> | ||
friend class weak_ptr; | ||
|
||
details::alive_tracker *incrementAliveTracker() const { | ||
auto current = _alive.load(); | ||
if (!current) { | ||
auto alive = std::make_unique<details::alive_tracker>(); | ||
if (_alive.compare_exchange_strong(current, alive.get())) { | ||
return alive.release(); | ||
} | ||
} | ||
++current->counter; | ||
return current; | ||
} | ||
|
||
mutable std::atomic<details::alive_tracker*> _alive = nullptr; | ||
|
||
}; | ||
|
||
template <typename T> | ||
class weak_ptr { | ||
public: | ||
weak_ptr() = default; | ||
weak_ptr(T *value) | ||
: _alive(value ? value->incrementAliveTracker() : nullptr) | ||
, _value(value) { | ||
} | ||
weak_ptr(const std::unique_ptr<T> &value) | ||
: weak_ptr(value.get()) { | ||
} | ||
weak_ptr(const std::shared_ptr<T> &value) | ||
: weak_ptr(value.get()) { | ||
} | ||
weak_ptr(const std::weak_ptr<T> &value) | ||
: weak_ptr(value.lock().get()) { | ||
} | ||
weak_ptr(const weak_ptr &other) noexcept | ||
: _alive(details::check_and_increment(other._alive)) | ||
, _value(other._value) { | ||
} | ||
weak_ptr(weak_ptr &&other) noexcept | ||
: _alive(std::exchange(other._alive, nullptr)) | ||
, _value(std::exchange(other._value, nullptr)) { | ||
} | ||
template < | ||
typename Other, | ||
typename = std::enable_if_t< | ||
std::is_base_of_v<T, Other> && !std::is_same_v<T, Other>>> | ||
weak_ptr(const weak_ptr<Other> &other) noexcept | ||
: _alive(details::check_and_increment(other._alive)) | ||
, _value(other._value) { | ||
} | ||
template < | ||
typename Other, | ||
typename = std::enable_if_t< | ||
std::is_base_of_v<T, Other> && !std::is_same_v<T, Other>>> | ||
weak_ptr(weak_ptr<Other> &&other) noexcept | ||
: _alive(std::exchange(other._alive, nullptr)) | ||
, _value(std::exchange(other._value, nullptr)) { | ||
} | ||
|
||
weak_ptr &operator=(T *value) { | ||
reset(value); | ||
return *this; | ||
} | ||
weak_ptr &operator=(const std::unique_ptr<T> &value) { | ||
reset(value.get()); | ||
return *this; | ||
} | ||
weak_ptr &operator=(const std::shared_ptr<T> &value) { | ||
reset(value.get()); | ||
return *this; | ||
} | ||
weak_ptr &operator=(const std::weak_ptr<T> &value) { | ||
reset(value.lock().get()); | ||
return *this; | ||
} | ||
weak_ptr &operator=(const weak_ptr &other) noexcept { | ||
if (_value != other._value) { | ||
destroy(); | ||
_alive = details::check_and_increment(other._alive); | ||
_value = other._value; | ||
} | ||
return *this; | ||
} | ||
weak_ptr &operator=(weak_ptr &&other) noexcept { | ||
if (_value != other._value) { | ||
destroy(); | ||
_alive = std::exchange(other._alive, nullptr); | ||
_value = std::exchange(other._value, nullptr); | ||
} | ||
return *this; | ||
} | ||
template < | ||
typename Other, | ||
typename = std::enable_if_t< | ||
std::is_base_of_v<T, Other> && !std::is_same_v<T, Other>>> | ||
weak_ptr &operator=(const weak_ptr<Other> &other) noexcept { | ||
if (_value != other._value) { | ||
destroy(); | ||
_alive = details::check_and_increment(other._alive); | ||
_value = other._value; | ||
} | ||
return *this; | ||
} | ||
template < | ||
typename Other, | ||
typename = std::enable_if_t< | ||
std::is_base_of_v<T, Other> && !std::is_same_v<T, Other>>> | ||
weak_ptr &operator=(weak_ptr<Other> &&other) noexcept { | ||
if (_value != other._value) { | ||
destroy(); | ||
_alive = std::exchange(other._alive, nullptr); | ||
_value = std::exchange(other._value, nullptr); | ||
} | ||
return *this; | ||
} | ||
|
||
~weak_ptr() { | ||
destroy(); | ||
} | ||
|
||
T *get() const noexcept { | ||
return (_alive && !_alive->dead) ? _value : nullptr; | ||
} | ||
explicit operator bool() const noexcept { | ||
return (_alive && !_alive->dead); | ||
} | ||
T &operator*() const noexcept { | ||
return *get(); | ||
} | ||
T *operator->() const noexcept { | ||
return get(); | ||
} | ||
|
||
void reset(T *value = nullptr) { | ||
if (_value != value) { | ||
destroy(); | ||
_alive = value ? value->incrementAliveTracker() : nullptr; | ||
_value = value; | ||
} | ||
} | ||
|
||
private: | ||
void destroy() noexcept { | ||
if (_alive) { | ||
details::decrement(_alive); | ||
} | ||
} | ||
|
||
details::alive_tracker *_alive = nullptr; | ||
T *_value = nullptr; | ||
|
||
template <typename Other> | ||
friend class weak_ptr; | ||
|
||
}; | ||
|
||
template <typename T> | ||
inline bool operator==(const weak_ptr<T> &pointer, std::nullptr_t) noexcept { | ||
return (pointer.get() == nullptr); | ||
} | ||
|
||
template <typename T> | ||
inline bool operator==(std::nullptr_t, const weak_ptr<T> &pointer) noexcept { | ||
return (pointer == nullptr); | ||
} | ||
|
||
template <typename T> | ||
inline bool operator!=(const weak_ptr<T> &pointer, std::nullptr_t) noexcept { | ||
return !(pointer == nullptr); | ||
} | ||
|
||
template <typename T> | ||
inline bool operator!=(std::nullptr_t, const weak_ptr<T> &pointer) noexcept { | ||
return !(pointer == nullptr); | ||
} | ||
|
||
template < | ||
typename T, | ||
typename = std::enable_if_t<std::is_base_of_v<has_weak_ptr, T>>> | ||
weak_ptr<T> make_weak(T *value) { | ||
return value; | ||
} | ||
|
||
template < | ||
typename T, | ||
typename = std::enable_if_t<std::is_base_of_v<has_weak_ptr, T>>> | ||
weak_ptr<T> make_weak(const std::unique_ptr<T> &value) { | ||
return value; | ||
} | ||
|
||
template < | ||
typename T, | ||
typename = std::enable_if_t<std::is_base_of_v<has_weak_ptr, T>>> | ||
weak_ptr<T> make_weak(const std::shared_ptr<T> &value) { | ||
return value; | ||
} | ||
|
||
template < | ||
typename T, | ||
typename = std::enable_if_t<std::is_base_of_v<has_weak_ptr, T>>> | ||
weak_ptr<T> make_weak(const std::weak_ptr<T> &value) { | ||
return value; | ||
} | ||
|
||
} // namespace base | ||
|
||
#ifdef QT_VERSION | ||
template <typename Lambda> | ||
inline void InvokeQueued(const base::has_weak_ptr *context, Lambda &&lambda) { | ||
auto callback = [ | ||
guard = base::make_weak(context), | ||
lambda = std::forward<Lambda>(lambda) | ||
] { | ||
if (guard) { | ||
lambda(); | ||
} | ||
}; | ||
QObject proxy; | ||
QObject::connect( | ||
&proxy, | ||
&QObject::destroyed, | ||
QCoreApplication::instance(), | ||
std::move(callback), | ||
Qt::QueuedConnection); | ||
} | ||
#endif // QT_VERSION |
Oops, something went wrong.