// Copyright (C) 2021 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0 #include #include #include static const int N = 1000; struct MyBase { MyBase(int i_) : i(i_) { } MyBase(const MyBase &other) : i(other.i) { } MyBase &operator=(const MyBase &other) { i = other.i; return *this; } bool operator==(const MyBase &other) const { return i == other.i; } protected: int i; }; struct MyPrimitive : public MyBase { MyPrimitive(int i_ = -1) : MyBase(i_) { } MyPrimitive(const MyPrimitive &other) : MyBase(other) { } MyPrimitive &operator=(const MyPrimitive &other) { MyBase::operator=(other); return *this; } }; struct MyMovable : public MyBase { MyMovable(int i_ = -1) : MyBase(i_) {} }; struct MyComplex : public MyBase { MyComplex(int i_ = -1) : MyBase(i_) {} }; QT_BEGIN_NAMESPACE Q_DECLARE_TYPEINFO(MyPrimitive, Q_PRIMITIVE_TYPE); Q_DECLARE_TYPEINFO(MyMovable, Q_RELOCATABLE_TYPE); Q_DECLARE_TYPEINFO(MyComplex, Q_COMPLEX_TYPE); QT_END_NAMESPACE class tst_QList: public QObject { Q_OBJECT const int million = 1000000; private Q_SLOTS: void removeAll_primitive_data(); void removeAll_primitive() { removeAll_impl(); } void removeAll_movable_data() { removeAll_primitive_data(); } void removeAll_movable() { removeAll_impl(); } void removeAll_complex_data() { removeAll_primitive_data(); } void removeAll_complex() { removeAll_impl(); } // append 1 element: void appendOne_int_data() const { commonBenchmark_data(); } void appendOne_primitive_data() const { commonBenchmark_data(); } void appendOne_movable_data() const { commonBenchmark_data(); } void appendOne_complex_data() const { commonBenchmark_data(); } void appendOne_QString_data() const { commonBenchmark_data(); } void appendOne_int() const { appendOne_impl(); } // QTBUG-87330 void appendOne_primitive() const { appendOne_impl(); } void appendOne_movable() const { appendOne_impl(); } void appendOne_complex() const { appendOne_impl(); } void appendOne_QString() const { appendOne_impl(); } // prepend 1 element: void prependOne_int_data() const { commonBenchmark_data(); } void prependOne_primitive_data() const { commonBenchmark_data(); } void prependOne_movable_data() const { commonBenchmark_data(); } void prependOne_complex_data() const { commonBenchmark_data(); } void prependOne_QString_data() const { commonBenchmark_data(); } void prependOne_int() const { prependOne_impl(); } void prependOne_primitive() const { prependOne_impl(); } void prependOne_movable() const { prependOne_impl(); } void prependOne_complex() const { prependOne_impl(); } void prependOne_QString() const { prependOne_impl(); } // insert in middle 1 element (quadratic, slow): void midInsertOne_int_data() const { commonBenchmark_data(million); } void midInsertOne_primitive_data() const { commonBenchmark_data(million); } void midInsertOne_movable_data() const { commonBenchmark_data(million); } void midInsertOne_complex_data() const { commonBenchmark_data(million / 10); } void midInsertOne_QString_data() const { commonBenchmark_data(million / 10); } void midInsertOne_int() const { midInsertOne_impl(); } void midInsertOne_primitive() const { midInsertOne_impl(); } void midInsertOne_movable() const { midInsertOne_impl(); } void midInsertOne_complex() const { midInsertOne_impl(); } void midInsertOne_QString() const { midInsertOne_impl(); } // append/prepend 1 element - hard times for branch predictor: void appendPrependOne_int_data() const { commonBenchmark_data(); } void appendPrependOne_primitive_data() const { commonBenchmark_data(); } void appendPrependOne_movable_data() const { commonBenchmark_data(); } void appendPrependOne_complex_data() const { commonBenchmark_data(); } void appendPrependOne_QString_data() const { commonBenchmark_data(); } void appendPrependOne_int() const { appendPrependOne_impl(); } void appendPrependOne_primitive() const { appendPrependOne_impl(); } void appendPrependOne_movable() const { appendPrependOne_impl(); } void appendPrependOne_complex() const { appendPrependOne_impl(); } void appendPrependOne_QString() const { appendPrependOne_impl(); } // prepend half elements, then appen another half: void prependAppendHalvesOne_int_data() const { commonBenchmark_data(); } void prependAppendHalvesOne_primitive_data() const { commonBenchmark_data(); } void prependAppendHalvesOne_movable_data() const { commonBenchmark_data(); } void prependAppendHalvesOne_complex_data() const { commonBenchmark_data(); } void prependAppendHalvesOne_QString_data() const { commonBenchmark_data(); } void prependAppendHalvesOne_int() const { prependAppendHalvesOne_impl(); } void prependAppendHalvesOne_primitive() const { prependAppendHalvesOne_impl(); } void prependAppendHalvesOne_movable() const { prependAppendHalvesOne_impl(); } void prependAppendHalvesOne_complex() const { prependAppendHalvesOne_impl(); } void prependAppendHalvesOne_QString() const { prependAppendHalvesOne_impl(); } // emplace in middle 1 element (quadratic, slow): void midEmplaceOne_int_data() const { commonBenchmark_data(million); } void midEmplaceOne_primitive_data() const { commonBenchmark_data(million); } void midEmplaceOne_movable_data() const { commonBenchmark_data(million); } void midEmplaceOne_complex_data() const { commonBenchmark_data(million / 10); } void midEmplaceOne_QString_data() const { commonBenchmark_data(million / 10); } void midEmplaceOne_int() const { midEmplaceOne_impl(); } void midEmplaceOne_primitive() const { midEmplaceOne_impl(); } void midEmplaceOne_movable() const { midEmplaceOne_impl(); } void midEmplaceOne_complex() const { midEmplaceOne_impl(); } void midEmplaceOne_QString() const { midEmplaceOne_impl(); } // remove from beginning in a general way void removeFirstGeneral_int_data() const { commonBenchmark_data(); } void removeFirstGeneral_primitive_data() const { commonBenchmark_data(); } void removeFirstGeneral_movable_data() const { commonBenchmark_data(); } void removeFirstGeneral_complex_data() const { commonBenchmark_data(); } void removeFirstGeneral_QString_data() const { commonBenchmark_data(); } void removeFirstGeneral_int() const { removeFirstGeneral_impl(); } void removeFirstGeneral_primitive() const { removeFirstGeneral_impl(); } void removeFirstGeneral_movable() const { removeFirstGeneral_impl(); } void removeFirstGeneral_complex() const { removeFirstGeneral_impl(); } void removeFirstGeneral_QString() const { removeFirstGeneral_impl(); } // remove from beginning in a special way (using fast part of QList::removeFirst()) void removeFirstSpecial_int_data() const { commonBenchmark_data(); } void removeFirstSpecial_primitive_data() const { commonBenchmark_data(); } void removeFirstSpecial_movable_data() const { commonBenchmark_data(); } void removeFirstSpecial_complex_data() const { commonBenchmark_data(); } void removeFirstSpecial_QString_data() const { commonBenchmark_data(); } void removeFirstSpecial_int() const { removeFirstSpecial_impl(); } void removeFirstSpecial_primitive() const { removeFirstSpecial_impl(); } void removeFirstSpecial_movable() const { removeFirstSpecial_impl(); } void removeFirstSpecial_complex() const { removeFirstSpecial_impl(); } void removeFirstSpecial_QString() const { removeFirstSpecial_impl(); } private: template void removeAll_impl() const; template void commonBenchmark_data(int max = 200000000) const; template typename, typename> void appendOne_impl() const; template typename, typename> void prependOne_impl() const; template typename, typename> void midInsertOne_impl() const; template typename, typename> void appendPrependOne_impl() const; template typename, typename> void prependAppendHalvesOne_impl() const; template typename, typename> void midEmplaceOne_impl() const; template typename, typename> void removeFirstGeneral_impl() const; template typename, typename> void removeFirstSpecial_impl() const; }; template void tst_QList::removeAll_impl() const { QFETCH(QList, i10); QFETCH(int, itemsToRemove); constexpr int valueToRemove = 5; QList list; for (int i = 0; i < 10 * N; ++i) { T t(i10.at(i % 10)); list.append(t); } T t(valueToRemove); qsizetype removedCount = 0; // make compiler happy by setting to 0 QList l; QBENCHMARK { l = list; removedCount = l.removeAll(t); } QCOMPARE(removedCount, itemsToRemove * N); QCOMPARE(l.size() + removedCount, list.size()); QVERIFY(!l.contains(valueToRemove)); } void tst_QList::removeAll_primitive_data() { qRegisterMetaType >(); QTest::addColumn >("i10"); QTest::addColumn("itemsToRemove"); QTest::newRow("0%") << QList(10, 0) << 0; QTest::newRow("10%") << (QList() << 0 << 0 << 0 << 0 << 5 << 0 << 0 << 0 << 0 << 0) << 1; QTest::newRow("90%") << (QList() << 5 << 5 << 5 << 5 << 0 << 5 << 5 << 5 << 5 << 5) << 9; QTest::newRow("100%") << QList(10, 5) << 10; } template void tst_QList::commonBenchmark_data(int max) const { QTest::addColumn("elemCount"); const auto addRow = [](int count, const char *text) { QTest::newRow(text) << count; }; const auto p = [](int i, const char *text) { return std::make_pair(i, text); }; // cap at 20m elements to allow 5.15/6.0 coverage to be the same for (auto pair : { p(100, "100"), p(1000, "1k"), p(10000, "10k"), p(100000, "100k"), p(1000000, "1m"), p(10000000, "10m"), p(20000000, "20m") }) { if (pair.first <= max) addRow(pair.first, pair.second); } } template typename Container, typename T> void tst_QList::appendOne_impl() const { QFETCH(int, elemCount); constexpr auto getValue = []() { return T {}; }; QBENCHMARK { Container container; auto lvalue = getValue(); for (int i = 0; i < elemCount; ++i) { container.append(lvalue); } } } template typename Container, typename T> void tst_QList::prependOne_impl() const { QFETCH(int, elemCount); constexpr auto getValue = []() { return T {}; }; QBENCHMARK { Container container; auto lvalue = getValue(); for (int i = 0; i < elemCount; ++i) { container.prepend(lvalue); } } } template typename Container, typename T> void tst_QList::midInsertOne_impl() const { QFETCH(int, elemCount); constexpr auto getValue = []() { return T {}; }; QBENCHMARK { Container container; auto lvalue = getValue(); for (int i = 0; i < elemCount; ++i) { const int remainder = i % 2; // use insert(i, n, t) as insert(i, t) calls emplace (implementation // detail) container.insert(container.size() / 2 + remainder, 1, lvalue); } } } template typename Container, typename T> void tst_QList::appendPrependOne_impl() const { QFETCH(int, elemCount); constexpr auto getValue = []() { return T {}; }; QBENCHMARK { Container container; auto lvalue = getValue(); for (int i = 0; i < elemCount; ++i) { if (i % 2 == 0) { container.append(lvalue); } else { container.prepend(lvalue); } } } } template typename Container, typename T> void tst_QList::prependAppendHalvesOne_impl() const { QFETCH(int, elemCount); constexpr auto getValue = []() { return T {}; }; QBENCHMARK { Container container; auto lvalue = getValue(); for (int i = 0; i < elemCount / 2; ++i) { container.prepend(lvalue); } for (int i = elemCount / 2; i < elemCount; ++i) { container.append(lvalue); } } } template typename Container, typename T> void tst_QList::midEmplaceOne_impl() const { QFETCH(int, elemCount); constexpr auto getValue = []() { return T {}; }; QBENCHMARK { Container container; auto lvalue = getValue(); for (int i = 0; i < elemCount; ++i) { const int remainder = i % 2; container.emplace(container.size() / 2 + remainder, lvalue); } } } template typename Container, typename T> void tst_QList::removeFirstGeneral_impl() const { QFETCH(int, elemCount); constexpr auto getValue = []() { return T {}; }; QBENCHMARK { Container container(elemCount, getValue()); for (int i = 0; i < elemCount - 1; ++i) { container.remove(0, 1); } } } template typename Container, typename T> void tst_QList::removeFirstSpecial_impl() const { QFETCH(int, elemCount); constexpr auto getValue = []() { return T {}; }; QBENCHMARK { Container container(elemCount, getValue()); for (int i = 0; i < elemCount; ++i) { container.removeFirst(); } } } QTEST_APPLESS_MAIN(tst_QList) #include "tst_bench_qlist.moc"