spv: Build internal representation of the SPIR-V

For convenience, use the SPIRV-Tools' optimizer representation.

Bug: tint:3
Change-Id: I1b046209584e1e907045d496b0f8d7b36fca79bd
Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/16660
Reviewed-by: dan sinclair <dsinclair@google.com>
This commit is contained in:
David Neto 2020-03-13 20:21:45 +00:00
parent e5dc367a7e
commit 9bb0dc543d
3 changed files with 110 additions and 20 deletions

View File

@ -306,14 +306,32 @@ set(TINT_TEST_SRCS
writer/wgsl/generator_impl_test.cc writer/wgsl/generator_impl_test.cc
) )
function(tint_spvtools_compile_options TARGET)
# We'll use the optimizer for its nice SPIR-V in-memory representation
target_link_libraries(libtint SPIRV-Tools-opt SPIRV-Tools)
# We'll be cheating: using internal interfaces to the SPIRV-Tools
# optimizer.
target_include_directories(libtint PRIVATE ${spirv-tools_SOURCE_DIR} ${spirv-tools_BINARY_DIR})
if (${CMAKE_CXX_COMPILER_ID} MATCHES Clang)
# The SPIRV-Tools code is conditioned against C++ and an older version of Clang.
# Suppress warnings triggered in our current compilation environment.
# TODO(dneto): Fix the issues upstream.
target_compile_options(${TARGET} PRIVATE
-Wno-newline-eof
-Wno-sign-conversion
-Wno-old-style-cast
-Wno-weak-vtables
)
endif()
endfunction()
## Tint library ## Tint library
add_library(libtint ${TINT_LIB_SRCS}) add_library(libtint ${TINT_LIB_SRCS})
tint_default_compile_options(libtint) tint_default_compile_options(libtint)
set_target_properties(libtint PROPERTIES OUTPUT_NAME "tint") set_target_properties(libtint PROPERTIES OUTPUT_NAME "tint")
if(${TINT_BUILD_SPV_PARSER}) if(${TINT_BUILD_SPV_PARSER})
# We'll use the optimizer for its nice SPIR-V in-memory representation tint_spvtools_compile_options(libtint)
target_link_libraries(libtint SPIRV-Tools-opt SPIRV-Tools)
endif() endif()
if(${TINT_BUILD_SPV_PARSER}) if(${TINT_BUILD_SPV_PARSER})
@ -325,6 +343,9 @@ if(${TINT_BUILD_SPV_PARSER})
endif() endif()
add_executable(tint_unittests ${TINT_TEST_SRCS}) add_executable(tint_unittests ${TINT_TEST_SRCS})
if (${TINT_BUILD_SPV_PARSER})
tint_spvtools_compile_options(tint_unittests)
endif()
if (NOT MSVC) if (NOT MSVC)
target_compile_options(tint_unittests PRIVATE target_compile_options(tint_unittests PRIVATE
-Wno-global-constructors -Wno-global-constructors
@ -335,6 +356,10 @@ endif()
## Test executable ## Test executable
target_include_directories( target_include_directories(
tint_unittests PRIVATE ${gmock_SOURCE_DIR}/include) tint_unittests PRIVATE ${gmock_SOURCE_DIR}/include)
if(${TINT_BUILD_SPV_PARSER})
target_include_directories(tint_unittests
PRIVATE ${spirv-tools_SOURCE_DIR} ${spirv-tools_BINARY_DIR})
endif()
target_link_libraries(tint_unittests libtint gmock_main) target_link_libraries(tint_unittests libtint gmock_main)
tint_default_compile_options(tint_unittests) tint_default_compile_options(tint_unittests)

View File

@ -14,6 +14,7 @@
#include <cstring> #include <cstring>
#include "source/opt/build_module.h"
#include "spirv-tools/libspirv.hpp" #include "spirv-tools/libspirv.hpp"
#include "src/reader/spv/parser_impl.h" #include "src/reader/spv/parser_impl.h"
@ -21,8 +22,37 @@ namespace tint {
namespace reader { namespace reader {
namespace spv { namespace spv {
namespace {
const spv_target_env kTargetEnv = SPV_ENV_WEBGPU_0;
} // namespace
ParserImpl::ParserImpl(const std::vector<uint32_t>& spv_binary) ParserImpl::ParserImpl(const std::vector<uint32_t>& spv_binary)
: Reader(), spv_binary_(spv_binary), fail_stream_(&success_, &errors_) {} : Reader(),
spv_binary_(spv_binary),
fail_stream_(&success_, &errors_),
tools_context_(kTargetEnv),
tools_(kTargetEnv) {
// Create a message consumer to propagate error messages from SPIRV-Tools
// out as our own failures.
message_consumer_ = [this](spv_message_level_t level, const char* /*source*/,
const spv_position_t& position,
const char* message) {
switch (level) {
// Ignore info and warning message.
case SPV_MSG_WARNING:
case SPV_MSG_INFO:
break;
// Otherwise, propagate the error.
default:
// For binary validation errors, we only have the instruction
// number. It's not text, so there is no column number.
this->Fail() << "line:" << position.index << ": " << message;
this->Fail() << "error: line " << position.index << ": " << message;
}
};
}
ParserImpl::~ParserImpl() = default; ParserImpl::~ParserImpl() = default;
@ -32,30 +62,20 @@ bool ParserImpl::Parse() {
} }
// Set up use of SPIRV-Tools utilities. // Set up use of SPIRV-Tools utilities.
// TODO(dneto): Add option to handle other environments. spvtools::SpirvTools spv_tools(kTargetEnv);
spvtools::SpirvTools spv_tools(SPV_ENV_WEBGPU_0);
// Error messages from SPIRV-Tools are forwarded as failures. // Error messages from SPIRV-Tools are forwarded as failures.
auto message_consumer = spv_tools.SetMessageConsumer(message_consumer_);
[this](spv_message_level_t level, const char* /*source*/,
const spv_position_t& position, const char* message) {
switch (level) {
// Drop info and warning message.
case SPV_MSG_WARNING:
case SPV_MSG_INFO:
default:
// For binary validation errors, we only have the instruction
// number. It's not text, so there is no column number.
this->Fail() << "line:" << position.index << ": " << message;
}
};
spv_tools.SetMessageConsumer(message_consumer);
// Only consider valid modules. // Only consider valid modules.
if (success_) { if (success_) {
success_ = spv_tools.Validate(spv_binary_); success_ = spv_tools.Validate(spv_binary_);
} }
if (success_) {
success_ = BuildInternalModule();
}
return success_; return success_;
} }
@ -65,6 +85,25 @@ ast::Module ParserImpl::module() {
return std::move(ast_module_); return std::move(ast_module_);
} }
bool ParserImpl::BuildInternalModule() {
tools_.SetMessageConsumer(message_consumer_);
const spv_context& context = tools_context_.CContext();
ir_context_ = spvtools::BuildModule(context->target_env, context->consumer,
spv_binary_.data(), spv_binary_.size());
if (!ir_context_) {
return Fail() << "internal error: couldn't build the internal "
"representation of the module";
}
module_ = ir_context_->module();
def_use_mgr_ = ir_context_->get_def_use_mgr();
constant_mgr_ = ir_context_->get_constant_mgr();
type_mgr_ = ir_context_->get_type_mgr();
deco_mgr_ = ir_context_->get_decoration_mgr();
return true;
}
} // namespace spv } // namespace spv
} // namespace reader } // namespace reader
} // namespace tint } // namespace tint

View File

@ -20,6 +20,12 @@
#include <sstream> #include <sstream>
#include <vector> #include <vector>
#include "source/opt/constants.h"
#include "source/opt/decoration_manager.h"
#include "source/opt/ir_context.h"
#include "source/opt/module.h"
#include "source/opt/type_manager.h"
#include "spirv-tools/libspirv.hpp"
#include "src/reader/reader.h" #include "src/reader/reader.h"
#include "src/reader/spv/fail_stream.h" #include "src/reader/spv/fail_stream.h"
@ -56,6 +62,13 @@ class ParserImpl : Reader {
const std::string error() { return errors_.str(); } const std::string error() { return errors_.str(); }
private: private:
/// Builds the internal representation of the SPIR-V module.
/// Assumes the module is somewhat well-formed. Normally you
/// would want to validate the SPIR-V module before attempting
/// to build this internal representation.
/// @returns true if successful.
bool BuildInternalModule();
// The SPIR-V binary we're parsing // The SPIR-V binary we're parsing
std::vector<uint32_t> spv_binary_; std::vector<uint32_t> spv_binary_;
@ -67,6 +80,19 @@ class ParserImpl : Reader {
// Collector for diagnostic messages. // Collector for diagnostic messages.
std::stringstream errors_; std::stringstream errors_;
FailStream fail_stream_; FailStream fail_stream_;
spvtools::MessageConsumer message_consumer_;
// The internal representation of the SPIR-V module and its context.
spvtools::Context tools_context_;
spvtools::SpirvTools tools_;
// All the state is owned by ir_context_.
std::unique_ptr<spvtools::opt::IRContext> ir_context_;
// The following are borrowed pointers to the internal state of ir_context_.
spvtools::opt::Module* module_ = nullptr;
spvtools::opt::analysis::DefUseManager* def_use_mgr_ = nullptr;
spvtools::opt::analysis::ConstantManager* constant_mgr_ = nullptr;
spvtools::opt::analysis::TypeManager* type_mgr_ = nullptr;
spvtools::opt::analysis::DecorationManager* deco_mgr_ = nullptr;
}; };
} // namespace spv } // namespace spv