45e855f89d
I can't maintain anymore this amount of features while keeping the library constantly updated and without bugs. Every merge was taking me multiple hours of revisioning the code. I give up. From this commit onwards TDLight will only have small useful customizations that are easy to maintain. Now the people relying on the OptimizeMemory method can restart the session every N hours to free up the memory. The real way to keep a low memory usage must involve a huge refactoring to allow the unloading of the caches into the sqlite database, similar to what's already happening with messages data. Only Levlam has the ability to implement this without needing to merge the upstream everytime.
229 lines
4.3 KiB
C++
229 lines
4.3 KiB
C++
//
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2021
|
|
//
|
|
// 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/algorithm.h"
|
|
#include "td/utils/common.h"
|
|
|
|
#include <algorithm>
|
|
#include <set>
|
|
#include <utility>
|
|
|
|
namespace td {
|
|
|
|
template <class T>
|
|
class FastSetWithPosition {
|
|
public:
|
|
std::vector<T> get_some_elements() const {
|
|
std::vector<T> res;
|
|
res.reserve(4);
|
|
if (!checked_.empty()) {
|
|
res.push_back(*checked_.begin());
|
|
res.push_back(*checked_.rbegin());
|
|
}
|
|
if (!not_checked_.empty()) {
|
|
res.push_back(*not_checked_.begin());
|
|
res.push_back(*not_checked_.rbegin());
|
|
}
|
|
td::unique(res);
|
|
if (res.size() > 2) {
|
|
res.erase(res.begin() + 1, res.end() - 1);
|
|
}
|
|
return res;
|
|
}
|
|
|
|
bool add(T x) {
|
|
if (checked_.count(x) != 0) {
|
|
return false;
|
|
}
|
|
return not_checked_.insert(x).second;
|
|
}
|
|
|
|
bool remove(T x) {
|
|
return checked_.erase(x) != 0 || not_checked_.erase(x) != 0;
|
|
}
|
|
|
|
bool has_next() const {
|
|
return !not_checked_.empty();
|
|
}
|
|
|
|
void reset_position() {
|
|
if (not_checked_.empty()) {
|
|
not_checked_ = std::move(checked_);
|
|
} else {
|
|
not_checked_.insert(checked_.begin(), checked_.end());
|
|
}
|
|
reset_to_empty(checked_);
|
|
}
|
|
|
|
T next() {
|
|
CHECK(has_next());
|
|
auto it = not_checked_.begin();
|
|
auto res = *it;
|
|
not_checked_.erase(it);
|
|
checked_.insert(res);
|
|
return res;
|
|
}
|
|
|
|
void merge(FastSetWithPosition &&other) {
|
|
if (this == &other) {
|
|
return;
|
|
}
|
|
|
|
if (size() < other.size()) {
|
|
std::swap(*this, other);
|
|
}
|
|
for (auto x : other.checked_) {
|
|
not_checked_.erase(x);
|
|
checked_.insert(x);
|
|
}
|
|
|
|
for (auto x : other.not_checked_) {
|
|
if (checked_.count(x) != 0) {
|
|
continue;
|
|
}
|
|
not_checked_.insert(x);
|
|
}
|
|
}
|
|
|
|
size_t size() const {
|
|
return checked_.size() + not_checked_.size();
|
|
}
|
|
|
|
bool empty() const {
|
|
return size() == 0;
|
|
}
|
|
|
|
private:
|
|
std::set<T> checked_;
|
|
std::set<T> not_checked_;
|
|
};
|
|
|
|
template <class T>
|
|
class SetWithPosition {
|
|
public:
|
|
std::vector<T> get_some_elements() const {
|
|
if (fast_) {
|
|
return fast_->get_some_elements();
|
|
}
|
|
if (has_value_) {
|
|
return {value_};
|
|
}
|
|
return {};
|
|
}
|
|
|
|
bool add(T x) {
|
|
if (fast_) {
|
|
return fast_->add(x);
|
|
}
|
|
if (!has_value_) {
|
|
value_ = x;
|
|
has_value_ = true;
|
|
is_checked_ = false;
|
|
return true;
|
|
}
|
|
if (value_ == x) {
|
|
return false;
|
|
}
|
|
make_fast();
|
|
return fast_->add(x);
|
|
}
|
|
|
|
bool remove(T x) {
|
|
if (fast_) {
|
|
return fast_->remove(x);
|
|
}
|
|
if (has_value_ && value_ == x) {
|
|
has_value_ = false;
|
|
is_checked_ = false;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool has_next() const {
|
|
if (fast_) {
|
|
return fast_->has_next();
|
|
}
|
|
return has_value_ && !is_checked_;
|
|
}
|
|
|
|
void reset_position() {
|
|
if (fast_) {
|
|
fast_->reset_position();
|
|
return;
|
|
}
|
|
is_checked_ = false;
|
|
}
|
|
|
|
T next() {
|
|
CHECK(has_next());
|
|
if (fast_) {
|
|
return fast_->next();
|
|
}
|
|
is_checked_ = true;
|
|
return value_;
|
|
}
|
|
|
|
void merge(SetWithPosition &&other) {
|
|
if (this == &other) {
|
|
return;
|
|
}
|
|
if (size() < other.size()) {
|
|
std::swap(*this, other);
|
|
}
|
|
if (other.size() == 0) {
|
|
return;
|
|
}
|
|
if (other.fast_ == nullptr && fast_ == nullptr && value_ == other.value_) {
|
|
is_checked_ |= other.is_checked_;
|
|
other.value_ = T();
|
|
other.has_value_ = false;
|
|
other.is_checked_ = false;
|
|
return;
|
|
}
|
|
make_fast();
|
|
other.make_fast();
|
|
fast_->merge(std::move(*other.fast_));
|
|
reset_to_empty(other);
|
|
}
|
|
|
|
size_t size() const {
|
|
if (fast_) {
|
|
return fast_->size();
|
|
}
|
|
return static_cast<size_t>(has_value_);
|
|
}
|
|
|
|
bool empty() const {
|
|
if (fast_) {
|
|
return false;
|
|
}
|
|
return !has_value_;
|
|
}
|
|
|
|
private:
|
|
T value_{};
|
|
bool has_value_{false};
|
|
bool is_checked_{false};
|
|
unique_ptr<FastSetWithPosition<T>> fast_;
|
|
|
|
void make_fast() {
|
|
if (fast_) {
|
|
return;
|
|
}
|
|
fast_ = make_unique<FastSetWithPosition<T>>();
|
|
CHECK(has_value_);
|
|
fast_->add(value_);
|
|
if (is_checked_) {
|
|
fast_->next();
|
|
}
|
|
}
|
|
};
|
|
|
|
} // namespace td
|