blob: d955b014a3ffe8fcedc009ec2ba723cbd1ec0c12 [file] [log] [blame]
// Copyright 2021 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.
#ifndef FUZZERS_TINT_COMMON_FUZZER_H_
#define FUZZERS_TINT_COMMON_FUZZER_H_
#include <cstring>
#include <memory>
#include <string>
#include <utility>
#include <vector>
#include "include/tint/tint.h"
namespace tint {
namespace fuzzers {
class Reader {
public:
Reader(const uint8_t* data, size_t size);
bool failed() const { return failed_; }
const uint8_t* data() { return data_; }
size_t size() const { return size_; }
template <typename T>
T read() {
T out{};
read(&out, sizeof(T));
return out;
}
std::string string();
template <typename T>
std::vector<T> vector() {
auto count = read<uint8_t>();
auto size = static_cast<size_t>(count) * sizeof(T);
if (failed_ || size_ < size) {
mark_failed();
return {};
}
std::vector<T> out(count);
if (!out.empty()) {
memcpy(out.data(), data_, size);
data_ += size;
size_ -= size;
}
return out;
}
template <typename T>
std::vector<T> vector(T (*extract)(Reader*)) {
auto count = read<uint8_t>();
if (failed_) {
return {};
}
std::vector<T> out(count);
for (uint8_t i = 0; i < count; i++) {
out[i] = extract(this);
if (failed_) {
return {};
}
}
return out;
}
template <typename T>
T enum_class(uint8_t count) {
auto val = read<uint8_t>();
return static_cast<T>(val % count);
}
private:
void mark_failed();
void read(void* out, size_t n);
const uint8_t* data_;
size_t size_;
bool failed_ = false;
};
void ExtractBindingRemapperInputs(Reader* r, tint::transform::DataMap* inputs);
void ExtractFirstIndexOffsetInputs(Reader* r, tint::transform::DataMap* inputs);
void ExtractSingleEntryPointInputs(Reader* r, tint::transform::DataMap* inputs);
void ExtractVertexPullingInputs(Reader* r, tint::transform::DataMap* inputs);
void ExtractSpirvOptions(Reader* r, writer::spirv::Options* options);
void ExtractWgslOptions(Reader* r, writer::wgsl::Options* options);
void ExtractHlslOptions(Reader* r, writer::hlsl::Options* options);
void ExtractMslOptions(Reader* r, writer::msl::Options* options);
enum class InputFormat { kWGSL, kSpv, kNone };
enum class OutputFormat { kWGSL, kSpv, kHLSL, kMSL, kNone };
class CommonFuzzer {
public:
explicit CommonFuzzer(InputFormat input, OutputFormat output);
~CommonFuzzer();
void SetTransformManager(transform::Manager* tm, transform::DataMap inputs) {
transform_manager_ = tm;
transform_inputs_ = std::move(inputs);
}
void EnableInspector() { inspector_enabled_ = true; }
void SetDumpInput(bool enabled) { dump_input_ = enabled; }
int Run(const uint8_t* data, size_t size);
const tint::diag::List& Diagnostics() const { return diagnostics_; }
bool HasErrors() const { return diagnostics_.contains_errors(); }
const std::vector<uint32_t>& GetGeneratedSpirv() const {
return generated_spirv_;
}
const std::string& GetGeneratedWgsl() const { return generated_wgsl_; }
const std::string& GetGeneratedHlsl() const { return generated_hlsl_; }
const std::string& GetGeneratedMsl() const { return generated_msl_; }
void SetOptionsSpirv(const writer::spirv::Options& options) {
options_spirv_ = options;
}
void SetOptionsWgsl(const writer::wgsl::Options& options) {
options_wgsl_ = options;
}
void SetOptionsHlsl(const writer::hlsl::Options& options) {
options_hlsl_ = options;
}
void SetOptionsMsl(const writer::msl::Options& options) {
options_msl_ = options;
}
private:
InputFormat input_;
OutputFormat output_;
transform::Manager* transform_manager_;
transform::DataMap transform_inputs_;
bool inspector_enabled_;
bool dump_input_ = false;
tint::diag::List diagnostics_;
std::vector<uint32_t> generated_spirv_;
std::string generated_wgsl_;
std::string generated_hlsl_;
std::string generated_msl_;
writer::spirv::Options options_spirv_;
writer::wgsl::Options options_wgsl_;
writer::hlsl::Options options_hlsl_;
writer::msl::Options options_msl_;
#if TINT_BUILD_WGSL_READER
/// The source file needs to live at least as long as #diagnostics_
std::unique_ptr<Source::File> file_;
#endif // TINT_BUILD_WGSL_READER
};
} // namespace fuzzers
} // namespace tint
#endif // FUZZERS_TINT_COMMON_FUZZER_H_