151c0b2f7Stbbdev /* 251c0b2f7Stbbdev Copyright (c) 2020 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 1751c0b2f7Stbbdev 1851c0b2f7Stbbdev #include "common/test.h" 1951c0b2f7Stbbdev 2051c0b2f7Stbbdev #include "common/utils.h" 2151c0b2f7Stbbdev #include "common/graph_utils.h" 2251c0b2f7Stbbdev 23*49e08aacStbbdev #include "oneapi/tbb/flow_graph.h" 24*49e08aacStbbdev #include "oneapi/tbb/task_arena.h" 25*49e08aacStbbdev #include "oneapi/tbb/global_control.h" 2651c0b2f7Stbbdev 2751c0b2f7Stbbdev #include "conformance_flowgraph.h" 2851c0b2f7Stbbdev 2951c0b2f7Stbbdev //! \file conformance_queue_node.cpp 3051c0b2f7Stbbdev //! \brief Test for [flow_graph.queue_node] specification 3151c0b2f7Stbbdev 3251c0b2f7Stbbdev /* 3351c0b2f7Stbbdev TODO: implement missing conformance tests for queue_node: 3451c0b2f7Stbbdev - [ ] The copy constructor and copy assignment are called for the node's type template parameter. 3551c0b2f7Stbbdev - [ ] Improve `test_forwarding()'. 3651c0b2f7Stbbdev - [ ] Improve tests of the constructors. 3751c0b2f7Stbbdev - [ ] Improve `test_buffering' by checking that additional `try_get()' does not receive the same 3851c0b2f7Stbbdev value. 3951c0b2f7Stbbdev - [ ] Based on the decision about the details for `try_put()' and `try_get()' write corresponding 4051c0b2f7Stbbdev tests. 4151c0b2f7Stbbdev */ 4251c0b2f7Stbbdev 4351c0b2f7Stbbdev template<typename T> 4451c0b2f7Stbbdev void test_inheritance(){ 45*49e08aacStbbdev using namespace oneapi::tbb::flow; 4651c0b2f7Stbbdev 4751c0b2f7Stbbdev CHECK_MESSAGE( (std::is_base_of<graph_node, queue_node<T>>::value), "queue_node should be derived from graph_node"); 4851c0b2f7Stbbdev CHECK_MESSAGE( (std::is_base_of<receiver<T>, queue_node<T>>::value), "queue_node should be derived from receiver<T>"); 4951c0b2f7Stbbdev CHECK_MESSAGE( (std::is_base_of<sender<T>, queue_node<T>>::value), "queue_node should be derived from sender<T>"); 5051c0b2f7Stbbdev } 5151c0b2f7Stbbdev 5251c0b2f7Stbbdev void test_copies(){ 53*49e08aacStbbdev using namespace oneapi::tbb::flow; 5451c0b2f7Stbbdev 5551c0b2f7Stbbdev graph g; 5651c0b2f7Stbbdev queue_node<int> n(g); 5751c0b2f7Stbbdev queue_node<int> n2(n); 5851c0b2f7Stbbdev 5951c0b2f7Stbbdev } 6051c0b2f7Stbbdev 6151c0b2f7Stbbdev void test_buffering(){ 62*49e08aacStbbdev oneapi::tbb::flow::graph g; 6351c0b2f7Stbbdev 64*49e08aacStbbdev oneapi::tbb::flow::queue_node<int> node(g); 65*49e08aacStbbdev oneapi::tbb::flow::limiter_node<int> rejecter(g, 0); 6651c0b2f7Stbbdev 67*49e08aacStbbdev oneapi::tbb::flow::make_edge(node, rejecter); 6851c0b2f7Stbbdev node.try_put(1); 6951c0b2f7Stbbdev g.wait_for_all(); 7051c0b2f7Stbbdev 7151c0b2f7Stbbdev int tmp = -1; 7251c0b2f7Stbbdev CHECK_MESSAGE( (node.try_get(tmp) == true), "try_get after rejection should succeed"); 7351c0b2f7Stbbdev CHECK_MESSAGE( (tmp == 1), "try_get after rejection should set value"); 7451c0b2f7Stbbdev } 7551c0b2f7Stbbdev 7651c0b2f7Stbbdev void test_forwarding(){ 77*49e08aacStbbdev oneapi::tbb::flow::graph g; 7851c0b2f7Stbbdev 79*49e08aacStbbdev oneapi::tbb::flow::queue_node<int> node1(g); 8051c0b2f7Stbbdev test_push_receiver<int> node2(g); 8151c0b2f7Stbbdev test_push_receiver<int> node3(g); 8251c0b2f7Stbbdev 83*49e08aacStbbdev oneapi::tbb::flow::make_edge(node1, node2); 84*49e08aacStbbdev oneapi::tbb::flow::make_edge(node1, node3); 8551c0b2f7Stbbdev 8651c0b2f7Stbbdev node1.try_put(1); 8751c0b2f7Stbbdev g.wait_for_all(); 8851c0b2f7Stbbdev 8951c0b2f7Stbbdev int c2 = get_count(node2), c3 = get_count(node3); 9051c0b2f7Stbbdev CHECK_MESSAGE( (c2 != c3 ), "Only one descendant the node needs to receive"); 9151c0b2f7Stbbdev CHECK_MESSAGE( (c2 + c3 == 1 ), "All messages need to be received"); 9251c0b2f7Stbbdev } 9351c0b2f7Stbbdev 9451c0b2f7Stbbdev void test_queue_node(){ 95*49e08aacStbbdev oneapi::tbb::flow::graph g; 9651c0b2f7Stbbdev 97*49e08aacStbbdev oneapi::tbb::flow::queue_node<int> node(g); 9851c0b2f7Stbbdev node.try_put(1); 9951c0b2f7Stbbdev node.try_put(2); 10051c0b2f7Stbbdev g.wait_for_all(); 10151c0b2f7Stbbdev 10251c0b2f7Stbbdev int tmp = -1; 10351c0b2f7Stbbdev CHECK_MESSAGE( (node.try_get(tmp) == true), "try_get should succeed"); 10451c0b2f7Stbbdev CHECK_MESSAGE( (tmp == 1), "try_get should set correct value"); 10551c0b2f7Stbbdev 10651c0b2f7Stbbdev tmp = -1; 10751c0b2f7Stbbdev CHECK_MESSAGE( (node.try_get(tmp) == true), "try_get should succeed"); 10851c0b2f7Stbbdev CHECK_MESSAGE( (tmp == 2), "try_get should set correct value"); 10951c0b2f7Stbbdev } 11051c0b2f7Stbbdev 11151c0b2f7Stbbdev void test_double_reserve(){ 112*49e08aacStbbdev oneapi::tbb::flow::graph g; 11351c0b2f7Stbbdev 114*49e08aacStbbdev oneapi::tbb::flow::queue_node<int> node(g); 11551c0b2f7Stbbdev 11651c0b2f7Stbbdev int tmp = -1; 11751c0b2f7Stbbdev node.try_reserve(tmp); 11851c0b2f7Stbbdev CHECK_MESSAGE((tmp == -1), "Should not be delivered"); 11951c0b2f7Stbbdev node.try_reserve(tmp); 12051c0b2f7Stbbdev CHECK_MESSAGE((tmp == -1), "Should not be delivered"); 12151c0b2f7Stbbdev 12251c0b2f7Stbbdev g.reset(); 12351c0b2f7Stbbdev 12451c0b2f7Stbbdev node.try_reserve(tmp); 12551c0b2f7Stbbdev CHECK_MESSAGE((tmp == -1), "Should not be delivered"); 12651c0b2f7Stbbdev node.try_reserve(tmp); 12751c0b2f7Stbbdev CHECK_MESSAGE((tmp == -1), "Should not be delivered"); 12851c0b2f7Stbbdev } 12951c0b2f7Stbbdev 13051c0b2f7Stbbdev //! Test multiple reserves 13151c0b2f7Stbbdev //! \brief \ref error_guessing 13251c0b2f7Stbbdev TEST_CASE("queue_node double reserve"){ 13351c0b2f7Stbbdev test_double_reserve(); 13451c0b2f7Stbbdev } 13551c0b2f7Stbbdev 13651c0b2f7Stbbdev //! Test message logic 13751c0b2f7Stbbdev //! \brief \ref requirement 13851c0b2f7Stbbdev TEST_CASE("queue_node messages"){ 13951c0b2f7Stbbdev test_queue_node(); 14051c0b2f7Stbbdev } 14151c0b2f7Stbbdev 14251c0b2f7Stbbdev //! Test single-push 14351c0b2f7Stbbdev //! \brief \ref requirement 14451c0b2f7Stbbdev TEST_CASE("queue_node buffering"){ 14551c0b2f7Stbbdev test_forwarding(); 14651c0b2f7Stbbdev } 14751c0b2f7Stbbdev 14851c0b2f7Stbbdev //! Test buffering 14951c0b2f7Stbbdev //! \brief \ref requirement 15051c0b2f7Stbbdev TEST_CASE("queue_node buffering"){ 15151c0b2f7Stbbdev test_buffering(); 15251c0b2f7Stbbdev } 15351c0b2f7Stbbdev 15451c0b2f7Stbbdev //! Test copy constructor 15551c0b2f7Stbbdev //! \brief \ref interface 15651c0b2f7Stbbdev TEST_CASE("queue_node copy constructor"){ 15751c0b2f7Stbbdev test_copies(); 15851c0b2f7Stbbdev } 15951c0b2f7Stbbdev 16051c0b2f7Stbbdev //! Test inheritance relations 16151c0b2f7Stbbdev //! \brief \ref interface 16251c0b2f7Stbbdev TEST_CASE("queue_node superclasses"){ 16351c0b2f7Stbbdev test_inheritance<int>(); 16451c0b2f7Stbbdev test_inheritance<void*>(); 16551c0b2f7Stbbdev } 16651c0b2f7Stbbdev 167