File indexing completed on 2025-09-17 09:09:13
0001
0002
0003
0004
0005
0006 #if 0
0007 #pragma qt_sync_skip_header_check
0008 #pragma qt_sync_stop_processing
0009 #endif
0010
0011 #ifndef QCONTAINERTOOLS_IMPL_H
0012 #define QCONTAINERTOOLS_IMPL_H
0013
0014 #include <QtCore/qglobal.h>
0015 #include <QtCore/qtypeinfo.h>
0016
0017 #include <QtCore/qxptype_traits.h>
0018
0019 #include <cstring>
0020 #include <iterator>
0021 #include <memory>
0022 #include <algorithm>
0023
0024 QT_BEGIN_NAMESPACE
0025
0026 namespace QtPrivate
0027 {
0028
0029
0030
0031
0032
0033
0034
0035 template<typename T, typename Cmp = std::less<>>
0036 static constexpr bool q_points_into_range(const T *p, const T *b, const T *e,
0037 Cmp less = {}) noexcept
0038 {
0039 return !less(p, b) && less(p, e);
0040 }
0041
0042
0043
0044
0045
0046
0047
0048 template <typename C, typename T>
0049 static constexpr bool q_points_into_range(const T &p, const C &c) noexcept
0050 {
0051 static_assert(std::is_same_v<decltype(std::data(c)), T>);
0052
0053
0054
0055 return q_points_into_range(p, std::data(c),
0056 std::data(c) + std::distance(std::begin(c), std::end(c)));
0057 }
0058
0059 QT_WARNING_PUSH
0060 QT_WARNING_DISABLE_GCC("-Wmaybe-uninitialized")
0061
0062 template <typename T, typename N>
0063 void q_uninitialized_move_if_noexcept_n(T* first, N n, T* out)
0064 {
0065 if constexpr (std::is_nothrow_move_constructible_v<T> || !std::is_copy_constructible_v<T>)
0066 std::uninitialized_move_n(first, n, out);
0067 else
0068 std::uninitialized_copy_n(first, n, out);
0069 }
0070
0071 template <typename T, typename N>
0072 void q_uninitialized_relocate_n(T* first, N n, T* out)
0073 {
0074 if constexpr (QTypeInfo<T>::isRelocatable) {
0075 static_assert(std::is_copy_constructible_v<T> || std::is_move_constructible_v<T>,
0076 "Refusing to relocate this non-copy/non-move-constructible type.");
0077 if (n != N(0)) {
0078 std::memcpy(static_cast<void *>(out),
0079 static_cast<const void *>(first),
0080 n * sizeof(T));
0081 }
0082 } else {
0083 q_uninitialized_move_if_noexcept_n(first, n, out);
0084 if constexpr (QTypeInfo<T>::isComplex)
0085 std::destroy_n(first, n);
0086 }
0087 }
0088
0089 QT_WARNING_POP
0090
0091
0092
0093
0094
0095
0096
0097
0098
0099
0100 template <typename T>
0101 void q_rotate(T *first, T *mid, T *last)
0102 {
0103 if constexpr (QTypeInfo<T>::isRelocatable) {
0104 const auto cast = [](T *p) { return reinterpret_cast<uchar*>(p); };
0105 std::rotate(cast(first), cast(mid), cast(last));
0106 } else {
0107 std::rotate(first, mid, last);
0108 }
0109 }
0110
0111
0112
0113
0114
0115
0116
0117
0118
0119
0120
0121
0122
0123 template <typename T, typename Predicate>
0124 T *q_uninitialized_remove_copy_if(T *first, T *last, T *out, Predicate &pred)
0125 {
0126 static_assert(std::is_nothrow_destructible_v<T>,
0127 "This algorithm requires that T has a non-throwing destructor");
0128 Q_ASSERT(!q_points_into_range(out, first, last));
0129
0130 T *dest_begin = out;
0131 QT_TRY {
0132 while (first != last) {
0133 if (!pred(*first)) {
0134 new (std::addressof(*out)) T(*first);
0135 ++out;
0136 }
0137 ++first;
0138 }
0139 } QT_CATCH (...) {
0140 std::destroy(std::reverse_iterator(out), std::reverse_iterator(dest_begin));
0141 QT_RETHROW;
0142 }
0143 return out;
0144 }
0145
0146 template<typename iterator, typename N>
0147 void q_relocate_overlap_n_left_move(iterator first, N n, iterator d_first)
0148 {
0149
0150
0151
0152
0153
0154 Q_ASSERT(n);
0155 Q_ASSERT(d_first < first);
0156 using T = typename std::iterator_traits<iterator>::value_type;
0157
0158
0159
0160
0161
0162
0163
0164
0165 struct Destructor
0166 {
0167 iterator *iter;
0168 iterator end;
0169 iterator intermediate;
0170
0171 Destructor(iterator &it) noexcept : iter(std::addressof(it)), end(it) { }
0172 void commit() noexcept { iter = std::addressof(end); }
0173 void freeze() noexcept
0174 {
0175 intermediate = *iter;
0176 iter = std::addressof(intermediate);
0177 }
0178 ~Destructor() noexcept
0179 {
0180 for (const int step = *iter < end ? 1 : -1; *iter != end;) {
0181 std::advance(*iter, step);
0182 (*iter)->~T();
0183 }
0184 }
0185 } destroyer(d_first);
0186
0187 const iterator d_last = d_first + n;
0188
0189
0190
0191
0192 auto pair = std::minmax(d_last, first);
0193
0194
0195
0196 iterator overlapBegin = pair.first;
0197 iterator overlapEnd = pair.second;
0198
0199
0200 while (d_first != overlapBegin) {
0201
0202 new (std::addressof(*d_first)) T(std::move_if_noexcept(*first));
0203 ++d_first;
0204 ++first;
0205 }
0206
0207
0208
0209 destroyer.freeze();
0210
0211
0212 while (d_first != d_last) {
0213 *d_first = std::move_if_noexcept(*first);
0214 ++d_first;
0215 ++first;
0216 }
0217
0218 Q_ASSERT(d_first == destroyer.end + n);
0219 destroyer.commit();
0220
0221 while (first != overlapEnd)
0222 (--first)->~T();
0223 }
0224
0225
0226
0227
0228
0229
0230
0231
0232
0233
0234
0235 template<typename T, typename N>
0236 void q_relocate_overlap_n(T *first, N n, T *d_first)
0237 {
0238 static_assert(std::is_nothrow_destructible_v<T>,
0239 "This algorithm requires that T has a non-throwing destructor");
0240
0241 if (n == N(0) || first == d_first || first == nullptr || d_first == nullptr)
0242 return;
0243
0244 if constexpr (QTypeInfo<T>::isRelocatable) {
0245 std::memmove(static_cast<void *>(d_first), static_cast<const void *>(first), n * sizeof(T));
0246 } else {
0247 if (d_first < first) {
0248 q_relocate_overlap_n_left_move(first, n, d_first);
0249 } else {
0250 auto rfirst = std::make_reverse_iterator(first + n);
0251 auto rd_first = std::make_reverse_iterator(d_first + n);
0252 q_relocate_overlap_n_left_move(rfirst, n, rd_first);
0253 }
0254 }
0255 }
0256
0257 template <typename T>
0258 struct ArrowProxy
0259 {
0260 T t;
0261 T *operator->() noexcept { return &t; }
0262 };
0263
0264 template <typename Iterator>
0265 using IfIsInputIterator = typename std::enable_if<
0266 std::is_convertible<typename std::iterator_traits<Iterator>::iterator_category, std::input_iterator_tag>::value,
0267 bool>::type;
0268
0269 template <typename Iterator>
0270 using IfIsForwardIterator = typename std::enable_if<
0271 std::is_convertible<typename std::iterator_traits<Iterator>::iterator_category, std::forward_iterator_tag>::value,
0272 bool>::type;
0273
0274 template <typename Iterator>
0275 using IfIsNotForwardIterator = typename std::enable_if<
0276 !std::is_convertible<typename std::iterator_traits<Iterator>::iterator_category, std::forward_iterator_tag>::value,
0277 bool>::type;
0278
0279 template <typename Container,
0280 typename InputIterator,
0281 IfIsNotForwardIterator<InputIterator> = true>
0282 void reserveIfForwardIterator(Container *, InputIterator, InputIterator)
0283 {
0284 }
0285
0286 template <typename Container,
0287 typename ForwardIterator,
0288 IfIsForwardIterator<ForwardIterator> = true>
0289 void reserveIfForwardIterator(Container *c, ForwardIterator f, ForwardIterator l)
0290 {
0291 c->reserve(static_cast<typename Container::size_type>(std::distance(f, l)));
0292 }
0293
0294 template <typename Iterator>
0295 using KeyAndValueTest = decltype(
0296 std::declval<Iterator &>().key(),
0297 std::declval<Iterator &>().value()
0298 );
0299
0300 template <typename Iterator>
0301 using FirstAndSecondTest = decltype(
0302 (*std::declval<Iterator &>()).first,
0303 (*std::declval<Iterator &>()).second
0304 );
0305
0306 template <typename Iterator>
0307 using IfAssociativeIteratorHasKeyAndValue =
0308 std::enable_if_t<qxp::is_detected_v<KeyAndValueTest, Iterator>, bool>;
0309
0310 template <typename Iterator>
0311 using IfAssociativeIteratorHasFirstAndSecond =
0312 std::enable_if_t<
0313 std::conjunction_v<
0314 std::negation<qxp::is_detected<KeyAndValueTest, Iterator>>,
0315 qxp::is_detected<FirstAndSecondTest, Iterator>
0316 >, bool>;
0317
0318 template <typename Iterator>
0319 using MoveBackwardsTest = decltype(
0320 std::declval<Iterator &>().operator--()
0321 );
0322
0323 template <typename Iterator>
0324 using IfIteratorCanMoveBackwards =
0325 std::enable_if_t<qxp::is_detected_v<MoveBackwardsTest, Iterator>, bool>;
0326
0327 template <typename T, typename U>
0328 using IfIsNotSame =
0329 typename std::enable_if<!std::is_same<T, U>::value, bool>::type;
0330
0331 template<typename T, typename U>
0332 using IfIsNotConvertible = typename std::enable_if<!std::is_convertible<T, U>::value, bool>::type;
0333
0334 template <typename Container, typename Predicate>
0335 auto sequential_erase_if(Container &c, Predicate &pred)
0336 {
0337
0338
0339
0340
0341
0342 const auto cbegin = c.cbegin();
0343 const auto cend = c.cend();
0344 const auto t_it = std::find_if(cbegin, cend, pred);
0345 auto result = std::distance(cbegin, t_it);
0346 if (result == c.size())
0347 return result - result;
0348
0349
0350 const auto e = c.end();
0351
0352 auto it = std::next(c.begin(), result);
0353 auto dest = it;
0354
0355
0356
0357
0358
0359 while (++it != e) {
0360 if (!pred(*it)) {
0361 *dest = std::move(*it);
0362 ++dest;
0363 }
0364 }
0365
0366 result = std::distance(dest, e);
0367 c.erase(dest, e);
0368 return result;
0369 }
0370
0371 template <typename Container, typename T>
0372 auto sequential_erase(Container &c, const T &t)
0373 {
0374
0375 auto cmp = [&](const auto &e) -> bool { return e == t; };
0376 return sequential_erase_if(c, cmp);
0377 }
0378
0379 template <typename Container, typename T>
0380 auto sequential_erase_with_copy(Container &c, const T &t)
0381 {
0382 using CopyProxy = std::conditional_t<std::is_copy_constructible_v<T>, T, const T &>;
0383 return sequential_erase(c, CopyProxy(t));
0384 }
0385
0386 template <typename Container, typename T>
0387 auto sequential_erase_one(Container &c, const T &t)
0388 {
0389 const auto cend = c.cend();
0390 const auto it = std::find(c.cbegin(), cend, t);
0391 if (it == cend)
0392 return false;
0393 c.erase(it);
0394 return true;
0395 }
0396
0397 template <typename T, typename Predicate>
0398 qsizetype qset_erase_if(QSet<T> &set, Predicate &pred)
0399 {
0400 qsizetype result = 0;
0401 auto it = set.cbegin();
0402 auto e = set.cend();
0403 while (it != e) {
0404 if (pred(*it)) {
0405 ++result;
0406 it = set.erase(it);
0407 e = set.cend();
0408 } else {
0409 ++it;
0410 }
0411 }
0412 return result;
0413 }
0414
0415
0416
0417 template <typename R, typename F, typename ... ArgTypes>
0418 struct is_invoke_result_explicitly_convertible : std::is_constructible<R, std::invoke_result_t<F, ArgTypes...>>
0419 {};
0420
0421
0422
0423 template <typename R, typename F, typename ... ArgTypes>
0424 constexpr bool is_invocable_explicit_r_v = std::conjunction_v<
0425 std::is_invocable<F, ArgTypes...>,
0426 is_invoke_result_explicitly_convertible<R, F, ArgTypes...>
0427 >;
0428
0429 template <typename Container, typename Predicate>
0430 auto associative_erase_if(Container &c, Predicate &pred)
0431 {
0432
0433
0434 using Iterator = typename Container::iterator;
0435 using Key = typename Container::key_type;
0436 using Value = typename Container::mapped_type;
0437 using KeyValuePair = std::pair<const Key &, Value &>;
0438
0439 typename Container::size_type result = 0;
0440
0441 auto it = c.begin();
0442 const auto e = c.end();
0443 while (it != e) {
0444 if constexpr (is_invocable_explicit_r_v<bool, Predicate &, Iterator &>) {
0445 if (pred(it)) {
0446 it = c.erase(it);
0447 ++result;
0448 } else {
0449 ++it;
0450 }
0451 } else if constexpr (is_invocable_explicit_r_v<bool, Predicate &, KeyValuePair &&>) {
0452 KeyValuePair p(it.key(), it.value());
0453 if (pred(std::move(p))) {
0454 it = c.erase(it);
0455 ++result;
0456 } else {
0457 ++it;
0458 }
0459 } else {
0460 static_assert(sizeof(Container) == 0, "Predicate has an incompatible signature");
0461 }
0462 }
0463
0464 return result;
0465 }
0466
0467 }
0468
0469 QT_END_NAMESPACE
0470
0471 #endif