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
20*b15aabb3Stbbdev 
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"
2851c0b2f7Stbbdev 
2949e08aacStbbdev #include "oneapi/tbb/global_control.h"
3051c0b2f7Stbbdev #include "conformance_flowgraph.h"
3151c0b2f7Stbbdev 
3251c0b2f7Stbbdev //! \file conformance_continue_node.cpp
3351c0b2f7Stbbdev //! \brief Test for [flow_graph.continue_node] specification
3451c0b2f7Stbbdev 
3551c0b2f7Stbbdev /*
3651c0b2f7Stbbdev TODO: implement missing conformance tests for continue_node:
3751c0b2f7Stbbdev   - [ ] For `test_forwarding' check that the value passed is the actual one received.
3851c0b2f7Stbbdev   - [ ] The `copy_body' function copies altered body (e.g. after its successful invocation).
3951c0b2f7Stbbdev   - [ ] Improve CTAD test.
4051c0b2f7Stbbdev   - [ ] Improve constructors test, including addition of calls to constructors with
4151c0b2f7Stbbdev     `number_of_predecessors' parameter.
4251c0b2f7Stbbdev   - [ ] Explicit test for copy constructor of the node.
4351c0b2f7Stbbdev   - [ ] Rewrite test_priority.
4451c0b2f7Stbbdev   - [ ] Check `Output' type indeed copy-constructed and copy-assigned while working with the node.
4551c0b2f7Stbbdev   - [ ] Explicit test for correct working of `number_of_predecessors' constructor parameter,
4651c0b2f7Stbbdev     including taking it into account when making and removing edges.
4751c0b2f7Stbbdev   - [ ] Add testing of `try_put' statement. In particular that it does not wait for the execution of
4851c0b2f7Stbbdev     the body to complete.
4951c0b2f7Stbbdev */
5051c0b2f7Stbbdev 
5151c0b2f7Stbbdev void test_cont_body(){
5249e08aacStbbdev     oneapi::tbb::flow::graph g;
5351c0b2f7Stbbdev     inc_functor<int> cf;
5451c0b2f7Stbbdev     cf.execute_count = 0;
5551c0b2f7Stbbdev 
5649e08aacStbbdev     oneapi::tbb::flow::continue_node<int> node1(g, cf);
5751c0b2f7Stbbdev 
5851c0b2f7Stbbdev     const size_t n = 10;
5951c0b2f7Stbbdev     for(size_t i = 0; i < n; ++i) {
6049e08aacStbbdev         CHECK_MESSAGE((node1.try_put(oneapi::tbb::flow::continue_msg()) == true),
6151c0b2f7Stbbdev                       "continue_node::try_put() should never reject a message.");
6251c0b2f7Stbbdev     }
6351c0b2f7Stbbdev     g.wait_for_all();
6451c0b2f7Stbbdev 
6551c0b2f7Stbbdev     CHECK_MESSAGE( (cf.execute_count == n), "Body of the first node needs to be executed N times");
6651c0b2f7Stbbdev }
6751c0b2f7Stbbdev 
6851c0b2f7Stbbdev template<typename O>
6951c0b2f7Stbbdev void test_inheritance(){
7049e08aacStbbdev     using namespace oneapi::tbb::flow;
7151c0b2f7Stbbdev 
7251c0b2f7Stbbdev     CHECK_MESSAGE( (std::is_base_of<graph_node, continue_node<O>>::value), "continue_node should be derived from graph_node");
7351c0b2f7Stbbdev     CHECK_MESSAGE( (std::is_base_of<receiver<continue_msg>, continue_node<O>>::value), "continue_node should be derived from receiver<Input>");
7451c0b2f7Stbbdev     CHECK_MESSAGE( (std::is_base_of<sender<O>, continue_node<O>>::value), "continue_node should be derived from sender<Output>");
7551c0b2f7Stbbdev }
7651c0b2f7Stbbdev 
7751c0b2f7Stbbdev #if __TBB_CPP17_DEDUCTION_GUIDES_PRESENT
7851c0b2f7Stbbdev void test_deduction_guides(){
7949e08aacStbbdev     oneapi::tbb::flow::graph g;
8051c0b2f7Stbbdev     inc_functor<int> fun;
8149e08aacStbbdev     oneapi::tbb::flow::continue_node node1(g, fun);
8251c0b2f7Stbbdev }
8351c0b2f7Stbbdev #endif
8451c0b2f7Stbbdev 
8551c0b2f7Stbbdev void test_forwarding(){
8649e08aacStbbdev     oneapi::tbb::flow::graph g;
8751c0b2f7Stbbdev     inc_functor<int> fun;
8851c0b2f7Stbbdev     fun.execute_count = 0;
8951c0b2f7Stbbdev 
9049e08aacStbbdev     oneapi::tbb::flow::continue_node<int> node1(g, fun);
9151c0b2f7Stbbdev     test_push_receiver<int> node2(g);
9251c0b2f7Stbbdev     test_push_receiver<int> node3(g);
9351c0b2f7Stbbdev 
9449e08aacStbbdev     oneapi::tbb::flow::make_edge(node1, node2);
9549e08aacStbbdev     oneapi::tbb::flow::make_edge(node1, node3);
9651c0b2f7Stbbdev 
9749e08aacStbbdev     node1.try_put(oneapi::tbb::flow::continue_msg());
9851c0b2f7Stbbdev     g.wait_for_all();
9951c0b2f7Stbbdev 
10051c0b2f7Stbbdev     CHECK_MESSAGE( (get_count(node2) == 1), "Descendant of the node must receive one message.");
10151c0b2f7Stbbdev     CHECK_MESSAGE( (get_count(node3) == 1), "Descendant of the node must receive one message.");
10251c0b2f7Stbbdev }
10351c0b2f7Stbbdev 
10451c0b2f7Stbbdev void test_buffering(){
10549e08aacStbbdev     oneapi::tbb::flow::graph g;
10651c0b2f7Stbbdev     inc_functor<int> fun;
10751c0b2f7Stbbdev 
10849e08aacStbbdev     oneapi::tbb::flow::continue_node<int> node(g, fun);
10949e08aacStbbdev     oneapi::tbb::flow::limiter_node<int> rejecter(g, 0);
11051c0b2f7Stbbdev 
11149e08aacStbbdev     oneapi::tbb::flow::make_edge(node, rejecter);
11249e08aacStbbdev     node.try_put(oneapi::tbb::flow::continue_msg());
11351c0b2f7Stbbdev 
11451c0b2f7Stbbdev     int tmp = -1;
11551c0b2f7Stbbdev     CHECK_MESSAGE( (node.try_get(tmp) == false), "try_get after rejection should not succeed");
11651c0b2f7Stbbdev     CHECK_MESSAGE( (tmp == -1), "try_get after rejection should not alter passed value");
11751c0b2f7Stbbdev     g.wait_for_all();
11851c0b2f7Stbbdev }
11951c0b2f7Stbbdev 
12051c0b2f7Stbbdev void test_policy_ctors(){
12149e08aacStbbdev     using namespace oneapi::tbb::flow;
12251c0b2f7Stbbdev     graph g;
12351c0b2f7Stbbdev 
12451c0b2f7Stbbdev     inc_functor<int> fun;
12551c0b2f7Stbbdev 
12651c0b2f7Stbbdev     continue_node<int, lightweight> lw_node(g, fun);
12751c0b2f7Stbbdev }
12851c0b2f7Stbbdev 
12951c0b2f7Stbbdev void test_ctors(){
13049e08aacStbbdev     using namespace oneapi::tbb::flow;
13151c0b2f7Stbbdev     graph g;
13251c0b2f7Stbbdev 
13351c0b2f7Stbbdev     inc_functor<int> fun;
13451c0b2f7Stbbdev 
13549e08aacStbbdev     continue_node<int> proto1(g, 2, fun, oneapi::tbb::flow::node_priority_t(1));
13651c0b2f7Stbbdev }
13751c0b2f7Stbbdev 
13851c0b2f7Stbbdev template<typename O>
13951c0b2f7Stbbdev struct CopyCounterBody{
14051c0b2f7Stbbdev     size_t copy_count;
14151c0b2f7Stbbdev 
14251c0b2f7Stbbdev     CopyCounterBody():
14351c0b2f7Stbbdev         copy_count(0) {}
14451c0b2f7Stbbdev 
14551c0b2f7Stbbdev     CopyCounterBody(const CopyCounterBody<O>& other):
14651c0b2f7Stbbdev         copy_count(other.copy_count + 1) {}
14751c0b2f7Stbbdev 
14851c0b2f7Stbbdev     CopyCounterBody& operator=(const CopyCounterBody<O>& other){
14951c0b2f7Stbbdev         copy_count = other.copy_count + 1;
15051c0b2f7Stbbdev         return *this;
15151c0b2f7Stbbdev     }
15251c0b2f7Stbbdev 
15349e08aacStbbdev     O operator()(oneapi::tbb::flow::continue_msg){
15451c0b2f7Stbbdev         return 1;
15551c0b2f7Stbbdev     }
15651c0b2f7Stbbdev };
15751c0b2f7Stbbdev 
15851c0b2f7Stbbdev void test_copies(){
15949e08aacStbbdev     using namespace oneapi::tbb::flow;
16051c0b2f7Stbbdev 
16151c0b2f7Stbbdev     CopyCounterBody<int> b;
16251c0b2f7Stbbdev 
16351c0b2f7Stbbdev     graph g;
16451c0b2f7Stbbdev     continue_node<int> fn(g, b);
16551c0b2f7Stbbdev 
16651c0b2f7Stbbdev     CopyCounterBody<int> b2 = copy_body<CopyCounterBody<int>,
16751c0b2f7Stbbdev                                              continue_node<int>>(fn);
16851c0b2f7Stbbdev 
16951c0b2f7Stbbdev     CHECK_MESSAGE( (b.copy_count + 2 <= b2.copy_count), "copy_body and constructor should copy bodies");
17051c0b2f7Stbbdev }
17151c0b2f7Stbbdev 
17251c0b2f7Stbbdev 
17351c0b2f7Stbbdev void test_priority(){
17451c0b2f7Stbbdev     size_t concurrency_limit = 1;
17549e08aacStbbdev     oneapi::tbb::global_control control(oneapi::tbb::global_control::max_allowed_parallelism, concurrency_limit);
17651c0b2f7Stbbdev 
17749e08aacStbbdev     oneapi::tbb::flow::graph g;
17851c0b2f7Stbbdev 
17949e08aacStbbdev     oneapi::tbb::flow::continue_node<oneapi::tbb::flow::continue_msg> source(g,
18049e08aacStbbdev                                                              [](oneapi::tbb::flow::continue_msg){ return oneapi::tbb::flow::continue_msg();});
18149e08aacStbbdev     source.try_put(oneapi::tbb::flow::continue_msg());
18251c0b2f7Stbbdev 
18351c0b2f7Stbbdev     first_functor<int>::first_id = -1;
18451c0b2f7Stbbdev     first_functor<int> low_functor(1);
18551c0b2f7Stbbdev     first_functor<int> high_functor(2);
18651c0b2f7Stbbdev 
18749e08aacStbbdev     oneapi::tbb::flow::continue_node<int, int> high(g, high_functor, oneapi::tbb::flow::node_priority_t(1));
18849e08aacStbbdev     oneapi::tbb::flow::continue_node<int, int> low(g, low_functor);
18951c0b2f7Stbbdev 
19051c0b2f7Stbbdev     make_edge(source, low);
19151c0b2f7Stbbdev     make_edge(source, high);
19251c0b2f7Stbbdev 
19351c0b2f7Stbbdev     g.wait_for_all();
19451c0b2f7Stbbdev 
19551c0b2f7Stbbdev     CHECK_MESSAGE( (first_functor<int>::first_id == 2), "High priority node should execute first");
19651c0b2f7Stbbdev }
19751c0b2f7Stbbdev 
19851c0b2f7Stbbdev //! Test node costructors
19951c0b2f7Stbbdev //! \brief \ref requirement
20051c0b2f7Stbbdev TEST_CASE("continue_node constructors"){
20151c0b2f7Stbbdev     test_ctors();
20251c0b2f7Stbbdev }
20351c0b2f7Stbbdev 
20451c0b2f7Stbbdev //! Test priorities work in single-threaded configuration
20551c0b2f7Stbbdev //! \brief \ref requirement
20651c0b2f7Stbbdev TEST_CASE("continue_node priority support"){
20751c0b2f7Stbbdev     test_priority();
20851c0b2f7Stbbdev }
20951c0b2f7Stbbdev 
21051c0b2f7Stbbdev //! Test body copying and copy_body logic
21151c0b2f7Stbbdev //! \brief \ref interface
21251c0b2f7Stbbdev TEST_CASE("continue_node and body copying"){
21351c0b2f7Stbbdev     test_copies();
21451c0b2f7Stbbdev }
21551c0b2f7Stbbdev 
21651c0b2f7Stbbdev //! Test constructors
21751c0b2f7Stbbdev //! \brief \ref interface
21851c0b2f7Stbbdev TEST_CASE("continue_node constructors"){
21951c0b2f7Stbbdev     test_policy_ctors();
22051c0b2f7Stbbdev }
22151c0b2f7Stbbdev 
22251c0b2f7Stbbdev //! Test continue_node buffering
22351c0b2f7Stbbdev //! \brief \ref requirement
22451c0b2f7Stbbdev TEST_CASE("continue_node buffering"){
22551c0b2f7Stbbdev     test_buffering();
22651c0b2f7Stbbdev }
22751c0b2f7Stbbdev 
22851c0b2f7Stbbdev //! Test function_node broadcasting
22951c0b2f7Stbbdev //! \brief \ref requirement
23051c0b2f7Stbbdev TEST_CASE("continue_node broadcast"){
23151c0b2f7Stbbdev     test_forwarding();
23251c0b2f7Stbbdev }
23351c0b2f7Stbbdev 
23451c0b2f7Stbbdev //! Test deduction guides
23551c0b2f7Stbbdev //! \brief \ref interface \ref requirement
23651c0b2f7Stbbdev TEST_CASE("Deduction guides"){
23751c0b2f7Stbbdev #if __TBB_CPP17_DEDUCTION_GUIDES_PRESENT
23851c0b2f7Stbbdev     test_deduction_guides();
23951c0b2f7Stbbdev #endif
24051c0b2f7Stbbdev }
24151c0b2f7Stbbdev 
24251c0b2f7Stbbdev //! Test inheritance relations
24351c0b2f7Stbbdev //! \brief \ref interface
24451c0b2f7Stbbdev TEST_CASE("continue_node superclasses"){
24551c0b2f7Stbbdev     test_inheritance<int>();
24651c0b2f7Stbbdev     test_inheritance<void*>();
24751c0b2f7Stbbdev }
24851c0b2f7Stbbdev 
24951c0b2f7Stbbdev //! Test body execution
25051c0b2f7Stbbdev //! \brief \ref interface \ref requirement
25151c0b2f7Stbbdev TEST_CASE("continue body") {
25251c0b2f7Stbbdev     test_cont_body();
25351c0b2f7Stbbdev }
254