# -*- mode: python -*-
# =============================================================================
# @@-COPYRIGHT-START-@@
#
# Copyright (c) 2024, 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-@@
# =============================================================================
""" Quant Analyzer """
import os
import re
from typing import Union, Tuple, Dict, List, Iterable
import copy
from collections import defaultdict
import numpy as np
from onnx import ModelProto
import onnxruntime as ort
from onnxruntime.quantization.onnx_model import ONNXModel
from sklearn.metrics import mean_squared_error
from aimet_common.utils import AimetLogger, CallbackFunc
from aimet_common.defs import QuantScheme
from aimet_common.quant_analyzer import save_json, export_per_layer_sensitivity_analysis_plot, \
create_and_export_min_max_ranges_plot, export_per_layer_mse_plot, export_stats_histogram_plot
from aimet_onnx.qc_quantize_op import QcQuantizeOp
from aimet_onnx.quantsim import QuantizationSimModel
from aimet_onnx.batch_norm_fold import fold_all_batch_norms_to_weight
from aimet_onnx.adaround.activation_sampler import ModuleData
from aimet_onnx import utils
from aimet_onnx.meta.operations import Op
_logger = AimetLogger.get_area_logger(AimetLogger.LogAreas.QuantAnalyzer)
[docs]
class QuantAnalyzer:
"""
QuantAnalyzer provides following utilities:
1) model sensitivity to weight and activation quantization
2) per layer sensitivity analysis
3) per layer encoding (min - max range)
4) per layer quantizer historgram analysis and
5) per layer MSE analysis
"""
def __init__(self,
model: Union[ModelProto, ONNXModel],
dummy_input: Dict[str, np.ndarray],
forward_pass_callback: CallbackFunc,
eval_callback: CallbackFunc,
):
"""
:param model: FP32 model to analyze for quantization.
:param dummy_input: Dummy input to model.
:param forward_pass_callback: A callback function for model calibration that simply runs
forward passes on the model to compute encoding (delta/offset). This
callback function should use representative data and should be subset of
entire train/validation dataset (~1000 images/samples).
:param eval_callback: A callback function for model evaluation that determines model
performance. This callback function is expected to return scalar value
representing the model performance evaluated against entire test/evaluation dataset.
"""
if not isinstance(forward_pass_callback, CallbackFunc):
raise ValueError('forward_pass_callback and its argument(s) are not encapsulated by CallbackFunc class.')
if not isinstance(eval_callback, CallbackFunc):
raise ValueError('eval_callback and its argument(s) are not encapsulated by CallbackFunc class.')
self._onnx_model = model
if not isinstance(self._onnx_model, ONNXModel):
self._onnx_model = ONNXModel(self._onnx_model)
self._dummy_input = dummy_input
self._forward_pass_callback = forward_pass_callback
self._eval_callback = eval_callback
self._unlabeled_dataset_iterable = None
self._num_batches = None
[docs]
def analyze(self,
quant_scheme: QuantScheme = QuantScheme.post_training_tf_enhanced,
default_param_bw: int = 8,
default_activation_bw: int = 8,
config_file: str = None,
results_dir: str = "./tmp/",
):
"""
Analyzes model for quantization and point out sensitive parts/hotspots of the model by performing
1) model sensitivity to quantization,
2) perform per layer sensitivity analysis by enabling and disabling quantizers,
3) export per layer encodings min - max ranges,
4) export per layer quantizer stats histogram,
5) per layer MSE analysis
:param quant_scheme: Quantization scheme. Supported values are
QuantScheme.post_training_tf or QuantScheme.post_training_tf_enhanced.
:param default_param_bw: Default bitwidth (4-31) to use for quantizing layer parameters.
:param default_activation_bw: Default bitwidth (4-31) to use for quantizing layer inputs and outputs.
:param config_file: Path to configuration file for model quantizers.
:param results_dir: Directory to save the results.
"""
sim = self.create_quantsim_and_encodings(quant_scheme,
default_param_bw,
default_activation_bw,
config_file)
results_dir = os.path.abspath(results_dir)
os.makedirs(results_dir, exist_ok=True)
# Check model sensitivity to weight and activation quantization individually.
self.check_model_sensitivity_to_quantization(sim)
# Perform per layer analysis by enabling its quantizers (OPTION-1).
self.perform_per_layer_analysis_by_enabling_quantizers(sim, results_dir)
# Perform per layer analysis by disabling its quantizers (OPTION-2).
self.perform_per_layer_analysis_by_disabling_quantizers(sim, results_dir)
# Export encoding min-max range.
self.export_per_layer_encoding_min_max_range(sim, results_dir)
# Export PDF of statistics.
if quant_scheme == QuantScheme.post_training_tf_enhanced:
self.export_per_layer_stats_histogram(sim, results_dir)
# Export per layer MSE loss between fp32 and quantized output activations.
if self._unlabeled_dataset_iterable:
self.export_per_layer_mse_loss(sim, results_dir)
[docs]
def create_quantsim_and_encodings(self, quant_scheme: QuantScheme, default_param_bw: int,
default_activation_bw: int, config_file: str) \
-> QuantizationSimModel:
"""
Creates quantsim object and computes encodings.
:param quant_scheme: Quantization scheme.
:param default_param_bw: Default bitwidth (4-31) to use for quantizing layer parameters.
:param default_activation_bw: Default bitwidth (4-31) to use for quantizing layer inputs and outputs.
:param config_file: Path to configuration file for model quantizers.
:return: Quantsim object.
"""
_ = fold_all_batch_norms_to_weight(self._onnx_model)
kwargs = dict(
quant_scheme=quant_scheme,
default_activation_bw=default_activation_bw,
default_param_bw=default_param_bw,
config_file=config_file,
)
sim = QuantizationSimModel(copy.deepcopy(self._onnx_model), self._dummy_input, **kwargs)
sim.compute_encodings(self._forward_pass_callback.func, self._forward_pass_callback.args)
return sim
[docs]
def check_model_sensitivity_to_quantization(self,
sim: QuantizationSimModel,
) -> Tuple[float, float, float]:
"""
Performs model sensitivity analysis to weight and activation quantization individually.
:param sim: Quantsim model.
:return: FP32 eval score, weight-quantized eval score, act-quantized eval score.
"""
# pylint: disable=protected-access
fp32_eval_score = self._eval_model(self._onnx_model.model)
_logger.info("FP32 eval score (W32A32): %f", fp32_eval_score)
weight_quantized_eval_score = self._eval_weight_quantized_model(sim)
_logger.info("Weight-quantized eval score (W%dA32): %f", sim._default_param_bw,
weight_quantized_eval_score)
act_quantized_eval_score = self._eval_activation_quantized_model(sim)
_logger.info("Activation-quantized eval score (W32A%d): %f", sim._default_activation_bw,
act_quantized_eval_score)
return fp32_eval_score, weight_quantized_eval_score, act_quantized_eval_score
def _eval_model(self, model):
"""
Evaluate the model performance.
:param model: ONNX model to be evaluated.
:return: Scaler value representing model performance.
"""
if 'CUDAExecutionProvider' in ort.get_available_providers():
providers = [('CUDAExecutionProvider', {'cudnn_conv_algo_search': 'DEFAULT'}), 'CPUExecutionProvider']
else:
providers = ['CPUExecutionProvider']
session = ort.InferenceSession(
path_or_bytes=model.SerializeToString(),
providers=providers,
)
return self._eval_callback.func(session, self._eval_callback.args)
def _eval_weight_quantized_model(self, sim: QuantizationSimModel)-> float:
"""
Evaluate weight quantized model performance.
For weight quantized model performance, disable enabled activation quantizers, measure
eval score and enable again.
:param sim: Quantsim model.
:return: Quantized model performance.
"""
enabled_activation_quantizers = self._get_enabled_activation_quantizers(sim)
self._enable_disable_quantizers(enabled_activation_quantizers, enabled=False)
eval_score = self._eval_callback.func(sim.session, self._eval_callback.args)
self._enable_disable_quantizers(enabled_activation_quantizers, enabled=True)
return eval_score
def _eval_activation_quantized_model(self, sim: QuantizationSimModel)-> float:
"""
Evaluate activation quantized model performance.
For activation quantized model performance, disable enabled param quantizers, measure
eval score and enable again.
:param sim: Quantsim model.
:return: Quantized model performance.
"""
enabled_param_quantizers = self._get_enabled_param_quantizers(sim)
self._enable_disable_quantizers(enabled_param_quantizers, enabled=False)
eval_score = self._eval_callback.func(sim.session, self._eval_callback.args)
self._enable_disable_quantizers(enabled_param_quantizers, enabled=True)
return eval_score
@staticmethod
def _get_enabled_param_quantizers(sim: QuantizationSimModel) -> List[QcQuantizeOp]:
"""
For given quantsim model, get all enabled param quantizers.
:param sim: Quantsim model.
:return: List of enabled param quantizers.
"""
param_quantizers, _ = sim.get_all_quantizers()
enabled_param_quantizers = [quantizer for quantizer in param_quantizers if quantizer.enabled]
return enabled_param_quantizers
@staticmethod
def _get_enabled_activation_quantizers(sim: QuantizationSimModel) -> List[QcQuantizeOp]:
"""
For given quantsim model, get all enabled activation quantizers.
:param sim: Quantsim model.
:return: List of enabled activation quantizers.
"""
_, act_quantizers = sim.get_all_quantizers()
enabled_activation_quantizers = [quantizer for quantizer in act_quantizers if quantizer.enabled]
return enabled_activation_quantizers
@staticmethod
def _enable_disable_quantizers(quantizers: List[QcQuantizeOp], enabled: bool):
"""
For given list of quantizers, set (enable/disable) quantizer's enabled.
:param quantizers: List of quantizers.
:param enabled: Enabled flag.
"""
for quantizer in quantizers:
quantizer.enabled = enabled
def _perform_per_layer_analysis(self,
sim: QuantizationSimModel,
disable_all_quantizers: bool,
enabled_before: bool,
enabled_after: bool,
) -> Dict:
"""
Helper function for perform_per_layer_analysis_by_enabling_quantizers() and
perform_per_layer_analysis_by_disabling_quantizers()
:param sim: Quantsim model.
:param disable_all_quantizers: Flag to disable all the quantizers before per-layer analysis.
:param enabled_before: Flag to set enabled for quantizers before computing encodings.
:param enabled_after: Flag to set enabled for quantizers after computing encodings.
:return: layer wise eval score dictionary. dict[layer_name] = eval_score.
"""
# Mapping of ops and their enabled quantizers
op_to_quantizers_dict = defaultdict(list)
for op in sim.connected_graph.ordered_ops:
input_quantizers, output_quantizers, param_quantizers = self._get_op_quantizers(op, sim)
if not input_quantizers and not output_quantizers and not param_quantizers:
continue
op_to_quantizers_dict[op.name_op].extend(input_quantizers)
op_to_quantizers_dict[op.name_op].extend(output_quantizers)
op_to_quantizers_dict[op.name_op].extend(list(param_quantizers.values()))
if disable_all_quantizers:
for enabled_quantizers in op_to_quantizers_dict.values():
self._enable_disable_quantizers(enabled_quantizers, enabled=False)
eval_score_dict = {}
for op_name, enabled_quantizers in op_to_quantizers_dict.items():
self._enable_disable_quantizers(enabled_quantizers, enabled=enabled_before)
# Record eval score.
eval_score_dict[op_name] = self._eval_callback.func(sim.session, self._eval_callback.args)
_logger.debug("For layer: %s, the eval score is: %f", op_name, eval_score_dict[op_name])
self._enable_disable_quantizers(enabled_quantizers, enabled=enabled_after)
if disable_all_quantizers:
for enabled_quantizers in op_to_quantizers_dict.values():
self._enable_disable_quantizers(enabled_quantizers, enabled=True)
return eval_score_dict
@staticmethod
def _get_op_quantizers(op: Op, sim: QuantizationSimModel) -> (List, List, Dict):
"""
This function returns the enabled input, output and param quantizers of the given connected graph op.
:param op: Connected Graph Op
:param sim: QuantSim object
:return: list of input quantizers, list of output quantizers and dictionary of param quantizers
"""
input_quantizers, output_quantizers, param_quantizers = sim.get_op_quantizers(op)
input_quantizers = [q for q in input_quantizers if q.enabled]
output_quantizers = [q for q in output_quantizers if q.enabled]
param_quantizers = {name: q for name, q in param_quantizers.items() if q.enabled}
return input_quantizers, output_quantizers, param_quantizers
# pylint: disable=no-self-use, too-many-branches, too-many-locals
[docs]
def export_per_layer_encoding_min_max_range(self, sim: QuantizationSimModel, results_dir: str) -> Tuple[Dict, Dict]:
"""
Exports encoding min and max range for all weights and activations. results_dir has
html files in following format.
-results_dir
-activations.html,
-weights.html
If per channel quantization(PCQ) is enabled then,
-results_dir
-activations.html,
-{layer_name}_{param_name}.html
:param sim: Quantsim model.
:param results_dir: Directory to save the results.
:return: layer wise min-max range for weights and activations.
"""
min_max_ranges_dir = os.path.join(results_dir, "min_max_ranges")
min_max_range_for_activations_dict = {}
min_max_range_for_weights_dict = {}
for op in sim.connected_graph.ordered_ops:
input_quantizers, output_quantizers, param_quantizers = self._get_op_quantizers(op, sim)
op_name = re.sub(r'\W+', '_', op.name_op)
# Get input activations' encodings if starting op
for index, quantizer in enumerate(input_quantizers):
name = f"{op_name}_input_{index}"
encodings = quantizer.get_encodings()
min_max_range_for_activations_dict[name] = (encodings[0].min, encodings[0].max)
# Get output activations' encodings
for index, quantizer in enumerate(output_quantizers):
name = f"{op_name}_output_{index}"
encodings = quantizer.get_encodings()
min_max_range_for_activations_dict[name] = (encodings[0].min, encodings[0].max)
# Get parameters' encodings
for param_name, quantizer in param_quantizers.items():
name = re.sub(r'\W+', '_', f"{op_name}_{param_name}")
encodings = quantizer.get_encodings()
if len(encodings) > 1: # per-channel
per_channel_encodings = {}
for index, encoding in enumerate(encodings):
per_channel_encodings[f"{name}_{index}"] = (encoding.min, encoding.max)
min_max_range_for_weights_dict[name] = per_channel_encodings
else: # per-tensor
min_max_range_for_weights_dict[name] = (encodings[0].min, encodings[0].max)
create_and_export_min_max_ranges_plot(min_max_range_for_weights_dict, min_max_ranges_dir, title="weights")
create_and_export_min_max_ranges_plot(min_max_range_for_activations_dict, min_max_ranges_dir, title="activations")
save_json(min_max_range_for_weights_dict, min_max_ranges_dir, title="weights.json")
save_json(min_max_range_for_activations_dict, min_max_ranges_dir, title="activations.json")
_logger.info("Exported per layer encodings min-max ranges plot(s).")
return min_max_range_for_weights_dict, min_max_range_for_activations_dict
# pylint: disable=too-many-branches, too-many-locals
[docs]
def export_per_layer_stats_histogram(self,
sim: QuantizationSimModel,
results_dir: str,
):
"""
NOTE: Not to invoke when quantization scheme is not TF-Enhanced.
Exports histogram that represents a PDF of collected statistics by a quantizer.
After invoking this API, results_dir should have html files in following
format for every quantizers in the model.
-results_dir
-activations_pdf
name_{input/output}_{index}.html
-weights_pdf
-name
param_name_{channel_index}.html
:param sim: Quantsim model.
:param results_dir: Directory to save the results.
"""
weights_pdf_dir = os.path.join(results_dir, "weights_pdf")
activations_pdf_dir = os.path.join(results_dir, "activations_pdf")
for op in sim.connected_graph.ordered_ops:
input_quantizers, output_quantizers, param_quantizers = self._get_op_quantizers(op, sim)
op_name = re.sub(r'\W+', '_', op.name_op)
# Collect stats histogram of input activation quantizers
for index, quantizer in enumerate(input_quantizers):
self._create_and_export_stats_histogram_plot(quantizer,
activations_pdf_dir,
title=f"{op_name}_input_q{index}")
# Collect stats histogram of output activation quantizers
for index, quantizer in enumerate(output_quantizers):
self._create_and_export_stats_histogram_plot(quantizer,
activations_pdf_dir,
title=f"{op_name}_output_q{index}")
# Collect stats histogram of param quantizers
for param_name, quantizer in param_quantizers.items():
sanitized_param_name = re.sub(r'\W+', '_', param_name)
self._create_and_export_stats_histogram_plot(quantizer,
os.path.join(weights_pdf_dir, op_name),
title=f"{op_name}_{sanitized_param_name}")
_logger.info("Exported per layer stats histogram plot(s).")
@staticmethod
def _create_and_export_stats_histogram_plot(quantizer: QcQuantizeOp, results_dir: str, title: str):
"""
For given quantizer, create and export histogram (PDF) of statistics in html format.
:param quantizer: Quantizer.
:param results_dir: Directory to save the results.
:param title: Title of the plot.
"""
os.makedirs(results_dir, exist_ok=True)
histograms = quantizer.get_stats_histogram()
encodings = quantizer.get_encodings()
if not isinstance(encodings, List):
encodings = [encodings]
for index, (histogram, encoding) in enumerate(zip(histograms, encodings)):
export_stats_histogram_plot(histogram, encoding, results_dir, title=f"{title}_{index}")
[docs]
def enable_per_layer_mse_loss(self, unlabeled_dataset_iterable: Iterable, num_batches: int):
"""
Enables per layer MSE loss analysis.
:param unlabeled_dataset_iterable: A collection (i.e. iterable with `__len__`)
that iterates over an unlabeled dataset. The values yielded by this iterable are expected
to be able to be passed directly to the model.
:param num_batches: Number of batches. Approximately 256 samples/images are recommended,
so if batch size of data loader is 64, then 4 number of batches leads to 256 samples/images.
"""
if len(unlabeled_dataset_iterable) < num_batches:
raise ValueError(f'Can not fetch {num_batches} batches from '
f'a data loader of length {len(unlabeled_dataset_iterable)}.')
self._unlabeled_dataset_iterable = unlabeled_dataset_iterable
self._num_batches = num_batches
[docs]
def export_per_layer_mse_loss(self, sim: QuantizationSimModel, results_dir: str) -> Dict:
"""
Exports MSE loss between fp32 and quantized output activations for each layer.
:param sim: Quantsim model.
:param results_dir: Directory to save the results.
:return: layer wise MSE loss. dict[layer_name] = MSE loss.
"""
results_dir = os.path.abspath(results_dir)
os.makedirs(results_dir, exist_ok=True)
mse_loss_dict = {}
for op_node in self._onnx_model.nodes():
if op_node.op_type == 'Constant':
continue
op_output = op_node.output[0]
if op_output in sim.qc_quantize_op_dict:
quantized_op_output = op_output + '_updated'
loss = self._compute_mse_loss(op_output, quantized_op_output, self._onnx_model, sim.model)
mse_loss_dict[op_node.name] = loss
export_per_layer_mse_plot(mse_loss_dict,
results_dir,
title="per_layer_mse_loss")
save_json(mse_loss_dict, results_dir, title="per_layer_mse_loss.json")
_logger.info("Exported per layer MSE loss plot.")
return mse_loss_dict
# pylint: disable=too-many-locals
def _compute_mse_loss(self, fp32_act_name: str, quantized_act_name: str,
fp32_model: ONNXModel, quantized_model: ONNXModel) -> float:
"""
Compute MSE loss between fp32 and quantized output activations for each batch, add for
all the batches and return averaged mse loss.
:param fp32_act_name: module from the fp32_model.
:param quantized_act_name: Corresponding quant wrapper from the QuantSim model.
:param fp32_model: PyTorch model.
:param sim: Quantsim model.
:return: MSE loss between fp32 and quantized output activations.
"""
providers = ['CPUExecutionProvider']
if 'CUDAExecutionProvider' in ort.get_available_providers():
providers = [('CUDAExecutionProvider', {'cudnn_conv_algo_search': 'DEFAULT'}), 'CPUExecutionProvider']
# output activations collector.
orig_module_collector = ModuleData(fp32_model, fp32_act_name, providers)
quant_module_collector = ModuleData(quantized_model, quantized_act_name, providers)
total = 0
loss = 0.0
batch_index = 0
for model_inputs in self._unlabeled_dataset_iterable:
model_inputs = utils.create_input_dict(fp32_model.model, model_inputs)
_, quantized_out_acts = quant_module_collector.collect_inp_out_data(model_inputs,
collect_input=False,
collect_output=True)
_, fp32_out_acts = orig_module_collector.collect_inp_out_data(model_inputs,
collect_input=False,
collect_output=True)
loss += mean_squared_error(fp32_out_acts[0].reshape(fp32_out_acts[0].shape[0], -1),
quantized_out_acts[0].reshape(fp32_out_acts[0].shape[0], -1))
total += fp32_out_acts[0].shape[0]
batch_index += 1
if batch_index == self._num_batches:
break
average_loss = loss/total
return float(average_loss)