Add the spirv-opt command line tool.

This commit is contained in:
Lei Zhang 2016-06-29 16:16:03 -04:00
parent 2c4c73cf37
commit 310bc84cf7
2 changed files with 178 additions and 1 deletions

View File

@ -25,7 +25,7 @@
# MATERIALS OR THE USE OR OTHER DEALINGS IN THE MATERIALS.
if (NOT ${SPIRV_SKIP_EXECUTABLES})
set(SPIRV_INSTALL_TARGETS spirv-as spirv-dis spirv-val)
set(SPIRV_INSTALL_TARGETS spirv-as spirv-dis spirv-val spirv-opt)
add_executable(spirv-as ${CMAKE_CURRENT_SOURCE_DIR}/as/as.cpp)
spvtools_default_compile_options(spirv-as)
@ -44,6 +44,12 @@ if (NOT ${SPIRV_SKIP_EXECUTABLES})
target_include_directories(spirv-val
PRIVATE ${spirv-tools_BINARY_DIR} ${spirv-tools_SOURCE_DIR}/source)
add_executable(spirv-opt ${CMAKE_CURRENT_SOURCE_DIR}/opt/opt.cpp)
spvtools_default_compile_options(spirv-opt)
target_link_libraries(spirv-opt PRIVATE SPIRV-Tools-opt ${SPIRV_TOOLS})
target_include_directories(spirv-opt PRIVATE
${spirv-tools_SOURCE_DIR}/source ${spirv-tools_BINARY_DIR})
install(TARGETS ${SPIRV_INSTALL_TARGETS}
RUNTIME DESTINATION bin
LIBRARY DESTINATION lib

171
tools/opt/opt.cpp Normal file
View File

@ -0,0 +1,171 @@
// Copyright (c) 2016 Google Inc.
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and/or associated documentation files (the
// "Materials"), to deal in the Materials without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Materials, and to
// permit persons to whom the Materials are furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be included
// in all copies or substantial portions of the Materials.
//
// MODIFICATIONS TO THIS FILE MAY MEAN IT NO LONGER ACCURATELY REFLECTS
// KHRONOS STANDARDS. THE UNMODIFIED, NORMATIVE VERSIONS OF KHRONOS
// SPECIFICATIONS AND HEADER INFORMATION ARE LOCATED AT
// https://www.khronos.org/registry/
//
// THE MATERIALS ARE PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
// IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
// CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
// TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
// MATERIALS OR THE USE OR OTHER DEALINGS IN THE MATERIALS.
#include <cstring>
#include <iostream>
#include <vector>
#include "opt/ir_loader.h"
#include "opt/libspirv.hpp"
#include "opt/pass_manager.h"
using namespace spvtools;
void PrintUsage(const char* program) {
printf(
R"(%s - Optimize a SPIR-V binary file.
USAGE: %s [options] [<input>] -o <output>
The SPIR-V binary is read from <input>. If no file is specified,
or if <input> is "-", then the binary is read from standard input.
if <output> is "-", then the optimized output is written to
standard output.
NOTE: The optimizer is a work in progress.
Options:
--strip-debug
Remove all debug instructions.
-h, --help Print this help.
--version Display optimizer version information.
)",
program, program);
}
int main(int argc, char** argv) {
const char* in_file = nullptr;
const char* out_file = nullptr;
spv_target_env target_env = SPV_ENV_UNIVERSAL_1_1;
opt::PassManager pass_manager;
for (int argi = 1; argi < argc; ++argi) {
const char* cur_arg = argv[argi];
if ('-' == cur_arg[0]) {
if (0 == strcmp(cur_arg, "--version")) {
printf("%s\n", spvSoftwareVersionDetailsString());
return 0;
} else if (0 == strcmp(cur_arg, "--help") || 0 == strcmp(cur_arg, "-h")) {
PrintUsage(argv[0]);
return 0;
} else if (0 == strcmp(cur_arg, "-o")) {
if (!out_file && argi + 1 < argc) {
out_file = argv[++argi];
} else {
PrintUsage(argv[0]);
return 1;
}
} else if (0 == strcmp(cur_arg, "--strip-debug")) {
pass_manager.AddPass<opt::StripDebugInfoPass>();
} else if ('\0' == cur_arg[1]) {
// Setting a filename of "-" to indicate stdin.
if (!in_file) {
in_file = cur_arg;
} else {
fprintf(stderr, "error: More than one input file specified\n");
return 1;
}
} else {
PrintUsage(argv[0]);
return 1;
}
} else {
if (!in_file) {
in_file = cur_arg;
} else {
fprintf(stderr, "error: More than one input file specified\n");
return 1;
}
}
}
if (out_file == nullptr) {
fprintf(stderr, "error: -o required\n");
return 1;
}
std::vector<uint32_t> source;
const bool use_file = in_file && strcmp("-", in_file);
if (FILE* fp = (use_file ? fopen(in_file, "rb") : stdin)) {
uint32_t buf[1024];
while (size_t len = fread(buf, sizeof(uint32_t),
sizeof(buf) / sizeof(uint32_t), fp)) {
source.insert(source.end(), buf, buf + len);
}
if (ftell(fp) == -1L) {
if (ferror(fp)) {
fprintf(stderr, "error: error reading file '%s'\n", in_file);
return 1;
}
} else {
if (ftell(fp) % sizeof(uint32_t)) {
fprintf(stderr, "error: corrupted word found in file '%s'\n", in_file);
return 1;
}
}
if (use_file) fclose(fp);
} else {
fprintf(stderr, "error: file does not exist '%s'\n", in_file);
return 1;
}
// Let's do validation first.
spv_context context = spvContextCreate(target_env);
spv_diagnostic diagnostic = nullptr;
spv_const_binary_t binary = {source.data(), source.size()};
spv_result_t error = spvValidate(context, &binary, &diagnostic);
if (error) {
spvDiagnosticPrint(diagnostic);
spvDiagnosticDestroy(diagnostic);
spvContextDestroy(context);
return error;
}
spvDiagnosticDestroy(diagnostic);
spvContextDestroy(context);
std::unique_ptr<ir::Module> module = SpvTools(target_env).BuildModule(source);
pass_manager.Run(module.get());
std::vector<uint32_t> target;
module->ToBinary(&target, /* skip_nop = */ true);
const bool use_stdout = out_file[0] == '-' && out_file[1] == 0;
if (FILE* fp = (use_stdout ? stdout : fopen(out_file, "wb"))) {
size_t written = fwrite(target.data(), sizeof(uint32_t), target.size(), fp);
if (target.size() != written) {
fprintf(stderr, "error: could not write to file '%s'\n", out_file);
return 1;
}
if (!use_stdout) fclose(fp);
} else {
fprintf(stderr, "error: could not open file '%s'\n", out_file);
return 1;
}
return 0;
}