2021-01-12 16:23:48 +00:00
|
|
|
// 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_
|
|
|
|
|
2021-06-07 08:07:11 +00:00
|
|
|
#include <cstring>
|
2021-05-06 15:43:33 +00:00
|
|
|
#include <string>
|
2021-04-26 20:38:36 +00:00
|
|
|
#include <utility>
|
2021-05-06 15:43:33 +00:00
|
|
|
#include <vector>
|
2021-04-26 20:38:36 +00:00
|
|
|
|
2021-01-12 16:23:48 +00:00
|
|
|
#include "include/tint/tint.h"
|
|
|
|
|
|
|
|
namespace tint {
|
|
|
|
namespace fuzzers {
|
|
|
|
|
2021-05-06 15:43:33 +00:00
|
|
|
class Reader {
|
|
|
|
public:
|
|
|
|
Reader(const uint8_t* data, size_t size);
|
|
|
|
|
2021-07-16 09:25:14 +00:00
|
|
|
bool failed() const { return failed_; }
|
2021-05-06 15:43:33 +00:00
|
|
|
const uint8_t* data() { return data_; }
|
2021-07-16 09:25:14 +00:00
|
|
|
size_t size() const { return size_; }
|
2021-05-06 15:43:33 +00:00
|
|
|
|
|
|
|
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>();
|
2021-07-20 18:59:10 +00:00
|
|
|
auto size = static_cast<size_t>(count) * sizeof(T);
|
|
|
|
if (failed_ || size_ < size) {
|
2021-05-06 15:43:33 +00:00
|
|
|
mark_failed();
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
std::vector<T> out(count);
|
2021-07-19 09:33:19 +00:00
|
|
|
if (!out.empty()) {
|
2021-07-20 18:59:10 +00:00
|
|
|
memcpy(out.data(), data_, size);
|
|
|
|
data_ += size;
|
|
|
|
size_ -= size;
|
2021-07-19 09:33:19 +00:00
|
|
|
}
|
2021-05-06 15:43:33 +00:00
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
std::vector<T> vector(T (*extract)(Reader*)) {
|
|
|
|
auto count = read<uint8_t>();
|
2021-07-20 18:59:10 +00:00
|
|
|
if (failed_) {
|
2021-05-06 15:43:33 +00:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
std::vector<T> out(count);
|
|
|
|
for (uint8_t i = 0; i < count; i++) {
|
|
|
|
out[i] = extract(this);
|
2021-07-20 18:59:10 +00:00
|
|
|
if (failed_) {
|
|
|
|
return {};
|
|
|
|
}
|
2021-05-06 15:43:33 +00:00
|
|
|
}
|
|
|
|
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);
|
2021-04-28 15:35:43 +00:00
|
|
|
|
2021-05-06 15:43:33 +00:00
|
|
|
void ExtractVertexPullingInputs(Reader* r, tint::transform::DataMap* inputs);
|
2021-04-29 20:49:25 +00:00
|
|
|
|
2021-01-12 16:23:48 +00:00
|
|
|
enum class InputFormat { kWGSL, kSpv, kNone };
|
|
|
|
|
|
|
|
enum class OutputFormat { kWGSL, kSpv, kHLSL, kMSL, kNone };
|
|
|
|
|
|
|
|
class CommonFuzzer {
|
|
|
|
public:
|
|
|
|
explicit CommonFuzzer(InputFormat input, OutputFormat output);
|
|
|
|
~CommonFuzzer();
|
|
|
|
|
2021-04-26 20:38:36 +00:00
|
|
|
void SetTransformManager(transform::Manager* tm, transform::DataMap inputs) {
|
|
|
|
transform_manager_ = tm;
|
|
|
|
transform_inputs_ = std::move(inputs);
|
|
|
|
}
|
2021-01-13 15:50:11 +00:00
|
|
|
void EnableInspector() { inspector_enabled_ = true; }
|
2021-01-12 16:23:48 +00:00
|
|
|
|
|
|
|
int Run(const uint8_t* data, size_t size);
|
|
|
|
|
2021-07-16 09:25:14 +00:00
|
|
|
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_; }
|
|
|
|
|
2021-07-20 14:39:50 +00:00
|
|
|
const tint::diag::List& Diagnostics() const { return diagnostics_; }
|
|
|
|
|
|
|
|
bool HasErrors() const { return diagnostics_.contains_errors(); }
|
2021-06-24 18:10:46 +00:00
|
|
|
|
2021-01-12 16:23:48 +00:00
|
|
|
private:
|
|
|
|
InputFormat input_;
|
|
|
|
OutputFormat output_;
|
|
|
|
transform::Manager* transform_manager_;
|
2021-04-26 20:38:36 +00:00
|
|
|
transform::DataMap transform_inputs_;
|
2021-01-13 15:50:11 +00:00
|
|
|
bool inspector_enabled_;
|
2021-07-20 14:39:50 +00:00
|
|
|
tint::diag::List diagnostics_;
|
2021-07-16 09:25:14 +00:00
|
|
|
|
|
|
|
std::vector<uint32_t> generated_spirv_;
|
|
|
|
std::string generated_wgsl_;
|
|
|
|
std::string generated_hlsl_;
|
|
|
|
std::string generated_msl_;
|
2021-01-12 16:23:48 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace fuzzers
|
|
|
|
} // namespace tint
|
|
|
|
|
|
|
|
#endif // FUZZERS_TINT_COMMON_FUZZER_H_
|