blob: 0092e0c3ad6da5a8a70d2fc6bbd87eeee6b49bc5 [file] [log] [blame]
// Copyright 2023 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
// Dumps parsed parameter file information.
// We intentionally use the same flags and parsing behavior as the rest of the
// runtime tools so that users can pass their parameter flags and see exactly
// what the runtime tools would see post-indexing. We don't try to dump original
// file metadata as we only support parsing enough information for what we need.
//
// We also support basic extraction of individual parameters in cases where
// users want to dump them out. Since we don't parse or preserve metadata we
// can't easily dump them to
//
// # List all available parameters and their index information:
// $ iree-dump-parameters --parameters=my_scope=my_file.gguf [--parameters=...]
// # Extract parameter binary contents from a file:
// $ iree-dump-parameters ... --extract=scope::key0=file0.bin [--extract=...]
#include <ctype.h>
#include <stdio.h>
#include "iree/base/api.h"
#include "iree/base/internal/file_io.h"
#include "iree/base/internal/flags.h"
#include "iree/io/parameter_index.h"
#include "iree/io/scope_map.h"
#include "iree/tooling/parameter_util.h"
//===----------------------------------------------------------------------===//
// Parameter extraction
//===----------------------------------------------------------------------===//
IREE_FLAG_LIST(string, extract,
"Extracts a parameter to a file as `[scope::]key=file.bin`.");
static iree_status_t iree_tooling_extract_parameter(
iree_io_scope_map_t* scope_map, iree_string_view_t scope,
iree_string_view_t key, iree_string_view_t path,
iree_allocator_t host_allocator) {
// Lookup the index for the given scope.
iree_io_parameter_index_t* index = NULL; // unowned
IREE_RETURN_IF_ERROR(iree_io_scope_map_lookup(scope_map, scope, &index));
// Lookup the entry within the index.
const iree_io_parameter_index_entry_t* entry = NULL; // unowned
IREE_RETURN_IF_ERROR(iree_io_parameter_index_lookup(index, key, &entry));
fprintf(stdout, "Extracting parameter `");
if (!iree_string_view_is_empty(scope)) {
fprintf(stdout, "%.*s::", (int)scope.size, scope.data);
}
fprintf(stdout, "%.*s` (%" PRIu64 "b) to `%.*s`...\n", (int)key.size,
key.data, entry->length, (int)path.size, path.data);
if (entry->type != IREE_IO_PARAMETER_INDEX_ENTRY_STORAGE_TYPE_FILE) {
return iree_make_status(IREE_STATUS_FAILED_PRECONDITION,
"cannot extract parameters of type %d",
(int)entry->type);
}
// TODO(benvanik): support generic file handle IO instead of memory-only.
if (iree_io_file_handle_type(entry->storage.file.handle) !=
IREE_IO_FILE_HANDLE_TYPE_HOST_ALLOCATION) {
return iree_make_status(
IREE_STATUS_UNIMPLEMENTED,
"only host allocation file handles are supported today");
}
iree_byte_span_t file_contents =
iree_io_file_handle_value(entry->storage.file.handle).host_allocation;
iree_const_byte_span_t entry_contents = iree_make_const_byte_span(
file_contents.data + entry->storage.file.offset, entry->length);
char* path_str = (char*)iree_alloca(path.size + 1);
memcpy(path_str, path.data, path.size);
path_str[path.size] = 0;
return iree_file_write_contents(path_str, entry_contents);
}
static iree_status_t iree_tooling_extract_parameters(
iree_io_scope_map_t* scope_map, iree_allocator_t host_allocator) {
for (iree_host_size_t i = 0; i < FLAG_extract_list().count; ++i) {
iree_string_view_t flag = FLAG_extract_list().values[i];
iree_string_view_t identifier, path;
iree_string_view_split(flag, '=', &identifier, &path);
iree_host_size_t separator_pos =
iree_string_view_find_first_of(identifier, IREE_SV("::"), 0);
iree_string_view_t scope = iree_string_view_empty();
iree_string_view_t key = iree_string_view_empty();
if (separator_pos != IREE_STRING_VIEW_NPOS) {
scope = iree_string_view_substr(identifier, 0, separator_pos);
key = iree_string_view_substr(identifier, separator_pos + 2,
IREE_HOST_SIZE_MAX);
} else {
key = identifier;
}
IREE_RETURN_IF_ERROR(iree_tooling_extract_parameter(scope_map, scope, key,
path, host_allocator),
"extracting parameter with flag `%.*s`",
(int)flag.size, flag.data);
}
return iree_ok_status();
}
//===----------------------------------------------------------------------===//
// main
//===----------------------------------------------------------------------===//
int main(int argc, char** argv) {
IREE_TRACE_APP_ENTER();
IREE_TRACE_ZONE_BEGIN(z0);
iree_allocator_t host_allocator = iree_allocator_system();
int exit_code = EXIT_SUCCESS;
// Parse command line flags.
iree_flags_set_usage("iree-dump-parameters",
"Dumps information about parsed parameter files.\n");
iree_flags_parse_checked(IREE_FLAGS_PARSE_MODE_DEFAULT, &argc, &argv);
if (argc > 1) {
fprintf(stderr, "Error: no positional arguments expected.\n");
fprintf(stderr,
"Use one or more --parameters=file.ext flags to specify parameter "
"files.\n");
IREE_TRACE_ZONE_END(z0);
IREE_TRACE_APP_EXIT(exit_code);
return EXIT_FAILURE;
}
iree_io_scope_map_t scope_map = {0};
iree_io_scope_map_initialize(host_allocator, &scope_map);
// Parse parameters using the common tooling flags.
iree_status_t status =
iree_tooling_build_parameter_indices_from_flags(&scope_map);
// Dump parameter information.
if (iree_status_is_ok(status)) {
iree_string_builder_t sb;
iree_string_builder_initialize(host_allocator, &sb);
status = iree_io_scope_map_dump(&scope_map, &sb);
if (iree_status_is_ok(status)) {
fprintf(stdout, "%.*s", (int)iree_string_builder_size(&sb),
iree_string_builder_buffer(&sb));
}
iree_string_builder_deinitialize(&sb);
}
// Extract parameters as requested, if any.
if (iree_status_is_ok(status)) {
status = iree_tooling_extract_parameters(&scope_map, host_allocator);
}
iree_io_scope_map_deinitialize(&scope_map);
fflush(stdout);
if (!iree_status_is_ok(status)) {
iree_status_fprint(stderr, status);
iree_status_free(status);
exit_code = EXIT_FAILURE;
}
fflush(stderr);
IREE_TRACE_ZONE_END(z0);
IREE_TRACE_APP_EXIT(exit_code);
return exit_code;
}