ArmNN
 21.11
ElementwiseBaseLayer.cpp
Go to the documentation of this file.
1 //
2 // Copyright © 2017 Arm Ltd and Contributors. All rights reserved.
3 // SPDX-License-Identifier: MIT
4 //
5 
7 
8 #include "InternalTypes.hpp"
9 #include "armnn/Exceptions.hpp"
10 #include <armnn/TypesUtils.hpp>
11 #include <armnn/utility/Assert.hpp>
12 
13 namespace armnn
14 {
15 
16 ElementwiseBaseLayer::ElementwiseBaseLayer(unsigned int numInputSlots,
17  unsigned int numOutputSlots,
18  LayerType type,
19  const char* name)
20  : Layer(numInputSlots, numOutputSlots, type, name)
21 {}
22 
23 std::vector<TensorShape> ElementwiseBaseLayer::InferOutputShapes(const std::vector<TensorShape>& inputShapes) const
24 {
25  ARMNN_ASSERT(inputShapes.size() == 2);
26  TensorShape input0 = inputShapes[0];
27  TensorShape input1 = inputShapes[1];
28 
30  {
31  if (input0.GetNumDimensions() != input1.GetNumDimensions())
32  {
33  std::stringstream errorMessage;
34  errorMessage << GetLayerTypeAsCString(GetType()) << " layer \"" << GetName() << "\": ";
35  errorMessage << "The tensor inputs to an element-wise operator are expected to have equal number of "
36  "dimensions. First = "
37  << input0.GetNumDimensions() << " second = " << input1.GetNumDimensions();
38  throw InvalidArgumentException(errorMessage.str(), CHECK_LOCATION());
39  }
40  }
42  inputShapes[0].GetNumDimensions() < inputShapes[1].GetNumDimensions())
43  {
44  input1 = inputShapes[0];
45  input0 = inputShapes[1];
46  }
47 
48  unsigned int numDims = input0.GetNumDimensions();
49  unsigned int shiftedDims = input0.GetNumDimensions() - input1.GetNumDimensions();
50 
51  // Get the max of the inputs.
52  std::vector<unsigned int> dims(numDims);
53  for (unsigned int i = shiftedDims; i < numDims; i++)
54  {
55  unsigned int dim0 = input0[i];
56  unsigned int dim1 = input1[i - shiftedDims];
57 
58 #if !NDEBUG
59  // Validate inputs are broadcast compatible.
60  ARMNN_ASSERT_MSG(dim0 == dim1 || dim0 == 1 || dim1 == 1,
61  "Dimensions should either match or one should be of size 1.");
62 #endif
63 
64  dims[i] = std::max(dim0, dim1);
65  }
66 
67  // Fill in the rest of the shifted dimensions.
68  for (unsigned int i = 0; i < shiftedDims; i++)
69  {
70  dims[i] = input0[i];
71  }
72 
73  return std::vector<TensorShape>({ TensorShape(numDims, dims.data()) });
74 }
75 
77 {
79 
80  const TensorShape& outputShape = GetOutputSlot(0).GetTensorInfo().GetShape();
81 
83 
84  auto inferredShapes = InferOutputShapes({ GetInputSlot(0).GetConnection()->GetTensorInfo().GetShape(),
86 
87  ARMNN_ASSERT(inferredShapes.size() == 1);
88 
89  ValidateAndCopyShape(outputShape, inferredShapes[0], m_ShapeInferenceMethod, GetLayerTypeAsCString(GetType()));
90 }
91 
93 {
94  strategy.ExecuteStrategy(this, BaseDescriptor(), {}, GetName());
95 }
96 
97 } // namespace armnn
const TensorShape & GetShape() const
Definition: Tensor.hpp:191
std::vector< TensorShape > InferOutputShapes(const std::vector< TensorShape > &inputShapes) const override
By default returns inputShapes if the number of inputs are equal to number of outputs, otherwise infers the output shapes from given input shapes and layer properties.
virtual void ExecuteStrategy(const armnn::IConnectableLayer *layer, const armnn::BaseDescriptor &descriptor, const std::vector< armnn::ConstTensor > &constants, const char *name, const armnn::LayerBindingId id=0)=0
void VerifyShapeInferenceType(const TensorShape &outputShape, ShapeInferenceMethod shapeInferenceMethod)
Definition: Layer.cpp:433
Copyright (c) 2021 ARM Limited and Contributors.
const IOutputSlot * GetConnection() const override
Definition: Layer.hpp:199
void ValidateAndCopyShape(const TensorShape &outputShape, const TensorShape &inferredShape, const ShapeInferenceMethod shapeInferenceMethod, const std::string &layerName, const unsigned int outputSlotIndex=0)
Definition: Layer.cpp:393
Base class for all descriptors.
Definition: Descriptors.hpp:22
void VerifyLayerConnections(unsigned int expectedConnections, const CheckLocation &location) const
Definition: Layer.cpp:349
const InputSlot & GetInputSlot(unsigned int index) const override
Get a const input slot handle by slot index.
Definition: Layer.hpp:316
#define ARMNN_ASSERT_MSG(COND, MSG)
Definition: Assert.hpp:15
Validate all output shapes.
LayerType GetType() const override
Returns the armnn::LayerType of this layer.
Definition: Layer.hpp:265
#define ARMNN_ASSERT(COND)
Definition: Assert.hpp:14
#define CHECK_LOCATION()
Definition: Exceptions.hpp:209
ElementwiseBaseLayer(unsigned int numInputSlots, unsigned int numOutputSlots, LayerType type, const char *name)
unsigned int GetNumDimensions() const
Function that returns the tensor rank.
Definition: Tensor.cpp:174
const OutputSlot & GetOutputSlot(unsigned int index=0) const override
Get the const output slot handle by slot index.
Definition: Layer.hpp:318
Infer missing output shapes and validate all output shapes.
virtual const TensorInfo & GetTensorInfo() const =0
const char * GetName() const override
Returns the name of the layer.
Definition: Layer.hpp:311
void ExecuteStrategy(IStrategy &strategy) const override
Apply a visitor to this layer.
const TensorInfo & GetTensorInfo() const override
Definition: Layer.cpp:63
const char * GetLayerTypeAsCString(LayerType type)
void ValidateTensorShapesFromInputs() override
Check if the input tensor shape(s) will lead to a valid configuration of the element wise operation...
ShapeInferenceMethod m_ShapeInferenceMethod
Definition: Layer.hpp:408
LayerType
When adding a new layer, adapt also the LastLayer enum value in the enum class LayerType below...
Definition: Types.hpp:443