mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2024-11-23 04:00:05 +00:00
73e8dac5b9
Command line application is located at tools/spirv-markv API at include/spirv-tools/markv.h At the moment only very basic compression is implemented, mostly varint. Scope of supported SPIR-V opcodes is also limited. Using a simple move-to-front implementation instead of encoding mapped ids. Work in progress: - Does not cover all of SPIR-V - Does not promise compatibility of compression/decompression across different versions of the code.
248 lines
7.2 KiB
C++
248 lines
7.2 KiB
C++
// Copyright (c) 2017 Google Inc.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
#include <cassert>
|
|
#include <cstdio>
|
|
#include <cstring>
|
|
#include <functional>
|
|
#include <iostream>
|
|
#include <memory>
|
|
#include <vector>
|
|
|
|
#include "source/spirv_target_env.h"
|
|
#include "source/table.h"
|
|
#include "spirv-tools/markv.h"
|
|
#include "tools/io.h"
|
|
|
|
namespace {
|
|
|
|
enum Task {
|
|
kNoTask = 0,
|
|
kEncode,
|
|
kDecode,
|
|
};
|
|
|
|
struct ScopedContext {
|
|
ScopedContext(spv_target_env env) : context(spvContextCreate(env)) {}
|
|
~ScopedContext() { spvContextDestroy(context); }
|
|
spv_context context;
|
|
};
|
|
|
|
void print_usage(char* argv0) {
|
|
printf(
|
|
R"(%s - Encodes or decodes a SPIR-V binary to or from a MARK-V binary.
|
|
|
|
USAGE: %s [e|d] [options] [<filename>]
|
|
|
|
The input binary is read from <filename>. If no file is specified,
|
|
or if the filename is "-", then the binary is read from standard input.
|
|
|
|
If no output is specified then the output is printed to stdout in a human
|
|
readable format.
|
|
|
|
WIP: MARK-V codec is in early stages of development. At the moment it only
|
|
can encode and decode some SPIR-V files and only if exacly the same build of
|
|
software is used (is doesn't write or handle version numbers yet).
|
|
|
|
Tasks:
|
|
e Encode SPIR-V to MARK-V.
|
|
d Decode MARK-V to SPIR-V.
|
|
|
|
Options:
|
|
-h, --help Print this help.
|
|
--comments Write codec comments to stdout.
|
|
--version Display MARK-V codec version.
|
|
|
|
-o <filename> Set the output filename.
|
|
Output goes to standard output if this option is
|
|
not specified, or if the filename is "-".
|
|
)",
|
|
argv0, argv0);
|
|
}
|
|
|
|
void DiagnosticsMessageHandler(spv_message_level_t level, const char*,
|
|
const spv_position_t& position,
|
|
const char* message) {
|
|
switch (level) {
|
|
case SPV_MSG_FATAL:
|
|
case SPV_MSG_INTERNAL_ERROR:
|
|
case SPV_MSG_ERROR:
|
|
std::cerr << "error: " << position.index << ": " << message
|
|
<< std::endl;
|
|
break;
|
|
case SPV_MSG_WARNING:
|
|
std::cout << "warning: " << position.index << ": " << message
|
|
<< std::endl;
|
|
break;
|
|
case SPV_MSG_INFO:
|
|
std::cout << "info: " << position.index << ": " << message << std::endl;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
} // namespace
|
|
|
|
int main(int argc, char** argv) {
|
|
const char* input_filename = nullptr;
|
|
const char* output_filename = nullptr;
|
|
|
|
Task task = kNoTask;
|
|
|
|
if (argc < 3) {
|
|
print_usage(argv[0]);
|
|
return 0;
|
|
}
|
|
|
|
const char* task_char = argv[1];
|
|
if (0 == strcmp("e", task_char)) {
|
|
task = kEncode;
|
|
} else if (0 == strcmp("d", task_char)) {
|
|
task = kDecode;
|
|
}
|
|
|
|
if (task == kNoTask) {
|
|
print_usage(argv[0]);
|
|
return 1;
|
|
}
|
|
|
|
bool want_comments = false;
|
|
|
|
for (int argi = 2; argi < argc; ++argi) {
|
|
if ('-' == argv[argi][0]) {
|
|
switch (argv[argi][1]) {
|
|
case 'h':
|
|
print_usage(argv[0]);
|
|
return 0;
|
|
case 'o': {
|
|
if (!output_filename && argi + 1 < argc) {
|
|
output_filename = argv[++argi];
|
|
} else {
|
|
print_usage(argv[0]);
|
|
return 1;
|
|
}
|
|
} break;
|
|
case '-': {
|
|
if (0 == strcmp(argv[argi], "--help")) {
|
|
print_usage(argv[0]);
|
|
return 0;
|
|
} else if (0 == strcmp(argv[argi], "--comments")) {
|
|
want_comments = true;
|
|
} else if (0 == strcmp(argv[argi], "--version")) {
|
|
fprintf(stderr, "error: Not implemented\n");
|
|
return 1;
|
|
} else {
|
|
print_usage(argv[0]);
|
|
return 1;
|
|
}
|
|
} break;
|
|
case '\0': {
|
|
// Setting a filename of "-" to indicate stdin.
|
|
if (!input_filename) {
|
|
input_filename = argv[argi];
|
|
} else {
|
|
fprintf(stderr, "error: More than one input file specified\n");
|
|
return 1;
|
|
}
|
|
} break;
|
|
default:
|
|
print_usage(argv[0]);
|
|
return 1;
|
|
}
|
|
} else {
|
|
if (!input_filename) {
|
|
input_filename = argv[argi];
|
|
} else {
|
|
fprintf(stderr, "error: More than one input file specified\n");
|
|
return 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (task == kDecode && want_comments) {
|
|
fprintf(stderr, "warning: Decoder comments not yet implemented\n");
|
|
want_comments = false;
|
|
}
|
|
|
|
const bool write_to_stdout = output_filename == nullptr ||
|
|
0 == strcmp(output_filename, "-");
|
|
|
|
spv_text comments = nullptr;
|
|
spv_text* comments_ptr = want_comments ? &comments : nullptr;
|
|
|
|
ScopedContext ctx(SPV_ENV_UNIVERSAL_1_2);
|
|
SetContextMessageConsumer(ctx.context, DiagnosticsMessageHandler);
|
|
|
|
if (task == kEncode) {
|
|
std::vector<uint32_t> contents;
|
|
if (!ReadFile<uint32_t>(input_filename, "rb", &contents)) return 1;
|
|
|
|
std::unique_ptr<spv_markv_encoder_options_t,
|
|
std::function<void(spv_markv_encoder_options_t*)>> options(
|
|
spvMarkvEncoderOptionsCreate(), &spvMarkvEncoderOptionsDestroy);
|
|
spv_markv_binary markv_binary = nullptr;
|
|
|
|
if (SPV_SUCCESS !=
|
|
spvSpirvToMarkv(ctx.context, contents.data(), contents.size(),
|
|
options.get(), &markv_binary, comments_ptr, nullptr)) {
|
|
std::cerr << "error: Failed to encode " << input_filename << " to MARK-V "
|
|
<< std::endl;
|
|
return 1;
|
|
}
|
|
|
|
if (want_comments) {
|
|
if (!WriteFile<char>(nullptr, "w", comments->str,
|
|
comments->length)) return 1;
|
|
}
|
|
|
|
if (!want_comments || !write_to_stdout) {
|
|
if (!WriteFile<uint8_t>(output_filename, "wb", markv_binary->data,
|
|
markv_binary->length)) return 1;
|
|
}
|
|
} else if (task == kDecode) {
|
|
std::vector<uint8_t> contents;
|
|
if (!ReadFile<uint8_t>(input_filename, "rb", &contents)) return 1;
|
|
|
|
std::unique_ptr<spv_markv_decoder_options_t,
|
|
std::function<void(spv_markv_decoder_options_t*)>> options(
|
|
spvMarkvDecoderOptionsCreate(), &spvMarkvDecoderOptionsDestroy);
|
|
spv_binary spirv_binary = nullptr;
|
|
|
|
if (SPV_SUCCESS !=
|
|
spvMarkvToSpirv(ctx.context, contents.data(), contents.size(),
|
|
options.get(), &spirv_binary, comments_ptr, nullptr)) {
|
|
std::cerr << "error: Failed to encode " << input_filename << " to MARK-V "
|
|
<< std::endl;
|
|
return 1;
|
|
}
|
|
|
|
if (want_comments) {
|
|
if (!WriteFile<char>(nullptr, "w", comments->str,
|
|
comments->length)) return 1;
|
|
}
|
|
|
|
if (!want_comments || !write_to_stdout) {
|
|
if (!WriteFile<uint32_t>(output_filename, "wb", spirv_binary->code,
|
|
spirv_binary->wordCount)) return 1;
|
|
}
|
|
} else {
|
|
assert(false && "Unknown task");
|
|
}
|
|
|
|
spvTextDestroy(comments);
|
|
|
|
return 0;
|
|
}
|