151c0b2f7Stbbdev /*
2*eac94650SAlex     Copyright (c) 2005-2022 Intel Corporation
351c0b2f7Stbbdev 
451c0b2f7Stbbdev     Licensed under the Apache License, Version 2.0 (the "License");
551c0b2f7Stbbdev     you may not use this file except in compliance with the License.
651c0b2f7Stbbdev     You may obtain a copy of the License at
751c0b2f7Stbbdev 
851c0b2f7Stbbdev         http://www.apache.org/licenses/LICENSE-2.0
951c0b2f7Stbbdev 
1051c0b2f7Stbbdev     Unless required by applicable law or agreed to in writing, software
1151c0b2f7Stbbdev     distributed under the License is distributed on an "AS IS" BASIS,
1251c0b2f7Stbbdev     WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1351c0b2f7Stbbdev     See the License for the specific language governing permissions and
1451c0b2f7Stbbdev     limitations under the License.
1551c0b2f7Stbbdev */
1651c0b2f7Stbbdev 
17478de5b1Stbbdev #if _WIN32 || _WIN64
18478de5b1Stbbdev #define _CRT_SECURE_NO_WARNINGS
19478de5b1Stbbdev #endif
20478de5b1Stbbdev 
21b15aabb3Stbbdev #if _MSC_VER && !defined(__INTEL_COMPILER)
22b15aabb3Stbbdev // structure was padded due to alignment specifier
23b15aabb3Stbbdev #pragma warning( disable: 4324 )
24b15aabb3Stbbdev #endif
2551c0b2f7Stbbdev 
2651c0b2f7Stbbdev #include "common/test.h"
2751c0b2f7Stbbdev #include "common/utils.h"
2851c0b2f7Stbbdev #define __TBB_TEST_DEFINE_PRIVATE_PUBLIC 1
2951c0b2f7Stbbdev #include "common/inject_scheduler.h"
3051c0b2f7Stbbdev #define private public
3151c0b2f7Stbbdev #define protected public
3251c0b2f7Stbbdev #include "tbb/concurrent_queue.h"
3351c0b2f7Stbbdev #undef protected
3451c0b2f7Stbbdev #undef private
3551c0b2f7Stbbdev 
3651c0b2f7Stbbdev #include <limits>
3751c0b2f7Stbbdev 
3851c0b2f7Stbbdev //! \file test_concurrent_queue_whitebox.cpp
3951c0b2f7Stbbdev //! \brief Test for [internal] functionality
4051c0b2f7Stbbdev 
4151c0b2f7Stbbdev template <typename Q>
4251c0b2f7Stbbdev class FloggerBody {
4351c0b2f7Stbbdev public:
4451c0b2f7Stbbdev     FloggerBody& operator=( const FloggerBody& ) = delete;
4551c0b2f7Stbbdev 
FloggerBody(Q & queue,std::size_t el_num)4651c0b2f7Stbbdev     FloggerBody( Q& queue, std::size_t el_num )
4751c0b2f7Stbbdev         : q(queue), elem_num(el_num) {}
4851c0b2f7Stbbdev 
operator ()(const int thread_id) const4951c0b2f7Stbbdev     void operator()( const int thread_id ) const {
5051c0b2f7Stbbdev         using value_type = typename Q::value_type;
5151c0b2f7Stbbdev         value_type elem = value_type(thread_id);
5251c0b2f7Stbbdev         for (std::size_t i = 0; i < elem_num; ++i) {
5351c0b2f7Stbbdev             q.push(elem);
54*eac94650SAlex             bool res = q.try_pop(elem);
55*eac94650SAlex             CHECK_FAST(res);
5651c0b2f7Stbbdev         }
5751c0b2f7Stbbdev     }
5851c0b2f7Stbbdev 
5951c0b2f7Stbbdev private:
6051c0b2f7Stbbdev     Q& q;
6151c0b2f7Stbbdev     std::size_t elem_num;
6251c0b2f7Stbbdev }; // class FloggerBody
6351c0b2f7Stbbdev 
6451c0b2f7Stbbdev template <typename Q>
test_flogger_help(Q & q,std::size_t items_per_page)6551c0b2f7Stbbdev void test_flogger_help( Q& q, std::size_t items_per_page ) {
6651c0b2f7Stbbdev     std::size_t nq = q.my_queue_representation->n_queue;
6751c0b2f7Stbbdev     std::size_t reserved_elem_num = nq * items_per_page - 1;
6851c0b2f7Stbbdev     std::size_t hack_val = std::numeric_limits<std::size_t>::max() & ~reserved_elem_num;
6951c0b2f7Stbbdev 
7051c0b2f7Stbbdev     q.my_queue_representation->head_counter = hack_val;
7151c0b2f7Stbbdev     q.my_queue_representation->tail_counter = hack_val;
7251c0b2f7Stbbdev 
7351c0b2f7Stbbdev     std::size_t k = q.my_queue_representation->tail_counter & -(std::ptrdiff_t)nq;
7451c0b2f7Stbbdev 
7551c0b2f7Stbbdev     for (std::size_t i = 0; i < nq; ++i) {
7651c0b2f7Stbbdev         q.my_queue_representation->array[i].head_counter = k;
7751c0b2f7Stbbdev         q.my_queue_representation->array[i].tail_counter = k;
7851c0b2f7Stbbdev     }
7951c0b2f7Stbbdev 
8051c0b2f7Stbbdev     // To induce the overflow occurrence
8155f9b178SIvan Kochin     utils::NativeParallelFor(static_cast<typename Q::value_type>(utils::MaxThread), FloggerBody<Q>(q, reserved_elem_num + 20));
8251c0b2f7Stbbdev 
8351c0b2f7Stbbdev     REQUIRE_MESSAGE(q.empty(), "Failed flogger/empty test");
8451c0b2f7Stbbdev     REQUIRE_MESSAGE(q.my_queue_representation->head_counter < hack_val, "Failed wraparound test");
8551c0b2f7Stbbdev }
8651c0b2f7Stbbdev 
87*eac94650SAlex //! \brief \ref error_guessing
88*eac94650SAlex TEST_CASE("Test CQ Wrapparound") {
89*eac94650SAlex     for (int i = 0; i < 1000; ++i) {
90*eac94650SAlex         tbb::concurrent_queue<int> q;
9151c0b2f7Stbbdev         test_flogger_help(q, q.my_queue_representation->items_per_page);
9251c0b2f7Stbbdev     }
9351c0b2f7Stbbdev }
9451c0b2f7Stbbdev 
9551c0b2f7Stbbdev //! \brief \ref error_guessing
96*eac94650SAlex TEST_CASE("Test CBQ Wrapparound") {
97*eac94650SAlex     for (int i = 0; i < 1000; ++i) {
98*eac94650SAlex         tbb::concurrent_bounded_queue<int> q;
99*eac94650SAlex         test_flogger_help(q, q.my_queue_representation->items_per_page);
100*eac94650SAlex     }
10151c0b2f7Stbbdev }
102