1*89c4ff92SAndroid Build Coastguard Worker //
2*89c4ff92SAndroid Build Coastguard Worker // Copyright © 2017 Arm Ltd. All rights reserved.
3*89c4ff92SAndroid Build Coastguard Worker // SPDX-License-Identifier: MIT
4*89c4ff92SAndroid Build Coastguard Worker //
5*89c4ff92SAndroid Build Coastguard Worker #pragma once
6*89c4ff92SAndroid Build Coastguard Worker
7*89c4ff92SAndroid Build Coastguard Worker #include <ResolveType.hpp>
8*89c4ff92SAndroid Build Coastguard Worker
9*89c4ff92SAndroid Build Coastguard Worker #include <armnn/INetwork.hpp>
10*89c4ff92SAndroid Build Coastguard Worker
11*89c4ff92SAndroid Build Coastguard Worker #include <CommonTestUtils.hpp>
12*89c4ff92SAndroid Build Coastguard Worker
13*89c4ff92SAndroid Build Coastguard Worker #include <doctest/doctest.h>
14*89c4ff92SAndroid Build Coastguard Worker
15*89c4ff92SAndroid Build Coastguard Worker namespace
16*89c4ff92SAndroid Build Coastguard Worker {
17*89c4ff92SAndroid Build Coastguard Worker template<typename armnn::DataType DataType>
CreatePreluNetwork(const armnn::TensorInfo & inputInfo,const armnn::TensorInfo & alphaInfo,const armnn::TensorInfo & outputInfo)18*89c4ff92SAndroid Build Coastguard Worker INetworkPtr CreatePreluNetwork(const armnn::TensorInfo& inputInfo,
19*89c4ff92SAndroid Build Coastguard Worker const armnn::TensorInfo& alphaInfo,
20*89c4ff92SAndroid Build Coastguard Worker const armnn::TensorInfo& outputInfo)
21*89c4ff92SAndroid Build Coastguard Worker {
22*89c4ff92SAndroid Build Coastguard Worker using namespace armnn;
23*89c4ff92SAndroid Build Coastguard Worker
24*89c4ff92SAndroid Build Coastguard Worker INetworkPtr net(INetwork::Create());
25*89c4ff92SAndroid Build Coastguard Worker
26*89c4ff92SAndroid Build Coastguard Worker IConnectableLayer* input = net->AddInputLayer(0, "input");
27*89c4ff92SAndroid Build Coastguard Worker IConnectableLayer* alpha = net->AddInputLayer(1, "alpha");
28*89c4ff92SAndroid Build Coastguard Worker IConnectableLayer* prelu = net->AddPreluLayer("Prelu");
29*89c4ff92SAndroid Build Coastguard Worker IConnectableLayer* output = net->AddOutputLayer(0, "output");
30*89c4ff92SAndroid Build Coastguard Worker
31*89c4ff92SAndroid Build Coastguard Worker Connect(input, prelu, inputInfo, 0, 0);
32*89c4ff92SAndroid Build Coastguard Worker Connect(alpha, prelu, alphaInfo, 0, 1);
33*89c4ff92SAndroid Build Coastguard Worker Connect(prelu, output, outputInfo, 0, 0);
34*89c4ff92SAndroid Build Coastguard Worker
35*89c4ff92SAndroid Build Coastguard Worker return net;
36*89c4ff92SAndroid Build Coastguard Worker }
37*89c4ff92SAndroid Build Coastguard Worker
38*89c4ff92SAndroid Build Coastguard Worker template<armnn::DataType ArmnnType, typename T = armnn::ResolveType<ArmnnType>>
PreluEndToEnd(const std::vector<BackendId> & backends,const std::vector<T> & inputData,const std::vector<T> & alphaData,const std::vector<T> & expectedOutputData,const float qScale,const int32_t qOffset)39*89c4ff92SAndroid Build Coastguard Worker void PreluEndToEnd(const std::vector<BackendId>& backends,
40*89c4ff92SAndroid Build Coastguard Worker const std::vector<T>& inputData,
41*89c4ff92SAndroid Build Coastguard Worker const std::vector<T>& alphaData,
42*89c4ff92SAndroid Build Coastguard Worker const std::vector<T>& expectedOutputData,
43*89c4ff92SAndroid Build Coastguard Worker const float qScale ,
44*89c4ff92SAndroid Build Coastguard Worker const int32_t qOffset)
45*89c4ff92SAndroid Build Coastguard Worker {
46*89c4ff92SAndroid Build Coastguard Worker using namespace armnn;
47*89c4ff92SAndroid Build Coastguard Worker
48*89c4ff92SAndroid Build Coastguard Worker armnn::TensorInfo inputInfo({ 2, 2, 2, 1 }, ArmnnType);
49*89c4ff92SAndroid Build Coastguard Worker armnn::TensorInfo alphaInfo({ 1, 2, 2, 1 }, ArmnnType);
50*89c4ff92SAndroid Build Coastguard Worker armnn::TensorInfo outputInfo({ 2, 2, 2, 1 }, ArmnnType);
51*89c4ff92SAndroid Build Coastguard Worker
52*89c4ff92SAndroid Build Coastguard Worker inputInfo.SetQuantizationOffset(qOffset);
53*89c4ff92SAndroid Build Coastguard Worker inputInfo.SetQuantizationScale(qScale);
54*89c4ff92SAndroid Build Coastguard Worker inputInfo.SetConstant(true);
55*89c4ff92SAndroid Build Coastguard Worker alphaInfo.SetQuantizationOffset(qOffset);
56*89c4ff92SAndroid Build Coastguard Worker alphaInfo.SetQuantizationScale(qScale);
57*89c4ff92SAndroid Build Coastguard Worker alphaInfo.SetConstant(true);
58*89c4ff92SAndroid Build Coastguard Worker outputInfo.SetQuantizationOffset(qOffset);
59*89c4ff92SAndroid Build Coastguard Worker outputInfo.SetQuantizationScale(qScale);
60*89c4ff92SAndroid Build Coastguard Worker
61*89c4ff92SAndroid Build Coastguard Worker INetworkPtr net = CreatePreluNetwork<ArmnnType>(inputInfo, alphaInfo, outputInfo);
62*89c4ff92SAndroid Build Coastguard Worker
63*89c4ff92SAndroid Build Coastguard Worker CHECK(net);
64*89c4ff92SAndroid Build Coastguard Worker
65*89c4ff92SAndroid Build Coastguard Worker std::map<int, std::vector<T>> inputTensorData = { { 0, inputData }, { 1, alphaData} };
66*89c4ff92SAndroid Build Coastguard Worker std::map<int, std::vector<T>> expectedOutputTensorData = { { 0, expectedOutputData } };
67*89c4ff92SAndroid Build Coastguard Worker
68*89c4ff92SAndroid Build Coastguard Worker EndToEndLayerTestImpl<ArmnnType, ArmnnType>(move(net),
69*89c4ff92SAndroid Build Coastguard Worker inputTensorData,
70*89c4ff92SAndroid Build Coastguard Worker expectedOutputTensorData,
71*89c4ff92SAndroid Build Coastguard Worker backends);
72*89c4ff92SAndroid Build Coastguard Worker }
73*89c4ff92SAndroid Build Coastguard Worker
74*89c4ff92SAndroid Build Coastguard Worker template<armnn::DataType ArmnnType, typename T = armnn::ResolveType<ArmnnType>>
PreluEndToEndPositiveTest(const std::vector<BackendId> & backends,const float qScale=1.0f,const int32_t qOffset=2)75*89c4ff92SAndroid Build Coastguard Worker void PreluEndToEndPositiveTest(const std::vector<BackendId>& backends, const float qScale = 1.0f,
76*89c4ff92SAndroid Build Coastguard Worker const int32_t qOffset = 2)
77*89c4ff92SAndroid Build Coastguard Worker {
78*89c4ff92SAndroid Build Coastguard Worker std::vector<T> inputData{ 1, 2, 3, 4, 5, 6, 7, 8 };
79*89c4ff92SAndroid Build Coastguard Worker std::vector<T> alphaData{ 2, 1, 1, 1 };
80*89c4ff92SAndroid Build Coastguard Worker
81*89c4ff92SAndroid Build Coastguard Worker std::vector<T> expectedOutputData{ 2, 2, 3, 4, 5, 6, 7, 8 };
82*89c4ff92SAndroid Build Coastguard Worker
83*89c4ff92SAndroid Build Coastguard Worker PreluEndToEnd<ArmnnType>(backends, inputData, alphaData, expectedOutputData, qScale, qOffset);
84*89c4ff92SAndroid Build Coastguard Worker }
85*89c4ff92SAndroid Build Coastguard Worker
86*89c4ff92SAndroid Build Coastguard Worker template<armnn::DataType ArmnnType, typename T = armnn::ResolveType<ArmnnType>>
PreluEndToEndNegativeTest(const std::vector<BackendId> & backends,const float qScale=1.0f,const int32_t qOffset=0)87*89c4ff92SAndroid Build Coastguard Worker void PreluEndToEndNegativeTest(const std::vector<BackendId>& backends, const float qScale = 1.0f,
88*89c4ff92SAndroid Build Coastguard Worker const int32_t qOffset = 0)
89*89c4ff92SAndroid Build Coastguard Worker {
90*89c4ff92SAndroid Build Coastguard Worker std::vector<T> inputData{ 1, -2, 3, 4, 5, 6, 7, 8 };
91*89c4ff92SAndroid Build Coastguard Worker std::vector<T> alphaData{ 1, 2, 1, 1 };
92*89c4ff92SAndroid Build Coastguard Worker
93*89c4ff92SAndroid Build Coastguard Worker std::vector<T> expectedOutputData{ 1, -4, 3, 4, 5, 6, 7, 8 };
94*89c4ff92SAndroid Build Coastguard Worker
95*89c4ff92SAndroid Build Coastguard Worker PreluEndToEnd<ArmnnType>(backends, inputData, alphaData, expectedOutputData, qScale, qOffset);
96*89c4ff92SAndroid Build Coastguard Worker }
97*89c4ff92SAndroid Build Coastguard Worker
98*89c4ff92SAndroid Build Coastguard Worker } // anonymous namespace