blob: 2034a65f6d5b1869e0aa51313ef603ddc1c8c5dd [file] [log] [blame]
Laurent Carlier749294b2020-06-01 09:03:17 +01001//
telsoa014fcda012018-03-09 14:13:49 +00002// 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
6#include "WorkloadTestUtils.hpp"
telsoa014fcda012018-03-09 14:13:49 +00007
8#include <armnn/Exceptions.hpp>
9
James Conroy1f58f032021-04-27 17:13:27 +010010#include <backendsCommon/TensorHandle.hpp>
Aron Virginas-Tarc9cc8042018-11-01 16:15:57 +000011#include <backendsCommon/Workload.hpp>
12
13#include <reference/workloads/RefWorkloads.hpp>
14#include <reference/RefWorkloadFactory.hpp>
15
Sadik Armagan1625efc2021-06-10 18:24:34 +010016#include <doctest/doctest.h>
telsoa014fcda012018-03-09 14:13:49 +000017
18using namespace armnn;
19
Sadik Armagan1625efc2021-06-10 18:24:34 +010020TEST_SUITE("WorkloadInfoValidation")
21{
22TEST_CASE("BatchNormalizationQueueDescriptor_Validate_DifferentQuantizationData")
Mike Kelly1ced4642020-03-04 18:01:13 +000023{
24 TensorShape inputShape { 1, 3, 2, 2 };
25 TensorShape outputShape { 1, 3, 2, 2 };
telsoa014fcda012018-03-09 14:13:49 +000026
Mike Kelly1ced4642020-03-04 18:01:13 +000027 TensorInfo inputTensorInfo(inputShape, armnn::DataType::QAsymmU8, .1f, 125);
28 TensorInfo outputTensorInfo(outputShape, armnn::DataType::QAsymmU8, .2f, 120);
29
30 BatchNormalizationQueueDescriptor invalidData;
31 WorkloadInfo invalidInfo;
32
33 unsigned int sameShape[] = { 10 };
34 TensorInfo sameInfo = armnn::TensorInfo(1, sameShape, armnn::DataType::QAsymmU8);
James Conroy1f58f032021-04-27 17:13:27 +010035 ScopedTensorHandle sameTensor(sameInfo);
Mike Kelly1ced4642020-03-04 18:01:13 +000036
37 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
38 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
39
40 invalidData.m_Mean = &sameTensor;
41 invalidData.m_Variance = &sameTensor;
42 invalidData.m_Beta= &sameTensor;
43 invalidData.m_Gamma = &sameTensor;
44
Sadik Armagan1625efc2021-06-10 18:24:34 +010045 CHECK_NOTHROW(RefBatchNormalizationWorkload(invalidData, invalidInfo));
Mike Kelly1ced4642020-03-04 18:01:13 +000046}
telsoa014fcda012018-03-09 14:13:49 +000047
Sadik Armagan1625efc2021-06-10 18:24:34 +010048TEST_CASE("QueueDescriptor_Validate_WrongNumOfInputsOutputs")
telsoa014fcda012018-03-09 14:13:49 +000049{
50 InputQueueDescriptor invalidData;
51 WorkloadInfo invalidInfo;
telsoa01c577f2c2018-08-31 09:22:23 +010052 //Invalid argument exception is expected, because no inputs and no outputs were defined.
Sadik Armagan1625efc2021-06-10 18:24:34 +010053 CHECK_THROWS_AS(RefWorkloadFactory().CreateInput(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +000054}
55
Sadik Armagan1625efc2021-06-10 18:24:34 +010056TEST_CASE("RefPooling2dFloat32Workload_Validate_WrongDimTensor")
telsoa014fcda012018-03-09 14:13:49 +000057{
58 armnn::TensorInfo inputTensorInfo;
59 armnn::TensorInfo outputTensorInfo;
60
telsoa01c577f2c2018-08-31 09:22:23 +010061 unsigned int inputShape[] = {2, 3, 4}; // <- Invalid - input tensor has to be 4D.
telsoa014fcda012018-03-09 14:13:49 +000062 unsigned int outputShape[] = {2, 3, 4, 5};
63
64 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
65 inputTensorInfo = armnn::TensorInfo(3, inputShape, armnn::DataType::Float32);
66
67 Pooling2dQueueDescriptor invalidData;
68 WorkloadInfo invalidInfo;
69
70 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
71 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
72
telsoa01c577f2c2018-08-31 09:22:23 +010073 // Invalid argument exception is expected, input tensor has to be 4D.
Sadik Armagan1625efc2021-06-10 18:24:34 +010074 CHECK_THROWS_AS(RefPooling2dWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +000075}
76
Sadik Armagan1625efc2021-06-10 18:24:34 +010077TEST_CASE("SoftmaxQueueDescriptor_Validate_WrongInputHeight")
telsoa014fcda012018-03-09 14:13:49 +000078{
79 unsigned int inputHeight = 1;
80 unsigned int inputWidth = 1;
81 unsigned int inputChannels = 4;
82 unsigned int inputNum = 2;
83
84 unsigned int outputChannels = inputChannels;
telsoa01c577f2c2018-08-31 09:22:23 +010085 unsigned int outputHeight = inputHeight + 1; //Makes data invalid - Softmax expects height and width to be 1.
telsoa014fcda012018-03-09 14:13:49 +000086 unsigned int outputWidth = inputWidth;
87 unsigned int outputNum = inputNum;
88
89 armnn::TensorInfo inputTensorInfo;
90 armnn::TensorInfo outputTensorInfo;
91
92 unsigned int inputShape[] = { inputNum, inputChannels, inputHeight, inputWidth };
93 unsigned int outputShape[] = { outputNum, outputChannels, outputHeight, outputWidth };
94
95 inputTensorInfo = armnn::TensorInfo(4, inputShape, armnn::DataType::Float32);
96 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
97
98 SoftmaxQueueDescriptor invalidData;
99 WorkloadInfo invalidInfo;
100
101 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
102 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
103
telsoa01c577f2c2018-08-31 09:22:23 +0100104 //Invalid argument exception is expected, because height != 1.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100105 CHECK_THROWS_AS(RefSoftmaxWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000106}
107
Sadik Armagan1625efc2021-06-10 18:24:34 +0100108TEST_CASE("FullyConnectedQueueDescriptor_Validate_RequiredDataMissing")
telsoa014fcda012018-03-09 14:13:49 +0000109{
110 unsigned int inputWidth = 1;
111 unsigned int inputHeight = 1;
112 unsigned int inputChannels = 5;
113 unsigned int inputNum = 2;
114
115 unsigned int outputWidth = 1;
116 unsigned int outputHeight = 1;
117 unsigned int outputChannels = 3;
118 unsigned int outputNum = 2;
119
telsoa01c577f2c2018-08-31 09:22:23 +0100120 // Define the tensor descriptors.
telsoa014fcda012018-03-09 14:13:49 +0000121 armnn::TensorInfo inputTensorInfo;
122 armnn::TensorInfo outputTensorInfo;
123 armnn::TensorInfo weightsDesc;
124 armnn::TensorInfo biasesDesc;
125
126 unsigned int inputShape[] = { inputNum, inputChannels, inputHeight, inputWidth };
127 unsigned int outputShape[] = { outputNum, outputChannels, outputHeight, outputWidth };
128 unsigned int weightsShape[] = { 1, 1, inputChannels, outputChannels };
129 unsigned int biasShape[] = { 1, outputChannels, outputHeight, outputWidth };
130
131 inputTensorInfo = armnn::TensorInfo(4, inputShape, armnn::DataType::Float32);
132 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
133 weightsDesc = armnn::TensorInfo(4, weightsShape, armnn::DataType::Float32);
134 biasesDesc = armnn::TensorInfo(4, biasShape, armnn::DataType::Float32);
135
136 FullyConnectedQueueDescriptor invalidData;
137 WorkloadInfo invalidInfo;
138
James Conroy1f58f032021-04-27 17:13:27 +0100139 ScopedTensorHandle weightTensor(weightsDesc);
140 ScopedTensorHandle biasTensor(biasesDesc);
telsoa014fcda012018-03-09 14:13:49 +0000141
142 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
143 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
144 invalidData.m_Weight = &weightTensor;
145 invalidData.m_Bias = &biasTensor;
146 invalidData.m_Parameters.m_BiasEnabled = true;
147 invalidData.m_Parameters.m_TransposeWeightMatrix = false;
148
149
telsoa01c577f2c2018-08-31 09:22:23 +0100150 //Invalid argument exception is expected, because not all required fields have been provided.
151 //In particular inputsData[0], outputsData[0] and weightsData can not be null.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100152 CHECK_THROWS_AS(RefFullyConnectedWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000153}
154
155
Sadik Armagan1625efc2021-06-10 18:24:34 +0100156TEST_CASE("NormalizationQueueDescriptor_Validate_WrongInputHeight")
telsoa014fcda012018-03-09 14:13:49 +0000157{
158 constexpr unsigned int inputNum = 5;
159 constexpr unsigned int inputHeight = 32;
160 constexpr unsigned int inputWidth = 24;
161 constexpr unsigned int inputChannels = 3;
162
163 constexpr unsigned int outputNum = inputNum;
164 constexpr unsigned int outputChannels = inputChannels;
telsoa01c577f2c2018-08-31 09:22:23 +0100165 constexpr unsigned int outputHeight = inputHeight + 1; //Makes data invalid - normalization requires.
166 //Input and output to have the same dimensions.
telsoa014fcda012018-03-09 14:13:49 +0000167 constexpr unsigned int outputWidth = inputWidth;
168
169
170 armnn::TensorInfo inputTensorInfo;
171 armnn::TensorInfo outputTensorInfo;
172
173 unsigned int inputShape[] = {inputNum, inputChannels, inputHeight, inputWidth};
174 unsigned int outputShape[] = {outputNum, outputChannels, outputHeight, outputWidth};
175
176 inputTensorInfo = armnn::TensorInfo(4, inputShape, armnn::DataType::Float32);
177 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
178
179
180 armnn::NormalizationAlgorithmMethod normMethod = armnn::NormalizationAlgorithmMethod::LocalBrightness;
181 armnn::NormalizationAlgorithmChannel normChannel = armnn::NormalizationAlgorithmChannel::Across;
182 float alpha = 1.f;
183 float beta = 1.f;
184 float kappa = 1.f;
185 uint32_t normSize = 5;
186
187 NormalizationQueueDescriptor invalidData;
188 WorkloadInfo invalidInfo;
189
190 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
191 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
192 invalidData.m_Parameters.m_NormChannelType = normChannel;
193 invalidData.m_Parameters.m_NormMethodType = normMethod;
194 invalidData.m_Parameters.m_NormSize = normSize;
195 invalidData.m_Parameters.m_Alpha = alpha;
196 invalidData.m_Parameters.m_Beta = beta;
197 invalidData.m_Parameters.m_K = kappa;
198
telsoa01c577f2c2018-08-31 09:22:23 +0100199 //Invalid argument exception is expected, because input height != output height.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100200 CHECK_THROWS_AS(RefNormalizationWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000201}
202
Sadik Armagan1625efc2021-06-10 18:24:34 +0100203TEST_CASE("SplitterQueueDescriptor_Validate_WrongWindow")
telsoa014fcda012018-03-09 14:13:49 +0000204{
205 constexpr unsigned int inputNum = 1;
206 constexpr unsigned int inputHeight = 32;
207 constexpr unsigned int inputWidth = 24;
208 constexpr unsigned int inputChannels = 3;
209
210 constexpr unsigned int outputNum = inputNum;
211 constexpr unsigned int outputChannels = inputChannels;
212 constexpr unsigned int outputHeight = 18;
213 constexpr unsigned int outputWidth = inputWidth;
214
215
216 armnn::TensorInfo inputTensorInfo;
217 armnn::TensorInfo outputTensorInfo;
218
219 unsigned int inputShape[] = {inputNum, inputChannels, inputHeight, inputWidth};
220 unsigned int outputShape[] = {outputNum, outputChannels, outputHeight, outputWidth};
221
222 inputTensorInfo = armnn::TensorInfo(4, inputShape, armnn::DataType::Float32);
223 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
224
225 SplitterQueueDescriptor invalidData;
226 WorkloadInfo invalidInfo;
227
228 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
229 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
230
telsoa01c577f2c2018-08-31 09:22:23 +0100231 // Invalid, since it has only 3 dimensions while the input tensor is 4d.
telsoa014fcda012018-03-09 14:13:49 +0000232 std::vector<unsigned int> wOrigin = {0, 0, 0};
233 armnn::SplitterQueueDescriptor::ViewOrigin window(wOrigin);
234 invalidData.m_ViewOrigins.push_back(window);
235
Sadik Armagan1625efc2021-06-10 18:24:34 +0100236 INFO("Invalid argument exception is expected, because split window dimensionality does not match input.");
237 CHECK_THROWS_AS(RefSplitterWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000238
telsoa01c577f2c2018-08-31 09:22:23 +0100239 // Invalid, since window extends past the boundary of input tensor.
telsoa014fcda012018-03-09 14:13:49 +0000240 std::vector<unsigned int> wOrigin3 = {0, 0, 15, 0};
241 armnn::SplitterQueueDescriptor::ViewOrigin window3(wOrigin3);
242 invalidData.m_ViewOrigins[0] = window3;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100243 INFO("Invalid argument exception is expected (wOrigin3[2]+ outputHeight > inputHeight");
244 CHECK_THROWS_AS(RefSplitterWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000245
246
247 std::vector<unsigned int> wOrigin4 = {0, 0, 0, 0};
248 armnn::SplitterQueueDescriptor::ViewOrigin window4(wOrigin4);
249 invalidData.m_ViewOrigins[0] = window4;
250
251 std::vector<unsigned int> wOrigin5 = {1, 16, 20, 2};
252 armnn::SplitterQueueDescriptor::ViewOrigin window5(wOrigin4);
253 invalidData.m_ViewOrigins.push_back(window5);
254
Sadik Armagan1625efc2021-06-10 18:24:34 +0100255 INFO("Invalid exception due to number of split windows not matching number of outputs.");
256 CHECK_THROWS_AS(RefSplitterWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000257}
258
259
Sadik Armagan1625efc2021-06-10 18:24:34 +0100260TEST_CASE("ConcatQueueDescriptor_Validate_WrongWindow")
telsoa014fcda012018-03-09 14:13:49 +0000261{
262 constexpr unsigned int inputNum = 1;
263 constexpr unsigned int inputChannels = 3;
264 constexpr unsigned int inputHeight = 32;
265 constexpr unsigned int inputWidth = 24;
266
267 constexpr unsigned int outputNum = 1;
268 constexpr unsigned int outputChannels = 3;
269 constexpr unsigned int outputHeight = 32;
270 constexpr unsigned int outputWidth = 24;
271
272
273 armnn::TensorInfo inputTensorInfo;
274 armnn::TensorInfo outputTensorInfo;
275
276 unsigned int inputShape[] = {inputNum, inputChannels, inputHeight, inputWidth};
277 unsigned int outputShape[] = {outputNum, outputChannels, outputHeight, outputWidth};
278
279 inputTensorInfo = armnn::TensorInfo(4, inputShape, armnn::DataType::Float32);
280 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
281
Jim Flynne242f2d2019-05-22 14:24:13 +0100282 ConcatQueueDescriptor invalidData;
telsoa014fcda012018-03-09 14:13:49 +0000283 WorkloadInfo invalidInfo;
284
285 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
286 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
287
telsoa01c577f2c2018-08-31 09:22:23 +0100288 // Invalid, since it has only 3 dimensions while the input tensor is 4d.
telsoa014fcda012018-03-09 14:13:49 +0000289 std::vector<unsigned int> wOrigin = {0, 0, 0};
Jim Flynne242f2d2019-05-22 14:24:13 +0100290 armnn::ConcatQueueDescriptor::ViewOrigin window(wOrigin);
telsoa014fcda012018-03-09 14:13:49 +0000291 invalidData.m_ViewOrigins.push_back(window);
292
Sadik Armagan1625efc2021-06-10 18:24:34 +0100293 INFO("Invalid argument exception is expected, because merge window dimensionality does not match input.");
294 CHECK_THROWS_AS(RefConcatWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000295
telsoa01c577f2c2018-08-31 09:22:23 +0100296 // Invalid, since window extends past the boundary of output tensor.
telsoa014fcda012018-03-09 14:13:49 +0000297 std::vector<unsigned int> wOrigin3 = {0, 0, 15, 0};
Jim Flynne242f2d2019-05-22 14:24:13 +0100298 armnn::ConcatQueueDescriptor::ViewOrigin window3(wOrigin3);
telsoa014fcda012018-03-09 14:13:49 +0000299 invalidData.m_ViewOrigins[0] = window3;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100300 INFO("Invalid argument exception is expected (wOrigin3[2]+ inputHeight > outputHeight");
301 CHECK_THROWS_AS(RefConcatWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000302
303
304 std::vector<unsigned int> wOrigin4 = {0, 0, 0, 0};
Jim Flynne242f2d2019-05-22 14:24:13 +0100305 armnn::ConcatQueueDescriptor::ViewOrigin window4(wOrigin4);
telsoa014fcda012018-03-09 14:13:49 +0000306 invalidData.m_ViewOrigins[0] = window4;
307
308 std::vector<unsigned int> wOrigin5 = {1, 16, 20, 2};
Jim Flynne242f2d2019-05-22 14:24:13 +0100309 armnn::ConcatQueueDescriptor::ViewOrigin window5(wOrigin4);
telsoa014fcda012018-03-09 14:13:49 +0000310 invalidData.m_ViewOrigins.push_back(window5);
311
Sadik Armagan1625efc2021-06-10 18:24:34 +0100312 INFO("Invalid exception due to number of merge windows not matching number of inputs.");
313 CHECK_THROWS_AS(RefConcatWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000314}
315
Sadik Armagan1625efc2021-06-10 18:24:34 +0100316TEST_CASE("AdditionQueueDescriptor_Validate_InputNumbers")
telsoa014fcda012018-03-09 14:13:49 +0000317{
318 armnn::TensorInfo input1TensorInfo;
319 armnn::TensorInfo input2TensorInfo;
320 armnn::TensorInfo input3TensorInfo;
321 armnn::TensorInfo outputTensorInfo;
322
323 unsigned int shape[] = {1, 1, 1, 1};
324
325 input1TensorInfo = armnn::TensorInfo(4, shape, armnn::DataType::Float32);
326 input2TensorInfo = armnn::TensorInfo(4, shape, armnn::DataType::Float32);
327 input3TensorInfo = armnn::TensorInfo(4, shape, armnn::DataType::Float32);
328 outputTensorInfo = armnn::TensorInfo(4, shape, armnn::DataType::Float32);
329
330 AdditionQueueDescriptor invalidData;
331 WorkloadInfo invalidInfo;
332
333 AddInputToWorkload(invalidData, invalidInfo, input1TensorInfo, nullptr);
334 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
335
telsoa01c577f2c2018-08-31 09:22:23 +0100336 // Too few inputs.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100337 CHECK_THROWS_AS(RefAdditionWorkload<>(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000338
339 AddInputToWorkload(invalidData, invalidInfo, input2TensorInfo, nullptr);
340
telsoa01c577f2c2018-08-31 09:22:23 +0100341 // Correct.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100342 CHECK_NOTHROW(RefAdditionWorkload<>(invalidData, invalidInfo));
telsoa014fcda012018-03-09 14:13:49 +0000343
344 AddInputToWorkload(invalidData, invalidInfo, input3TensorInfo, nullptr);
345
telsoa01c577f2c2018-08-31 09:22:23 +0100346 // Too many inputs.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100347 CHECK_THROWS_AS(RefAdditionWorkload<>(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000348}
349
Sadik Armagan1625efc2021-06-10 18:24:34 +0100350TEST_CASE("AdditionQueueDescriptor_Validate_InputShapes")
telsoa014fcda012018-03-09 14:13:49 +0000351{
352 armnn::TensorInfo input1TensorInfo;
353 armnn::TensorInfo input2TensorInfo;
354 armnn::TensorInfo outputTensorInfo;
355
356 unsigned int shape1[] = {1, 1, 2, 1};
357 unsigned int shape2[] = {1, 1, 3, 2};
358
telsoa01c577f2c2018-08-31 09:22:23 +0100359 // Incompatible shapes even with broadcasting.
telsoa014fcda012018-03-09 14:13:49 +0000360 {
361 input1TensorInfo = armnn::TensorInfo(4, shape1, armnn::DataType::Float32);
362 input2TensorInfo = armnn::TensorInfo(4, shape2, armnn::DataType::Float32);
363 outputTensorInfo = armnn::TensorInfo(4, shape1, armnn::DataType::Float32);
364
365 AdditionQueueDescriptor invalidData;
366 WorkloadInfo invalidInfo;
367
368 AddInputToWorkload(invalidData, invalidInfo, input1TensorInfo, nullptr);
369 AddInputToWorkload(invalidData, invalidInfo, input2TensorInfo, nullptr);
370 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
371
Sadik Armagan1625efc2021-06-10 18:24:34 +0100372 CHECK_THROWS_AS(RefAdditionWorkload<>(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000373 }
374
telsoa01c577f2c2018-08-31 09:22:23 +0100375 // Output size not compatible with input sizes.
telsoa014fcda012018-03-09 14:13:49 +0000376 {
377 input1TensorInfo = armnn::TensorInfo(4, shape1, armnn::DataType::Float32);
378 input2TensorInfo = armnn::TensorInfo(4, shape1, armnn::DataType::Float32);
379 outputTensorInfo = armnn::TensorInfo(4, shape2, armnn::DataType::Float32);
380
381 AdditionQueueDescriptor invalidData;
382 WorkloadInfo invalidInfo;
383
384 AddInputToWorkload(invalidData, invalidInfo, input1TensorInfo, nullptr);
385 AddInputToWorkload(invalidData, invalidInfo, input2TensorInfo, nullptr);
386 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
387
telsoa01c577f2c2018-08-31 09:22:23 +0100388 // Output differs.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100389 CHECK_THROWS_AS(RefAdditionWorkload<>(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000390 }
391}
392
Sadik Armagan1625efc2021-06-10 18:24:34 +0100393TEST_CASE("MultiplicationQueueDescriptor_Validate_InputTensorDimensionMismatch")
telsoa014fcda012018-03-09 14:13:49 +0000394{
395 armnn::TensorInfo input0TensorInfo;
396 armnn::TensorInfo input1TensorInfo;
397 armnn::TensorInfo outputTensorInfo;
398
399 constexpr unsigned int input0Shape[] = { 2, 2, 4, 4 };
400 constexpr std::size_t dimensionCount = std::extent<decltype(input0Shape)>::value;
401
telsoa01c577f2c2018-08-31 09:22:23 +0100402 // Checks dimension consistency for input tensors.
telsoa014fcda012018-03-09 14:13:49 +0000403 for (unsigned int dimIndex = 0; dimIndex < dimensionCount; ++dimIndex)
404 {
405 unsigned int input1Shape[dimensionCount];
406 for (unsigned int i = 0; i < dimensionCount; ++i)
407 {
408 input1Shape[i] = input0Shape[i];
409 }
410
411 ++input1Shape[dimIndex];
412
413 input0TensorInfo = armnn::TensorInfo(dimensionCount, input0Shape, armnn::DataType::Float32);
414 input1TensorInfo = armnn::TensorInfo(dimensionCount, input1Shape, armnn::DataType::Float32);
415 outputTensorInfo = armnn::TensorInfo(dimensionCount, input0Shape, armnn::DataType::Float32);
416
417 MultiplicationQueueDescriptor invalidData;
418 WorkloadInfo invalidInfo;
419
420 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
421 AddInputToWorkload(invalidData, invalidInfo, input0TensorInfo, nullptr);
422 AddInputToWorkload(invalidData, invalidInfo, input1TensorInfo, nullptr);
423
Sadik Armagan1625efc2021-06-10 18:24:34 +0100424 CHECK_THROWS_AS(RefMultiplicationWorkload<>(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000425 }
426
telsoa01c577f2c2018-08-31 09:22:23 +0100427 // Checks dimension consistency for input and output tensors.
telsoa014fcda012018-03-09 14:13:49 +0000428 for (unsigned int dimIndex = 0; dimIndex < dimensionCount; ++dimIndex)
429 {
430 unsigned int outputShape[dimensionCount];
431 for (unsigned int i = 0; i < dimensionCount; ++i)
432 {
433 outputShape[i] = input0Shape[i];
434 }
435
436 ++outputShape[dimIndex];
437
438 input0TensorInfo = armnn::TensorInfo(dimensionCount, input0Shape, armnn::DataType::Float32);
439 input1TensorInfo = armnn::TensorInfo(dimensionCount, input0Shape, armnn::DataType::Float32);
440 outputTensorInfo = armnn::TensorInfo(dimensionCount, outputShape, armnn::DataType::Float32);
441
442 MultiplicationQueueDescriptor invalidData;
443 WorkloadInfo invalidInfo;
444
445 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
446 AddInputToWorkload(invalidData, invalidInfo, input0TensorInfo, nullptr);
447 AddInputToWorkload(invalidData, invalidInfo, input1TensorInfo, nullptr);
448
Sadik Armagan1625efc2021-06-10 18:24:34 +0100449 CHECK_THROWS_AS(RefMultiplicationWorkload<>(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000450 }
451}
452
Sadik Armagan1625efc2021-06-10 18:24:34 +0100453TEST_CASE("ReshapeQueueDescriptor_Validate_MismatchingNumElements")
telsoa014fcda012018-03-09 14:13:49 +0000454{
455 armnn::TensorInfo inputTensorInfo;
456 armnn::TensorInfo outputTensorInfo;
457
telsoa01c577f2c2018-08-31 09:22:23 +0100458 // The input and output shapes should have the same number of elements, but these don't.
telsoa014fcda012018-03-09 14:13:49 +0000459 unsigned int inputShape[] = { 1, 1, 2, 3 };
460 unsigned int outputShape[] = { 1, 1, 1, 2 };
461
462 inputTensorInfo = armnn::TensorInfo(4, inputShape, armnn::DataType::Float32);
463 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
464
465 ReshapeQueueDescriptor invalidData;
466 WorkloadInfo invalidInfo;
467
468 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
469 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
470
telsoa01c577f2c2018-08-31 09:22:23 +0100471 // InvalidArgumentException is expected, because the number of elements don't match.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100472 CHECK_THROWS_AS(RefReshapeWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000473}
474
telsoa01c577f2c2018-08-31 09:22:23 +0100475
Sadik Armagan1625efc2021-06-10 18:24:34 +0100476TEST_CASE("LstmQueueDescriptor_Validate")
telsoa01c577f2c2018-08-31 09:22:23 +0100477{
Jan Eilers38e05bd2019-06-26 13:10:09 +0100478 armnn::DataType dataType = armnn::DataType::Float32;
telsoa01c577f2c2018-08-31 09:22:23 +0100479
Jan Eilers38e05bd2019-06-26 13:10:09 +0100480 float qScale = 0.0f;
481 int32_t qOffset = 0;
telsoa01c577f2c2018-08-31 09:22:23 +0100482
Jan Eilers38e05bd2019-06-26 13:10:09 +0100483 unsigned int batchSize = 2;
484 unsigned int outputSize = 3;
485 unsigned int inputSize = 5;
486 unsigned numUnits = 4;
telsoa01c577f2c2018-08-31 09:22:23 +0100487
Jan Eilers38e05bd2019-06-26 13:10:09 +0100488 armnn::TensorInfo inputTensorInfo({batchSize , inputSize}, dataType, qScale, qOffset );
489 armnn::TensorInfo outputStateInTensorInfo({batchSize , outputSize}, dataType, qScale, qOffset);
490 armnn::TensorInfo cellStateInTensorInfo({batchSize , numUnits}, dataType, qScale, qOffset);
telsoa01c577f2c2018-08-31 09:22:23 +0100491
Jan Eilers38e05bd2019-06-26 13:10:09 +0100492 // Scratch buffer size with CIFG [batchSize, numUnits * 4]
493 armnn::TensorInfo scratchBufferTensorInfo({batchSize, numUnits * 4}, dataType, qScale, qOffset);
494 armnn::TensorInfo cellStateOutTensorInfo({batchSize, numUnits}, dataType, qScale, qOffset);
495 armnn::TensorInfo outputStateOutTensorInfo({batchSize, outputSize}, dataType, qScale, qOffset);
496 armnn::TensorInfo outputTensorInfo({batchSize, outputSize}, dataType, qScale, qOffset);
telsoa01c577f2c2018-08-31 09:22:23 +0100497
Jan Eilers38e05bd2019-06-26 13:10:09 +0100498 armnn::TensorInfo tensorInfo3({outputSize}, dataType, qScale, qOffset);
499 armnn::TensorInfo tensorInfo4({numUnits}, dataType, qScale, qOffset);
500 armnn::TensorInfo tensorInfo4x5({numUnits, inputSize}, dataType, qScale, qOffset);
501 armnn::TensorInfo tensorInfo4x3({numUnits, outputSize}, dataType, qScale, qOffset);
502 armnn::TensorInfo tensorInfo3x4({outputSize, numUnits}, dataType, qScale, qOffset);
503
504 LstmQueueDescriptor data;
505 WorkloadInfo info;
506
507 AddInputToWorkload(data, info, inputTensorInfo, nullptr);
508 AddInputToWorkload(data, info, outputStateInTensorInfo, nullptr);
509 AddInputToWorkload(data, info, cellStateInTensorInfo, nullptr);
510
511 AddOutputToWorkload(data, info, scratchBufferTensorInfo, nullptr);
512 AddOutputToWorkload(data, info, outputStateOutTensorInfo, nullptr);
513 AddOutputToWorkload(data, info, cellStateOutTensorInfo, nullptr);
514 // AddOutputToWorkload(data, info, outputTensorInfo, nullptr); is left out
515
James Conroy1f58f032021-04-27 17:13:27 +0100516 armnn::ScopedTensorHandle inputToInputWeightsTensor(tensorInfo4x5);
517 armnn::ScopedTensorHandle inputToForgetWeightsTensor(tensorInfo4x5);
518 armnn::ScopedTensorHandle inputToCellWeightsTensor(tensorInfo4x5);
519 armnn::ScopedTensorHandle inputToOutputWeightsTensor(tensorInfo4x5);
520 armnn::ScopedTensorHandle recurrentToForgetWeightsTensor(tensorInfo4x3);
521 armnn::ScopedTensorHandle recurrentToInputWeightsTensor(tensorInfo4x3);
522 armnn::ScopedTensorHandle recurrentToCellWeightsTensor(tensorInfo4x3);
523 armnn::ScopedTensorHandle recurrentToOutputWeightsTensor(tensorInfo4x3);
524 armnn::ScopedTensorHandle cellToInputWeightsTensor(tensorInfo4);
525 armnn::ScopedTensorHandle inputGateBiasTensor(tensorInfo4);
526 armnn::ScopedTensorHandle forgetGateBiasTensor(tensorInfo4);
527 armnn::ScopedTensorHandle cellBiasTensor(tensorInfo4);
528 armnn::ScopedTensorHandle outputGateBiasTensor(tensorInfo4);
529 armnn::ScopedTensorHandle cellToForgetWeightsTensor(tensorInfo4);
530 armnn::ScopedTensorHandle cellToOutputWeightsTensor(tensorInfo4);
531 armnn::ScopedTensorHandle projectionWeightsTensor(tensorInfo3x4);
532 armnn::ScopedTensorHandle projectionBiasTensor(tensorInfo3);
533 armnn::ScopedTensorHandle inputLayerNormWeightsTensor(tensorInfo4);
534 armnn::ScopedTensorHandle forgetLayerNormWeightsTensor(tensorInfo4);
535 armnn::ScopedTensorHandle cellLayerNormWeightsTensor(tensorInfo4);
536 armnn::ScopedTensorHandle outputLayerNormWeightsTensor(tensorInfo4);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100537
538 data.m_InputToInputWeights = &inputToInputWeightsTensor;
539 data.m_InputToForgetWeights = &inputToForgetWeightsTensor;
540 data.m_InputToCellWeights = &inputToCellWeightsTensor;
541 data.m_InputToOutputWeights = &inputToOutputWeightsTensor;
542 data.m_RecurrentToInputWeights = &recurrentToInputWeightsTensor;
543 data.m_RecurrentToForgetWeights = &recurrentToForgetWeightsTensor;
544 data.m_RecurrentToCellWeights = &recurrentToCellWeightsTensor;
545 data.m_RecurrentToOutputWeights = &recurrentToOutputWeightsTensor;
546 data.m_CellToInputWeights = &cellToInputWeightsTensor;
547 data.m_InputGateBias = &inputGateBiasTensor;
548 data.m_ForgetGateBias = &forgetGateBiasTensor;
549 data.m_CellBias = &cellBiasTensor;
550 data.m_OutputGateBias = &outputGateBiasTensor;
551 data.m_CellToForgetWeights = &cellToForgetWeightsTensor;
552 data.m_CellToOutputWeights = &cellToOutputWeightsTensor;
553 data.m_ProjectionWeights = &projectionWeightsTensor;
554 data.m_ProjectionBias = &projectionBiasTensor;
555
556 data.m_InputLayerNormWeights = &inputLayerNormWeightsTensor;
557 data.m_ForgetLayerNormWeights = &forgetLayerNormWeightsTensor;
558 data.m_CellLayerNormWeights = &cellLayerNormWeightsTensor;
559 data.m_OutputLayerNormWeights = &outputLayerNormWeightsTensor;
560
561 // Flags to set test configuration
562 data.m_Parameters.m_ActivationFunc = 4;
563 data.m_Parameters.m_CifgEnabled = false;
564 data.m_Parameters.m_PeepholeEnabled = true;
565 data.m_Parameters.m_ProjectionEnabled = true;
566 data.m_Parameters.m_LayerNormEnabled = true;
567
568 // check wrong number of outputs
Sadik Armagan1625efc2021-06-10 18:24:34 +0100569 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100570 AddOutputToWorkload(data, info, outputTensorInfo, nullptr);
571
572 // check wrong cifg parameter configuration
573 data.m_Parameters.m_CifgEnabled = true;
574 armnn::TensorInfo scratchBufferTensorInfo2({batchSize, numUnits * 3}, dataType, qScale, qOffset);
575 SetWorkloadOutput(data, info, 0, scratchBufferTensorInfo2, nullptr);
Sadik Armagan1625efc2021-06-10 18:24:34 +0100576 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100577 data.m_Parameters.m_CifgEnabled = false;
578 SetWorkloadOutput(data, info, 0, scratchBufferTensorInfo, nullptr);
579
580 // check wrong inputGateBias configuration
581 data.m_InputGateBias = nullptr;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100582 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100583 data.m_InputGateBias = &inputGateBiasTensor;
584
585 // check inconsistant projection parameters
586 data.m_Parameters.m_ProjectionEnabled = false;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100587 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100588 data.m_Parameters.m_ProjectionEnabled = true;
589 data.m_ProjectionWeights = nullptr;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100590 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100591 data.m_ProjectionWeights = &projectionWeightsTensor;
592
593 // check missing input layer normalisation weights
594 data.m_InputLayerNormWeights = nullptr;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100595 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100596 data.m_InputLayerNormWeights = &inputLayerNormWeightsTensor;
597
598 // layer norm disabled but normalisation weights are present
599 data.m_Parameters.m_LayerNormEnabled = false;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100600 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100601 data.m_Parameters.m_LayerNormEnabled = true;
602
603 // check invalid outputTensor shape
604 armnn::TensorInfo incorrectOutputTensorInfo({batchSize, outputSize + 1}, dataType, qScale, qOffset);
605 SetWorkloadOutput(data, info, 3, incorrectOutputTensorInfo, nullptr);
Sadik Armagan1625efc2021-06-10 18:24:34 +0100606 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100607 SetWorkloadOutput(data, info, 3, outputTensorInfo, nullptr);
608
janeil0117d8d852019-11-15 15:00:16 +0000609 // check invalid cell clipping parameters
610 data.m_Parameters.m_ClippingThresCell = -1.0f;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100611 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
janeil0117d8d852019-11-15 15:00:16 +0000612 data.m_Parameters.m_ClippingThresCell = 0.0f;
613
614 // check invalid projection clipping parameters
615 data.m_Parameters.m_ClippingThresProj = -1.0f;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100616 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
janeil0117d8d852019-11-15 15:00:16 +0000617 data.m_Parameters.m_ClippingThresProj = 0.0f;
618
Jan Eilers38e05bd2019-06-26 13:10:09 +0100619 // check correct configuration
Sadik Armagan1625efc2021-06-10 18:24:34 +0100620 CHECK_NOTHROW(data.Validate(info));
telsoa01c577f2c2018-08-31 09:22:23 +0100621}
622
Sadik Armagan1625efc2021-06-10 18:24:34 +0100623TEST_CASE("BiasPerAxisQuantization_Validate")
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000624{
625 constexpr unsigned int nInput = 1u;
626 constexpr unsigned int cInput = 3u;
627 constexpr unsigned int hInput = 3u;
628 constexpr unsigned int wInput = 3u;
629
630 constexpr unsigned int nOutput = nInput;
631 constexpr unsigned int cOutput = cInput;
632 constexpr unsigned int hOutput = 1u;
633 constexpr unsigned int wOutput = 1u;
634
635 const TensorShape inputShape { nInput, cInput, hInput, wInput };
636 const TensorShape outputShape{ nOutput, cOutput, hOutput, wOutput };
637 const TensorShape weightShape{ cOutput, cInput, hInput, wInput };
638 const TensorShape biasShape { cOutput };
639
Derek Lambertif90c56d2020-01-10 17:14:08 +0000640 constexpr DataType inputType = DataType::QAsymmU8;
Derek Lambertid466a542020-01-22 15:37:29 +0000641 constexpr DataType weightType = DataType::QSymmS8;
Aron Virginas-Tar5edc8812019-11-05 18:00:21 +0000642 constexpr DataType biasType = DataType::Signed32;
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000643
644 constexpr float perTensorScale = 1.5f;
Aron Virginas-Tar5edc8812019-11-05 18:00:21 +0000645 const TensorInfo inputInfo (inputShape, inputType, perTensorScale);
646 const TensorInfo outputInfo(outputShape, inputType, perTensorScale);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000647
648 const std::vector<float> weightPerAxisScales = { 2.50f, 3.50f };
Aron Virginas-Tar5edc8812019-11-05 18:00:21 +0000649 const TensorInfo weightInfo(weightShape, weightType, weightPerAxisScales, 0);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000650
651 Convolution2dQueueDescriptor queueDescriptor;
652 queueDescriptor.m_Parameters.m_BiasEnabled = true;
653
654 WorkloadInfo workloadInfo;
655 AddInputToWorkload(queueDescriptor, workloadInfo, inputInfo, nullptr);
656 AddOutputToWorkload(queueDescriptor, workloadInfo, outputInfo, nullptr);
657
James Conroy1f58f032021-04-27 17:13:27 +0100658 ScopedTensorHandle weightTensor(weightInfo);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000659 queueDescriptor.m_Weight = &weightTensor;
660
661 // Test 1: correct per-axis quantization values
662 const std::vector<float> biasPerAxisScales1 = { 3.75f, 5.25f };
663 const TensorInfo biasInfo1(biasShape, biasType, biasPerAxisScales1, 0);
664
James Conroy1f58f032021-04-27 17:13:27 +0100665 ScopedTensorHandle biasHandle1(biasInfo1);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000666 queueDescriptor.m_Bias = &biasHandle1;
667
Sadik Armagan1625efc2021-06-10 18:24:34 +0100668 CHECK_NOTHROW(queueDescriptor.Validate(workloadInfo));
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000669
670 // Test 2: wrong per-axis quantization values
671 const std::vector<float> biasPerAxisScales2 = { 4.00f, 5.00f };
672 const TensorInfo biasInfo2(biasShape, biasType, biasPerAxisScales2, 0);
673
James Conroy1f58f032021-04-27 17:13:27 +0100674 ScopedTensorHandle biasHandle2(biasInfo2);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000675 queueDescriptor.m_Bias = &biasHandle2;
676
Sadik Armagan1625efc2021-06-10 18:24:34 +0100677 CHECK_NOTHROW(queueDescriptor.Validate(workloadInfo));
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000678
679 // Test 3: mismatched number of quantization scales
680 const std::vector<float> biasPerAxisScales3 = { 3.75f, 5.25f, 5.25f };
681 const TensorInfo biasInfo3(biasShape, biasType, biasPerAxisScales3, 0);
682
James Conroy1f58f032021-04-27 17:13:27 +0100683 ScopedTensorHandle biasHandle3(biasInfo3);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000684 queueDescriptor.m_Bias = &biasHandle3;
685
Sadik Armagan1625efc2021-06-10 18:24:34 +0100686 CHECK_THROWS_AS(queueDescriptor.Validate(workloadInfo), InvalidArgumentException);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000687}
688
Sadik Armagan1625efc2021-06-10 18:24:34 +0100689}