/* Copyright 2018 The TensorFlow Authors. All Rights Reserved. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. ==============================================================================*/ #include "tensorflow/contrib/lite/delegates/flex/delegate.h" #include #include "tensorflow/contrib/lite/context_util.h" #include "tensorflow/contrib/lite/delegates/flex/buffer_map.h" #include "tensorflow/contrib/lite/delegates/flex/kernel.h" #include "tensorflow/contrib/lite/delegates/flex/util.h" #include "tensorflow/contrib/lite/util.h" #include "tensorflow/core/lib/core/status.h" namespace tflite { namespace flex { namespace delegate { TfLiteStatus Prepare(TfLiteContext* context, TfLiteDelegate* delegate) { // Get the nodes in the current execution plan. Interpreter owns this array. TfLiteIntArray* plan; TF_LITE_ENSURE_STATUS(context->GetExecutionPlan(context, &plan)); // Add all custom ops starting with "Flex" to list of supported nodes. std::vector supported_nodes; for (int node_index : TfLiteIntArrayView(plan)) { TfLiteNode* node; TfLiteRegistration* registration; TF_LITE_ENSURE_STATUS(context->GetNodeAndRegistration( context, node_index, &node, ®istration)); if (IsFlexOp(registration->custom_name)) { supported_nodes.push_back(node_index); } } // Request TFLite to partition the graph and make kernels for each independent // subgraph. TfLiteIntArray* size_and_nodes = ConvertVectorToTfLiteIntArray(supported_nodes); context->ReplaceSubgraphsWithDelegateKernels(context, GetKernel(), size_and_nodes, delegate); TfLiteIntArrayFree(size_and_nodes); return kTfLiteOk; } TfLiteStatus CopyFromBufferHandle(TfLiteContext* context, TfLiteDelegate* delegate, TfLiteBufferHandle buffer_handle, void* data, size_t size) { BufferMap* buffer_map = reinterpret_cast(delegate->data_)->GetBufferMap(context); if (!buffer_map->HasTensor(buffer_handle)) { context->ReportError(context, "Invalid tensor index %d.", buffer_handle); return kTfLiteError; } tensorflow::Tensor t = buffer_map->GetTensor(buffer_handle); tensorflow::StringPiece t_data = t.tensor_data(); if (size != t_data.size()) { context->ReportError( context, "Not enough space to store TensorFlow's aligned buffer."); return kTfLiteError; } memcpy(data, t_data.data(), t_data.size()); return kTfLiteOk; } } // namespace delegate } // namespace flex // Corresponding weak declaration found in lite/model.cc. std::unique_ptr AcquireFlexDelegate() { return std::unique_ptr( tflite::FlexDelegate::Create().release(), [](TfLiteDelegate* delegate) { delete reinterpret_cast(delegate); }); } std::unique_ptr FlexDelegate::Create() { std::unique_ptr delegate_data; if (!flex::DelegateData::Create(&delegate_data).ok()) { fprintf(stderr, "Unable to initialize TensorFlow context.\n"); return nullptr; } return std::unique_ptr( new FlexDelegate(std::move(delegate_data))); } FlexDelegate::FlexDelegate(std::unique_ptr delegate_data) : TfLiteDelegate{ /*data_=*/delegate_data.get(), /*nullptr,*/ &flex::delegate::Prepare, /*CopyFromBufferHandle=*/&flex::delegate::CopyFromBufferHandle, /*CopyToBufferHandle=*/nullptr, /*FreeBufferHandle=*/nullptr}, delegate_data_(std::move(delegate_data)) {} FlexDelegate::~FlexDelegate() {} } // namespace tflite