| // Copyright 2020 The IREE Authors |
| // |
| // Licensed under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| |
| #include "iree/hal/local/local_executable_cache.h" |
| |
| #include <stdbool.h> |
| #include <stddef.h> |
| |
| typedef struct iree_hal_local_executable_cache_t { |
| iree_hal_resource_t resource; |
| iree_allocator_t host_allocator; |
| iree_string_view_t identifier; |
| iree_host_size_t worker_capacity; |
| iree_host_size_t loader_count; |
| iree_hal_executable_loader_t* loaders[]; |
| } iree_hal_local_executable_cache_t; |
| |
| static const iree_hal_executable_cache_vtable_t |
| iree_hal_local_executable_cache_vtable; |
| |
| static iree_hal_local_executable_cache_t* iree_hal_local_executable_cache_cast( |
| iree_hal_executable_cache_t* base_value) { |
| IREE_HAL_ASSERT_TYPE(base_value, &iree_hal_local_executable_cache_vtable); |
| return (iree_hal_local_executable_cache_t*)base_value; |
| } |
| |
| iree_status_t iree_hal_local_executable_cache_create( |
| iree_string_view_t identifier, iree_host_size_t worker_capacity, |
| iree_host_size_t loader_count, iree_hal_executable_loader_t** loaders, |
| iree_allocator_t host_allocator, |
| iree_hal_executable_cache_t** out_executable_cache) { |
| IREE_ASSERT_ARGUMENT(!loader_count || loaders); |
| IREE_ASSERT_ARGUMENT(out_executable_cache); |
| *out_executable_cache = NULL; |
| |
| IREE_TRACE_ZONE_BEGIN(z0); |
| |
| iree_hal_local_executable_cache_t* executable_cache = NULL; |
| iree_host_size_t total_size = |
| sizeof(*executable_cache) + |
| loader_count * sizeof(*executable_cache->loaders) + identifier.size; |
| iree_status_t status = iree_allocator_malloc(host_allocator, total_size, |
| (void**)&executable_cache); |
| if (iree_status_is_ok(status)) { |
| iree_hal_resource_initialize(&iree_hal_local_executable_cache_vtable, |
| &executable_cache->resource); |
| executable_cache->host_allocator = host_allocator; |
| iree_string_view_append_to_buffer( |
| identifier, &executable_cache->identifier, |
| (char*)executable_cache + total_size - identifier.size); |
| executable_cache->worker_capacity = worker_capacity; |
| |
| executable_cache->loader_count = loader_count; |
| for (iree_host_size_t i = 0; i < executable_cache->loader_count; ++i) { |
| executable_cache->loaders[i] = loaders[i]; |
| iree_hal_executable_loader_retain(executable_cache->loaders[i]); |
| } |
| |
| *out_executable_cache = (iree_hal_executable_cache_t*)executable_cache; |
| } |
| |
| IREE_TRACE_ZONE_END(z0); |
| return status; |
| } |
| |
| static void iree_hal_local_executable_cache_destroy( |
| iree_hal_executable_cache_t* base_executable_cache) { |
| iree_hal_local_executable_cache_t* executable_cache = |
| iree_hal_local_executable_cache_cast(base_executable_cache); |
| iree_allocator_t host_allocator = executable_cache->host_allocator; |
| IREE_TRACE_ZONE_BEGIN(z0); |
| |
| for (iree_host_size_t i = 0; i < executable_cache->loader_count; ++i) { |
| iree_hal_executable_loader_release(executable_cache->loaders[i]); |
| } |
| iree_allocator_free(host_allocator, executable_cache); |
| |
| IREE_TRACE_ZONE_END(z0); |
| } |
| |
| static iree_status_t iree_hal_local_executable_cache_infer_format( |
| iree_hal_executable_cache_t* base_executable_cache, |
| iree_hal_executable_caching_mode_t caching_mode, |
| iree_const_byte_span_t executable_data, |
| iree_host_size_t executable_format_capacity, char* executable_format, |
| iree_host_size_t* out_inferred_size) { |
| iree_hal_local_executable_cache_t* executable_cache = |
| iree_hal_local_executable_cache_cast(base_executable_cache); |
| IREE_TRACE_ZONE_BEGIN(z0); |
| |
| iree_status_t status = iree_ok_status(); |
| for (iree_host_size_t i = 0; i < executable_cache->loader_count; ++i) { |
| iree_status_t infer_status = iree_hal_executable_loader_infer_format( |
| executable_cache->loaders[i], caching_mode, executable_data, |
| executable_format_capacity, executable_format, out_inferred_size); |
| if (iree_status_is_ok(status)) { |
| // Successfully inferred. |
| status = iree_ok_status(); |
| break; |
| } else if (iree_status_is_incompatible(infer_status)) { |
| // Skip this loader and try another. |
| iree_status_ignore(infer_status); |
| status = iree_status_from_code(IREE_STATUS_INCOMPATIBLE); |
| continue; |
| } else { |
| status = infer_status; |
| break; |
| } |
| } |
| |
| IREE_TRACE_ZONE_END(z0); |
| return status; |
| } |
| |
| static bool iree_hal_local_executable_cache_can_prepare_format( |
| iree_hal_executable_cache_t* base_executable_cache, |
| iree_hal_executable_caching_mode_t caching_mode, |
| iree_string_view_t executable_format) { |
| iree_hal_local_executable_cache_t* executable_cache = |
| iree_hal_local_executable_cache_cast(base_executable_cache); |
| for (iree_host_size_t i = 0; i < executable_cache->loader_count; ++i) { |
| if (iree_hal_executable_loader_query_support( |
| executable_cache->loaders[i], caching_mode, executable_format)) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| static iree_status_t iree_hal_local_executable_cache_prepare_executable( |
| iree_hal_executable_cache_t* base_executable_cache, |
| const iree_hal_executable_params_t* executable_params, |
| iree_hal_executable_t** out_executable) { |
| iree_hal_local_executable_cache_t* executable_cache = |
| iree_hal_local_executable_cache_cast(base_executable_cache); |
| for (iree_host_size_t i = 0; i < executable_cache->loader_count; ++i) { |
| if (!iree_hal_executable_loader_query_support( |
| executable_cache->loaders[i], executable_params->caching_mode, |
| executable_params->executable_format)) { |
| // Loader definitely can't handle the executable; no use trying so skip. |
| continue; |
| } |
| // The loader _may_ handle the executable. |
| // If the specific executable is present but not supported then the try will |
| // fail with IREE_STATUS_CANCELLED and if and if the executable is not |
| // registered with the loader then it will fail with IREE_STATUS_NOT_FOUND. |
| // In either of those cases we continue trying to find a loader that can |
| // provide the executable. |
| iree_status_t status = iree_hal_executable_loader_try_load( |
| executable_cache->loaders[i], executable_params, |
| executable_cache->worker_capacity, out_executable); |
| if (iree_status_is_ok(status)) { |
| // Executable was successfully loaded. |
| return status; |
| } else if (!iree_status_is_cancelled(status) && |
| !iree_status_is_not_found(status)) { |
| // Error beyond just the try failing due to unsupported formats. |
| return status; |
| } |
| iree_status_ignore(status); |
| } |
| return iree_make_status( |
| IREE_STATUS_NOT_FOUND, |
| "no executable loader registered for the given executable format '%.*s'", |
| (int)executable_params->executable_format.size, |
| executable_params->executable_format.data); |
| } |
| |
| static const iree_hal_executable_cache_vtable_t |
| iree_hal_local_executable_cache_vtable = { |
| .destroy = iree_hal_local_executable_cache_destroy, |
| .infer_format = iree_hal_local_executable_cache_infer_format, |
| .can_prepare_format = |
| iree_hal_local_executable_cache_can_prepare_format, |
| .prepare_executable = |
| iree_hal_local_executable_cache_prepare_executable, |
| }; |