# /usr/bin/env python3.8
# -*- mode: python -*-
# =============================================================================
# @@-COPYRIGHT-START-@@
#
# Copyright (c) 2022, Qualcomm Innovation Center, Inc. All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
# 1. Redistributions of source code must retain the above copyright notice,
# this list of conditions and the following disclaimer.
#
# 2. Redistributions in binary form must reproduce the above copyright notice,
# this list of conditions and the following disclaimer in the documentation
# and/or other materials provided with the distribution.
#
# 3. Neither the name of the copyright holder nor the names of its contributors
# may be used to endorse or promote products derived from this software
# without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
#
# SPDX-License-Identifier: BSD-3-Clause
#
# @@-COPYRIGHT-END-@@
# =============================================================================
""" Implementation for simulating models running on Quantized hardware """
import os
from typing import Dict, List
import numpy as np
from onnx import helper, onnx_pb
import onnxruntime as ort
from onnxruntime import SessionOptions, GraphOptimizationLevel, InferenceSession
from onnxruntime.quantization.onnx_quantizer import ONNXModel
from aimet_common import libpymo
from aimet_common import libquant_info
from aimet_common.defs import QuantScheme, QuantizationDataType
from aimet_common.quantsim import encoding_version, extract_global_quantizer_args
from aimet_common.utils import save_json_yaml
from aimet_onnx import utils
from aimet_onnx.meta.operations import Op
from aimet_onnx.meta.connectedgraph import ConnectedGraph
from aimet_onnx.qc_quantize_op import QcQuantizeOp, OpMode
from aimet_onnx.quantsim_config.quantsim_config import QuantSimConfigurator
from aimet_onnx.utils import make_dummy_input, add_hook_to_get_activation, remove_activation_hooks
WORKING_DIR = '/tmp/quantsim/'
op_types_to_ignore = ["branch", "Flatten", "Gather", "Reshape", "Shape", "Unsqueeze", "Squeeze", "Split",
"Compress", "Tile", "Transpose", "Identity"]
data_types_to_quantize = [np.float32]
[docs]class QuantizationSimModel:
""" Creates a QuantizationSimModel model by adding quantization simulations ops to a given model """
# pylint: disable=too-many-arguments
# pylint: disable=too-many-instance-attributes
def __init__(self,
model: onnx_pb.ModelProto,
dummy_input: Dict[str, np.ndarray] = None,
quant_scheme: QuantScheme = QuantScheme.post_training_tf_enhanced,
rounding_mode: str = 'nearest',
default_param_bw: int = 8,
default_activation_bw: int = 8,
use_symmetric_encodings: bool = False, use_cuda: bool = True,
device: int = 0, config_file: str = None, default_data_type: QuantizationDataType = QuantizationDataType.int):
"""
Constructor
:param model: ONNX model or path to model
:param dummy_input: Dummy input to the model. If None, will attempt to auto-generate a dummy input
:param quant_scheme: Quantization scheme (e.g. QuantScheme.post_training_tf)
:param rounding_mode: Rounding mode (e.g. nearest)
:param default_param_bw: Quantization bitwidth for parameter
:param default_activation_bw: Quantization bitwidth for activation
:param use_symmetric_encodings: True if symmetric encoding is used. False otherwise.
:param use_cuda: True if using CUDA to run quantization op. False otherwise.
:param config_file: Path to Configuration file for model quantizers
:param default_data_type: Default data type to use for quantizing all layer inputs, outputs and parameters.
Possible options are QuantizationDataType.int and QuantizationDataType.float.
Note that the mode default_data_type=QuantizationDataType.float is only supported with
default_output_bw=16 and default_param_bw=16
"""
self.model = model
if not isinstance(model, ONNXModel):
self.model = ONNXModel(model)
if not dummy_input:
dummy_input = make_dummy_input(self.model.model)
self.qc_quantize_op_dict = {}
self.connected_graph = ConnectedGraph(self.model)
self._quant_scheme = quant_scheme
self._rounding_mode = rounding_mode
self._default_param_bw = default_param_bw
self._default_activation_bw = default_activation_bw
self._default_quantization_data_type = default_data_type
self._use_symmetric_encodings = use_symmetric_encodings
self._use_cuda = use_cuda
if 'CUDAExecutionProvider' not in ort.get_available_providers():
self._use_cuda = False
if self._use_cuda:
self._op_domain = "aimet.customop.cuda"
self.providers = [('CUDAExecutionProvider', {'device_id': device}), 'CPUExecutionProvider']
else:
self._op_domain = "aimet.customop.cpu"
self.providers = ['CPUExecutionProvider']
self.param_names = []
self.activation_names = []
self.activation_dtypes = {}
self._get_param_names()
self._get_activations_to_quantize(dummy_input)
self._add_quantization_nodes()
self.session = self._build_session(self.providers)
quantsim_configurator = self._add_configuration_(config_file)
self.quant_args = extract_global_quantizer_args(quant_scheme, quantsim_configurator)
def _add_configuration_(self, config_file: str):
"""
Add configuration based on config file
:param config_file: Path to Configuration file for model quantizers
"""
quantsim_configurator = QuantSimConfigurator(self.model, self.connected_graph, config_file,
self._default_activation_bw, self._default_param_bw,
self._default_quantization_data_type)
quantsim_configurator.configure_quantizers(self.qc_quantize_op_dict, self.param_names, self.activation_names)
return quantsim_configurator
def _get_param_names(self):
"""
Get the names of params
"""
valid_ops = self._get_ops_with_parameter()
for op in valid_ops:
for param_info in op.parameters.values():
param, _ = param_info
if param.name and param.name not in self.param_names:
self.param_names.append(param.name)
def _get_ops_with_parameter(self) -> List[Op]:
"""
Gets ops with parameters to add quantization nodes for
:return: Connected graph ops
"""
valid_ops = [op for op in self.connected_graph.get_all_ops().values() if op.type not in ['BatchNormalization']]
return valid_ops
def _get_activations_to_quantize(self, dummy_input: Dict[str, np.ndarray]):
"""
Get the names of activations to quantize
:param dummy_input: Sample input to be run through the model
"""
self.fill_activation_dtypes(dummy_input)
for node in self.model.nodes():
if node.op_type not in op_types_to_ignore:
for name in node.output:
if name not in self.activation_names and name not in self.param_names and \
self._is_op_quantizable(name):
self.activation_names.append(name)
for node in self.model.graph().input:
name = node.name
if name not in self.activation_names and name not in self.param_names and self._is_op_quantizable(name):
self.activation_names.append(name)
for node in self.model.graph().output:
if node.name in self.activation_names:
node.name += '_updated'
def _is_op_quantizable(self, name: str) -> bool:
"""
Checks whether the given activation should be quantized
:param name: Name of the activation
:return: True if the activation should be quantized
"""
# Check if activation is used as an input to another node
if name not in self.activation_dtypes.keys() or self.activation_dtypes[name] not in data_types_to_quantize:
return False
return True
def fill_activation_dtypes(self, dummy_input: Dict[str, np.ndarray]):
"""
Get the data type for each activation
:param dummy_input: Sample input to run through the model
"""
activations = utils.get_graph_intermediate_activations(self.model.graph())
hooks = []
for name in activations:
hooks.append(add_hook_to_get_activation(self.model.model, name))
sess = self._build_session(self.providers)
outputs = sess.run(None, dummy_input)
for idx in range(len(self.model.graph().output)):
act_name = self.model.graph().output[idx].name
dtype = outputs[idx].dtype
self.activation_dtypes[act_name] = dtype
remove_activation_hooks(self.model.model, hooks)
def _add_quantization_nodes(self):
"""
Call insert functions for quantization nodes
"""
self._insert_param_quantization_nodes()
self._insert_activation_quantization_nodes()
def _insert_param_quantization_nodes(self):
"""
Insert quantization node for each param tensor
"""
for name in self.param_names:
self.model.replace_input_of_all_nodes(name, name + '_qdq')
quant_info = libquant_info.QcQuantizeInfo()
custom_node = helper.make_node(
op_type='QcQuantizeOp',
inputs=[name],
outputs=[name + '_qdq'],
name='QcQuantizeOp_' + name,
domain=self._op_domain,
op_name=name,
quant_info=libpymo.PtrToInt64(quant_info),
)
self.model.add_node(custom_node)
self.qc_quantize_op_dict[name] = QcQuantizeOp(quant_info=quant_info,
quant_scheme=self._quant_scheme,
rounding_mode=self._rounding_mode,
encodings=None,
op_mode=OpMode.oneShotQuantizeDequantize,
bitwidth=self._default_param_bw,
use_symmetric_encodings=self._use_symmetric_encodings
)
def _insert_activation_quantization_nodes(self):
"""
Insert quantization node for each activation tensor
"""
for name in self.activation_names:
self.model.replace_input_of_all_nodes(name, name + '_updated')
quant_info = libquant_info.QcQuantizeInfo()
custom_node = helper.make_node(
op_type='QcQuantizeOp',
inputs=[name],
outputs=[name + '_updated'],
name='QcQuantizeOp_' + name,
domain=self._op_domain,
op_name=name,
quant_info=libpymo.PtrToInt64(quant_info)
)
self.model.add_node(custom_node)
self.qc_quantize_op_dict[name] = QcQuantizeOp(quant_info=quant_info,
quant_scheme=self._quant_scheme,
rounding_mode=self._rounding_mode,
encodings=None,
op_mode=OpMode.updateStats,
bitwidth=self._default_activation_bw,
use_symmetric_encodings=self._use_symmetric_encodings
)
def _build_session(self, providers):
"""
Build and return onnxruntime inference session
:param providers: providers to execute onnxruntime
"""
sess_options = SessionOptions()
shared_library = os.path.dirname(libquant_info.__file__)
shared_library = os.path.join(shared_library, "libaimet_onnxrt_ops.so")
sess_options.register_custom_ops_library(shared_library)
sess_options.graph_optimization_level = GraphOptimizationLevel.ORT_DISABLE_ALL
session = InferenceSession(
path_or_bytes=self.model.model.SerializeToString(),
sess_options=sess_options,
providers=providers,
)
return session
def get_qc_quantize_op(self):
"""
Return dict of qc quantize ops
"""
return self.qc_quantize_op_dict
def save_model_graph(self, filename_prefix: str):
"""
Save model to given path
:param filename_prefix: filename to save the onnx model
"""
if not os.path.exists(WORKING_DIR):
os.makedirs(WORKING_DIR)
self.model.save_model_to_file(os.path.join(WORKING_DIR, filename_prefix) + '.onnx')
[docs] def compute_encodings(self, forward_pass_callback, forward_pass_callback_args):
"""
Compute and return the encodings of each tensor quantizer
:param forward_pass_callback: A callback function that simply runs forward passes on the model. This callback
function should use representative data for the forward pass, so the calculated encodings work for all
data samples. This callback internally chooses the number of data samples it wants to use for calculating
encodings.
:param forward_pass_callback_args: These argument(s) are passed to the forward_pass_callback as-is. Up to
the user to determine the type of this parameter. E.g. could be simply an integer representing the number
of data samples to use. Or could be a tuple of parameters or an object representing something more complex.
If set to None, forward_pass_callback will be invoked with no parameters.
"""
for op_name, qc_op in self.qc_quantize_op_dict.items():
qc_op.reset_encoding_stats()
if op_name in self.activation_names:
qc_op.op_mode = OpMode.updateStats
forward_pass_callback(self.session, forward_pass_callback_args)
for op_name, qc_op in self.qc_quantize_op_dict.items():
if qc_op.data_type == QuantizationDataType.int:
qc_op.compute_encodings()
if op_name in self.activation_names:
qc_op.op_mode = OpMode.quantizeDequantize
def _export_encodings(self, encoding_file_path):
"""
Export encodings to json and yaml file
:param encoding_file_path: path to save the encoding files
"""
def update_encoding_dict_entry_int(encoding_dict: Dict, op_name: str):
qc_quantize_op = self.qc_quantize_op_dict[op_name]
if qc_quantize_op.data_type == QuantizationDataType.int:
encoding_dict[op_name] = {'min': qc_quantize_op.encodings.min, 'max': qc_quantize_op.encodings.max,
'scale': qc_quantize_op.encodings.delta,
'offset': qc_quantize_op.encodings.offset,
'bitwidth': qc_quantize_op.encodings.bw,
'is_symmetric': qc_quantize_op.use_symmetric_encodings,
'dtype': 'int'}
else:
encoding_dict[op_name] = {'bitwidth': qc_quantize_op.encodings.bw,
'dtype': 'float'}
param_encodings = {}
for name in self.param_names:
if self.qc_quantize_op_dict[name].enabled:
update_encoding_dict_entry_int(param_encodings, name)
activation_encodings = {}
for name in self.activation_names:
if self.qc_quantize_op_dict[name].enabled:
update_encoding_dict_entry_int(activation_encodings, name)
encodings_dict = {'version': encoding_version,
'activation_encodings': activation_encodings,
'param_encodings': param_encodings,
'quantizer_args': self.quant_args}
save_json_yaml(encoding_file_path, encodings_dict)
def _remove_nodes_and_save_model(self, file_path):
"""
Remove quantization nodes and save model to file
:param file_path: path to save onnx model
"""
nodes_to_remove = []
for node in self.model.nodes():
if node.op_type == 'QcQuantizeOp':
nodes_to_remove.append(node)
else:
for name in node.input:
self.model.replace_input_of_all_nodes(name, name.replace('_qdq', '').replace('_updated', ''))
self.model.remove_nodes(nodes_to_remove)
for node in self.model.graph().output:
node.name = node.name.replace('_updated', '')
self.model.save_model_to_file(file_path)
[docs] def export(self, path: str, filename_prefix: str):
"""
Compute encodings and export to files
:param path: dir to save encoding files
:param filename_prefix: filename to save encoding files
"""
self._export_encodings(os.path.join(path, filename_prefix) + '.encodings')
self._remove_nodes_and_save_model(os.path.join(path, filename_prefix) + '.onnx')