1*89c4ff92SAndroid Build Coastguard Worker //
2*89c4ff92SAndroid Build Coastguard Worker // Copyright © 2020 Arm Ltd and Contributors. All rights reserved.
3*89c4ff92SAndroid Build Coastguard Worker // SPDX-License-Identifier: MIT
4*89c4ff92SAndroid Build Coastguard Worker //
5*89c4ff92SAndroid Build Coastguard Worker
6*89c4ff92SAndroid Build Coastguard Worker #include "LogicalBinaryLayer.hpp"
7*89c4ff92SAndroid Build Coastguard Worker
8*89c4ff92SAndroid Build Coastguard Worker #include "LayerCloneBase.hpp"
9*89c4ff92SAndroid Build Coastguard Worker
10*89c4ff92SAndroid Build Coastguard Worker #include <armnn/backends/WorkloadData.hpp>
11*89c4ff92SAndroid Build Coastguard Worker #include <armnn/backends/WorkloadFactory.hpp>
12*89c4ff92SAndroid Build Coastguard Worker
13*89c4ff92SAndroid Build Coastguard Worker #include <algorithm>
14*89c4ff92SAndroid Build Coastguard Worker
15*89c4ff92SAndroid Build Coastguard Worker namespace armnn
16*89c4ff92SAndroid Build Coastguard Worker {
17*89c4ff92SAndroid Build Coastguard Worker
LogicalBinaryLayer(const LogicalBinaryDescriptor & param,const char * name)18*89c4ff92SAndroid Build Coastguard Worker LogicalBinaryLayer::LogicalBinaryLayer(const LogicalBinaryDescriptor& param, const char* name)
19*89c4ff92SAndroid Build Coastguard Worker : LayerWithParameters(2, 1, LayerType::LogicalBinary, param, name)
20*89c4ff92SAndroid Build Coastguard Worker {
21*89c4ff92SAndroid Build Coastguard Worker }
22*89c4ff92SAndroid Build Coastguard Worker
CreateWorkload(const IWorkloadFactory & factory) const23*89c4ff92SAndroid Build Coastguard Worker std::unique_ptr<IWorkload> LogicalBinaryLayer::CreateWorkload(const IWorkloadFactory& factory) const
24*89c4ff92SAndroid Build Coastguard Worker {
25*89c4ff92SAndroid Build Coastguard Worker LogicalBinaryQueueDescriptor descriptor;
26*89c4ff92SAndroid Build Coastguard Worker return factory.CreateWorkload(LayerType::LogicalBinary, descriptor, PrepInfoAndDesc(descriptor));
27*89c4ff92SAndroid Build Coastguard Worker }
28*89c4ff92SAndroid Build Coastguard Worker
Clone(Graph & graph) const29*89c4ff92SAndroid Build Coastguard Worker LogicalBinaryLayer* LogicalBinaryLayer::Clone(Graph& graph) const
30*89c4ff92SAndroid Build Coastguard Worker {
31*89c4ff92SAndroid Build Coastguard Worker return CloneBase<LogicalBinaryLayer>(graph, m_Param, GetName());
32*89c4ff92SAndroid Build Coastguard Worker }
33*89c4ff92SAndroid Build Coastguard Worker
InferOutputShapes(const std::vector<TensorShape> & inputShapes) const34*89c4ff92SAndroid Build Coastguard Worker std::vector<TensorShape> LogicalBinaryLayer::InferOutputShapes(const std::vector<TensorShape>& inputShapes) const
35*89c4ff92SAndroid Build Coastguard Worker {
36*89c4ff92SAndroid Build Coastguard Worker ARMNN_ASSERT(inputShapes.size() == 2);
37*89c4ff92SAndroid Build Coastguard Worker const TensorShape& input0 = inputShapes[0];
38*89c4ff92SAndroid Build Coastguard Worker const TensorShape& input1 = inputShapes[1];
39*89c4ff92SAndroid Build Coastguard Worker
40*89c4ff92SAndroid Build Coastguard Worker ARMNN_ASSERT(input0.GetNumDimensions() == input1.GetNumDimensions());
41*89c4ff92SAndroid Build Coastguard Worker unsigned int numDims = input0.GetNumDimensions();
42*89c4ff92SAndroid Build Coastguard Worker
43*89c4ff92SAndroid Build Coastguard Worker std::vector<unsigned int> dims(numDims);
44*89c4ff92SAndroid Build Coastguard Worker for (unsigned int i = 0; i < numDims; i++)
45*89c4ff92SAndroid Build Coastguard Worker {
46*89c4ff92SAndroid Build Coastguard Worker unsigned int dim0 = input0[i];
47*89c4ff92SAndroid Build Coastguard Worker unsigned int dim1 = input1[i];
48*89c4ff92SAndroid Build Coastguard Worker
49*89c4ff92SAndroid Build Coastguard Worker ARMNN_ASSERT_MSG(dim0 == dim1 || dim0 == 1 || dim1 == 1,
50*89c4ff92SAndroid Build Coastguard Worker "Dimensions should either match or one should be of size 1.");
51*89c4ff92SAndroid Build Coastguard Worker
52*89c4ff92SAndroid Build Coastguard Worker dims[i] = std::max(dim0, dim1);
53*89c4ff92SAndroid Build Coastguard Worker }
54*89c4ff92SAndroid Build Coastguard Worker
55*89c4ff92SAndroid Build Coastguard Worker return std::vector<TensorShape>({ TensorShape(numDims, dims.data()) });
56*89c4ff92SAndroid Build Coastguard Worker }
57*89c4ff92SAndroid Build Coastguard Worker
ValidateTensorShapesFromInputs()58*89c4ff92SAndroid Build Coastguard Worker void LogicalBinaryLayer::ValidateTensorShapesFromInputs()
59*89c4ff92SAndroid Build Coastguard Worker {
60*89c4ff92SAndroid Build Coastguard Worker VerifyLayerConnections(2, CHECK_LOCATION());
61*89c4ff92SAndroid Build Coastguard Worker
62*89c4ff92SAndroid Build Coastguard Worker const TensorShape& outputShape = GetOutputSlot(0).GetTensorInfo().GetShape();
63*89c4ff92SAndroid Build Coastguard Worker
64*89c4ff92SAndroid Build Coastguard Worker VerifyShapeInferenceType(outputShape, m_ShapeInferenceMethod);
65*89c4ff92SAndroid Build Coastguard Worker
66*89c4ff92SAndroid Build Coastguard Worker std::vector<TensorShape> inferredShapes = InferOutputShapes({
67*89c4ff92SAndroid Build Coastguard Worker GetInputSlot(0).GetConnection()->GetTensorInfo().GetShape(),
68*89c4ff92SAndroid Build Coastguard Worker GetInputSlot(1).GetConnection()->GetTensorInfo().GetShape()
69*89c4ff92SAndroid Build Coastguard Worker });
70*89c4ff92SAndroid Build Coastguard Worker ARMNN_ASSERT(inferredShapes.size() == 1);
71*89c4ff92SAndroid Build Coastguard Worker
72*89c4ff92SAndroid Build Coastguard Worker ValidateAndCopyShape(outputShape, inferredShapes[0], m_ShapeInferenceMethod, "LogicalBinaryLayer");
73*89c4ff92SAndroid Build Coastguard Worker }
74*89c4ff92SAndroid Build Coastguard Worker
ExecuteStrategy(IStrategy & strategy) const75*89c4ff92SAndroid Build Coastguard Worker void LogicalBinaryLayer::ExecuteStrategy(IStrategy& strategy) const
76*89c4ff92SAndroid Build Coastguard Worker {
77*89c4ff92SAndroid Build Coastguard Worker strategy.ExecuteStrategy(this, GetParameters(), {}, GetName());
78*89c4ff92SAndroid Build Coastguard Worker }
79*89c4ff92SAndroid Build Coastguard Worker
80*89c4ff92SAndroid Build Coastguard Worker } // namespace armnn
81