blob: 81acf025c726ebfc23734befb21d62fa3616672e [file] [log] [blame]
// Copyright 2020 The Tint Authors.
//
// 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 "src/tint/lang/spirv/writer/writer.h"
#include <memory>
#include <utility>
#include "src/tint/lang/core/ir/transform/binding_remapper.h"
#include "src/tint/lang/spirv/writer/ast_printer/ast_printer.h"
#include "src/tint/lang/spirv/writer/printer/printer.h"
#include "src/tint/lang/spirv/writer/raise/raise.h"
#include "src/tint/lang/wgsl/reader/program_to_ir/program_to_ir.h"
// Included by 'ast_printer.h', included again here for './tools/run gen' track the dependency.
#include "spirv/unified1/spirv.h"
namespace tint::spirv::writer {
Output::Output() = default;
Output::~Output() = default;
Output::Output(const Output&) = default;
Result<Output, std::string> Generate(const Program* program, const Options& options) {
if (!program->IsValid()) {
return std::string("input program is not valid");
}
bool zero_initialize_workgroup_memory =
!options.disable_workgroup_init && options.use_zero_initialize_workgroup_memory_extension;
Output output;
if (options.use_tint_ir) {
// Convert the AST program to an IR module.
auto converted = wgsl::reader::ProgramToIR(program);
if (!converted) {
return "IR converter: " + converted.Failure();
}
auto ir = converted.Move();
// Apply transforms as required by writer options.
auto remapper = core::ir::transform::BindingRemapper(&ir, options.binding_remapper_options);
if (!remapper) {
return remapper.Failure();
}
// Raise the IR to the SPIR-V dialect.
auto raised = raise::Raise(&ir, options);
if (!raised) {
return std::move(raised.Failure());
}
// Generate the SPIR-V code.
auto impl = std::make_unique<Printer>(&ir, zero_initialize_workgroup_memory);
auto spirv = impl->Generate();
if (!spirv) {
return std::move(spirv.Failure());
}
output.spirv = std::move(spirv.Get());
} else {
// Sanitize the program.
auto sanitized_result = Sanitize(program, options);
if (!sanitized_result.program.IsValid()) {
return sanitized_result.program.Diagnostics().str();
}
// Generate the SPIR-V code.
auto impl = std::make_unique<ASTPrinter>(
&sanitized_result.program, zero_initialize_workgroup_memory,
options.experimental_require_subgroup_uniform_control_flow);
if (!impl->Generate()) {
return impl->Diagnostics().str();
}
output.spirv = std::move(impl->Result());
}
return output;
}
} // namespace tint::spirv::writer