blob: 922e3483f019c179f761f9f52f03632b100063a0 [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.
#include <iostream>
#include "src/tint/cmd/fuzz/ir/fuzz.h"
#include "src/tint/lang/core/ir/module.h"
#include "src/tint/lang/core/ir/var.h"
#include "src/tint/lang/core/type/input_attachment.h"
#include "src/tint/lang/core/type/pointer.h"
#include "src/tint/lang/hlsl/validate/validate.h"
#include "src/tint/lang/hlsl/writer/helpers/generate_bindings.h"
#include "src/tint/lang/hlsl/writer/writer.h"
#include "src/tint/utils/command/command.h"
namespace tint::hlsl::writer {
namespace {
bool CanRun(const core::ir::Module& module) {
// Check for unsupported module-scope variable address spaces and types.
for (auto* inst : *module.root_block) {
auto* var = inst->As<core::ir::Var>();
auto* ptr = var->Result(0)->Type()->As<core::type::Pointer>();
if (ptr->AddressSpace() == core::AddressSpace::kPushConstant) {
return false;
}
if (ptr->AddressSpace() == core::AddressSpace::kPixelLocal) {
return false;
}
if (ptr->StoreType()->Is<core::type::InputAttachment>()) {
return false;
}
}
return true;
}
// Fuzzed options used to init tint::hlsl::writer::Options
struct FuzzedOptions {
bool disable_robustness;
bool disable_workgroup_init;
bool polyfill_reflect_vec2_f32;
bool polyfill_dot_4x8_packed;
bool disable_polyfill_integer_div_mod;
bool polyfill_pack_unpack_4x8;
bool compiler_is_dxc;
/// Reflect the fields of this class so that it can be used by tint::ForeachField()
TINT_REFLECT(FuzzedOptions,
disable_robustness,
disable_workgroup_init,
polyfill_reflect_vec2_f32,
polyfill_dot_4x8_packed,
disable_polyfill_integer_div_mod,
polyfill_pack_unpack_4x8,
compiler_is_dxc);
};
Result<SuccessType> IRFuzzer(core::ir::Module& module,
const fuzz::ir::Context& context,
FuzzedOptions fuzzed_options) {
if (!CanRun(module)) {
return Failure{"Cannot run module"};
}
Options options;
options.disable_robustness = fuzzed_options.disable_robustness;
options.disable_workgroup_init = fuzzed_options.disable_workgroup_init;
options.polyfill_reflect_vec2_f32 = fuzzed_options.polyfill_reflect_vec2_f32;
options.polyfill_dot_4x8_packed = fuzzed_options.polyfill_dot_4x8_packed;
options.disable_polyfill_integer_div_mod = fuzzed_options.disable_polyfill_integer_div_mod;
options.polyfill_pack_unpack_4x8 = fuzzed_options.polyfill_pack_unpack_4x8;
options.compiler =
fuzzed_options.compiler_is_dxc ? Options::Compiler::kDXC : Options::Compiler::kFXC;
options.bindings = GenerateBindings(module);
options.array_length_from_uniform.ubo_binding = {30, 0};
// Add array_length_from_uniform entries for all storage buffers with runtime sized arrays.
std::unordered_set<tint::BindingPoint> storage_bindings;
for (auto* inst : *module.root_block) {
auto* var = inst->As<core::ir::Var>();
if (!var->Result(0)->Type()->UnwrapPtr()->HasFixedFootprint()) {
if (auto bp = var->BindingPoint()) {
if (storage_bindings.insert(bp.value()).second) {
options.array_length_from_uniform.bindpoint_to_size_index.emplace(
bp.value(), static_cast<uint32_t>(storage_bindings.size() - 1));
}
}
}
}
auto output = Generate(module, options);
if (output == Success && context.options.dump) {
std::cout << "Dumping generated HLSL:\n" << output->hlsl << "\n";
}
// Run DXC against generated HLSL in order to fuzz it. Note that we ignore whether it succeeds
// as DXC may fail on valid HLSL emitted by Tint. For example: post optimization infinite loops
// will fail to compile, but these are beyond Tint's analysis capabilities.
const char* dxc_path = validate::kDxcDLLName;
if (!context.options.dxc.empty()) {
dxc_path = context.options.dxc.c_str();
}
auto dxc = tint::Command::LookPath(dxc_path);
if (dxc.Found()) {
uint32_t hlsl_shader_model = 66;
bool require_16bit_types = true;
[[maybe_unused]] auto validate_res = validate::ValidateUsingDXC(
dxc.Path(), output->hlsl, output->entry_points, require_16bit_types, hlsl_shader_model);
}
return Success;
}
} // namespace
} // namespace tint::hlsl::writer
TINT_IR_MODULE_FUZZER(tint::hlsl::writer::IRFuzzer, tint::core::ir::Capabilities{});