blob: 904ccd3c4b9671a634d11bcf255b144bdacebedb [file] [log] [blame]
// Copyright 2024 The Dawn & Tint Authors
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this
// list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice,
// this list of conditions and the following disclaimer in the documentation
// and/or other materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
// GEN_BUILD:CONDITION(tint_build_wgsl_reader)
#include <iostream>
#include <sstream>
#include <string>
#include <unordered_map>
#include "src/tint/cmd/fuzz/wgsl/fuzz.h"
#include "src/tint/lang/hlsl/validate/validate.h"
#include "src/tint/lang/hlsl/writer/writer.h"
#include "src/tint/lang/wgsl/ast/module.h"
#include "src/tint/lang/wgsl/ast/transform/renamer.h"
#include "src/tint/utils/command/command.h"
#include "src/tint/utils/text/string.h"
namespace tint::hlsl::writer {
namespace {
bool CanRun(const Program& program) {
if (program.AST().HasOverrides()) {
return false;
}
// The PixelLocal transform assumes only a single entry point, so check for multiple entry
// points if chromium_experimental_pixel_local is enabled.
uint32_t num_entry_points = 0;
for (auto* fn : program.AST().Functions()) {
if (fn->PipelineStage() != ast::PipelineStage::kNone) {
num_entry_points++;
}
}
for (auto* enable : program.AST().Enables()) {
if (enable->HasExtension(tint::wgsl::Extension::kChromiumExperimentalPixelLocal)) {
if (num_entry_points > 1) {
return false;
}
break;
}
}
return true;
}
void ASTFuzzer(const tint::Program& program,
const fuzz::wgsl::Context& context,
const Options& options) {
if (!CanRun(program)) {
return;
}
// Currently disabled, as DXC can error on HLSL emitted by Tint. For example: post optimization
// infinite loops will fail to compile, but these are beyond Tint's analysis capabilities.
constexpr bool must_validate = false;
const char* dxc_path = validate::kDxcDLLName;
Result<tint::hlsl::writer::Output> res;
if (!context.options.dxc.empty()) {
dxc_path = context.options.dxc.c_str();
ast::transform::DataMap inputs, outputs;
inputs.Add<ast::transform::Renamer::Config>(ast::transform::Renamer::Target::kHlslKeywords,
/* preserve_unicode */ false);
if (auto renamer_res = tint::ast::transform::Renamer{}.Apply(program, inputs, outputs)) {
if (!renamer_res->IsValid()) {
TINT_ICE() << renamer_res->Diagnostics();
}
res = tint::hlsl::writer::Generate(*renamer_res, options);
}
} else {
res = tint::hlsl::writer::Generate(program, options);
}
if (res != Success) {
return;
}
if (context.options.dump) {
std::cout << "Dumping generated HLSL:\n" << res->hlsl << std::endl;
}
auto dxc = tint::Command::LookPath(dxc_path);
if (dxc.Found()) {
uint32_t hlsl_shader_model = 60;
bool require_16bit_types = false;
auto enable_list = program.AST().Enables();
for (auto* enable : enable_list) {
if (enable->HasExtension(tint::wgsl::Extension::kF16)) {
hlsl_shader_model = 62;
require_16bit_types = true;
break;
}
}
auto validate_res = validate::ValidateUsingDXC(dxc.Path(), res->hlsl, res->entry_points,
require_16bit_types, hlsl_shader_model);
if (must_validate && validate_res.failed) {
size_t line_num = 1;
std::stringstream err;
err << "DXC was expected to succeed, but failed:\n\n";
for (auto line : Split(res->hlsl, "\n")) {
err << line_num++ << ": " << line << "\n";
}
err << "\n\n" << validate_res.output;
TINT_ICE() << err.str();
}
} else if (must_validate) {
TINT_ICE() << "cannot validate with DXC as it was not found at: " << dxc_path;
}
}
} // namespace
} // namespace tint::hlsl::writer
TINT_WGSL_PROGRAM_FUZZER(tint::hlsl::writer::ASTFuzzer);