2024-09-27 15:41:48 +02:00
|
|
|
"""Model Utils"""
|
|
|
|
|
|
|
|
import os
|
2024-10-11 18:47:23 +02:00
|
|
|
from typing import Any
|
2024-09-27 15:41:48 +02:00
|
|
|
|
|
|
|
import onnxruntime as ort
|
|
|
|
|
2024-10-11 18:47:23 +02:00
|
|
|
try:
|
|
|
|
import openvino as ov
|
|
|
|
except ImportError:
|
|
|
|
# openvino is not included
|
|
|
|
pass
|
|
|
|
|
2024-09-27 15:41:48 +02:00
|
|
|
|
|
|
|
def get_ort_providers(
|
2024-10-10 03:46:31 +02:00
|
|
|
force_cpu: bool = False, openvino_device: str = "AUTO", requires_fp16: bool = False
|
2024-09-27 15:41:48 +02:00
|
|
|
) -> tuple[list[str], list[dict[str, any]]]:
|
|
|
|
if force_cpu:
|
2024-10-11 20:03:47 +02:00
|
|
|
return (
|
|
|
|
["CPUExecutionProvider"],
|
|
|
|
[
|
|
|
|
{
|
2024-10-21 17:00:45 +02:00
|
|
|
"enable_cpu_mem_arena": False,
|
2024-10-11 20:03:47 +02:00
|
|
|
}
|
|
|
|
],
|
|
|
|
)
|
2024-09-27 15:41:48 +02:00
|
|
|
|
2024-10-14 04:34:51 +02:00
|
|
|
providers = []
|
2024-09-27 15:41:48 +02:00
|
|
|
options = []
|
|
|
|
|
2024-10-14 04:34:51 +02:00
|
|
|
for provider in ort.get_available_providers():
|
|
|
|
if provider == "CUDAExecutionProvider":
|
|
|
|
providers.append(provider)
|
|
|
|
options.append(
|
|
|
|
{
|
|
|
|
"arena_extend_strategy": "kSameAsRequested",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
elif provider == "TensorrtExecutionProvider":
|
|
|
|
# TensorrtExecutionProvider uses too much memory without options to control it
|
|
|
|
pass
|
2024-09-27 15:41:48 +02:00
|
|
|
elif provider == "OpenVINOExecutionProvider":
|
|
|
|
os.makedirs("/config/model_cache/openvino/ort", exist_ok=True)
|
2024-10-14 04:34:51 +02:00
|
|
|
providers.append(provider)
|
2024-09-27 15:41:48 +02:00
|
|
|
options.append(
|
|
|
|
{
|
2024-10-11 20:03:47 +02:00
|
|
|
"arena_extend_strategy": "kSameAsRequested",
|
2024-09-27 15:41:48 +02:00
|
|
|
"cache_dir": "/config/model_cache/openvino/ort",
|
|
|
|
"device_type": openvino_device,
|
|
|
|
}
|
|
|
|
)
|
2024-10-11 20:03:47 +02:00
|
|
|
elif provider == "CPUExecutionProvider":
|
2024-10-14 04:34:51 +02:00
|
|
|
providers.append(provider)
|
2024-10-11 20:03:47 +02:00
|
|
|
options.append(
|
|
|
|
{
|
2024-10-21 17:00:45 +02:00
|
|
|
"enable_cpu_mem_arena": False,
|
2024-10-11 20:03:47 +02:00
|
|
|
}
|
|
|
|
)
|
2024-09-27 15:41:48 +02:00
|
|
|
else:
|
2024-10-14 04:34:51 +02:00
|
|
|
providers.append(provider)
|
2024-09-27 15:41:48 +02:00
|
|
|
options.append({})
|
|
|
|
|
|
|
|
return (providers, options)
|
2024-10-11 18:47:23 +02:00
|
|
|
|
|
|
|
|
|
|
|
class ONNXModelRunner:
|
|
|
|
"""Run onnx models optimally based on available hardware."""
|
|
|
|
|
|
|
|
def __init__(self, model_path: str, device: str, requires_fp16: bool = False):
|
|
|
|
self.model_path = model_path
|
|
|
|
self.ort: ort.InferenceSession = None
|
|
|
|
self.ov: ov.Core = None
|
|
|
|
providers, options = get_ort_providers(device == "CPU", device, requires_fp16)
|
|
|
|
|
|
|
|
if "OpenVINOExecutionProvider" in providers:
|
|
|
|
# use OpenVINO directly
|
|
|
|
self.type = "ov"
|
|
|
|
self.ov = ov.Core()
|
|
|
|
self.ov.set_property(
|
|
|
|
{ov.properties.cache_dir: "/config/model_cache/openvino"}
|
|
|
|
)
|
|
|
|
self.interpreter = self.ov.compile_model(
|
|
|
|
model=model_path, device_name=device
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
# Use ONNXRuntime
|
|
|
|
self.type = "ort"
|
|
|
|
self.ort = ort.InferenceSession(
|
2024-10-20 22:14:51 +02:00
|
|
|
model_path,
|
|
|
|
providers=providers,
|
|
|
|
provider_options=options,
|
2024-10-11 18:47:23 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
def get_input_names(self) -> list[str]:
|
|
|
|
if self.type == "ov":
|
|
|
|
input_names = []
|
|
|
|
|
|
|
|
for input in self.interpreter.inputs:
|
|
|
|
input_names.extend(input.names)
|
|
|
|
|
|
|
|
return input_names
|
|
|
|
elif self.type == "ort":
|
|
|
|
return [input.name for input in self.ort.get_inputs()]
|
|
|
|
|
|
|
|
def run(self, input: dict[str, Any]) -> Any:
|
|
|
|
if self.type == "ov":
|
|
|
|
infer_request = self.interpreter.create_infer_request()
|
|
|
|
input_tensor = list(input.values())
|
|
|
|
|
|
|
|
if len(input_tensor) == 1:
|
|
|
|
input_tensor = ov.Tensor(array=input_tensor[0])
|
|
|
|
else:
|
|
|
|
input_tensor = ov.Tensor(array=input_tensor)
|
|
|
|
|
|
|
|
infer_request.infer(input_tensor)
|
|
|
|
return [infer_request.get_output_tensor().data]
|
|
|
|
elif self.type == "ort":
|
|
|
|
return self.ort.run(None, input)
|