blob: fed21eb91122773b9718389417d20835892fa62f [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
Colm Donelan0c479742021-12-10 12:43:54 +00006#include <armnnTestUtils/WorkloadTestUtils.hpp>
telsoa014fcda012018-03-09 14:13:49 +00007
8#include <armnn/Exceptions.hpp>
9
Colm Donelan0c479742021-12-10 12:43:54 +000010#include <armnn/backends/TensorHandle.hpp>
11#include <armnn/backends/Workload.hpp>
Aron Virginas-Tarc9cc8042018-11-01 16:15:57 +000012
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.
Teresa Charlin611c7fb2022-01-07 09:47:29 +000053 CHECK_THROWS_AS(RefWorkloadFactory().CreateWorkload(LayerType::Input, invalidData, invalidInfo),
54 armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +000055}
56
Sadik Armagan1625efc2021-06-10 18:24:34 +010057TEST_CASE("RefPooling2dFloat32Workload_Validate_WrongDimTensor")
telsoa014fcda012018-03-09 14:13:49 +000058{
59 armnn::TensorInfo inputTensorInfo;
60 armnn::TensorInfo outputTensorInfo;
61
telsoa01c577f2c2018-08-31 09:22:23 +010062 unsigned int inputShape[] = {2, 3, 4}; // <- Invalid - input tensor has to be 4D.
telsoa014fcda012018-03-09 14:13:49 +000063 unsigned int outputShape[] = {2, 3, 4, 5};
64
65 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
66 inputTensorInfo = armnn::TensorInfo(3, inputShape, armnn::DataType::Float32);
67
68 Pooling2dQueueDescriptor invalidData;
69 WorkloadInfo invalidInfo;
70
71 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
72 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
73
telsoa01c577f2c2018-08-31 09:22:23 +010074 // Invalid argument exception is expected, input tensor has to be 4D.
Sadik Armagan1625efc2021-06-10 18:24:34 +010075 CHECK_THROWS_AS(RefPooling2dWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +000076}
77
Tamás Nyíri7b885b32021-10-26 14:47:57 +010078TEST_CASE("RefPooling3dFloat32Workload_Validate_WrongDimTensor")
79{
80 armnn::TensorInfo inputTensorInfo;
81 armnn::TensorInfo outputTensorInfo;
82
83 unsigned int inputShape[] = {2, 3, 4, 5}; // <- Invalid - input tensor has to be 5D.
84 unsigned int outputShape[] = {2, 3, 4, 5, 6};
85
86 outputTensorInfo = armnn::TensorInfo(5, outputShape, armnn::DataType::Float32);
87 inputTensorInfo = armnn::TensorInfo(4, inputShape, armnn::DataType::Float32);
88
89 Pooling3dQueueDescriptor invalidData;
90 WorkloadInfo invalidInfo;
91
92 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
93 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
94
95 // Invalid argument exception is expected, input tensor has to be 5D.
96 CHECK_THROWS_AS(RefPooling3dWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
97}
98
Sadik Armagan1625efc2021-06-10 18:24:34 +010099TEST_CASE("SoftmaxQueueDescriptor_Validate_WrongInputHeight")
telsoa014fcda012018-03-09 14:13:49 +0000100{
101 unsigned int inputHeight = 1;
102 unsigned int inputWidth = 1;
103 unsigned int inputChannels = 4;
104 unsigned int inputNum = 2;
105
106 unsigned int outputChannels = inputChannels;
telsoa01c577f2c2018-08-31 09:22:23 +0100107 unsigned int outputHeight = inputHeight + 1; //Makes data invalid - Softmax expects height and width to be 1.
telsoa014fcda012018-03-09 14:13:49 +0000108 unsigned int outputWidth = inputWidth;
109 unsigned int outputNum = inputNum;
110
111 armnn::TensorInfo inputTensorInfo;
112 armnn::TensorInfo outputTensorInfo;
113
114 unsigned int inputShape[] = { inputNum, inputChannels, inputHeight, inputWidth };
115 unsigned int outputShape[] = { outputNum, outputChannels, outputHeight, outputWidth };
116
117 inputTensorInfo = armnn::TensorInfo(4, inputShape, armnn::DataType::Float32);
118 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
119
120 SoftmaxQueueDescriptor invalidData;
121 WorkloadInfo invalidInfo;
122
123 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
124 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
125
telsoa01c577f2c2018-08-31 09:22:23 +0100126 //Invalid argument exception is expected, because height != 1.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100127 CHECK_THROWS_AS(RefSoftmaxWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000128}
129
Sadik Armagan1625efc2021-06-10 18:24:34 +0100130TEST_CASE("FullyConnectedQueueDescriptor_Validate_RequiredDataMissing")
telsoa014fcda012018-03-09 14:13:49 +0000131{
132 unsigned int inputWidth = 1;
133 unsigned int inputHeight = 1;
134 unsigned int inputChannels = 5;
135 unsigned int inputNum = 2;
136
137 unsigned int outputWidth = 1;
138 unsigned int outputHeight = 1;
139 unsigned int outputChannels = 3;
140 unsigned int outputNum = 2;
141
telsoa01c577f2c2018-08-31 09:22:23 +0100142 // Define the tensor descriptors.
telsoa014fcda012018-03-09 14:13:49 +0000143 armnn::TensorInfo inputTensorInfo;
144 armnn::TensorInfo outputTensorInfo;
145 armnn::TensorInfo weightsDesc;
146 armnn::TensorInfo biasesDesc;
147
148 unsigned int inputShape[] = { inputNum, inputChannels, inputHeight, inputWidth };
149 unsigned int outputShape[] = { outputNum, outputChannels, outputHeight, outputWidth };
150 unsigned int weightsShape[] = { 1, 1, inputChannels, outputChannels };
151 unsigned int biasShape[] = { 1, outputChannels, outputHeight, outputWidth };
152
153 inputTensorInfo = armnn::TensorInfo(4, inputShape, armnn::DataType::Float32);
154 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
155 weightsDesc = armnn::TensorInfo(4, weightsShape, armnn::DataType::Float32);
156 biasesDesc = armnn::TensorInfo(4, biasShape, armnn::DataType::Float32);
157
158 FullyConnectedQueueDescriptor invalidData;
159 WorkloadInfo invalidInfo;
160
James Conroy1f58f032021-04-27 17:13:27 +0100161 ScopedTensorHandle weightTensor(weightsDesc);
162 ScopedTensorHandle biasTensor(biasesDesc);
telsoa014fcda012018-03-09 14:13:49 +0000163
164 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
165 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
166 invalidData.m_Weight = &weightTensor;
167 invalidData.m_Bias = &biasTensor;
168 invalidData.m_Parameters.m_BiasEnabled = true;
169 invalidData.m_Parameters.m_TransposeWeightMatrix = false;
170
171
telsoa01c577f2c2018-08-31 09:22:23 +0100172 //Invalid argument exception is expected, because not all required fields have been provided.
173 //In particular inputsData[0], outputsData[0] and weightsData can not be null.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100174 CHECK_THROWS_AS(RefFullyConnectedWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000175}
176
177
Sadik Armagan1625efc2021-06-10 18:24:34 +0100178TEST_CASE("NormalizationQueueDescriptor_Validate_WrongInputHeight")
telsoa014fcda012018-03-09 14:13:49 +0000179{
180 constexpr unsigned int inputNum = 5;
181 constexpr unsigned int inputHeight = 32;
182 constexpr unsigned int inputWidth = 24;
183 constexpr unsigned int inputChannels = 3;
184
185 constexpr unsigned int outputNum = inputNum;
186 constexpr unsigned int outputChannels = inputChannels;
telsoa01c577f2c2018-08-31 09:22:23 +0100187 constexpr unsigned int outputHeight = inputHeight + 1; //Makes data invalid - normalization requires.
188 //Input and output to have the same dimensions.
telsoa014fcda012018-03-09 14:13:49 +0000189 constexpr unsigned int outputWidth = inputWidth;
190
191
192 armnn::TensorInfo inputTensorInfo;
193 armnn::TensorInfo outputTensorInfo;
194
195 unsigned int inputShape[] = {inputNum, inputChannels, inputHeight, inputWidth};
196 unsigned int outputShape[] = {outputNum, outputChannels, outputHeight, outputWidth};
197
198 inputTensorInfo = armnn::TensorInfo(4, inputShape, armnn::DataType::Float32);
199 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
200
201
202 armnn::NormalizationAlgorithmMethod normMethod = armnn::NormalizationAlgorithmMethod::LocalBrightness;
203 armnn::NormalizationAlgorithmChannel normChannel = armnn::NormalizationAlgorithmChannel::Across;
204 float alpha = 1.f;
205 float beta = 1.f;
206 float kappa = 1.f;
207 uint32_t normSize = 5;
208
209 NormalizationQueueDescriptor invalidData;
210 WorkloadInfo invalidInfo;
211
212 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
213 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
214 invalidData.m_Parameters.m_NormChannelType = normChannel;
215 invalidData.m_Parameters.m_NormMethodType = normMethod;
216 invalidData.m_Parameters.m_NormSize = normSize;
217 invalidData.m_Parameters.m_Alpha = alpha;
218 invalidData.m_Parameters.m_Beta = beta;
219 invalidData.m_Parameters.m_K = kappa;
220
telsoa01c577f2c2018-08-31 09:22:23 +0100221 //Invalid argument exception is expected, because input height != output height.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100222 CHECK_THROWS_AS(RefNormalizationWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000223}
224
Sadik Armagan1625efc2021-06-10 18:24:34 +0100225TEST_CASE("SplitterQueueDescriptor_Validate_WrongWindow")
telsoa014fcda012018-03-09 14:13:49 +0000226{
227 constexpr unsigned int inputNum = 1;
228 constexpr unsigned int inputHeight = 32;
229 constexpr unsigned int inputWidth = 24;
230 constexpr unsigned int inputChannels = 3;
231
232 constexpr unsigned int outputNum = inputNum;
233 constexpr unsigned int outputChannels = inputChannels;
234 constexpr unsigned int outputHeight = 18;
235 constexpr unsigned int outputWidth = inputWidth;
236
237
238 armnn::TensorInfo inputTensorInfo;
239 armnn::TensorInfo outputTensorInfo;
240
241 unsigned int inputShape[] = {inputNum, inputChannels, inputHeight, inputWidth};
242 unsigned int outputShape[] = {outputNum, outputChannels, outputHeight, outputWidth};
243
244 inputTensorInfo = armnn::TensorInfo(4, inputShape, armnn::DataType::Float32);
245 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
246
247 SplitterQueueDescriptor invalidData;
248 WorkloadInfo invalidInfo;
249
250 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
251 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
252
telsoa01c577f2c2018-08-31 09:22:23 +0100253 // Invalid, since it has only 3 dimensions while the input tensor is 4d.
telsoa014fcda012018-03-09 14:13:49 +0000254 std::vector<unsigned int> wOrigin = {0, 0, 0};
255 armnn::SplitterQueueDescriptor::ViewOrigin window(wOrigin);
256 invalidData.m_ViewOrigins.push_back(window);
257
Sadik Armagan1625efc2021-06-10 18:24:34 +0100258 INFO("Invalid argument exception is expected, because split window dimensionality does not match input.");
259 CHECK_THROWS_AS(RefSplitterWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000260
telsoa01c577f2c2018-08-31 09:22:23 +0100261 // Invalid, since window extends past the boundary of input tensor.
telsoa014fcda012018-03-09 14:13:49 +0000262 std::vector<unsigned int> wOrigin3 = {0, 0, 15, 0};
263 armnn::SplitterQueueDescriptor::ViewOrigin window3(wOrigin3);
264 invalidData.m_ViewOrigins[0] = window3;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100265 INFO("Invalid argument exception is expected (wOrigin3[2]+ outputHeight > inputHeight");
266 CHECK_THROWS_AS(RefSplitterWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000267
268
269 std::vector<unsigned int> wOrigin4 = {0, 0, 0, 0};
270 armnn::SplitterQueueDescriptor::ViewOrigin window4(wOrigin4);
271 invalidData.m_ViewOrigins[0] = window4;
272
273 std::vector<unsigned int> wOrigin5 = {1, 16, 20, 2};
274 armnn::SplitterQueueDescriptor::ViewOrigin window5(wOrigin4);
275 invalidData.m_ViewOrigins.push_back(window5);
276
Sadik Armagan1625efc2021-06-10 18:24:34 +0100277 INFO("Invalid exception due to number of split windows not matching number of outputs.");
278 CHECK_THROWS_AS(RefSplitterWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000279}
280
281
Sadik Armagan1625efc2021-06-10 18:24:34 +0100282TEST_CASE("ConcatQueueDescriptor_Validate_WrongWindow")
telsoa014fcda012018-03-09 14:13:49 +0000283{
284 constexpr unsigned int inputNum = 1;
285 constexpr unsigned int inputChannels = 3;
286 constexpr unsigned int inputHeight = 32;
287 constexpr unsigned int inputWidth = 24;
288
289 constexpr unsigned int outputNum = 1;
290 constexpr unsigned int outputChannels = 3;
291 constexpr unsigned int outputHeight = 32;
292 constexpr unsigned int outputWidth = 24;
293
294
295 armnn::TensorInfo inputTensorInfo;
296 armnn::TensorInfo outputTensorInfo;
297
298 unsigned int inputShape[] = {inputNum, inputChannels, inputHeight, inputWidth};
299 unsigned int outputShape[] = {outputNum, outputChannels, outputHeight, outputWidth};
300
301 inputTensorInfo = armnn::TensorInfo(4, inputShape, armnn::DataType::Float32);
302 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
303
Jim Flynne242f2d2019-05-22 14:24:13 +0100304 ConcatQueueDescriptor invalidData;
telsoa014fcda012018-03-09 14:13:49 +0000305 WorkloadInfo invalidInfo;
306
307 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
308 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
309
telsoa01c577f2c2018-08-31 09:22:23 +0100310 // Invalid, since it has only 3 dimensions while the input tensor is 4d.
telsoa014fcda012018-03-09 14:13:49 +0000311 std::vector<unsigned int> wOrigin = {0, 0, 0};
Jim Flynne242f2d2019-05-22 14:24:13 +0100312 armnn::ConcatQueueDescriptor::ViewOrigin window(wOrigin);
telsoa014fcda012018-03-09 14:13:49 +0000313 invalidData.m_ViewOrigins.push_back(window);
314
Sadik Armagan1625efc2021-06-10 18:24:34 +0100315 INFO("Invalid argument exception is expected, because merge window dimensionality does not match input.");
316 CHECK_THROWS_AS(RefConcatWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000317
telsoa01c577f2c2018-08-31 09:22:23 +0100318 // Invalid, since window extends past the boundary of output tensor.
telsoa014fcda012018-03-09 14:13:49 +0000319 std::vector<unsigned int> wOrigin3 = {0, 0, 15, 0};
Jim Flynne242f2d2019-05-22 14:24:13 +0100320 armnn::ConcatQueueDescriptor::ViewOrigin window3(wOrigin3);
telsoa014fcda012018-03-09 14:13:49 +0000321 invalidData.m_ViewOrigins[0] = window3;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100322 INFO("Invalid argument exception is expected (wOrigin3[2]+ inputHeight > outputHeight");
323 CHECK_THROWS_AS(RefConcatWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000324
325
326 std::vector<unsigned int> wOrigin4 = {0, 0, 0, 0};
Jim Flynne242f2d2019-05-22 14:24:13 +0100327 armnn::ConcatQueueDescriptor::ViewOrigin window4(wOrigin4);
telsoa014fcda012018-03-09 14:13:49 +0000328 invalidData.m_ViewOrigins[0] = window4;
329
330 std::vector<unsigned int> wOrigin5 = {1, 16, 20, 2};
Jim Flynne242f2d2019-05-22 14:24:13 +0100331 armnn::ConcatQueueDescriptor::ViewOrigin window5(wOrigin4);
telsoa014fcda012018-03-09 14:13:49 +0000332 invalidData.m_ViewOrigins.push_back(window5);
333
Sadik Armagan1625efc2021-06-10 18:24:34 +0100334 INFO("Invalid exception due to number of merge windows not matching number of inputs.");
335 CHECK_THROWS_AS(RefConcatWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000336}
337
Sadik Armagan1625efc2021-06-10 18:24:34 +0100338TEST_CASE("AdditionQueueDescriptor_Validate_InputNumbers")
telsoa014fcda012018-03-09 14:13:49 +0000339{
340 armnn::TensorInfo input1TensorInfo;
341 armnn::TensorInfo input2TensorInfo;
342 armnn::TensorInfo input3TensorInfo;
343 armnn::TensorInfo outputTensorInfo;
344
345 unsigned int shape[] = {1, 1, 1, 1};
346
347 input1TensorInfo = armnn::TensorInfo(4, shape, armnn::DataType::Float32);
348 input2TensorInfo = armnn::TensorInfo(4, shape, armnn::DataType::Float32);
349 input3TensorInfo = armnn::TensorInfo(4, shape, armnn::DataType::Float32);
350 outputTensorInfo = armnn::TensorInfo(4, shape, armnn::DataType::Float32);
351
352 AdditionQueueDescriptor invalidData;
353 WorkloadInfo invalidInfo;
354
355 AddInputToWorkload(invalidData, invalidInfo, input1TensorInfo, nullptr);
356 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
357
telsoa01c577f2c2018-08-31 09:22:23 +0100358 // Too few inputs.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100359 CHECK_THROWS_AS(RefAdditionWorkload<>(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000360
361 AddInputToWorkload(invalidData, invalidInfo, input2TensorInfo, nullptr);
362
telsoa01c577f2c2018-08-31 09:22:23 +0100363 // Correct.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100364 CHECK_NOTHROW(RefAdditionWorkload<>(invalidData, invalidInfo));
telsoa014fcda012018-03-09 14:13:49 +0000365
366 AddInputToWorkload(invalidData, invalidInfo, input3TensorInfo, nullptr);
367
telsoa01c577f2c2018-08-31 09:22:23 +0100368 // Too many inputs.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100369 CHECK_THROWS_AS(RefAdditionWorkload<>(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000370}
371
Sadik Armagan1625efc2021-06-10 18:24:34 +0100372TEST_CASE("AdditionQueueDescriptor_Validate_InputShapes")
telsoa014fcda012018-03-09 14:13:49 +0000373{
374 armnn::TensorInfo input1TensorInfo;
375 armnn::TensorInfo input2TensorInfo;
376 armnn::TensorInfo outputTensorInfo;
377
378 unsigned int shape1[] = {1, 1, 2, 1};
379 unsigned int shape2[] = {1, 1, 3, 2};
380
telsoa01c577f2c2018-08-31 09:22:23 +0100381 // Incompatible shapes even with broadcasting.
telsoa014fcda012018-03-09 14:13:49 +0000382 {
383 input1TensorInfo = armnn::TensorInfo(4, shape1, armnn::DataType::Float32);
384 input2TensorInfo = armnn::TensorInfo(4, shape2, armnn::DataType::Float32);
385 outputTensorInfo = armnn::TensorInfo(4, shape1, armnn::DataType::Float32);
386
387 AdditionQueueDescriptor invalidData;
388 WorkloadInfo invalidInfo;
389
390 AddInputToWorkload(invalidData, invalidInfo, input1TensorInfo, nullptr);
391 AddInputToWorkload(invalidData, invalidInfo, input2TensorInfo, nullptr);
392 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
393
Sadik Armagan1625efc2021-06-10 18:24:34 +0100394 CHECK_THROWS_AS(RefAdditionWorkload<>(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000395 }
396
telsoa01c577f2c2018-08-31 09:22:23 +0100397 // Output size not compatible with input sizes.
telsoa014fcda012018-03-09 14:13:49 +0000398 {
399 input1TensorInfo = armnn::TensorInfo(4, shape1, armnn::DataType::Float32);
400 input2TensorInfo = armnn::TensorInfo(4, shape1, armnn::DataType::Float32);
401 outputTensorInfo = armnn::TensorInfo(4, shape2, armnn::DataType::Float32);
402
403 AdditionQueueDescriptor invalidData;
404 WorkloadInfo invalidInfo;
405
406 AddInputToWorkload(invalidData, invalidInfo, input1TensorInfo, nullptr);
407 AddInputToWorkload(invalidData, invalidInfo, input2TensorInfo, nullptr);
408 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
409
telsoa01c577f2c2018-08-31 09:22:23 +0100410 // Output differs.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100411 CHECK_THROWS_AS(RefAdditionWorkload<>(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000412 }
413}
414
Sadik Armagan1625efc2021-06-10 18:24:34 +0100415TEST_CASE("MultiplicationQueueDescriptor_Validate_InputTensorDimensionMismatch")
telsoa014fcda012018-03-09 14:13:49 +0000416{
417 armnn::TensorInfo input0TensorInfo;
418 armnn::TensorInfo input1TensorInfo;
419 armnn::TensorInfo outputTensorInfo;
420
421 constexpr unsigned int input0Shape[] = { 2, 2, 4, 4 };
422 constexpr std::size_t dimensionCount = std::extent<decltype(input0Shape)>::value;
423
telsoa01c577f2c2018-08-31 09:22:23 +0100424 // Checks dimension consistency for input tensors.
telsoa014fcda012018-03-09 14:13:49 +0000425 for (unsigned int dimIndex = 0; dimIndex < dimensionCount; ++dimIndex)
426 {
427 unsigned int input1Shape[dimensionCount];
428 for (unsigned int i = 0; i < dimensionCount; ++i)
429 {
430 input1Shape[i] = input0Shape[i];
431 }
432
433 ++input1Shape[dimIndex];
434
435 input0TensorInfo = armnn::TensorInfo(dimensionCount, input0Shape, armnn::DataType::Float32);
436 input1TensorInfo = armnn::TensorInfo(dimensionCount, input1Shape, armnn::DataType::Float32);
437 outputTensorInfo = armnn::TensorInfo(dimensionCount, input0Shape, armnn::DataType::Float32);
438
439 MultiplicationQueueDescriptor invalidData;
440 WorkloadInfo invalidInfo;
441
442 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
443 AddInputToWorkload(invalidData, invalidInfo, input0TensorInfo, nullptr);
444 AddInputToWorkload(invalidData, invalidInfo, input1TensorInfo, nullptr);
445
Sadik Armagan1625efc2021-06-10 18:24:34 +0100446 CHECK_THROWS_AS(RefMultiplicationWorkload<>(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000447 }
448
telsoa01c577f2c2018-08-31 09:22:23 +0100449 // Checks dimension consistency for input and output tensors.
telsoa014fcda012018-03-09 14:13:49 +0000450 for (unsigned int dimIndex = 0; dimIndex < dimensionCount; ++dimIndex)
451 {
452 unsigned int outputShape[dimensionCount];
453 for (unsigned int i = 0; i < dimensionCount; ++i)
454 {
455 outputShape[i] = input0Shape[i];
456 }
457
458 ++outputShape[dimIndex];
459
460 input0TensorInfo = armnn::TensorInfo(dimensionCount, input0Shape, armnn::DataType::Float32);
461 input1TensorInfo = armnn::TensorInfo(dimensionCount, input0Shape, armnn::DataType::Float32);
462 outputTensorInfo = armnn::TensorInfo(dimensionCount, outputShape, armnn::DataType::Float32);
463
464 MultiplicationQueueDescriptor invalidData;
465 WorkloadInfo invalidInfo;
466
467 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
468 AddInputToWorkload(invalidData, invalidInfo, input0TensorInfo, nullptr);
469 AddInputToWorkload(invalidData, invalidInfo, input1TensorInfo, nullptr);
470
Sadik Armagan1625efc2021-06-10 18:24:34 +0100471 CHECK_THROWS_AS(RefMultiplicationWorkload<>(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000472 }
473}
474
Sadik Armagan1625efc2021-06-10 18:24:34 +0100475TEST_CASE("ReshapeQueueDescriptor_Validate_MismatchingNumElements")
telsoa014fcda012018-03-09 14:13:49 +0000476{
477 armnn::TensorInfo inputTensorInfo;
478 armnn::TensorInfo outputTensorInfo;
479
telsoa01c577f2c2018-08-31 09:22:23 +0100480 // The input and output shapes should have the same number of elements, but these don't.
telsoa014fcda012018-03-09 14:13:49 +0000481 unsigned int inputShape[] = { 1, 1, 2, 3 };
482 unsigned int outputShape[] = { 1, 1, 1, 2 };
483
484 inputTensorInfo = armnn::TensorInfo(4, inputShape, armnn::DataType::Float32);
485 outputTensorInfo = armnn::TensorInfo(4, outputShape, armnn::DataType::Float32);
486
487 ReshapeQueueDescriptor invalidData;
488 WorkloadInfo invalidInfo;
489
490 AddInputToWorkload(invalidData, invalidInfo, inputTensorInfo, nullptr);
491 AddOutputToWorkload(invalidData, invalidInfo, outputTensorInfo, nullptr);
492
telsoa01c577f2c2018-08-31 09:22:23 +0100493 // InvalidArgumentException is expected, because the number of elements don't match.
Sadik Armagan1625efc2021-06-10 18:24:34 +0100494 CHECK_THROWS_AS(RefReshapeWorkload(invalidData, invalidInfo), armnn::InvalidArgumentException);
telsoa014fcda012018-03-09 14:13:49 +0000495}
496
telsoa01c577f2c2018-08-31 09:22:23 +0100497
Sadik Armagan1625efc2021-06-10 18:24:34 +0100498TEST_CASE("LstmQueueDescriptor_Validate")
telsoa01c577f2c2018-08-31 09:22:23 +0100499{
Jan Eilers38e05bd2019-06-26 13:10:09 +0100500 armnn::DataType dataType = armnn::DataType::Float32;
telsoa01c577f2c2018-08-31 09:22:23 +0100501
Jan Eilers38e05bd2019-06-26 13:10:09 +0100502 float qScale = 0.0f;
503 int32_t qOffset = 0;
telsoa01c577f2c2018-08-31 09:22:23 +0100504
Jan Eilers38e05bd2019-06-26 13:10:09 +0100505 unsigned int batchSize = 2;
506 unsigned int outputSize = 3;
507 unsigned int inputSize = 5;
508 unsigned numUnits = 4;
telsoa01c577f2c2018-08-31 09:22:23 +0100509
Jan Eilers38e05bd2019-06-26 13:10:09 +0100510 armnn::TensorInfo inputTensorInfo({batchSize , inputSize}, dataType, qScale, qOffset );
511 armnn::TensorInfo outputStateInTensorInfo({batchSize , outputSize}, dataType, qScale, qOffset);
512 armnn::TensorInfo cellStateInTensorInfo({batchSize , numUnits}, dataType, qScale, qOffset);
telsoa01c577f2c2018-08-31 09:22:23 +0100513
Jan Eilers38e05bd2019-06-26 13:10:09 +0100514 // Scratch buffer size with CIFG [batchSize, numUnits * 4]
515 armnn::TensorInfo scratchBufferTensorInfo({batchSize, numUnits * 4}, dataType, qScale, qOffset);
516 armnn::TensorInfo cellStateOutTensorInfo({batchSize, numUnits}, dataType, qScale, qOffset);
517 armnn::TensorInfo outputStateOutTensorInfo({batchSize, outputSize}, dataType, qScale, qOffset);
518 armnn::TensorInfo outputTensorInfo({batchSize, outputSize}, dataType, qScale, qOffset);
telsoa01c577f2c2018-08-31 09:22:23 +0100519
Jan Eilers38e05bd2019-06-26 13:10:09 +0100520 armnn::TensorInfo tensorInfo3({outputSize}, dataType, qScale, qOffset);
521 armnn::TensorInfo tensorInfo4({numUnits}, dataType, qScale, qOffset);
522 armnn::TensorInfo tensorInfo4x5({numUnits, inputSize}, dataType, qScale, qOffset);
523 armnn::TensorInfo tensorInfo4x3({numUnits, outputSize}, dataType, qScale, qOffset);
524 armnn::TensorInfo tensorInfo3x4({outputSize, numUnits}, dataType, qScale, qOffset);
525
526 LstmQueueDescriptor data;
527 WorkloadInfo info;
528
529 AddInputToWorkload(data, info, inputTensorInfo, nullptr);
530 AddInputToWorkload(data, info, outputStateInTensorInfo, nullptr);
531 AddInputToWorkload(data, info, cellStateInTensorInfo, nullptr);
532
533 AddOutputToWorkload(data, info, scratchBufferTensorInfo, nullptr);
534 AddOutputToWorkload(data, info, outputStateOutTensorInfo, nullptr);
535 AddOutputToWorkload(data, info, cellStateOutTensorInfo, nullptr);
536 // AddOutputToWorkload(data, info, outputTensorInfo, nullptr); is left out
537
James Conroy1f58f032021-04-27 17:13:27 +0100538 armnn::ScopedTensorHandle inputToInputWeightsTensor(tensorInfo4x5);
539 armnn::ScopedTensorHandle inputToForgetWeightsTensor(tensorInfo4x5);
540 armnn::ScopedTensorHandle inputToCellWeightsTensor(tensorInfo4x5);
541 armnn::ScopedTensorHandle inputToOutputWeightsTensor(tensorInfo4x5);
542 armnn::ScopedTensorHandle recurrentToForgetWeightsTensor(tensorInfo4x3);
543 armnn::ScopedTensorHandle recurrentToInputWeightsTensor(tensorInfo4x3);
544 armnn::ScopedTensorHandle recurrentToCellWeightsTensor(tensorInfo4x3);
545 armnn::ScopedTensorHandle recurrentToOutputWeightsTensor(tensorInfo4x3);
546 armnn::ScopedTensorHandle cellToInputWeightsTensor(tensorInfo4);
547 armnn::ScopedTensorHandle inputGateBiasTensor(tensorInfo4);
548 armnn::ScopedTensorHandle forgetGateBiasTensor(tensorInfo4);
549 armnn::ScopedTensorHandle cellBiasTensor(tensorInfo4);
550 armnn::ScopedTensorHandle outputGateBiasTensor(tensorInfo4);
551 armnn::ScopedTensorHandle cellToForgetWeightsTensor(tensorInfo4);
552 armnn::ScopedTensorHandle cellToOutputWeightsTensor(tensorInfo4);
553 armnn::ScopedTensorHandle projectionWeightsTensor(tensorInfo3x4);
554 armnn::ScopedTensorHandle projectionBiasTensor(tensorInfo3);
555 armnn::ScopedTensorHandle inputLayerNormWeightsTensor(tensorInfo4);
556 armnn::ScopedTensorHandle forgetLayerNormWeightsTensor(tensorInfo4);
557 armnn::ScopedTensorHandle cellLayerNormWeightsTensor(tensorInfo4);
558 armnn::ScopedTensorHandle outputLayerNormWeightsTensor(tensorInfo4);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100559
560 data.m_InputToInputWeights = &inputToInputWeightsTensor;
561 data.m_InputToForgetWeights = &inputToForgetWeightsTensor;
562 data.m_InputToCellWeights = &inputToCellWeightsTensor;
563 data.m_InputToOutputWeights = &inputToOutputWeightsTensor;
564 data.m_RecurrentToInputWeights = &recurrentToInputWeightsTensor;
565 data.m_RecurrentToForgetWeights = &recurrentToForgetWeightsTensor;
566 data.m_RecurrentToCellWeights = &recurrentToCellWeightsTensor;
567 data.m_RecurrentToOutputWeights = &recurrentToOutputWeightsTensor;
568 data.m_CellToInputWeights = &cellToInputWeightsTensor;
569 data.m_InputGateBias = &inputGateBiasTensor;
570 data.m_ForgetGateBias = &forgetGateBiasTensor;
571 data.m_CellBias = &cellBiasTensor;
572 data.m_OutputGateBias = &outputGateBiasTensor;
573 data.m_CellToForgetWeights = &cellToForgetWeightsTensor;
574 data.m_CellToOutputWeights = &cellToOutputWeightsTensor;
575 data.m_ProjectionWeights = &projectionWeightsTensor;
576 data.m_ProjectionBias = &projectionBiasTensor;
577
578 data.m_InputLayerNormWeights = &inputLayerNormWeightsTensor;
579 data.m_ForgetLayerNormWeights = &forgetLayerNormWeightsTensor;
580 data.m_CellLayerNormWeights = &cellLayerNormWeightsTensor;
581 data.m_OutputLayerNormWeights = &outputLayerNormWeightsTensor;
582
583 // Flags to set test configuration
584 data.m_Parameters.m_ActivationFunc = 4;
585 data.m_Parameters.m_CifgEnabled = false;
586 data.m_Parameters.m_PeepholeEnabled = true;
587 data.m_Parameters.m_ProjectionEnabled = true;
588 data.m_Parameters.m_LayerNormEnabled = true;
589
590 // check wrong number of outputs
Sadik Armagan1625efc2021-06-10 18:24:34 +0100591 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100592 AddOutputToWorkload(data, info, outputTensorInfo, nullptr);
593
594 // check wrong cifg parameter configuration
595 data.m_Parameters.m_CifgEnabled = true;
596 armnn::TensorInfo scratchBufferTensorInfo2({batchSize, numUnits * 3}, dataType, qScale, qOffset);
597 SetWorkloadOutput(data, info, 0, scratchBufferTensorInfo2, nullptr);
Sadik Armagan1625efc2021-06-10 18:24:34 +0100598 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100599 data.m_Parameters.m_CifgEnabled = false;
600 SetWorkloadOutput(data, info, 0, scratchBufferTensorInfo, nullptr);
601
602 // check wrong inputGateBias configuration
603 data.m_InputGateBias = nullptr;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100604 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100605 data.m_InputGateBias = &inputGateBiasTensor;
606
607 // check inconsistant projection parameters
608 data.m_Parameters.m_ProjectionEnabled = false;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100609 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100610 data.m_Parameters.m_ProjectionEnabled = true;
611 data.m_ProjectionWeights = nullptr;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100612 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100613 data.m_ProjectionWeights = &projectionWeightsTensor;
614
615 // check missing input layer normalisation weights
616 data.m_InputLayerNormWeights = nullptr;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100617 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100618 data.m_InputLayerNormWeights = &inputLayerNormWeightsTensor;
619
620 // layer norm disabled but normalisation weights are present
621 data.m_Parameters.m_LayerNormEnabled = false;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100622 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100623 data.m_Parameters.m_LayerNormEnabled = true;
624
625 // check invalid outputTensor shape
626 armnn::TensorInfo incorrectOutputTensorInfo({batchSize, outputSize + 1}, dataType, qScale, qOffset);
627 SetWorkloadOutput(data, info, 3, incorrectOutputTensorInfo, nullptr);
Sadik Armagan1625efc2021-06-10 18:24:34 +0100628 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
Jan Eilers38e05bd2019-06-26 13:10:09 +0100629 SetWorkloadOutput(data, info, 3, outputTensorInfo, nullptr);
630
janeil0117d8d852019-11-15 15:00:16 +0000631 // check invalid cell clipping parameters
632 data.m_Parameters.m_ClippingThresCell = -1.0f;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100633 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
janeil0117d8d852019-11-15 15:00:16 +0000634 data.m_Parameters.m_ClippingThresCell = 0.0f;
635
636 // check invalid projection clipping parameters
637 data.m_Parameters.m_ClippingThresProj = -1.0f;
Sadik Armagan1625efc2021-06-10 18:24:34 +0100638 CHECK_THROWS_AS(data.Validate(info), armnn::InvalidArgumentException);
janeil0117d8d852019-11-15 15:00:16 +0000639 data.m_Parameters.m_ClippingThresProj = 0.0f;
640
Jan Eilers38e05bd2019-06-26 13:10:09 +0100641 // check correct configuration
Sadik Armagan1625efc2021-06-10 18:24:34 +0100642 CHECK_NOTHROW(data.Validate(info));
telsoa01c577f2c2018-08-31 09:22:23 +0100643}
644
Keith Davisb4dd5cc2022-04-07 11:32:00 +0100645TEST_CASE("BiasPerAxisQuantization_ValidateCorrectValues")
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000646{
647 constexpr unsigned int nInput = 1u;
648 constexpr unsigned int cInput = 3u;
649 constexpr unsigned int hInput = 3u;
650 constexpr unsigned int wInput = 3u;
651
652 constexpr unsigned int nOutput = nInput;
653 constexpr unsigned int cOutput = cInput;
654 constexpr unsigned int hOutput = 1u;
655 constexpr unsigned int wOutput = 1u;
656
657 const TensorShape inputShape { nInput, cInput, hInput, wInput };
658 const TensorShape outputShape{ nOutput, cOutput, hOutput, wOutput };
659 const TensorShape weightShape{ cOutput, cInput, hInput, wInput };
660 const TensorShape biasShape { cOutput };
661
Derek Lambertif90c56d2020-01-10 17:14:08 +0000662 constexpr DataType inputType = DataType::QAsymmU8;
Derek Lambertid466a542020-01-22 15:37:29 +0000663 constexpr DataType weightType = DataType::QSymmS8;
Aron Virginas-Tar5edc8812019-11-05 18:00:21 +0000664 constexpr DataType biasType = DataType::Signed32;
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000665
666 constexpr float perTensorScale = 1.5f;
Aron Virginas-Tar5edc8812019-11-05 18:00:21 +0000667 const TensorInfo inputInfo (inputShape, inputType, perTensorScale);
668 const TensorInfo outputInfo(outputShape, inputType, perTensorScale);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000669
670 const std::vector<float> weightPerAxisScales = { 2.50f, 3.50f };
Aron Virginas-Tar5edc8812019-11-05 18:00:21 +0000671 const TensorInfo weightInfo(weightShape, weightType, weightPerAxisScales, 0);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000672
673 Convolution2dQueueDescriptor queueDescriptor;
674 queueDescriptor.m_Parameters.m_BiasEnabled = true;
675
676 WorkloadInfo workloadInfo;
677 AddInputToWorkload(queueDescriptor, workloadInfo, inputInfo, nullptr);
Keith Davisb4dd5cc2022-04-07 11:32:00 +0100678 AddInputToWorkload(queueDescriptor, workloadInfo, weightInfo, nullptr);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000679 AddOutputToWorkload(queueDescriptor, workloadInfo, outputInfo, nullptr);
680
James Conroy1f58f032021-04-27 17:13:27 +0100681 ScopedTensorHandle weightTensor(weightInfo);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000682 queueDescriptor.m_Weight = &weightTensor;
683
684 // Test 1: correct per-axis quantization values
685 const std::vector<float> biasPerAxisScales1 = { 3.75f, 5.25f };
686 const TensorInfo biasInfo1(biasShape, biasType, biasPerAxisScales1, 0);
687
James Conroy1f58f032021-04-27 17:13:27 +0100688 ScopedTensorHandle biasHandle1(biasInfo1);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000689 queueDescriptor.m_Bias = &biasHandle1;
690
Keith Davisb4dd5cc2022-04-07 11:32:00 +0100691 AddInputToWorkload(queueDescriptor, workloadInfo, biasInfo1, nullptr);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000692
Keith Davisb4dd5cc2022-04-07 11:32:00 +0100693 CHECK_NOTHROW(queueDescriptor.Validate(workloadInfo));
694}
695
696TEST_CASE("BiasPerAxisQuantization_ValidateIncorrectValues")
697{
698 constexpr unsigned int nInput = 1u;
699 constexpr unsigned int cInput = 3u;
700 constexpr unsigned int hInput = 3u;
701 constexpr unsigned int wInput = 3u;
702
703 constexpr unsigned int nOutput = nInput;
704 constexpr unsigned int cOutput = cInput;
705 constexpr unsigned int hOutput = 1u;
706 constexpr unsigned int wOutput = 1u;
707
708 const TensorShape inputShape { nInput, cInput, hInput, wInput };
709 const TensorShape outputShape{ nOutput, cOutput, hOutput, wOutput };
710 const TensorShape weightShape{ cOutput, cInput, hInput, wInput };
711 const TensorShape biasShape { cOutput };
712
713 constexpr DataType inputType = DataType::QAsymmU8;
714 constexpr DataType weightType = DataType::QSymmS8;
715 constexpr DataType biasType = DataType::Signed32;
716
717 constexpr float perTensorScale = 1.5f;
718 const TensorInfo inputInfo (inputShape, inputType, perTensorScale);
719 const TensorInfo outputInfo(outputShape, inputType, perTensorScale);
720
721 const std::vector<float> weightPerAxisScales = { 2.50f, 3.50f };
722 const TensorInfo weightInfo(weightShape, weightType, weightPerAxisScales, 0);
723
724 Convolution2dQueueDescriptor queueDescriptor;
725 queueDescriptor.m_Parameters.m_BiasEnabled = true;
726
727 WorkloadInfo workloadInfo;
728 AddInputToWorkload(queueDescriptor, workloadInfo, inputInfo, nullptr);
729 AddInputToWorkload(queueDescriptor, workloadInfo, weightInfo, nullptr);
730 AddOutputToWorkload(queueDescriptor, workloadInfo, outputInfo, nullptr);
731
732 ScopedTensorHandle weightTensor(weightInfo);
733 queueDescriptor.m_Weight = &weightTensor;
734
735 // Test 2: wrong per-axis quantization values
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000736 const std::vector<float> biasPerAxisScales2 = { 4.00f, 5.00f };
737 const TensorInfo biasInfo2(biasShape, biasType, biasPerAxisScales2, 0);
738
James Conroy1f58f032021-04-27 17:13:27 +0100739 ScopedTensorHandle biasHandle2(biasInfo2);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000740 queueDescriptor.m_Bias = &biasHandle2;
741
Keith Davisb4dd5cc2022-04-07 11:32:00 +0100742 AddInputToWorkload(queueDescriptor, workloadInfo, biasInfo2, nullptr);
743
Sadik Armagan1625efc2021-06-10 18:24:34 +0100744 CHECK_NOTHROW(queueDescriptor.Validate(workloadInfo));
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000745
Keith Davisb4dd5cc2022-04-07 11:32:00 +0100746}
747
748TEST_CASE("BiasPerAxisQuantization_ValidateInvalidArgumentException")
749{
750 constexpr unsigned int nInput = 1u;
751 constexpr unsigned int cInput = 3u;
752 constexpr unsigned int hInput = 3u;
753 constexpr unsigned int wInput = 3u;
754
755 constexpr unsigned int nOutput = nInput;
756 constexpr unsigned int cOutput = cInput;
757 constexpr unsigned int hOutput = 1u;
758 constexpr unsigned int wOutput = 1u;
759
760 const TensorShape inputShape { nInput, cInput, hInput, wInput };
761 const TensorShape outputShape{ nOutput, cOutput, hOutput, wOutput };
762 const TensorShape weightShape{ cOutput, cInput, hInput, wInput };
763 const TensorShape biasShape { cOutput };
764
765 constexpr DataType inputType = DataType::QAsymmU8;
766 constexpr DataType weightType = DataType::QSymmS8;
767 constexpr DataType biasType = DataType::Signed32;
768
769 constexpr float perTensorScale = 1.5f;
770 const TensorInfo inputInfo (inputShape, inputType, perTensorScale);
771 const TensorInfo outputInfo(outputShape, inputType, perTensorScale);
772
773 const std::vector<float> weightPerAxisScales = { 2.50f, 3.50f };
774 const TensorInfo weightInfo(weightShape, weightType, weightPerAxisScales, 0);
775
776 Convolution2dQueueDescriptor queueDescriptor;
777 queueDescriptor.m_Parameters.m_BiasEnabled = true;
778
779 WorkloadInfo workloadInfo;
780 AddInputToWorkload(queueDescriptor, workloadInfo, inputInfo, nullptr);
781 AddInputToWorkload(queueDescriptor, workloadInfo, weightInfo, nullptr);
782 AddOutputToWorkload(queueDescriptor, workloadInfo, outputInfo, nullptr);
783
784 ScopedTensorHandle weightTensor(weightInfo);
785 queueDescriptor.m_Weight = &weightTensor;
786
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000787 // Test 3: mismatched number of quantization scales
788 const std::vector<float> biasPerAxisScales3 = { 3.75f, 5.25f, 5.25f };
789 const TensorInfo biasInfo3(biasShape, biasType, biasPerAxisScales3, 0);
790
James Conroy1f58f032021-04-27 17:13:27 +0100791 ScopedTensorHandle biasHandle3(biasInfo3);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000792 queueDescriptor.m_Bias = &biasHandle3;
793
Keith Davisb4dd5cc2022-04-07 11:32:00 +0100794 AddInputToWorkload(queueDescriptor, workloadInfo, biasInfo3, nullptr);
795
Sadik Armagan1625efc2021-06-10 18:24:34 +0100796 CHECK_THROWS_AS(queueDescriptor.Validate(workloadInfo), InvalidArgumentException);
Aron Virginas-Tard9053072019-10-30 16:03:19 +0000797}
798
Keith Davisb4dd5cc2022-04-07 11:32:00 +0100799
Sadik Armagan1625efc2021-06-10 18:24:34 +0100800}