151c0b2f7Stbbdev /*
2*b15aabb3Stbbdev     Copyright (c) 2020-2021 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 
17*b15aabb3Stbbdev #if __INTEL_COMPILER && _MSC_VER
18*b15aabb3Stbbdev #pragma warning(disable : 2586) // decorated name length exceeded, name was truncated
19*b15aabb3Stbbdev #endif
2051c0b2f7Stbbdev 
2151c0b2f7Stbbdev #include "common/test.h"
2251c0b2f7Stbbdev 
2351c0b2f7Stbbdev #include "common/utils.h"
2451c0b2f7Stbbdev #include "common/graph_utils.h"
2551c0b2f7Stbbdev 
2649e08aacStbbdev #include "oneapi/tbb/flow_graph.h"
2749e08aacStbbdev #include "oneapi/tbb/task_arena.h"
2849e08aacStbbdev #include "oneapi/tbb/global_control.h"
2951c0b2f7Stbbdev 
3051c0b2f7Stbbdev #include "conformance_flowgraph.h"
3151c0b2f7Stbbdev 
3251c0b2f7Stbbdev //! \file conformance_queue_node.cpp
3351c0b2f7Stbbdev //! \brief Test for [flow_graph.queue_node] specification
3451c0b2f7Stbbdev 
3551c0b2f7Stbbdev /*
3651c0b2f7Stbbdev TODO: implement missing conformance tests for queue_node:
3751c0b2f7Stbbdev   - [ ] The copy constructor and copy assignment are called for the node's type template parameter.
3851c0b2f7Stbbdev   - [ ] Improve `test_forwarding()'.
3951c0b2f7Stbbdev   - [ ] Improve tests of the constructors.
4051c0b2f7Stbbdev   - [ ] Improve `test_buffering' by checking that additional `try_get()' does not receive the same
4151c0b2f7Stbbdev     value.
4251c0b2f7Stbbdev - [ ] Based on the decision about the details for `try_put()' and `try_get()' write corresponding
4351c0b2f7Stbbdev     tests.
4451c0b2f7Stbbdev */
4551c0b2f7Stbbdev 
4651c0b2f7Stbbdev template<typename T>
4751c0b2f7Stbbdev void test_inheritance(){
4849e08aacStbbdev     using namespace oneapi::tbb::flow;
4951c0b2f7Stbbdev 
5051c0b2f7Stbbdev     CHECK_MESSAGE( (std::is_base_of<graph_node, queue_node<T>>::value), "queue_node should be derived from graph_node");
5151c0b2f7Stbbdev     CHECK_MESSAGE( (std::is_base_of<receiver<T>, queue_node<T>>::value), "queue_node should be derived from receiver<T>");
5251c0b2f7Stbbdev     CHECK_MESSAGE( (std::is_base_of<sender<T>, queue_node<T>>::value), "queue_node should be derived from sender<T>");
5351c0b2f7Stbbdev }
5451c0b2f7Stbbdev 
5551c0b2f7Stbbdev void test_copies(){
5649e08aacStbbdev     using namespace oneapi::tbb::flow;
5751c0b2f7Stbbdev 
5851c0b2f7Stbbdev     graph g;
5951c0b2f7Stbbdev     queue_node<int> n(g);
6051c0b2f7Stbbdev     queue_node<int> n2(n);
6151c0b2f7Stbbdev 
6251c0b2f7Stbbdev }
6351c0b2f7Stbbdev 
6451c0b2f7Stbbdev void test_buffering(){
6549e08aacStbbdev     oneapi::tbb::flow::graph g;
6651c0b2f7Stbbdev 
6749e08aacStbbdev     oneapi::tbb::flow::queue_node<int> node(g);
6849e08aacStbbdev     oneapi::tbb::flow::limiter_node<int> rejecter(g, 0);
6951c0b2f7Stbbdev 
7049e08aacStbbdev     oneapi::tbb::flow::make_edge(node, rejecter);
7151c0b2f7Stbbdev     node.try_put(1);
7251c0b2f7Stbbdev     g.wait_for_all();
7351c0b2f7Stbbdev 
7451c0b2f7Stbbdev     int tmp = -1;
7551c0b2f7Stbbdev     CHECK_MESSAGE( (node.try_get(tmp) == true), "try_get after rejection should succeed");
7651c0b2f7Stbbdev     CHECK_MESSAGE( (tmp == 1), "try_get after rejection should set value");
7751c0b2f7Stbbdev }
7851c0b2f7Stbbdev 
7951c0b2f7Stbbdev void test_forwarding(){
8049e08aacStbbdev     oneapi::tbb::flow::graph g;
8151c0b2f7Stbbdev 
8249e08aacStbbdev     oneapi::tbb::flow::queue_node<int> node1(g);
8351c0b2f7Stbbdev     test_push_receiver<int> node2(g);
8451c0b2f7Stbbdev     test_push_receiver<int> node3(g);
8551c0b2f7Stbbdev 
8649e08aacStbbdev     oneapi::tbb::flow::make_edge(node1, node2);
8749e08aacStbbdev     oneapi::tbb::flow::make_edge(node1, node3);
8851c0b2f7Stbbdev 
8951c0b2f7Stbbdev     node1.try_put(1);
9051c0b2f7Stbbdev     g.wait_for_all();
9151c0b2f7Stbbdev 
9251c0b2f7Stbbdev     int c2 = get_count(node2), c3 = get_count(node3);
9351c0b2f7Stbbdev     CHECK_MESSAGE( (c2 != c3 ), "Only one descendant the node needs to receive");
9451c0b2f7Stbbdev     CHECK_MESSAGE( (c2 + c3 == 1 ), "All messages need to be received");
9551c0b2f7Stbbdev }
9651c0b2f7Stbbdev 
9751c0b2f7Stbbdev void test_queue_node(){
9849e08aacStbbdev     oneapi::tbb::flow::graph g;
9951c0b2f7Stbbdev 
10049e08aacStbbdev     oneapi::tbb::flow::queue_node<int> node(g);
10151c0b2f7Stbbdev     node.try_put(1);
10251c0b2f7Stbbdev     node.try_put(2);
10351c0b2f7Stbbdev     g.wait_for_all();
10451c0b2f7Stbbdev 
10551c0b2f7Stbbdev     int tmp = -1;
10651c0b2f7Stbbdev     CHECK_MESSAGE( (node.try_get(tmp) == true), "try_get should succeed");
10751c0b2f7Stbbdev     CHECK_MESSAGE( (tmp == 1), "try_get should set correct value");
10851c0b2f7Stbbdev 
10951c0b2f7Stbbdev     tmp = -1;
11051c0b2f7Stbbdev     CHECK_MESSAGE( (node.try_get(tmp) == true), "try_get should succeed");
11151c0b2f7Stbbdev     CHECK_MESSAGE( (tmp == 2), "try_get should set correct value");
11251c0b2f7Stbbdev }
11351c0b2f7Stbbdev 
11451c0b2f7Stbbdev void test_double_reserve(){
11549e08aacStbbdev     oneapi::tbb::flow::graph g;
11651c0b2f7Stbbdev 
11749e08aacStbbdev     oneapi::tbb::flow::queue_node<int> node(g);
11851c0b2f7Stbbdev 
11951c0b2f7Stbbdev     int tmp = -1;
12051c0b2f7Stbbdev     node.try_reserve(tmp);
12151c0b2f7Stbbdev     CHECK_MESSAGE((tmp == -1), "Should not be delivered");
12251c0b2f7Stbbdev     node.try_reserve(tmp);
12351c0b2f7Stbbdev     CHECK_MESSAGE((tmp == -1), "Should not be delivered");
12451c0b2f7Stbbdev 
12551c0b2f7Stbbdev     g.reset();
12651c0b2f7Stbbdev 
12751c0b2f7Stbbdev     node.try_reserve(tmp);
12851c0b2f7Stbbdev     CHECK_MESSAGE((tmp == -1), "Should not be delivered");
12951c0b2f7Stbbdev     node.try_reserve(tmp);
13051c0b2f7Stbbdev     CHECK_MESSAGE((tmp == -1), "Should not be delivered");
13151c0b2f7Stbbdev }
13251c0b2f7Stbbdev 
13351c0b2f7Stbbdev //! Test multiple reserves
13451c0b2f7Stbbdev //! \brief \ref error_guessing
13551c0b2f7Stbbdev TEST_CASE("queue_node double reserve"){
13651c0b2f7Stbbdev     test_double_reserve();
13751c0b2f7Stbbdev }
13851c0b2f7Stbbdev 
13951c0b2f7Stbbdev //! Test message logic
14051c0b2f7Stbbdev //! \brief \ref requirement
14151c0b2f7Stbbdev TEST_CASE("queue_node messages"){
14251c0b2f7Stbbdev     test_queue_node();
14351c0b2f7Stbbdev }
14451c0b2f7Stbbdev 
14551c0b2f7Stbbdev //! Test single-push
14651c0b2f7Stbbdev //! \brief \ref requirement
14751c0b2f7Stbbdev TEST_CASE("queue_node buffering"){
14851c0b2f7Stbbdev     test_forwarding();
14951c0b2f7Stbbdev }
15051c0b2f7Stbbdev 
15151c0b2f7Stbbdev //! Test buffering
15251c0b2f7Stbbdev //! \brief \ref requirement
15351c0b2f7Stbbdev TEST_CASE("queue_node buffering"){
15451c0b2f7Stbbdev     test_buffering();
15551c0b2f7Stbbdev }
15651c0b2f7Stbbdev 
15751c0b2f7Stbbdev //! Test copy constructor
15851c0b2f7Stbbdev //! \brief \ref interface
15951c0b2f7Stbbdev TEST_CASE("queue_node copy constructor"){
16051c0b2f7Stbbdev     test_copies();
16151c0b2f7Stbbdev }
16251c0b2f7Stbbdev 
16351c0b2f7Stbbdev //! Test inheritance relations
16451c0b2f7Stbbdev //! \brief \ref interface
16551c0b2f7Stbbdev TEST_CASE("queue_node superclasses"){
16651c0b2f7Stbbdev     test_inheritance<int>();
16751c0b2f7Stbbdev     test_inheritance<void*>();
16851c0b2f7Stbbdev }
16951c0b2f7Stbbdev 
170