1*89c4ff92SAndroid Build Coastguard Worker //
2*89c4ff92SAndroid Build Coastguard Worker // Copyright © 2022-2023 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 #pragma once
7*89c4ff92SAndroid Build Coastguard Worker
8*89c4ff92SAndroid Build Coastguard Worker #include <ClassicDelegateUtils.hpp>
9*89c4ff92SAndroid Build Coastguard Worker
10*89c4ff92SAndroid Build Coastguard Worker #include <algorithm>
11*89c4ff92SAndroid Build Coastguard Worker #include <iterator>
12*89c4ff92SAndroid Build Coastguard Worker #include <string>
13*89c4ff92SAndroid Build Coastguard Worker #include <vector>
14*89c4ff92SAndroid Build Coastguard Worker
15*89c4ff92SAndroid Build Coastguard Worker namespace armnnDelegate
16*89c4ff92SAndroid Build Coastguard Worker {
VisitGatherNdOperator(DelegateData & delegateData,TfLiteContext * tfLiteContext,TfLiteNode * tfLiteNode,int nodeIndex,int32_t operatorCode)17*89c4ff92SAndroid Build Coastguard Worker TfLiteStatus VisitGatherNdOperator(DelegateData& delegateData,
18*89c4ff92SAndroid Build Coastguard Worker TfLiteContext* tfLiteContext,
19*89c4ff92SAndroid Build Coastguard Worker TfLiteNode* tfLiteNode,
20*89c4ff92SAndroid Build Coastguard Worker int nodeIndex,
21*89c4ff92SAndroid Build Coastguard Worker int32_t operatorCode)
22*89c4ff92SAndroid Build Coastguard Worker {
23*89c4ff92SAndroid Build Coastguard Worker TF_LITE_ENSURE_STATUS(ValidateNumInputs(tfLiteContext, tfLiteNode, 2, nodeIndex));
24*89c4ff92SAndroid Build Coastguard Worker TF_LITE_ENSURE_STATUS(ValidateNumOutputs(tfLiteContext, tfLiteNode, 1, nodeIndex));
25*89c4ff92SAndroid Build Coastguard Worker
26*89c4ff92SAndroid Build Coastguard Worker const TfLiteTensor* tfLiteTensors = tfLiteContext->tensors;
27*89c4ff92SAndroid Build Coastguard Worker
28*89c4ff92SAndroid Build Coastguard Worker const TfLiteTensor& tfLiteInputTensor = tfLiteTensors[tfLiteNode->inputs->data[0]];
29*89c4ff92SAndroid Build Coastguard Worker if (!IsValid(tfLiteContext, tfLiteInputTensor, operatorCode, nodeIndex))
30*89c4ff92SAndroid Build Coastguard Worker {
31*89c4ff92SAndroid Build Coastguard Worker return kTfLiteError;
32*89c4ff92SAndroid Build Coastguard Worker }
33*89c4ff92SAndroid Build Coastguard Worker
34*89c4ff92SAndroid Build Coastguard Worker const TfLiteTensor& tfLiteIndicesTensor = tfLiteTensors[tfLiteNode->inputs->data[1]];
35*89c4ff92SAndroid Build Coastguard Worker if (!IsValid(tfLiteContext, tfLiteIndicesTensor, operatorCode, nodeIndex))
36*89c4ff92SAndroid Build Coastguard Worker {
37*89c4ff92SAndroid Build Coastguard Worker return kTfLiteError;
38*89c4ff92SAndroid Build Coastguard Worker }
39*89c4ff92SAndroid Build Coastguard Worker
40*89c4ff92SAndroid Build Coastguard Worker const TfLiteTensor& tfLiteOutputTensor = tfLiteTensors[tfLiteNode->outputs->data[0]];
41*89c4ff92SAndroid Build Coastguard Worker if (!IsValid(tfLiteContext, tfLiteOutputTensor, operatorCode, nodeIndex))
42*89c4ff92SAndroid Build Coastguard Worker {
43*89c4ff92SAndroid Build Coastguard Worker return kTfLiteError;
44*89c4ff92SAndroid Build Coastguard Worker }
45*89c4ff92SAndroid Build Coastguard Worker
46*89c4ff92SAndroid Build Coastguard Worker const armnn::TensorInfo& inputTensorInfo = GetTensorInfoForTfLiteTensor(tfLiteInputTensor);
47*89c4ff92SAndroid Build Coastguard Worker const armnn::TensorInfo& indicesTensorInfo = GetTensorInfoForTfLiteTensor(tfLiteIndicesTensor);
48*89c4ff92SAndroid Build Coastguard Worker const armnn::TensorInfo& outputTensorInfo = GetTensorInfoForTfLiteTensor(tfLiteOutputTensor, true);
49*89c4ff92SAndroid Build Coastguard Worker
50*89c4ff92SAndroid Build Coastguard Worker armnn::BackendId setBackend;
51*89c4ff92SAndroid Build Coastguard Worker if (!delegateData.m_Network)
52*89c4ff92SAndroid Build Coastguard Worker {
53*89c4ff92SAndroid Build Coastguard Worker // Check if supported
54*89c4ff92SAndroid Build Coastguard Worker bool isSupported = false;
55*89c4ff92SAndroid Build Coastguard Worker FORWARD_LAYER_SUPPORT_FUNC("GATHER_ND",
56*89c4ff92SAndroid Build Coastguard Worker tfLiteContext,
57*89c4ff92SAndroid Build Coastguard Worker IsGatherNdSupported,
58*89c4ff92SAndroid Build Coastguard Worker delegateData.m_Backends,
59*89c4ff92SAndroid Build Coastguard Worker isSupported,
60*89c4ff92SAndroid Build Coastguard Worker setBackend,
61*89c4ff92SAndroid Build Coastguard Worker inputTensorInfo,
62*89c4ff92SAndroid Build Coastguard Worker indicesTensorInfo,
63*89c4ff92SAndroid Build Coastguard Worker outputTensorInfo);
64*89c4ff92SAndroid Build Coastguard Worker return isSupported ? kTfLiteOk : kTfLiteError;
65*89c4ff92SAndroid Build Coastguard Worker }
66*89c4ff92SAndroid Build Coastguard Worker
67*89c4ff92SAndroid Build Coastguard Worker armnn::IConnectableLayer* layer = delegateData.m_Network->AddGatherNdLayer();
68*89c4ff92SAndroid Build Coastguard Worker layer->SetBackendId(setBackend);
69*89c4ff92SAndroid Build Coastguard Worker ARMNN_ASSERT(layer != nullptr);
70*89c4ff92SAndroid Build Coastguard Worker layer->GetOutputSlot(0).SetTensorInfo(outputTensorInfo);
71*89c4ff92SAndroid Build Coastguard Worker
72*89c4ff92SAndroid Build Coastguard Worker auto inputsTensorsProcess = ProcessInputs(layer,
73*89c4ff92SAndroid Build Coastguard Worker delegateData,
74*89c4ff92SAndroid Build Coastguard Worker tfLiteContext,
75*89c4ff92SAndroid Build Coastguard Worker tfLiteNode);
76*89c4ff92SAndroid Build Coastguard Worker if (inputsTensorsProcess == kTfLiteError)
77*89c4ff92SAndroid Build Coastguard Worker {
78*89c4ff92SAndroid Build Coastguard Worker return inputsTensorsProcess;
79*89c4ff92SAndroid Build Coastguard Worker }
80*89c4ff92SAndroid Build Coastguard Worker
81*89c4ff92SAndroid Build Coastguard Worker return Connect(layer, tfLiteNode, delegateData);
82*89c4ff92SAndroid Build Coastguard Worker }
83*89c4ff92SAndroid Build Coastguard Worker } // namespace armnnDelegate