diff options
Diffstat (limited to 'src/mlia/target/ethos_u/data_collection.py')
-rw-r--r-- | src/mlia/target/ethos_u/data_collection.py | 135 |
1 files changed, 18 insertions, 117 deletions
diff --git a/src/mlia/target/ethos_u/data_collection.py b/src/mlia/target/ethos_u/data_collection.py index 4ea6120..2b41d7d 100644 --- a/src/mlia/target/ethos_u/data_collection.py +++ b/src/mlia/target/ethos_u/data_collection.py @@ -6,30 +6,23 @@ from __future__ import annotations import logging from pathlib import Path from typing import Any +from typing import cast from mlia.backend.vela.compat import Operators from mlia.backend.vela.compat import supported_operators -from mlia.core.context import Context from mlia.core.data_collection import ContextAwareDataCollector -from mlia.core.errors import FunctionalityNotSupportedError -from mlia.core.performance import estimate_performance -from mlia.nn.select import get_optimizer -from mlia.nn.select import OptimizationSettings -from mlia.nn.tensorflow.config import get_keras_model +from mlia.core.performance import P +from mlia.core.performance import PerformanceEstimator from mlia.nn.tensorflow.config import get_tflite_model -from mlia.nn.tensorflow.config import KerasModel -from mlia.nn.tensorflow.config import TFLiteModel from mlia.nn.tensorflow.tflite_compat import TFLiteChecker from mlia.nn.tensorflow.tflite_compat import TFLiteCompatibilityInfo from mlia.nn.tensorflow.utils import is_tflite_model -from mlia.nn.tensorflow.utils import save_keras_model -from mlia.nn.tensorflow.utils import save_tflite_model +from mlia.target.common.optimization import OptimizingPerformaceDataCollector from mlia.target.ethos_u.config import EthosUConfiguration from mlia.target.ethos_u.performance import EthosUPerformanceEstimator from mlia.target.ethos_u.performance import OptimizationPerformanceMetrics from mlia.target.ethos_u.performance import PerformanceMetrics from mlia.utils.logging import log_action -from mlia.utils.types import is_list_of logger = logging.getLogger(__name__) @@ -96,118 +89,26 @@ class EthosUPerformance(ContextAwareDataCollector): return "ethos_u_performance" -class OptimizeModel: - """Helper class for model optimization.""" +# pylint: disable=too-many-ancestors +class EthosUOptimizationPerformance(OptimizingPerformaceDataCollector): + """Collect performance metrics for performance optimizations.""" - def __init__( - self, context: Context, opt_settings: list[OptimizationSettings] - ) -> None: - """Init helper.""" - self.context = context - self.opt_settings = opt_settings - - def __call__(self, model: KerasModel | TFLiteModel) -> Any: - """Run optimization.""" - optimizer = get_optimizer(model, self.opt_settings) - - opts_as_str = ", ".join(str(opt) for opt in self.opt_settings) - logger.info("Applying model optimizations - [%s]", opts_as_str) - optimizer.apply_optimization() - model = optimizer.get_model() # type: ignore - - if isinstance(model, Path): - return model - - if isinstance(model, TFLiteModel): - model_path = self.context.get_model_path("optimized_model.tflite") - with open(model.model_path, "rb") as file_handle: - model_data = bytearray(file_handle.read()) - save_tflite_model(model_data, model_path) - return TFLiteModel(model_path) - - model_path = self.context.get_model_path("optimized_model.h5") - save_keras_model(model, model_path) - return KerasModel(model_path) - - -class EthosUOptimizationPerformance(ContextAwareDataCollector): - """Collect performance metrics for the optimizations.""" - - def __init__( - self, - model: Path, - target_config: EthosUConfiguration, - optimizations: list[list[dict]], - backends: list[str] | None = None, - ) -> None: - """Init performance optimizations data collector.""" - self.model = model - self.target = target_config - self.optimizations = optimizations - self.backends = backends - - def collect_data(self) -> OptimizationPerformanceMetrics | None: - """Collect performance metrics for the optimizations.""" - logger.info("Estimate performance ...") - - if not self.optimizations: - raise FunctionalityNotSupportedError( - reason="Unable to estimate model optimizations impact", - description="No optimization targets provided", - ) - - opt_settings = self._parse_optimization_params(self.optimizations) - - if opt_settings[0][0].optimization_type != "rewrite": - try: - model = get_keras_model(self.model, self.context) - except NotImplementedError as err: - raise FunctionalityNotSupportedError( - reason="Unable to run model optimizations", - description=f"{self.model} is not a Keras model and " - "could not be converted to a Keras model", - ) from err - else: - model = self.model # type: ignore - - optimizers = [OptimizeModel(self.context, opts) for opts in opt_settings] - - estimator = EthosUPerformanceEstimator( + def create_estimator(self) -> PerformanceEstimator: + """Create a PerformanceEstimator, to be overridden in subclasses.""" + return EthosUPerformanceEstimator( self.context, - self.target, + cast(EthosUConfiguration, self.target), self.backends, ) - original_metrics, *optimized_metrics = estimate_performance( - model, estimator, optimizers # type: ignore - ) - - result = OptimizationPerformanceMetrics( - original_perf_metrics=original_metrics, - optimizations_perf_metrics=list(zip(opt_settings, optimized_metrics)), + def create_optimization_performance_metrics( + self, original_metrics: P, optimizations_perf_metrics: list[P] + ) -> Any: + """Create an optimization metrics object.""" + return OptimizationPerformanceMetrics( + original_perf_metrics=original_metrics, # type: ignore + optimizations_perf_metrics=optimizations_perf_metrics, # type: ignore ) - return result - - @staticmethod - def _parse_optimization_params( - optimizations: list[list[dict]], - ) -> list[list[OptimizationSettings]]: - """Parse optimization parameters.""" - if not is_list_of(optimizations, list): - raise ValueError("Optimization parameters expected to be a list.") - - return [ - [ - OptimizationSettings( - item.get("optimization_type"), # type: ignore - item.get("optimization_target"), # type: ignore - item.get("layers_to_optimize"), - item.get("dataset"), - ) - for item in opt_configuration - ] - for opt_configuration in optimizations - ] @classmethod def name(cls) -> str: |