xref: /aosp_15_r20/external/armnn/delegate/classic/src/LogicalBinary.hpp (revision 89c4ff92f2867872bb9e2354d150bf0c8c502810)
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 <tensorflow/lite/builtin_ops.h>
9*89c4ff92SAndroid Build Coastguard Worker #include <tensorflow/lite/c/builtin_op_data.h>
10*89c4ff92SAndroid Build Coastguard Worker #include <tensorflow/lite/c/common.h>
11*89c4ff92SAndroid Build Coastguard Worker #include <tensorflow/lite/minimal_logging.h>
12*89c4ff92SAndroid Build Coastguard Worker 
13*89c4ff92SAndroid Build Coastguard Worker namespace armnnDelegate
14*89c4ff92SAndroid Build Coastguard Worker {
15*89c4ff92SAndroid Build Coastguard Worker 
VisitLogicalBinaryOperator(DelegateData & delegateData,TfLiteContext * tfLiteContext,TfLiteNode * tfLiteNode,int nodeIndex,int32_t logicalOperatorCode,armnn::LogicalBinaryOperation binaryOperation)16*89c4ff92SAndroid Build Coastguard Worker TfLiteStatus VisitLogicalBinaryOperator(DelegateData& delegateData,
17*89c4ff92SAndroid Build Coastguard Worker                                         TfLiteContext* tfLiteContext,
18*89c4ff92SAndroid Build Coastguard Worker                                         TfLiteNode* tfLiteNode,
19*89c4ff92SAndroid Build Coastguard Worker                                         int nodeIndex,
20*89c4ff92SAndroid Build Coastguard Worker                                         int32_t logicalOperatorCode,
21*89c4ff92SAndroid Build Coastguard Worker                                         armnn::LogicalBinaryOperation binaryOperation)
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     const TfLiteTensor& tfLiteInputTensor0 = tfLiteTensors[tfLiteNode->inputs->data[0]];
28*89c4ff92SAndroid Build Coastguard Worker     if (!IsValid(tfLiteContext, tfLiteInputTensor0, logicalOperatorCode, nodeIndex))
29*89c4ff92SAndroid Build Coastguard Worker     {
30*89c4ff92SAndroid Build Coastguard Worker         return kTfLiteError;
31*89c4ff92SAndroid Build Coastguard Worker     }
32*89c4ff92SAndroid Build Coastguard Worker 
33*89c4ff92SAndroid Build Coastguard Worker     const TfLiteTensor& tfLiteInputTensor1 = tfLiteTensors[tfLiteNode->inputs->data[1]];
34*89c4ff92SAndroid Build Coastguard Worker     if (!IsValid(tfLiteContext, tfLiteInputTensor1, logicalOperatorCode, nodeIndex))
35*89c4ff92SAndroid Build Coastguard Worker     {
36*89c4ff92SAndroid Build Coastguard Worker         return kTfLiteError;
37*89c4ff92SAndroid Build Coastguard Worker     }
38*89c4ff92SAndroid Build Coastguard Worker 
39*89c4ff92SAndroid Build Coastguard Worker     const TfLiteTensor& tfLiteOutputTensor = tfLiteTensors[tfLiteNode->outputs->data[0]];
40*89c4ff92SAndroid Build Coastguard Worker     if (!IsValid(tfLiteContext, tfLiteOutputTensor, logicalOperatorCode, nodeIndex))
41*89c4ff92SAndroid Build Coastguard Worker     {
42*89c4ff92SAndroid Build Coastguard Worker         return kTfLiteError;
43*89c4ff92SAndroid Build Coastguard Worker     }
44*89c4ff92SAndroid Build Coastguard Worker 
45*89c4ff92SAndroid Build Coastguard Worker     armnn::TensorInfo inputTensorInfo0 = GetTensorInfoForTfLiteTensor(tfLiteInputTensor0);
46*89c4ff92SAndroid Build Coastguard Worker     armnn::TensorInfo inputTensorInfo1 = GetTensorInfoForTfLiteTensor(tfLiteInputTensor1);
47*89c4ff92SAndroid Build Coastguard Worker     const armnn::TensorInfo& outputTensorInfo = GetTensorInfoForTfLiteTensor(tfLiteOutputTensor, true);
48*89c4ff92SAndroid Build Coastguard Worker 
49*89c4ff92SAndroid Build Coastguard Worker     // Check if we need to expand the dims of any of the input tensor infos.
50*89c4ff92SAndroid Build Coastguard Worker     // This is required for a few of the backends.
51*89c4ff92SAndroid Build Coastguard Worker     if(inputTensorInfo0.GetNumDimensions() != inputTensorInfo1.GetNumDimensions())
52*89c4ff92SAndroid Build Coastguard Worker     {
53*89c4ff92SAndroid Build Coastguard Worker         ExpandTensorRankToEqual(inputTensorInfo0, inputTensorInfo1);
54*89c4ff92SAndroid Build Coastguard Worker     }
55*89c4ff92SAndroid Build Coastguard Worker 
56*89c4ff92SAndroid Build Coastguard Worker     // Setup descriptor and assign operation
57*89c4ff92SAndroid Build Coastguard Worker     armnn::LogicalBinaryDescriptor desc;
58*89c4ff92SAndroid Build Coastguard Worker     desc.m_Operation = binaryOperation;
59*89c4ff92SAndroid Build Coastguard Worker 
60*89c4ff92SAndroid Build Coastguard Worker     // Check if supported
61*89c4ff92SAndroid Build Coastguard Worker     bool isSupported = false;
62*89c4ff92SAndroid Build Coastguard Worker     armnn::BackendId setBackend;
63*89c4ff92SAndroid Build Coastguard Worker     auto validateFunc = [&](const armnn::TensorInfo& outputTensorInfo, bool& isSupported)
64*89c4ff92SAndroid Build Coastguard Worker     {
65*89c4ff92SAndroid Build Coastguard Worker         FORWARD_LAYER_SUPPORT_FUNC("LOGICAL_BINARY",
66*89c4ff92SAndroid Build Coastguard Worker                                    tfLiteContext,
67*89c4ff92SAndroid Build Coastguard Worker                                    IsLogicalBinarySupported,
68*89c4ff92SAndroid Build Coastguard Worker                                    delegateData.m_Backends,
69*89c4ff92SAndroid Build Coastguard Worker                                    isSupported,
70*89c4ff92SAndroid Build Coastguard Worker                                    setBackend,
71*89c4ff92SAndroid Build Coastguard Worker                                    inputTensorInfo0,
72*89c4ff92SAndroid Build Coastguard Worker                                    inputTensorInfo1,
73*89c4ff92SAndroid Build Coastguard Worker                                    outputTensorInfo,
74*89c4ff92SAndroid Build Coastguard Worker                                    desc);
75*89c4ff92SAndroid Build Coastguard Worker     };
76*89c4ff92SAndroid Build Coastguard Worker 
77*89c4ff92SAndroid Build Coastguard Worker     if (!delegateData.m_Network)
78*89c4ff92SAndroid Build Coastguard Worker     {
79*89c4ff92SAndroid Build Coastguard Worker         validateFunc(outputTensorInfo, isSupported);
80*89c4ff92SAndroid Build Coastguard Worker         return isSupported ? kTfLiteOk : kTfLiteError;
81*89c4ff92SAndroid Build Coastguard Worker     }
82*89c4ff92SAndroid Build Coastguard Worker 
83*89c4ff92SAndroid Build Coastguard Worker     armnn::IConnectableLayer* logicalBinaryLayer = delegateData.m_Network->AddLogicalBinaryLayer(desc);
84*89c4ff92SAndroid Build Coastguard Worker     logicalBinaryLayer->SetBackendId(setBackend);
85*89c4ff92SAndroid Build Coastguard Worker     ARMNN_ASSERT(logicalBinaryLayer != nullptr);
86*89c4ff92SAndroid Build Coastguard Worker 
87*89c4ff92SAndroid Build Coastguard Worker     armnn::IOutputSlot& outputSlot = logicalBinaryLayer->GetOutputSlot(0);
88*89c4ff92SAndroid Build Coastguard Worker     outputSlot.SetTensorInfo(outputTensorInfo);
89*89c4ff92SAndroid Build Coastguard Worker 
90*89c4ff92SAndroid Build Coastguard Worker     auto inputsTensorsProcess = ProcessInputs(logicalBinaryLayer,
91*89c4ff92SAndroid Build Coastguard Worker                                               delegateData,
92*89c4ff92SAndroid Build Coastguard Worker                                               tfLiteContext,
93*89c4ff92SAndroid Build Coastguard Worker                                               tfLiteNode);
94*89c4ff92SAndroid Build Coastguard Worker     if (inputsTensorsProcess == kTfLiteError)
95*89c4ff92SAndroid Build Coastguard Worker     {
96*89c4ff92SAndroid Build Coastguard Worker         return inputsTensorsProcess;
97*89c4ff92SAndroid Build Coastguard Worker     }
98*89c4ff92SAndroid Build Coastguard Worker 
99*89c4ff92SAndroid Build Coastguard Worker     return Connect(logicalBinaryLayer, tfLiteNode, delegateData);
100*89c4ff92SAndroid Build Coastguard Worker }
101*89c4ff92SAndroid Build Coastguard Worker 
102*89c4ff92SAndroid Build Coastguard Worker } // namespace armnnDelegate
103