blob: 8ea923d599b324a5b8f6b2a9f73aa4fa48a86fb0 [file] [log] [blame]
telsoa014fcda012018-03-09 14:13:49 +00001//
2// Copyright © 2017 Arm Ltd. All rights reserved.
David Beckecb56cd2018-09-05 12:52:57 +01003// SPDX-License-Identifier: MIT
telsoa014fcda012018-03-09 14:13:49 +00004//
Aron Virginas-Tarc9cc8042018-11-01 16:15:57 +00005#include <backendsCommon/CpuTensorHandle.hpp>
6#include <backendsCommon/MemCopyWorkload.hpp>
7#include <backendsCommon/MakeWorkloadHelper.hpp>
telsoa014fcda012018-03-09 14:13:49 +00008#include "RefWorkloadFactory.hpp"
David Beck79141b92018-10-23 16:09:36 +01009#include "RefBackendId.hpp"
David Beckb4540be2018-09-24 13:18:27 +010010#include "workloads/RefWorkloads.hpp"
telsoa014fcda012018-03-09 14:13:49 +000011#include "Layer.hpp"
telsoa014fcda012018-03-09 14:13:49 +000012
13#include <boost/log/trivial.hpp>
14
15namespace armnn
16{
17
David Beck79141b92018-10-23 16:09:36 +010018namespace
19{
20static const BackendId s_Id{RefBackendId()};
21}
telsoa014fcda012018-03-09 14:13:49 +000022template <typename F32Workload, typename U8Workload, typename QueueDescriptorType>
23std::unique_ptr<IWorkload> RefWorkloadFactory::MakeWorkload(const QueueDescriptorType& descriptor,
24 const WorkloadInfo& info) const
25{
kevmay012b4d88e2019-01-24 14:05:09 +000026 return armnn::MakeWorkloadHelper<NullWorkload, F32Workload, U8Workload, NullWorkload, NullWorkload>(descriptor,
27 info);
telsoa014fcda012018-03-09 14:13:49 +000028}
29
telsoa01c577f2c2018-08-31 09:22:23 +010030RefWorkloadFactory::RefWorkloadFactory()
telsoa014fcda012018-03-09 14:13:49 +000031{
32}
33
David Beck79141b92018-10-23 16:09:36 +010034const BackendId& RefWorkloadFactory::GetBackendId() const
35{
36 return s_Id;
37}
38
David Beck29c75de2018-10-23 13:35:58 +010039bool RefWorkloadFactory::IsLayerSupported(const Layer& layer,
40 Optional<DataType> dataType,
telsoa01c577f2c2018-08-31 09:22:23 +010041 std::string& outReasonIfUnsupported)
telsoa014fcda012018-03-09 14:13:49 +000042{
David Beck79141b92018-10-23 16:09:36 +010043 return IWorkloadFactory::IsLayerSupported(s_Id, layer, dataType, outReasonIfUnsupported);
telsoa014fcda012018-03-09 14:13:49 +000044}
45
46std::unique_ptr<ITensorHandle> RefWorkloadFactory::CreateTensorHandle(const TensorInfo& tensorInfo) const
47{
48 return std::make_unique<ScopedCpuTensorHandle>(tensorInfo);
49}
50
Francis Murtagh351d13d2018-09-24 15:01:18 +010051std::unique_ptr<ITensorHandle> RefWorkloadFactory::CreateTensorHandle(const TensorInfo& tensorInfo,
52 DataLayout dataLayout) const
53{
54 return std::make_unique<ScopedCpuTensorHandle>(tensorInfo);
55}
56
telsoa014fcda012018-03-09 14:13:49 +000057std::unique_ptr<IWorkload> RefWorkloadFactory::CreateInput(const InputQueueDescriptor& descriptor,
58 const WorkloadInfo& info) const
59{
60 if (info.m_InputTensorInfos.empty() )
61 {
62 throw InvalidArgumentException("RefWorkloadFactory::CreateInput: Input cannot be zero length");
63 }
64 if (info.m_OutputTensorInfos.empty())
65 {
66 throw InvalidArgumentException("RefWorkloadFactory::CreateInput: Output cannot be zero length");
67 }
68
69 if (info.m_InputTensorInfos[0].GetNumBytes() != info.m_OutputTensorInfos[0].GetNumBytes())
70 {
71 throw InvalidArgumentException("RefWorkloadFactory::CreateInput: data input and output differ in byte count.");
72 }
73
telsoa01c577f2c2018-08-31 09:22:23 +010074 return MakeWorkload<CopyMemGenericWorkload, CopyMemGenericWorkload>(descriptor, info);
telsoa014fcda012018-03-09 14:13:49 +000075}
76
77std::unique_ptr<IWorkload> RefWorkloadFactory::CreateOutput(const OutputQueueDescriptor& descriptor,
78 const WorkloadInfo& info) const
79{
80 if (info.m_InputTensorInfos.empty() )
81 {
82 throw InvalidArgumentException("RefWorkloadFactory::CreateOutput: Input cannot be zero length");
83 }
84 if (info.m_OutputTensorInfos.empty())
85 {
86 throw InvalidArgumentException("RefWorkloadFactory::CreateOutput: Output cannot be zero length");
87 }
88 if (info.m_InputTensorInfos[0].GetNumBytes() != info.m_OutputTensorInfos[0].GetNumBytes())
89 {
90 throw InvalidArgumentException("RefWorkloadFactory::CreateOutput: data input and output differ in byte count.");
91 }
92
kevmay012b4d88e2019-01-24 14:05:09 +000093 return MakeWorkloadHelper<CopyMemGenericWorkload, CopyMemGenericWorkload,
94 CopyMemGenericWorkload, NullWorkload, CopyMemGenericWorkload>(descriptor, info);
telsoa014fcda012018-03-09 14:13:49 +000095}
96
97std::unique_ptr<IWorkload> RefWorkloadFactory::CreateActivation(const ActivationQueueDescriptor& descriptor,
98 const WorkloadInfo& info) const
99{
100 return MakeWorkload<RefActivationFloat32Workload, RefActivationUint8Workload>(descriptor, info);
101}
102
103std::unique_ptr<IWorkload> RefWorkloadFactory::CreateSoftmax(const SoftmaxQueueDescriptor& descriptor,
104 const WorkloadInfo& info) const
105{
106 return MakeWorkload<RefSoftmaxFloat32Workload, RefSoftmaxUint8Workload>(descriptor, info);
107}
108
109std::unique_ptr<IWorkload> RefWorkloadFactory::CreateSplitter(const SplitterQueueDescriptor& descriptor,
110 const WorkloadInfo& info) const
111{
112 return MakeWorkload<RefSplitterFloat32Workload, RefSplitterUint8Workload>(descriptor, info);
113}
114
115std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreateMerger(const MergerQueueDescriptor& descriptor,
116 const WorkloadInfo& info) const
117{
118 return MakeWorkload<RefMergerFloat32Workload, RefMergerUint8Workload>(descriptor, info);
119}
120
121std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreateFullyConnected(
122 const FullyConnectedQueueDescriptor& descriptor, const WorkloadInfo& info) const
123{
124 return MakeWorkload<RefFullyConnectedFloat32Workload, RefFullyConnectedUint8Workload>(descriptor, info);
125}
126
127std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreatePermute(const PermuteQueueDescriptor& descriptor,
128 const WorkloadInfo& info) const
129{
narpra01db2b1602019-01-23 15:23:11 +0000130 return MakeWorkloadHelper<RefPermuteFloat16Workload, RefPermuteFloat32Workload, RefPermuteUint8Workload,
kevmay012b4d88e2019-01-24 14:05:09 +0000131 NullWorkload, NullWorkload>(descriptor, info);
telsoa014fcda012018-03-09 14:13:49 +0000132}
133
134std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreatePooling2d(const Pooling2dQueueDescriptor& descriptor,
135 const WorkloadInfo& info) const
136{
137 return MakeWorkload<RefPooling2dFloat32Workload, RefPooling2dUint8Workload>(descriptor, info);
138}
139
140std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreateConvolution2d(
141 const Convolution2dQueueDescriptor& descriptor, const WorkloadInfo& info) const
142{
143 return MakeWorkload<RefConvolution2dFloat32Workload, RefConvolution2dUint8Workload>(descriptor, info);
144}
145
146std::unique_ptr<IWorkload> RefWorkloadFactory::CreateDepthwiseConvolution2d(
147 const DepthwiseConvolution2dQueueDescriptor& descriptor, const WorkloadInfo& info) const
148{
149 return MakeWorkload<RefDepthwiseConvolution2dFloat32Workload,
150 RefDepthwiseConvolution2dUint8Workload>(descriptor, info);
151}
152
Narumol Prangnawarat94dd5d82019-01-23 18:06:26 +0000153std::unique_ptr<IWorkload> RefWorkloadFactory::CreateDetectionPostProcess(
154 const armnn::DetectionPostProcessQueueDescriptor& descriptor, const armnn::WorkloadInfo& info) const
155{
Narumol Prangnawarat6d302bf2019-02-04 11:46:26 +0000156 const DataType dataType = info.m_InputTensorInfos[0].GetDataType();
157 switch (dataType)
158 {
159 case DataType::Float32:
160 return std::make_unique<RefDetectionPostProcessFloat32Workload>(descriptor, info);
161 case DataType::QuantisedAsymm8:
162 return std::make_unique<RefDetectionPostProcessUint8Workload>(descriptor, info);
163 default:
164 return MakeWorkload<NullWorkload, NullWorkload>(descriptor, info);
165 }
Narumol Prangnawarat94dd5d82019-01-23 18:06:26 +0000166}
167
telsoa014fcda012018-03-09 14:13:49 +0000168std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreateNormalization(
169 const NormalizationQueueDescriptor& descriptor, const WorkloadInfo& info) const
170{
171 return MakeWorkload<RefNormalizationFloat32Workload, NullWorkload>(descriptor, info);
172}
173
174std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreateAddition(const AdditionQueueDescriptor& descriptor,
175 const WorkloadInfo& info) const
176{
Sadik Armagan2e6dc3a2019-04-03 17:48:18 +0100177 return std::make_unique<RefAdditionWorkload>(descriptor, info);
telsoa014fcda012018-03-09 14:13:49 +0000178}
179
180std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreateMultiplication(
181 const MultiplicationQueueDescriptor& descriptor, const WorkloadInfo& info) const
182{
Sadik Armagan2e6dc3a2019-04-03 17:48:18 +0100183 return std::make_unique<RefMultiplicationWorkload>(descriptor, info);
telsoa014fcda012018-03-09 14:13:49 +0000184}
185
186std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreateBatchNormalization(
187 const BatchNormalizationQueueDescriptor& descriptor, const WorkloadInfo& info) const
188{
189 return MakeWorkload<RefBatchNormalizationFloat32Workload, RefBatchNormalizationUint8Workload>(descriptor, info);
190}
191
192std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreateMemCopy(const MemCopyQueueDescriptor& descriptor,
193 const WorkloadInfo& info) const
194{
195 if (descriptor.m_Inputs.empty())
196 {
197 throw InvalidArgumentException("RefWorkloadFactory: CreateMemCopy() expected an input tensor.");
198 }
telsoa01c577f2c2018-08-31 09:22:23 +0100199 return std::make_unique<CopyMemGenericWorkload>(descriptor, info);
telsoa014fcda012018-03-09 14:13:49 +0000200}
201
202std::unique_ptr<IWorkload> RefWorkloadFactory::CreateResizeBilinear(const ResizeBilinearQueueDescriptor& descriptor,
203 const WorkloadInfo& info) const
204{
205 return MakeWorkload<RefResizeBilinearFloat32Workload, RefResizeBilinearUint8Workload>(descriptor, info);
206}
207
208std::unique_ptr<IWorkload> RefWorkloadFactory::CreateFakeQuantization(
209 const FakeQuantizationQueueDescriptor& descriptor,
210 const WorkloadInfo& info) const
211{
212 return MakeWorkload<RefFakeQuantizationFloat32Workload, NullWorkload>(descriptor, info);
213}
214
215std::unique_ptr<IWorkload> RefWorkloadFactory::CreateL2Normalization(const L2NormalizationQueueDescriptor& descriptor,
216 const WorkloadInfo& info) const
217{
218 return MakeWorkload<RefL2NormalizationFloat32Workload, NullWorkload>(descriptor, info);
219}
220
221std::unique_ptr<IWorkload> RefWorkloadFactory::CreateConstant(const ConstantQueueDescriptor& descriptor,
222 const WorkloadInfo& info) const
223{
narpra01db2b1602019-01-23 15:23:11 +0000224 return MakeWorkloadHelper<NullWorkload, RefConstantFloat32Workload, RefConstantUint8Workload,
kevmay012b4d88e2019-01-24 14:05:09 +0000225 RefConstantInt32Workload, NullWorkload>(descriptor, info);
telsoa014fcda012018-03-09 14:13:49 +0000226}
227
228std::unique_ptr<IWorkload> RefWorkloadFactory::CreateReshape(const ReshapeQueueDescriptor& descriptor,
229 const WorkloadInfo& info) const
230{
231 return MakeWorkload<RefReshapeFloat32Workload, RefReshapeUint8Workload>(descriptor, info);
232}
233
Nattapat Chaimanowong207ef9a2018-11-02 10:57:25 +0000234std::unique_ptr<IWorkload> RefWorkloadFactory::CreateSpaceToBatchNd(const SpaceToBatchNdQueueDescriptor& descriptor,
235 const WorkloadInfo& info) const
236{
Nattapat Chaimanowong3ea76d52018-11-09 14:10:38 +0000237 return MakeWorkload<RefSpaceToBatchNdFloat32Workload, RefSpaceToBatchNdUint8Workload>(descriptor, info);
Nattapat Chaimanowong207ef9a2018-11-02 10:57:25 +0000238}
239
telsoa014fcda012018-03-09 14:13:49 +0000240std::unique_ptr<IWorkload> RefWorkloadFactory::CreateFloor(const FloorQueueDescriptor& descriptor,
telsoa01c577f2c2018-08-31 09:22:23 +0100241 const WorkloadInfo& info) const
telsoa014fcda012018-03-09 14:13:49 +0000242{
243 return MakeWorkload<RefFloorFloat32Workload, NullWorkload>(descriptor, info);
244}
245
telsoa01c577f2c2018-08-31 09:22:23 +0100246std::unique_ptr<IWorkload> RefWorkloadFactory::CreateLstm(const LstmQueueDescriptor& descriptor,
247 const WorkloadInfo& info) const
248{
249 return MakeWorkload<RefLstmFloat32Workload, NullWorkload>(descriptor, info);
250}
251
252std::unique_ptr<IWorkload> RefWorkloadFactory::CreateConvertFp16ToFp32(
253 const ConvertFp16ToFp32QueueDescriptor& descriptor,
254 const WorkloadInfo& info) const
255{
256 return std::make_unique<RefConvertFp16ToFp32Workload>(descriptor, info);
257}
258
259std::unique_ptr<IWorkload> RefWorkloadFactory::CreateConvertFp32ToFp16(
260 const ConvertFp32ToFp16QueueDescriptor& descriptor,
261 const WorkloadInfo& info) const
262{
263 return std::make_unique<RefConvertFp32ToFp16Workload>(descriptor, info);
264}
265
Francis Murtaghe7a86a42018-08-29 12:42:10 +0100266std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreateDivision(
267 const DivisionQueueDescriptor& descriptor, const WorkloadInfo& info) const
268{
Sadik Armagan2e6dc3a2019-04-03 17:48:18 +0100269 return std::make_unique<RefDivisionWorkload>(descriptor, info);
Francis Murtaghe7a86a42018-08-29 12:42:10 +0100270}
271
David Beckc2044fe2018-09-05 15:00:38 +0100272std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreateSubtraction(
273 const SubtractionQueueDescriptor& descriptor, const WorkloadInfo& info) const
274{
Sadik Armagan2e6dc3a2019-04-03 17:48:18 +0100275 return std::make_unique<RefSubtractionWorkload>(descriptor, info);
David Beckc2044fe2018-09-05 15:00:38 +0100276}
277
Nattapat Chaimanowong5a4304a2018-11-28 10:44:37 +0000278std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreateMaximum(
279 const MaximumQueueDescriptor& descriptor, const WorkloadInfo& info) const
280{
Sadik Armagan2e6dc3a2019-04-03 17:48:18 +0100281 return std::make_unique<RefMaximumWorkload>(descriptor, info);
Nattapat Chaimanowong5a4304a2018-11-28 10:44:37 +0000282}
283
narpra01a6bf9122018-09-10 09:50:09 +0100284std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreateMean(
285 const MeanQueueDescriptor& descriptor, const WorkloadInfo& info) const
286{
narpra011e4c31d2018-09-28 11:07:51 +0100287 return MakeWorkload<RefMeanFloat32Workload, RefMeanUint8Workload>(descriptor, info);
narpra01a6bf9122018-09-10 09:50:09 +0100288}
289
Éanna Ó Catháin20e58802018-12-04 10:29:06 +0000290std::unique_ptr<armnn::IWorkload> RefWorkloadFactory::CreateMinimum(
291 const MinimumQueueDescriptor& descriptor, const WorkloadInfo& info) const
292{
Sadik Armagan2e6dc3a2019-04-03 17:48:18 +0100293 return std::make_unique<RefMinimumWorkload>(descriptor, info);
Éanna Ó Catháin20e58802018-12-04 10:29:06 +0000294}
295
jimfly012c9322a2018-09-19 10:59:49 +0100296std::unique_ptr<IWorkload> RefWorkloadFactory::CreatePad(const PadQueueDescriptor& descriptor,
297 const WorkloadInfo& info) const
298{
Mohamed Nour Abouelseouddd6acea2018-10-18 12:26:19 +0100299 return MakeWorkload<RefPadFloat32Workload, RefPadUint8Workload>(descriptor, info);
jimfly012c9322a2018-09-19 10:59:49 +0100300}
301
FrancisMurtagh20995952018-12-17 12:11:36 +0000302std::unique_ptr<IWorkload> RefWorkloadFactory::CreateEqual(const EqualQueueDescriptor& descriptor,
303 const WorkloadInfo& info) const
304{
Sadik Armagan2e6dc3a2019-04-03 17:48:18 +0100305 return std::make_unique<RefEqualWorkload>(descriptor, info);
FrancisMurtagh20995952018-12-17 12:11:36 +0000306}
307
Éanna Ó Catháin4e1e1362018-11-12 11:36:34 +0000308std::unique_ptr<IWorkload> RefWorkloadFactory::CreateBatchToSpaceNd(const BatchToSpaceNdQueueDescriptor& descriptor,
309 const WorkloadInfo& info) const
310{
311 return MakeWorkload<RefBatchToSpaceNdFloat32Workload, RefBatchToSpaceNdUint8Workload>(descriptor, info);
312}
jimfly012c9322a2018-09-19 10:59:49 +0100313
Conor Kennedy430b5d82018-11-14 15:28:28 +0000314std::unique_ptr<IWorkload> RefWorkloadFactory::CreateStridedSlice(const StridedSliceQueueDescriptor& descriptor,
315 const WorkloadInfo& info) const
316{
Nattapat Chaimanowong1216b582018-11-23 15:33:41 +0000317 return MakeWorkload<RefStridedSliceFloat32Workload, RefStridedSliceUint8Workload>(descriptor, info);
Conor Kennedy430b5d82018-11-14 15:28:28 +0000318}
319
Matteo Martincigh59a950c2018-12-13 12:48:25 +0000320std::unique_ptr<IWorkload> RefWorkloadFactory::CreateGreater(const GreaterQueueDescriptor& descriptor,
321 const WorkloadInfo& info) const
322{
Sadik Armagan2e6dc3a2019-04-03 17:48:18 +0100323 return std::make_unique<RefGreaterWorkload>(descriptor, info);
Matteo Martincigh59a950c2018-12-13 12:48:25 +0000324}
325
Nattapat Chaimanowonga9a1cf12018-12-03 16:06:49 +0000326std::unique_ptr<IWorkload> RefWorkloadFactory::CreateDebug(const DebugQueueDescriptor& descriptor,
327 const WorkloadInfo& info) const
328{
Nattapat Chaimanowongcfdcadf2018-12-06 11:54:33 +0000329 return MakeWorkload<RefDebugFloat32Workload, RefDebugUint8Workload>(descriptor, info);
Nattapat Chaimanowonga9a1cf12018-12-03 16:06:49 +0000330}
331
Mohamed Nour Abouelseouda1d3c6a2018-12-27 12:39:16 +0000332std::unique_ptr<IWorkload> RefWorkloadFactory::CreateRsqrt(const RsqrtQueueDescriptor& descriptor,
333 const WorkloadInfo& info) const
334{
335 return MakeWorkload<RefRsqrtFloat32Workload, NullWorkload>(descriptor, info);
336}
337
narpra014951d842019-01-18 16:53:53 +0000338std::unique_ptr<IWorkload> RefWorkloadFactory::CreateGather(const armnn::GatherQueueDescriptor& descriptor,
339 const armnn::WorkloadInfo& info) const
340{
341 return MakeWorkload<RefGatherFloat32Workload, RefGatherUint8Workload>(descriptor, info);
342}
343
Matteo Martincigh49124022019-01-11 13:25:59 +0000344std::unique_ptr<IWorkload> RefWorkloadFactory::CreatePreCompiled(const PreCompiledQueueDescriptor& descriptor,
345 const WorkloadInfo& info) const
346{
347 return nullptr;
348}
349
Derek Lamberti5f400d62019-03-25 15:41:58 +0000350std::unique_ptr<IWorkload> RefWorkloadFactory::CreateQuantize(const QuantizeQueueDescriptor& descriptor,
351 const WorkloadInfo& info) const
352{
353 return std::make_unique<RefQuantizeWorkload>(descriptor, info);
354}
355
Nattapat Chaimanowong8a54ac02019-03-29 15:25:04 +0000356std::unique_ptr<IWorkload> RefWorkloadFactory::CreateDequantize(const DequantizeQueueDescriptor& descriptor,
357 const WorkloadInfo& info) const
358{
Nattapat Chaimanowongafa4e3a2019-04-02 11:41:45 +0100359 return std::make_unique<RefDequantizeWorkload>(descriptor, info);
Nattapat Chaimanowong8a54ac02019-03-29 15:25:04 +0000360}
361
Matteo Martincigh49124022019-01-11 13:25:59 +0000362} // namespace armnn