/* Copyright 2019 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/lite/kernels/internal/reference/quantize.h" #include "tensorflow/lite/c/common.h" #include "tensorflow/lite/kernels/internal/quantization_util.h" #include "tensorflow/lite/kernels/internal/reference/requantize.h" #include "tensorflow/lite/kernels/internal/tensor_ctypes.h" #include "tensorflow/lite/kernels/kernel_util.h" #include "tensorflow/lite/micro/micro_utils.h" namespace tflite { namespace ops { namespace micro { namespace quantize { TfLiteStatus Prepare(TfLiteContext* context, TfLiteNode* node) { TF_LITE_ENSURE_EQ(context, NumInputs(node), 1); TF_LITE_ENSURE_EQ(context, NumOutputs(node), 1); const TfLiteTensor* input = GetInput(context, node, 0); TfLiteTensor* output = GetOutput(context, node, 0); // TODO(b/128934713): Add support for fixed-point per-channel quantization. // Currently this only support affine per-layer quantization. TF_LITE_ENSURE_EQ(context, output->quantization.type, kTfLiteAffineQuantization); const auto* affine_quantization = reinterpret_cast(output->quantization.params); TF_LITE_ENSURE(context, affine_quantization); TF_LITE_ENSURE(context, affine_quantization->scale); TF_LITE_ENSURE(context, affine_quantization->scale->size == 1); TF_LITE_ENSURE(context, input->type == kTfLiteFloat32 || input->type == kTfLiteInt16); TF_LITE_ENSURE(context, output->type == kTfLiteUInt8 || output->type == kTfLiteInt8); return kTfLiteOk; } TfLiteStatus Eval(TfLiteContext* context, TfLiteNode* node) { const TfLiteTensor* input = GetInput(context, node, 0); TfLiteTensor* output = GetOutput(context, node, 0); tflite::QuantizationParams op_params; op_params.zero_point = output->params.zero_point; op_params.scale = static_cast(output->params.scale); if (input->type == kTfLiteFloat32) { switch (output->type) { case kTfLiteInt8: reference_ops::AffineQuantize( op_params, GetTensorShape(input), GetTensorData(input), GetTensorShape(output), GetTensorData(output)); break; case kTfLiteUInt8: reference_ops::AffineQuantize( op_params, GetTensorShape(input), GetTensorData(input), GetTensorShape(output), GetTensorData(output)); break; default: TF_LITE_KERNEL_LOG(context, "Input %s, output %s not supported.", TfLiteTypeGetName(input->type), TfLiteTypeGetName(output->type)); return kTfLiteError; } } else if (input->type == kTfLiteInt16) { size_t size = ElementCount(*input->dims); int32_t output_multiplier; int output_shift; double effective_scale = static_cast(input->params.scale / output->params.scale); switch (output->type) { case kTfLiteInt8: QuantizeMultiplier(effective_scale, &output_multiplier, &output_shift); reference_ops::Requantize( GetTensorData(input), size, output_multiplier, output_shift, input->params.zero_point, output->params.zero_point, GetTensorData(output)); break; default: TF_LITE_KERNEL_LOG(context, "Input %s, output %s not supported.", TfLiteTypeGetName(input->type), TfLiteTypeGetName(output->type)); return kTfLiteError; } } else { TF_LITE_KERNEL_LOG(context, "Input %s, output %s not supported.", TfLiteTypeGetName(input->type), TfLiteTypeGetName(output->type)); return kTfLiteError; } return kTfLiteOk; } } // namespace quantize // This Op (QUANTIZE) quantizes the input and produces quantized output. // AffineQuantize takes scale and zero point and quantizes the float value to // quantized output, in int8 or uint8 format. TfLiteRegistration* Register_QUANTIZE() { static TfLiteRegistration r = {/*init=*/nullptr, /*free=*/nullptr, /*prepare=*/quantize::Prepare, /*invoke=*/quantize::Eval, /*profiling_string=*/nullptr, /*builtin_code=*/0, /*custom_name=*/nullptr, /*version=*/0}; return &r; } } // namespace micro } // namespace ops } // namespace tflite