Files
scylladb/utils/stall_free.hh
Petr Gusev 79c6bf0885 clear_gently: remove noexcept for rvalue references overload
We use this overload in vnode_erm, one of the
arguments is boost::icl::interval_map,
whose move constructor is not noexcept.
2023-05-24 12:08:19 +04:00

276 lines
7.5 KiB
C++

/*
* Copyright (C) 2020-present ScyllaDB
*/
/*
* SPDX-License-Identifier: AGPL-3.0-or-later
*/
#pragma once
#include <list>
#include <algorithm>
#include <seastar/core/thread.hh>
#include <seastar/core/future.hh>
#include <seastar/core/future-util.hh>
#include <seastar/core/sharded.hh>
#include <seastar/core/when_all.hh>
#include <seastar/core/do_with.hh>
#include "utils/collection-concepts.hh"
using namespace seastar;
namespace utils {
// Similar to std::merge but it does not stall. Must run inside a seastar
// thread. It merges items from list2 into list1. Items from list2 can only be copied.
template<class T, class Compare>
requires LessComparable<T, T, Compare>
void merge_to_gently(std::list<T>& list1, const std::list<T>& list2, Compare comp) {
auto first1 = list1.begin();
auto first2 = list2.begin();
auto last1 = list1.end();
auto last2 = list2.end();
while (first2 != last2) {
seastar::thread::maybe_yield();
if (first1 == last1) {
// Copy remaining items of list2 into list1
list1.insert(last1, *first2);
++first2;
continue;
}
if (comp(*first2, *first1)) {
list1.insert(first1, *first2);
++first2;
} else {
++first1;
}
}
}
// The clear_gently functions are meant for
// gently destroying the contents of containers.
// The containers can be re-used after clear_gently
// or may be destroyed. But unlike e.g. std::vector::clear(),
// clear_gently will not necessarily keep the object allocation.
template <typename T>
concept HasClearGentlyMethod = requires (T x) {
{ x.clear_gently() } -> std::same_as<future<>>;
};
template <typename T>
concept SmartPointer = requires (T x) {
{ x.get() } -> std::same_as<typename T::element_type*>;
{ *x } -> std::same_as<typename T::element_type&>;
};
template <typename T>
concept SharedPointer = SmartPointer<T> && requires (T x) {
{ x.use_count() } -> std::convertible_to<long>;
};
template <typename T>
concept StringLike = requires (T x) {
std::is_same_v<typename T::traits_type, std::char_traits<typename T::value_type>>;
};
template <typename T>
concept Iterable = requires (T x) {
{ x.empty() } -> std::same_as<bool>;
{ x.begin() } -> std::same_as<typename T::iterator>;
{ x.end() } -> std::same_as<typename T::iterator>;
};
template <typename T>
concept Sequence = Iterable<T> && requires (T x, size_t n) {
{ x.back() } -> std::same_as<typename T::value_type&>;
{ x.pop_back() } -> std::same_as<void>;
};
template <typename T>
concept TriviallyClearableSequence =
Sequence<T> && std::is_trivially_destructible_v<typename T::value_type> && !HasClearGentlyMethod<typename T::value_type> && requires (T s) {
{ s.clear() } -> std::same_as<void>;
};
template <typename T>
concept Container = Iterable<T> && requires (T x, typename T::iterator it) {
x.erase(it);
};
template <typename T>
concept MapLike = Container<T> && requires (T x) {
std::is_same_v<typename T::value_type, std::pair<const typename T::key_type, typename T::mapped_type>>;
};
template <HasClearGentlyMethod T>
future<> clear_gently(T& o) noexcept;
template <typename T>
future<> clear_gently(foreign_ptr<T>& o) noexcept;
template <SharedPointer T>
future<> clear_gently(T& o) noexcept;
template <SmartPointer T>
future<> clear_gently(T& o) noexcept;
template <typename T, std::size_t N>
future<> clear_gently(std::array<T, N>&a) noexcept;
template <typename T>
requires (StringLike<T> || TriviallyClearableSequence<T>)
future<> clear_gently(T& s) noexcept;
template <Sequence T>
requires (!StringLike<T> && !TriviallyClearableSequence<T>)
future<> clear_gently(T& v) noexcept;
template <MapLike T>
future<> clear_gently(T& c) noexcept;
template <Container T>
requires (!StringLike<T> && !Sequence<T> && !MapLike<T>)
future<> clear_gently(T& c) noexcept;
template <typename T>
future<> clear_gently(std::optional<T>& opt) noexcept;
template <typename T>
future<> clear_gently(seastar::optimized_optional<T>& opt) noexcept;
namespace internal {
template <typename T>
concept HasClearGentlyImpl = requires (T x) {
{ clear_gently(x) } -> std::same_as<future<>>;
};
template <typename T>
requires HasClearGentlyImpl<T>
future<> clear_gently(T& x) noexcept {
return utils::clear_gently(x);
}
// This default implementation of clear_gently
// is required to "terminate" recursive clear_gently calls
// at trivial objects
template <typename T>
future<> clear_gently(T&) noexcept {
return make_ready_future<>();
}
} // namespace internal
template <HasClearGentlyMethod T>
future<> clear_gently(T& o) noexcept {
return futurize_invoke(std::bind(&T::clear_gently, &o));
}
template <typename T>
future<> clear_gently(foreign_ptr<T>& o) noexcept {
return smp::submit_to(o.get_owner_shard(), [&o] {
return internal::clear_gently(*o);
});
}
template <typename... T>
requires (std::is_rvalue_reference_v<T&&> && ...)
future<> clear_gently(T&&... o) {
return do_with(std::move(o)..., [](auto&... args) {
return when_all(clear_gently(args)...).discard_result();
});
}
template <SharedPointer T>
future<> clear_gently(T& o) noexcept {
if (o.use_count() == 1) {
return internal::clear_gently(*o);
}
return make_ready_future<>();
}
template <SmartPointer T>
future<> clear_gently(T& o) noexcept {
if (auto p = o.get()) {
return internal::clear_gently(*p);
} else {
return make_ready_future<>();
}
}
template <typename T, std::size_t N>
future<> clear_gently(std::array<T, N>& a) noexcept {
return do_for_each(a, [] (T& o) {
return internal::clear_gently(o);
});
}
// Trivially destructible elements can be safely cleared in bulk
template <typename T>
requires (StringLike<T> || TriviallyClearableSequence<T>)
future<> clear_gently(T& s) noexcept {
// Note: clear() is pointless in this case since it keeps the allocation
// and since the values are trivially destructible it achieves nothing.
// `s = {}` will free the vector/string allocation.
s = {};
return make_ready_future<>();
}
// Clear the elements gently and destroy them one-by-one
// in reverse order, to avoid copying.
template <Sequence T>
requires (!StringLike<T> && !TriviallyClearableSequence<T>)
future<> clear_gently(T& v) noexcept {
return do_until([&v] { return v.empty(); }, [&v] {
return internal::clear_gently(v.back()).then([&v] {
v.pop_back();
});
});
return make_ready_future<>();
}
template <MapLike T>
future<> clear_gently(T& c) noexcept {
return do_until([&c] { return c.empty(); }, [&c] {
auto it = c.begin();
return internal::clear_gently(it->second).then([&c, it = std::move(it)] () mutable {
c.erase(it);
});
});
}
template <Container T>
requires (!StringLike<T> && !Sequence<T> && !MapLike<T>)
future<> clear_gently(T& c) noexcept {
return do_until([&c] { return c.empty(); }, [&c] {
auto it = c.begin();
return internal::clear_gently(*it).then([&c, it = std::move(it)] () mutable {
c.erase(it);
});
});
}
template <typename T>
future<> clear_gently(std::optional<T>& opt) noexcept {
if (opt) {
return utils::clear_gently(*opt);
} else {
return make_ready_future<>();
}
}
template <typename T>
future<> clear_gently(seastar::optimized_optional<T>& opt) noexcept {
if (opt) {
return utils::clear_gently(*opt);
} else {
return make_ready_future<>();
}
}
}