blob: 3b7417da3fc149dbddbcebb8d392e82aa6c715c5 [file] [log] [blame]
Georgios Pinitasd8734b52017-12-22 15:27:52 +00001/*
2 * Copyright (c) 2018 ARM Limited.
3 *
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/backends/NEON/NEFunctionFactory.h"
Georgios Pinitasd8734b52017-12-22 15:27:52 +000025
26#include "arm_compute/core/utils/misc/Cast.h"
Georgios Pinitasd9eb2752018-04-03 13:44:29 +010027#include "arm_compute/graph/Graph.h"
28#include "arm_compute/graph/GraphContext.h"
29#include "arm_compute/graph/Logger.h"
30#include "arm_compute/graph/TypePrinter.h"
Georgios Pinitasda2491f2018-06-01 17:49:09 +010031#include "arm_compute/graph/backends/FunctionHelpers.h"
Georgios Pinitasd9eb2752018-04-03 13:44:29 +010032#include "arm_compute/graph/backends/Utils.h"
33#include "arm_compute/graph/nodes/Nodes.h"
Georgios Pinitasd8734b52017-12-22 15:27:52 +000034#include "arm_compute/runtime/NEON/NEFunctions.h"
35#include "support/ToolchainSupport.h"
36
37using namespace arm_compute::utils::cast;
38
39namespace arm_compute
40{
Georgios Pinitasd9eb2752018-04-03 13:44:29 +010041namespace graph
Georgios Pinitasd8734b52017-12-22 15:27:52 +000042{
43namespace backends
44{
Georgios Pinitasda2491f2018-06-01 17:49:09 +010045/** Target specific information structure used to pass information to the layer templates */
46struct NETargetInfo
Georgios Pinitasd8734b52017-12-22 15:27:52 +000047{
Georgios Pinitasda2491f2018-06-01 17:49:09 +010048 using TensorType = arm_compute::ITensor;
49 static Target TargetType;
50};
Georgios Pinitasd8734b52017-12-22 15:27:52 +000051
Georgios Pinitasda2491f2018-06-01 17:49:09 +010052Target NETargetInfo::TargetType = Target::NEON;
53
54/** Collection of CL convolution functions */
55struct NEConvolutionLayerFunctions
Georgios Pinitasd8734b52017-12-22 15:27:52 +000056{
Georgios Pinitasda2491f2018-06-01 17:49:09 +010057 using GenericConvolutionLayer = NEConvolutionLayer;
58 using GEMMConvolutionLayer = NEGEMMConvolutionLayer;
59 using DirectConvolutionLayer = NEDirectConvolutionLayer;
60 using WinogradConvolutionLayer = NEWinogradConvolutionLayer;
61};
62
63/** Collection of CL depthwise convolution functions */
64struct NEDepthwiseConvolutionLayerFunctions
65{
66 using GenericDepthwiseConvolutionLayer = NEDepthwiseConvolutionLayer;
67 using DepthwiseConvolutionLayer3x3 = NEDepthwiseConvolutionLayer3x3;
68};
69
70/** Collection of CL element-wise functions */
71struct NEEltwiseFunctions
72{
73 using Addition = NEArithmeticAddition;
74 using Subtraction = NEArithmeticSubtraction;
75 using Multiplication = NEPixelWiseMultiplication;
76};
77
78namespace detail
79{
80// Specialize functions
81template <>
82std::unique_ptr<IFunction> create_convolution_layer<NEConvolutionLayerFunctions, NETargetInfo>(ConvolutionLayerNode &node,
83 GraphContext &ctx)
84{
85 validate_node<NETargetInfo>(node, 3 /* expected inputs */, 1 /* expected outputs */);
Georgios Pinitasd8734b52017-12-22 15:27:52 +000086
87 // Extract IO and info
Georgios Pinitasda2491f2018-06-01 17:49:09 +010088 NETargetInfo::TensorType *input = get_backing_tensor<NETargetInfo>(node.input(0));
89 NETargetInfo::TensorType *weights = get_backing_tensor<NETargetInfo>(node.input(1));
90 NETargetInfo::TensorType *biases = get_backing_tensor<NETargetInfo>(node.input(2));
91 NETargetInfo::TensorType *output = get_backing_tensor<NETargetInfo>(node.output(0));
Giorgio Arenabb54e4e2018-04-05 17:20:34 +010092
93 if(is_data_type_quantized_asymmetric(input->info()->data_type()))
94 {
95 biases->info()->set_data_type(DataType::S32);
96 }
97
Georgios Pinitasd8734b52017-12-22 15:27:52 +000098 const PadStrideInfo conv_info = node.convolution_info();
99 const ConvolutionMethod conv_algorithm = node.convolution_method();
100
101 // Create and configure function (we assume that functions have been validated before creation)
102 std::shared_ptr<IMemoryManager> mm = get_memory_manager(ctx, Target::NEON);
103 std::unique_ptr<IFunction> func;
104 std::string func_name;
105 if(conv_algorithm == ConvolutionMethod::DIRECT)
106 {
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100107 std::tie(func, func_name) = create_named_memory_managed_function<NEDirectConvolutionLayer>(
108 std::string("DirectConvolutionLayer"), mm, input, weights, biases, output, conv_info);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000109 }
110 else if(conv_algorithm == ConvolutionMethod::GEMM)
111 {
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100112 std::tie(func, func_name) = create_named_memory_managed_function<NEGEMMConvolutionLayer>(
113 std::string("GEMMConvolutionLayer"), mm, input, weights, biases, output, conv_info);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000114 }
115 else if(conv_algorithm == ConvolutionMethod::WINOGRAD)
116 {
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100117 std::tie(func, func_name) = create_named_memory_managed_function<NEWinogradConvolutionLayer>(
118 std::string("WinogradConvolutionLayer"), mm, input, weights, biases, output, conv_info);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000119 }
120 else
121 {
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100122 std::tie(func, func_name) = create_named_memory_managed_function<NEConvolutionLayer>(
123 std::string("ConvolutionLayer"), mm, input, weights, biases, output, conv_info);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000124 }
125
126 // Log info
127 ARM_COMPUTE_LOG_GRAPH_INFO("Instantiated " << func_name
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100128 << " Target " << NETargetInfo::TargetType
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000129 << " Data Type: " << input->info()->data_type()
Giorgio Arenabb54e4e2018-04-05 17:20:34 +0100130 << " Input QuantInfo: " << input->info()->quantization_info()
131 << " Weights QuantInfo: " << weights->info()->quantization_info()
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000132 << " Input shape: " << input->info()->tensor_shape()
133 << " Weights shape: " << weights->info()->tensor_shape()
134 << " Output shape: " << output->info()->tensor_shape()
135 << std::endl);
136 return func;
137}
138
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100139template <>
140std::unique_ptr<IFunction> create_normalization_layer<NENormalizationLayer, NETargetInfo>(NormalizationLayerNode &node, GraphContext &ctx)
Georgios Pinitas087eaf62018-05-16 15:52:35 +0100141{
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100142 validate_node<NETargetInfo>(node, 1 /* expected inputs */, 1 /* expected outputs */);
Georgios Pinitas087eaf62018-05-16 15:52:35 +0100143
144 // Extract IO and info
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100145 NETargetInfo::TensorType *input = get_backing_tensor<NETargetInfo>(node.input(0));
146 NETargetInfo::TensorType *output = get_backing_tensor<NETargetInfo>(node.output(0));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000147 const NormalizationLayerInfo norm_info = node.normalization_info();
148 ARM_COMPUTE_ERROR_ON(input == nullptr);
149 ARM_COMPUTE_ERROR_ON(output == nullptr);
150
151 // Create and configure function
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100152 auto func = support::cpp14::make_unique<NENormalizationLayer>(get_memory_manager(ctx, NETargetInfo::TargetType));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000153 func->configure(input, output, norm_info);
154
155 // Log info
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100156 ARM_COMPUTE_LOG_GRAPH_INFO("Instantiated " << node.type()
157 << " Target " << NETargetInfo::TargetType
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000158 << " Data Type: " << input->info()->data_type()
159 << " Input shape: " << input->info()->tensor_shape()
160 << " Output shape: " << output->info()->tensor_shape()
161 << " Normalization info: " << norm_info.type()
162 << std::endl);
163
164 return std::move(func);
165}
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100166} // namespace detail
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000167
168std::unique_ptr<IFunction> NEFunctionFactory::create(INode *node, GraphContext &ctx)
169{
170 if(node == nullptr)
171 {
172 return nullptr;
173 }
174
175 NodeType type = node->type();
176 switch(type)
177 {
178 case NodeType::ActivationLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100179 return detail::create_activation_layer<NEActivationLayer, NETargetInfo>(*polymorphic_downcast<ActivationLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000180 case NodeType::BatchNormalizationLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100181 return detail::create_batch_normalization_layer<NEBatchNormalizationLayer, NETargetInfo>(*polymorphic_downcast<BatchNormalizationLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000182 case NodeType::ConvolutionLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100183 return detail::create_convolution_layer<NEConvolutionLayerFunctions, NETargetInfo>(*polymorphic_downcast<ConvolutionLayerNode *>(node), ctx);
Georgios Pinitas087eaf62018-05-16 15:52:35 +0100184 case NodeType::DeconvolutionLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100185 return detail::create_deconvolution_layer<NEDeconvolutionLayer, NETargetInfo>(*polymorphic_downcast<DeconvolutionLayerNode *>(node), ctx);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000186 case NodeType::DepthConcatenateLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100187 return detail::create_depth_concatenate_layer<NEDepthConcatenateLayer, NETargetInfo>(*polymorphic_downcast<DepthConcatenateLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000188 case NodeType::DepthwiseConvolutionLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100189 return detail::create_depthwise_convolution_layer<NEDepthwiseConvolutionLayerFunctions, NETargetInfo>(*polymorphic_downcast<DepthwiseConvolutionLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000190 case NodeType::EltwiseLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100191 return detail::create_eltwise_layer<NEEltwiseFunctions, NETargetInfo>(*polymorphic_downcast<EltwiseLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000192 case NodeType::FlattenLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100193 return detail::create_flatten_layer<NEFlattenLayer, NETargetInfo>(*polymorphic_downcast<FlattenLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000194 case NodeType::FullyConnectedLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100195 return detail::create_fully_connected_layer<NEFullyConnectedLayer, NETargetInfo>(*polymorphic_downcast<FullyConnectedLayerNode *>(node), ctx);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000196 case NodeType::NormalizationLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100197 return detail::create_normalization_layer<NENormalizationLayer, NETargetInfo>(*polymorphic_downcast<NormalizationLayerNode *>(node), ctx);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000198 case NodeType::PoolingLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100199 return detail::create_pooling_layer<NEPoolingLayer, NETargetInfo>(*polymorphic_downcast<PoolingLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000200 case NodeType::ReshapeLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100201 return detail::create_reshape_layer<NEReshapeLayer, NETargetInfo>(*polymorphic_downcast<ReshapeLayerNode *>(node));
Georgios Pinitas087eaf62018-05-16 15:52:35 +0100202 case NodeType::ResizeLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100203 return detail::create_resize_layer<NEScale, NETargetInfo>(*polymorphic_downcast<ResizeLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000204 case NodeType::SoftmaxLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100205 return detail::create_softmax_layer<NESoftmaxLayer, NETargetInfo>(*polymorphic_downcast<SoftmaxLayerNode *>(node), ctx);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000206 default:
207 return nullptr;
208 }
209}
210} // namespace backends
Georgios Pinitasd9eb2752018-04-03 13:44:29 +0100211} // namespace graph
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000212} // namespace arm_compute