ArmNN
 20.05
Pooling2dLayer.cpp
Go to the documentation of this file.
1 //
2 // Copyright © 2017 Arm Ltd. All rights reserved.
3 // SPDX-License-Identifier: MIT
4 //
5 
6 #include "Pooling2dLayer.hpp"
7 
8 #include "LayerCloneBase.hpp"
9 
10 #include <armnn/TypesUtils.hpp>
11 
13 
16 
17 using namespace armnnUtils;
18 
19 namespace armnn
20 {
21 
22 Pooling2dLayer::Pooling2dLayer(const Pooling2dDescriptor& param, const char* name)
23  : LayerWithParameters(1, 1, LayerType::Pooling2d, param, name)
24 {
25 }
26 
27 std::unique_ptr<IWorkload> Pooling2dLayer::CreateWorkload(const IWorkloadFactory& factory) const
28 {
29  Pooling2dQueueDescriptor descriptor;
30  return factory.CreatePooling2d(descriptor, PrepInfoAndDesc(descriptor));
31 }
32 
34 {
35  return CloneBase<Pooling2dLayer>(graph, m_Param, GetName());
36 }
37 
38 std::vector<TensorShape> Pooling2dLayer::InferOutputShapes(const std::vector<TensorShape>& inputShapes) const
39 {
40  ARMNN_ASSERT(inputShapes.size() == 1);
41  const TensorShape& inputShape = inputShapes[0];
42  const DataLayoutIndexed dimensionIndices = m_Param.m_DataLayout;
43 
44  // If we support multiple batch dimensions in the future, then this assert will need to change.
45  ARMNN_ASSERT_MSG(inputShape.GetNumDimensions() == 4, "Pooling2dLayer will always have 4D input.");
46 
47  unsigned int inWidth = inputShape[dimensionIndices.GetWidthIndex()];
48  unsigned int inHeight = inputShape[dimensionIndices.GetHeightIndex()];
49  unsigned int inChannels = inputShape[dimensionIndices.GetChannelsIndex()];
50  unsigned int inBatchSize = inputShape[0];
51 
52  bool isGlobalPooling = (m_Param.m_StrideX==0 && m_Param.m_StrideY==0);
53  unsigned int outWidth = 1;
54  unsigned int outHeight = 1;
55  if (!isGlobalPooling)
56  {
58  "Stride can only be zero when performing global pooling");
59 
60  auto CalcSize = [](auto inSize, auto lowPad, auto highPad, auto poolSize, auto stride, auto outputShapeRounding)
61  {
62  unsigned int readSize = inSize + lowPad + highPad - poolSize;
63  float div = static_cast<float>(readSize) / static_cast<float>(stride);
64 
65  unsigned int size = 0;
66  switch (outputShapeRounding)
67  {
69  size = static_cast<unsigned int>(ceil(div)) + 1;
70  break;
72  size = static_cast<unsigned int>(floor(div)) + 1;
73  break;
74  default:
75  ARMNN_ASSERT_MSG(false, "Unsupported Output Shape Rounding");
76  }
77 
78  // MakeS sure that border operations will start from inside the input and not the padded area.
79  // This is what both Caffe and CL do...
80  if ((size - 1)*stride >= inSize + lowPad)
81  {
82  --size;
83  }
84 
85  return size;
86  };
87 
88  outWidth = CalcSize(inWidth, m_Param.m_PadLeft, m_Param.m_PadRight, m_Param.m_PoolWidth, m_Param.m_StrideX,
90  outHeight = CalcSize(inHeight, m_Param.m_PadTop, m_Param.m_PadBottom, m_Param.m_PoolHeight, m_Param.m_StrideY,
92  }
93  unsigned int outChannels = inChannels;
94  unsigned int outBatchSize = inBatchSize;
95 
97  TensorShape( { outBatchSize, outHeight, outWidth, outChannels } ) :
98  TensorShape( { outBatchSize, outChannels, outHeight, outWidth });
99 
100  return std::vector<TensorShape>({ tensorShape });
101 }
102 
104 {
106 
107  auto inferredShapes = InferOutputShapes({ GetInputSlot(0).GetConnection()->GetTensorInfo().GetShape() });
108 
109  ARMNN_ASSERT(inferredShapes.size() == 1);
110 
111  ConditionalThrowIfNotEqual<LayerValidationException>(
112  "Pooling2dLayer: TensorShape set on OutputSlot[0] does not match the inferred shape.",
114  inferredShapes[0]);
115 }
116 
118 {
119  visitor.VisitPooling2dLayer(this, GetParameters(), GetName());
120 }
121 
122 } // namespace armnn
uint32_t m_PadBottom
Padding bottom value in the height dimension.
Pooling2dDescriptor m_Param
The parameters for the layer (not including tensor-valued weights etc.).
const Pooling2dDescriptor & GetParameters() const
unsigned int GetWidthIndex() const
const TensorShape & GetShape() const
Definition: Tensor.hpp:88
uint32_t m_PadLeft
Padding left value in the width dimension.
Pooling2dLayer(const Pooling2dDescriptor &param, const char *name)
Constructor to create a Pooling2dLayer.
uint32_t m_PoolWidth
Pooling width value.
uint32_t m_PadTop
Padding top value in the height dimension.
Copyright (c) 2020 ARM Limited.
virtual void VisitPooling2dLayer(const IConnectableLayer *layer, const Pooling2dDescriptor &pooling2dDescriptor, const char *name=nullptr)=0
Function that a pooling layer should call back to when its Accept(ILayerVisitor&) function is invoked...
const IOutputSlot * GetConnection() const override
Definition: Layer.hpp:199
uint32_t m_StrideX
Stride value when proceeding through input for the width dimension.
unsigned int GetHeightIndex() const
void Accept(ILayerVisitor &visitor) const override
Apply a visitor to this layer.
void VerifyLayerConnections(unsigned int expectedConnections, const CheckLocation &location) const
Definition: Layer.cpp:339
uint32_t m_PoolHeight
Pooling height value.
const InputSlot & GetInputSlot(unsigned int index) const override
Get a const input slot handle by slot index.
Definition: Layer.hpp:310
Pooling2dLayer * Clone(Graph &graph) const override
Creates a dynamically-allocated copy of this layer.
uint32_t m_PadRight
Padding right value in the width dimension.
#define ARMNN_ASSERT_MSG(COND, MSG)
Definition: Assert.hpp:15
Provides access to the appropriate indexes for Channels, Height and Width based on DataLayout...
#define ARMNN_ASSERT(COND)
Definition: Assert.hpp:14
#define CHECK_LOCATION()
Definition: Exceptions.hpp:192
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.
This layer represents a pooling 2d operation.
DataLayout m_DataLayout
The data layout to be used (NCHW, NHWC).
virtual std::unique_ptr< IWorkload > CreatePooling2d(const Pooling2dQueueDescriptor &descriptor, const WorkloadInfo &info) const
void ValidateTensorShapesFromInputs() override
Check if the input tensor shape(s) will lead to a valid configuration of Pooling2dLayer.
OutputShapeRounding m_OutputShapeRounding
The rounding method for the output shape. (Floor, Ceiling).
virtual std::unique_ptr< IWorkload > CreateWorkload(const IWorkloadFactory &factory) const override
Makes a workload for the Pooling2d type.
WorkloadInfo PrepInfoAndDesc(QueueDescriptor &descriptor) const
Helper function to reduce duplication in *LayerCreateWorkload.
const OutputSlot & GetOutputSlot(unsigned int index=0) const override
Get the const output slot handle by slot index.
Definition: Layer.hpp:312
virtual const TensorInfo & GetTensorInfo() const =0
const char * GetName() const override
Returns the name of the layer.
Definition: Layer.hpp:305
A Pooling2dDescriptor for the Pooling2dLayer.
void Pooling2d(Decoder< float > &rInputDecoder, Encoder< float > &rOutputEncoder, const TensorInfo &inputInfo, const TensorInfo &outputInfo, const Pooling2dDescriptor &params)
Computes the Pooling2d operation.
Definition: Pooling2d.cpp:143
const TensorInfo & GetTensorInfo() const override
Definition: Layer.cpp:63
unsigned int GetChannelsIndex() const
uint32_t m_StrideY
Stride value when proceeding through input for the height dimension.