blob: 0fb819db0b31a884874a9d25ee81ae0aa65afeb8 [file] [log] [blame]
James Conroyd47a0642019-09-17 14:22:06 +01001//
2// Copyright © 2019 Arm Ltd. All rights reserved.
3// SPDX-License-Identifier: MIT
4//
5
6#include "NeonArgMinMaxWorkload.hpp"
7#include "NeonWorkloadUtils.hpp"
8
9#include <aclCommon/ArmComputeTensorUtils.hpp>
Matteo Martincighe011d202019-11-28 11:35:47 +000010
James Conroyd47a0642019-09-17 14:22:06 +010011#include <backendsCommon/CpuTensorHandle.hpp>
Matteo Martincighe011d202019-11-28 11:35:47 +000012
Jan Eilersbb446e52020-04-02 13:56:54 +010013#include <armnn/utility/PolymorphicDowncast.hpp>
Matteo Martincighe011d202019-11-28 11:35:47 +000014#include <armnnUtils/TensorUtils.hpp>
James Conroyd47a0642019-09-17 14:22:06 +010015
16#include <arm_compute/runtime/NEON/functions/NEArgMinMaxLayer.h>
17
18namespace
19{
20unsigned int CalcAclAxis(unsigned int numDimensions, unsigned int axisIndex)
21{
22 return (numDimensions - axisIndex) - 1;
23}
24
25} //namespace
26
27namespace armnn
28{
29
30arm_compute::Status NeonArgMinMaxWorkloadValidate(const TensorInfo& input,
31 const TensorInfo& output,
32 const ArgMinMaxDescriptor& descriptor)
33{
34 const arm_compute::TensorInfo aclInput = armcomputetensorutils::BuildArmComputeTensorInfo(input);
35 const arm_compute::TensorInfo aclOutput = armcomputetensorutils::BuildArmComputeTensorInfo(output);
36
37 auto numDims = input.GetNumDimensions();
38 auto unsignedAxis = armnnUtils::GetUnsignedAxis(numDims, descriptor.m_Axis);
39 int aclAxis = boost::numeric_cast<int>(CalcAclAxis(numDims, unsignedAxis));
40
41 if (descriptor.m_Function == ArgMinMaxFunction::Max)
42 {
43 return arm_compute::NEArgMinMaxLayer::validate(&aclInput, aclAxis, &aclOutput,
44 arm_compute::ReductionOperation::ARG_IDX_MAX);
45 }
46 else
47 {
48 return arm_compute::NEArgMinMaxLayer::validate(&aclInput, aclAxis, &aclOutput,
49 arm_compute::ReductionOperation::ARG_IDX_MIN);
50 }
51}
52
53
54NeonArgMinMaxWorkload::NeonArgMinMaxWorkload(const ArgMinMaxQueueDescriptor& descriptor,
55 const WorkloadInfo& info)
56 : BaseWorkload<ArgMinMaxQueueDescriptor>(descriptor, info)
57{
Jan Eilersbb446e52020-04-02 13:56:54 +010058 arm_compute::ITensor& input = PolymorphicDowncast<IAclTensorHandle*>(m_Data.m_Inputs[0])->GetTensor();
59 arm_compute::ITensor& output = PolymorphicDowncast<IAclTensorHandle*>(m_Data.m_Outputs[0])->GetTensor();
James Conroyd47a0642019-09-17 14:22:06 +010060
61 auto numDims = info.m_InputTensorInfos[0].GetNumDimensions();
62 auto unsignedAxis = armnnUtils::GetUnsignedAxis(numDims, m_Data.m_Parameters.m_Axis);
63 int aclAxis = boost::numeric_cast<int>(CalcAclAxis(numDims, unsignedAxis));
64
James Conroy663c1842019-11-01 15:21:48 +000065 auto layer = std::make_unique<arm_compute::NEArgMinMaxLayer>();
66
James Conroyd47a0642019-09-17 14:22:06 +010067 if (m_Data.m_Parameters.m_Function == ArgMinMaxFunction::Max)
68 {
James Conroy663c1842019-11-01 15:21:48 +000069 layer->configure(&input, aclAxis, &output, arm_compute::ReductionOperation::ARG_IDX_MAX);
James Conroyd47a0642019-09-17 14:22:06 +010070 }
71 else
72 {
James Conroy663c1842019-11-01 15:21:48 +000073 layer->configure(&input, aclAxis, &output, arm_compute::ReductionOperation::ARG_IDX_MIN);
James Conroyd47a0642019-09-17 14:22:06 +010074 }
James Conroy663c1842019-11-01 15:21:48 +000075
76 m_ArgMinMaxLayer.reset(layer.release());
James Conroyd47a0642019-09-17 14:22:06 +010077}
78
79void NeonArgMinMaxWorkload::Execute() const
80{
81 ARMNN_SCOPED_PROFILING_EVENT_NEON("NeonArgMinMaxWorkload_Execute");
James Conroy663c1842019-11-01 15:21:48 +000082 m_ArgMinMaxLayer->run();
James Conroyd47a0642019-09-17 14:22:06 +010083}
84
85} //namespace armnn
86