blob: 45b608c70a82970cbe3a176a15227087d0cf8512 [file] [log] [blame]
Georgios Pinitasd8734b52017-12-22 15:27:52 +00001/*
Michalis Spyrou402740d2021-04-20 11:26:21 +01002 * Copyright (c) 2018-2021 Arm Limited.
Georgios Pinitasd8734b52017-12-22 15:27:52 +00003 *
4 * SPDX-License-Identifier: MIT
5 *
6 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 * of this software and associated documentation files (the "Software"), to
8 * deal in the Software without restriction, including without limitation the
9 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10 * sell copies of the Software, and to permit persons to whom the Software is
11 * furnished to do so, subject to the following conditions:
12 *
13 * The above copyright notice and this permission notice shall be included in all
14 * copies or substantial portions of the Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
22 * SOFTWARE.
23 */
Georgios Pinitasd9eb2752018-04-03 13:44:29 +010024#include "arm_compute/graph/GraphManager.h"
Georgios Pinitasd8734b52017-12-22 15:27:52 +000025
Georgios Pinitasd9eb2752018-04-03 13:44:29 +010026#include "arm_compute/graph/Graph.h"
27#include "arm_compute/graph/GraphContext.h"
28#include "arm_compute/graph/Logger.h"
29#include "arm_compute/graph/PassManager.h"
Georgios Pinitas12be7ab2018-07-03 12:06:23 +010030#include "arm_compute/graph/TypePrinter.h"
Georgios Pinitasd9eb2752018-04-03 13:44:29 +010031#include "arm_compute/graph/Utils.h"
Georgios Pinitas2b147ee2021-07-08 18:14:45 +010032#include "arm_compute/graph/algorithms/TopologicalSort.h"
Georgios Pinitas3d1489d2018-05-03 20:47:16 +010033#include "arm_compute/graph/detail/CrossLayerMemoryManagerHelpers.h"
Georgios Pinitasd9eb2752018-04-03 13:44:29 +010034#include "arm_compute/graph/detail/ExecutionHelpers.h"
Georgios Pinitasd8734b52017-12-22 15:27:52 +000035
Georgios Pinitas2b147ee2021-07-08 18:14:45 +010036#include "src/common/utils/Log.h"
Georgios Pinitas2a2db592018-08-15 12:14:46 +010037
Georgios Pinitasd8734b52017-12-22 15:27:52 +000038namespace arm_compute
39{
Georgios Pinitasd9eb2752018-04-03 13:44:29 +010040namespace graph
Georgios Pinitasd8734b52017-12-22 15:27:52 +000041{
42GraphManager::GraphManager()
43 : _workloads()
44{
Georgios Pinitasd8734b52017-12-22 15:27:52 +000045}
46
47void GraphManager::finalize_graph(Graph &graph, GraphContext &ctx, PassManager &pm, Target target)
48{
Georgios Pinitas2b147ee2021-07-08 18:14:45 +010049 ARM_COMPUTE_LOG_INFO_WITH_FUNCNAME_ACL("Initiate graph configuration!");
50
Georgios Pinitasd8734b52017-12-22 15:27:52 +000051 // Check if graph has been registered
Georgios Pinitasae700722018-04-27 18:10:01 +010052 if(_workloads.find(graph.id()) != std::end(_workloads))
53 {
54 ARM_COMPUTE_ERROR("Graph is already registered!");
55 }
Georgios Pinitasd8734b52017-12-22 15:27:52 +000056
Georgios Pinitasf4261ad2019-12-02 11:58:19 +000057 // Apply IR mutating passes
58 pm.run_type(graph, IGraphMutator::MutationType::IR);
59
Georgios Pinitasd8734b52017-12-22 15:27:52 +000060 // Force target to all graph construct
Georgios Pinitas12be7ab2018-07-03 12:06:23 +010061 Target forced_target = target;
Michalis Spyrou402740d2021-04-20 11:26:21 +010062
63 // In case CLVK is selected, use the CL backend and
64 // update config
65 if(target == Target::CLVK)
66 {
67 forced_target = Target::CL;
68 GraphConfig config = ctx.config();
69 config.backend_type = CLBackendType::Clvk;
70
71 ctx.set_config(config);
72 }
73
Georgios Pinitas12be7ab2018-07-03 12:06:23 +010074 if(!is_target_supported(target))
75 {
76 forced_target = get_default_target();
77 ARM_COMPUTE_LOG_GRAPH_INFO("Switching target from " << target << " to " << forced_target << std::endl);
78 }
Georgios Pinitasd8734b52017-12-22 15:27:52 +000079 force_target_to_graph(graph, forced_target);
80
Georgios Pinitas7097e3c2019-02-20 18:11:42 +000081 // Setup backend context
Georgios Pinitas7097e3c2019-02-20 18:11:42 +000082 setup_requested_backend_context(ctx, forced_target);
83
Georgios Pinitasd8734b52017-12-22 15:27:52 +000084 // Configure all tensors
85 detail::configure_all_tensors(graph);
86
Georgios Pinitasf4261ad2019-12-02 11:58:19 +000087 // Apply backend mutating passes
88 pm.run_type(graph, IGraphMutator::MutationType::Backend);
Georgios Pinitasd8734b52017-12-22 15:27:52 +000089
Georgios Pinitasd8734b52017-12-22 15:27:52 +000090 // Perform topological sort
Georgios Pinitas2a2db592018-08-15 12:14:46 +010091 std::vector<NodeID> topological_sorted_nodes = dfs(graph);
Georgios Pinitasd8734b52017-12-22 15:27:52 +000092
Georgios Pinitas28705162018-03-21 20:10:53 +000093 // Validate all nodes
94 detail::validate_all_nodes(graph);
95
Georgios Pinitasd8734b52017-12-22 15:27:52 +000096 // Configure all nodes
Georgios Pinitas2a2db592018-08-15 12:14:46 +010097 auto workload = detail::configure_all_nodes(graph, ctx, topological_sorted_nodes);
Georgios Pinitasd8734b52017-12-22 15:27:52 +000098 ARM_COMPUTE_ERROR_ON_MSG(workload.tasks.empty(), "Could not configure all nodes!");
99
Georgios Pinitas3d1489d2018-05-03 20:47:16 +0100100 // Allocate const tensors and call accessors
101 detail::allocate_const_tensors(graph);
102 detail::call_all_const_node_accessors(graph);
103
Georgios Pinitas72219332018-06-05 14:56:06 +0100104 // Prepare graph
105 detail::prepare_all_tasks(workload);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000106
Georgios Pinitas3d1489d2018-05-03 20:47:16 +0100107 // Setup tensor memory (Allocate all tensors or setup transition manager)
108 if(ctx.config().use_transition_memory_manager)
109 {
110 detail::configure_transition_manager(graph, ctx, workload);
Georgios Pinitase0437672018-05-02 14:07:55 +0100111 }
112 else
113 {
Georgios Pinitase0437672018-05-02 14:07:55 +0100114 detail::allocate_all_tensors(graph);
Georgios Pinitas3d1489d2018-05-03 20:47:16 +0100115 }
Georgios Pinitas1562be32018-03-08 19:09:19 +0000116
Georgios Pinitas3d1489d2018-05-03 20:47:16 +0100117 // Finalize Graph context
118 ctx.finalize();
Georgios Pinitase0437672018-05-02 14:07:55 +0100119
Georgios Pinitas3d1489d2018-05-03 20:47:16 +0100120 // Register graph
121 _workloads.insert(std::make_pair(graph.id(), std::move(workload)));
Georgios Pinitas54c92fd2018-11-05 12:16:15 +0000122 ARM_COMPUTE_LOG_GRAPH_VERBOSE("Created workload for graph with ID : " << graph.id() << std::endl);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000123}
124
125void GraphManager::execute_graph(Graph &graph)
126{
Georgios Pinitas2b147ee2021-07-08 18:14:45 +0100127 ARM_COMPUTE_LOG_INFO_WITH_FUNCNAME_ACL("Initiate graph execution!");
128
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000129 // Check if graph is finalized
130 auto it = _workloads.find(graph.id());
131 ARM_COMPUTE_ERROR_ON_MSG(it == std::end(_workloads), "Graph is not registered!");
132
Georgios Pinitas12be7ab2018-07-03 12:06:23 +0100133 while(true)
134 {
135 // Call input accessors
136 if(!detail::call_all_input_node_accessors(it->second))
137 {
138 return;
139 }
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000140
Georgios Pinitas12be7ab2018-07-03 12:06:23 +0100141 // Run graph
142 detail::call_all_tasks(it->second);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000143
Georgios Pinitas12be7ab2018-07-03 12:06:23 +0100144 // Call output accessors
145 if(!detail::call_all_output_node_accessors(it->second))
146 {
147 return;
148 }
149 }
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000150}
151
152void GraphManager::invalidate_graph(Graph &graph)
153{
154 auto it = _workloads.find(graph.id());
155 ARM_COMPUTE_ERROR_ON_MSG(it == std::end(_workloads), "Graph is not registered!");
156
157 _workloads.erase(it);
158}
Georgios Pinitasd9eb2752018-04-03 13:44:29 +0100159} // namespace graph
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000160} // namespace arm_compute