blob: dc987dd86e68b9f4508d9a44f3dd297b5286c9ee [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"
Isabella Gottardi7234ed82018-11-27 08:51:10 +000034#include "arm_compute/runtime/CPP/CPPFunctions.h"
Georgios Pinitasd8734b52017-12-22 15:27:52 +000035#include "arm_compute/runtime/NEON/NEFunctions.h"
36#include "support/ToolchainSupport.h"
37
38using namespace arm_compute::utils::cast;
39
40namespace arm_compute
41{
Georgios Pinitasd9eb2752018-04-03 13:44:29 +010042namespace graph
Georgios Pinitasd8734b52017-12-22 15:27:52 +000043{
44namespace backends
45{
Georgios Pinitasda2491f2018-06-01 17:49:09 +010046/** Target specific information structure used to pass information to the layer templates */
47struct NETargetInfo
Georgios Pinitasd8734b52017-12-22 15:27:52 +000048{
Georgios Pinitasda2491f2018-06-01 17:49:09 +010049 using TensorType = arm_compute::ITensor;
50 static Target TargetType;
51};
Georgios Pinitasd8734b52017-12-22 15:27:52 +000052
Georgios Pinitasda2491f2018-06-01 17:49:09 +010053Target NETargetInfo::TargetType = Target::NEON;
54
55/** Collection of CL convolution functions */
56struct NEConvolutionLayerFunctions
Georgios Pinitasd8734b52017-12-22 15:27:52 +000057{
Georgios Pinitasda2491f2018-06-01 17:49:09 +010058 using GenericConvolutionLayer = NEConvolutionLayer;
59 using GEMMConvolutionLayer = NEGEMMConvolutionLayer;
60 using DirectConvolutionLayer = NEDirectConvolutionLayer;
61 using WinogradConvolutionLayer = NEWinogradConvolutionLayer;
62};
63
64/** Collection of CL depthwise convolution functions */
65struct NEDepthwiseConvolutionLayerFunctions
66{
67 using GenericDepthwiseConvolutionLayer = NEDepthwiseConvolutionLayer;
68 using DepthwiseConvolutionLayer3x3 = NEDepthwiseConvolutionLayer3x3;
69};
70
71/** Collection of CL element-wise functions */
72struct NEEltwiseFunctions
73{
74 using Addition = NEArithmeticAddition;
75 using Subtraction = NEArithmeticSubtraction;
76 using Multiplication = NEPixelWiseMultiplication;
77};
78
79namespace detail
80{
Isabella Gottardi7234ed82018-11-27 08:51:10 +000081// Specialized functions
Georgios Pinitasda2491f2018-06-01 17:49:09 +010082template <>
83std::unique_ptr<IFunction> create_convolution_layer<NEConvolutionLayerFunctions, NETargetInfo>(ConvolutionLayerNode &node,
84 GraphContext &ctx)
85{
86 validate_node<NETargetInfo>(node, 3 /* expected inputs */, 1 /* expected outputs */);
Georgios Pinitasd8734b52017-12-22 15:27:52 +000087
88 // Extract IO and info
Georgios Pinitasda2491f2018-06-01 17:49:09 +010089 NETargetInfo::TensorType *input = get_backing_tensor<NETargetInfo>(node.input(0));
90 NETargetInfo::TensorType *weights = get_backing_tensor<NETargetInfo>(node.input(1));
91 NETargetInfo::TensorType *biases = get_backing_tensor<NETargetInfo>(node.input(2));
92 NETargetInfo::TensorType *output = get_backing_tensor<NETargetInfo>(node.output(0));
Giorgio Arenabb54e4e2018-04-05 17:20:34 +010093
Georgios Pinitasfd7e8532018-09-07 10:51:27 +010094 const bool is_quantized = is_data_type_quantized_asymmetric(input->info()->data_type());
95
96 if(is_quantized)
Giorgio Arenabb54e4e2018-04-05 17:20:34 +010097 {
98 biases->info()->set_data_type(DataType::S32);
99 }
100
Georgios Pinitas08346e92018-10-16 19:10:46 +0100101 const PadStrideInfo conv_info = node.convolution_info();
102 const ConvolutionMethod conv_algorithm = node.convolution_method();
103 const ActivationLayerInfo fused_act = node.fused_activation();
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000104
105 // Create and configure function (we assume that functions have been validated before creation)
106 std::shared_ptr<IMemoryManager> mm = get_memory_manager(ctx, Target::NEON);
107 std::unique_ptr<IFunction> func;
108 std::string func_name;
Georgios Pinitase2220552018-07-20 13:23:44 +0100109 if(conv_algorithm == ConvolutionMethod::Direct)
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000110 {
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100111 std::tie(func, func_name) = create_named_memory_managed_function<NEDirectConvolutionLayer>(
Georgios Pinitas08346e92018-10-16 19:10:46 +0100112 std::string("DirectConvolutionLayer"), mm, input, weights, biases, output, conv_info, fused_act);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000113 }
114 else if(conv_algorithm == ConvolutionMethod::GEMM)
115 {
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100116 std::tie(func, func_name) = create_named_memory_managed_function<NEGEMMConvolutionLayer>(
Georgios Pinitas08346e92018-10-16 19:10:46 +0100117 std::string("GEMMConvolutionLayer"), mm, input, weights, biases, output, conv_info, WeightsInfo(), Size2D(1, 1), fused_act);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000118 }
Georgios Pinitase2220552018-07-20 13:23:44 +0100119 else if(conv_algorithm == ConvolutionMethod::Winograd)
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000120 {
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100121 std::tie(func, func_name) = create_named_memory_managed_function<NEWinogradConvolutionLayer>(
Georgios Pinitas08346e92018-10-16 19:10:46 +0100122 std::string("WinogradConvolutionLayer"), mm, input, weights, biases, output, conv_info, fused_act);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000123 }
124 else
125 {
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100126 std::tie(func, func_name) = create_named_memory_managed_function<NEConvolutionLayer>(
Georgios Pinitas08346e92018-10-16 19:10:46 +0100127 std::string("ConvolutionLayer"), mm, input, weights, biases, output, conv_info, WeightsInfo(), Size2D(1, 1), fused_act);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000128 }
129
130 // Log info
Georgios Pinitasfd7e8532018-09-07 10:51:27 +0100131 std::ostringstream qss;
132 if(is_quantized)
133 {
134 qss << " Input QuantInfo: " << input->info()->quantization_info()
135 << " Weights QuantInfo: " << weights->info()->quantization_info()
136 << " Output QuantInfo: " << output->info()->quantization_info();
137 }
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000138 ARM_COMPUTE_LOG_GRAPH_INFO("Instantiated " << func_name
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100139 << " Target " << NETargetInfo::TargetType
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000140 << " Data Type: " << input->info()->data_type()
Georgios Pinitasfd7e8532018-09-07 10:51:27 +0100141 << qss.str()
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000142 << " Input shape: " << input->info()->tensor_shape()
143 << " Weights shape: " << weights->info()->tensor_shape()
144 << " Output shape: " << output->info()->tensor_shape()
Georgios Pinitas08346e92018-10-16 19:10:46 +0100145 << (fused_act.enabled() ? " " + to_string(fused_act.activation()) : "")
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000146 << std::endl);
147 return func;
148}
149
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100150template <>
151std::unique_ptr<IFunction> create_normalization_layer<NENormalizationLayer, NETargetInfo>(NormalizationLayerNode &node, GraphContext &ctx)
Georgios Pinitas087eaf62018-05-16 15:52:35 +0100152{
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100153 validate_node<NETargetInfo>(node, 1 /* expected inputs */, 1 /* expected outputs */);
Georgios Pinitas087eaf62018-05-16 15:52:35 +0100154
155 // Extract IO and info
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100156 NETargetInfo::TensorType *input = get_backing_tensor<NETargetInfo>(node.input(0));
157 NETargetInfo::TensorType *output = get_backing_tensor<NETargetInfo>(node.output(0));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000158 const NormalizationLayerInfo norm_info = node.normalization_info();
159 ARM_COMPUTE_ERROR_ON(input == nullptr);
160 ARM_COMPUTE_ERROR_ON(output == nullptr);
161
162 // Create and configure function
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100163 auto func = support::cpp14::make_unique<NENormalizationLayer>(get_memory_manager(ctx, NETargetInfo::TargetType));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000164 func->configure(input, output, norm_info);
165
166 // Log info
Pablo Tello32521432018-11-15 14:43:10 +0000167 ARM_COMPUTE_LOG_GRAPH_INFO("Instantiated "
168 << node.name()
169 << " Type: " << node.type()
170 << " Target: " << NETargetInfo::TargetType
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000171 << " Data Type: " << input->info()->data_type()
172 << " Input shape: " << input->info()->tensor_shape()
173 << " Output shape: " << output->info()->tensor_shape()
174 << " Normalization info: " << norm_info.type()
175 << std::endl);
176
177 return std::move(func);
178}
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100179} // namespace detail
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000180
181std::unique_ptr<IFunction> NEFunctionFactory::create(INode *node, GraphContext &ctx)
182{
183 if(node == nullptr)
184 {
185 return nullptr;
186 }
187
188 NodeType type = node->type();
189 switch(type)
190 {
191 case NodeType::ActivationLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100192 return detail::create_activation_layer<NEActivationLayer, NETargetInfo>(*polymorphic_downcast<ActivationLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000193 case NodeType::BatchNormalizationLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100194 return detail::create_batch_normalization_layer<NEBatchNormalizationLayer, NETargetInfo>(*polymorphic_downcast<BatchNormalizationLayerNode *>(node));
Georgios Pinitasf1adf112018-11-02 12:54:18 +0000195 case NodeType::ChannelShuffleLayer:
196 return detail::create_channel_shuffle_layer<NEChannelShuffleLayer, NETargetInfo>(*polymorphic_downcast<ChannelShuffleLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000197 case NodeType::ConvolutionLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100198 return detail::create_convolution_layer<NEConvolutionLayerFunctions, NETargetInfo>(*polymorphic_downcast<ConvolutionLayerNode *>(node), ctx);
Georgios Pinitas087eaf62018-05-16 15:52:35 +0100199 case NodeType::DeconvolutionLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100200 return detail::create_deconvolution_layer<NEDeconvolutionLayer, NETargetInfo>(*polymorphic_downcast<DeconvolutionLayerNode *>(node), ctx);
Georgios Pinitase2220552018-07-20 13:23:44 +0100201 case NodeType::ConcatenateLayer:
202 return detail::create_concatenate_layer<NEConcatenateLayer, NETargetInfo>(*polymorphic_downcast<ConcatenateLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000203 case NodeType::DepthwiseConvolutionLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100204 return detail::create_depthwise_convolution_layer<NEDepthwiseConvolutionLayerFunctions, NETargetInfo>(*polymorphic_downcast<DepthwiseConvolutionLayerNode *>(node));
Isabella Gottardi7234ed82018-11-27 08:51:10 +0000205 case NodeType::DetectionOutputLayer:
206 return detail::create_detection_output_layer<CPPDetectionOutputLayer, NETargetInfo>(*polymorphic_downcast<DetectionOutputLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000207 case NodeType::EltwiseLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100208 return detail::create_eltwise_layer<NEEltwiseFunctions, NETargetInfo>(*polymorphic_downcast<EltwiseLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000209 case NodeType::FlattenLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100210 return detail::create_flatten_layer<NEFlattenLayer, NETargetInfo>(*polymorphic_downcast<FlattenLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000211 case NodeType::FullyConnectedLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100212 return detail::create_fully_connected_layer<NEFullyConnectedLayer, NETargetInfo>(*polymorphic_downcast<FullyConnectedLayerNode *>(node), ctx);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000213 case NodeType::NormalizationLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100214 return detail::create_normalization_layer<NENormalizationLayer, NETargetInfo>(*polymorphic_downcast<NormalizationLayerNode *>(node), ctx);
Georgios Pinitas57c48242018-08-02 13:41:49 +0100215 case NodeType::PermuteLayer:
216 return detail::create_permute_layer<NEPermute, NETargetInfo>(*polymorphic_downcast<PermuteLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000217 case NodeType::PoolingLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100218 return detail::create_pooling_layer<NEPoolingLayer, NETargetInfo>(*polymorphic_downcast<PoolingLayerNode *>(node));
Pablo Tello32521432018-11-15 14:43:10 +0000219 case NodeType::PriorBoxLayer:
220 return detail::create_priorbox_layer<NEPriorBoxLayer, NETargetInfo>(*polymorphic_downcast<PriorBoxLayerNode *>(node));
Gian Marco Iodice23e24792018-09-07 15:32:14 +0100221 case NodeType::ReorgLayer:
222 return detail::create_reorg_layer<NEReorgLayer, NETargetInfo>(*polymorphic_downcast<ReorgLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000223 case NodeType::ReshapeLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100224 return detail::create_reshape_layer<NEReshapeLayer, NETargetInfo>(*polymorphic_downcast<ReshapeLayerNode *>(node));
Georgios Pinitas087eaf62018-05-16 15:52:35 +0100225 case NodeType::ResizeLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100226 return detail::create_resize_layer<NEScale, NETargetInfo>(*polymorphic_downcast<ResizeLayerNode *>(node));
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000227 case NodeType::SoftmaxLayer:
Georgios Pinitasda2491f2018-06-01 17:49:09 +0100228 return detail::create_softmax_layer<NESoftmaxLayer, NETargetInfo>(*polymorphic_downcast<SoftmaxLayerNode *>(node), ctx);
Michalis Spyrou7c9541c2018-09-20 17:40:04 +0100229 case NodeType::UpsampleLayer:
230 return detail::create_upsample_layer<NEUpsampleLayer, NETargetInfo>(*polymorphic_downcast<UpsampleLayerNode *>(node), ctx);
Michalis Spyroue22aa132018-09-13 10:35:33 +0100231 case NodeType::YOLOLayer:
232 return detail::create_yolo_layer<NEYOLOLayer, NETargetInfo>(*polymorphic_downcast<YOLOLayerNode *>(node), ctx);
Georgios Pinitasd8734b52017-12-22 15:27:52 +0000233 default:
234 return nullptr;
235 }
236}
237} // namespace backends
Georgios Pinitasd9eb2752018-04-03 13:44:29 +0100238} // namespace graph
Pablo Tello32521432018-11-15 14:43:10 +0000239} // namespace arm_compute