33#if defined(PROVIDE_FUZZ_MAIN_FUNCTION) && defined(__AFL_FUZZ_INIT)
46static std::vector<const char*>
g_args;
48static void SetArgs(
int argc,
char** argv) {
49 for (
int i = 1; i < argc; ++i) {
52 if (strlen(argv[i]) > 2 && argv[i][0] ==
'-' && argv[i][1] ==
'-') {
69 static std::map<std::string_view, FuzzTarget> g_fuzz_targets;
70 return g_fuzz_targets;
92#if defined(__clang__) && defined(__linux__)
93extern "C" void __llvm_profile_reset_counters(
void) __attribute__((weak));
94extern "C" void __gcov_reset(
void) __attribute__((weak));
98 if (__llvm_profile_reset_counters) {
99 __llvm_profile_reset_counters();
124 CreateSock = [](int, int, int) -> std::unique_ptr<Sock> { std::terminate(); };
134 bool should_exit{
false};
135 if (std::getenv(
"PRINT_ALL_FUZZ_TARGETS_AND_ABORT")) {
137 if (
t.opts.hidden)
continue;
138 std::cout <<
name << std::endl;
142 if (
const char* out_path = std::getenv(
"WRITE_ALL_FUZZ_TARGETS_AND_ABORT")) {
143 std::cout <<
"Writing all fuzz target names to '" << out_path <<
"'." << std::endl;
144 std::ofstream out_stream{out_path, std::ios::binary};
146 if (
t.opts.hidden)
continue;
147 out_stream <<
name << std::endl;
154 if (
const auto* env_fuzz{std::getenv(
"FUZZ")}) {
156 static std::string g_copy{env_fuzz};
159 std::cerr <<
"Must select fuzz target with the FUZZ env var." << std::endl;
160 std::cerr <<
"Hint: Set the PRINT_ALL_FUZZ_TARGETS_AND_ABORT=1 env var to see all compiled targets." << std::endl;
161 std::exit(EXIT_FAILURE);
165 std::cerr <<
"No fuzz target compiled for " <<
g_fuzz_target <<
"." << std::endl;
166 std::exit(EXIT_FAILURE);
169 std::cerr <<
"Must compile with -DBUILD_FOR_FUZZING=ON to execute a fuzz target." << std::endl;
170 std::exit(EXIT_FAILURE);
174 it->second.opts.init();
179#if defined(PROVIDE_FUZZ_MAIN_FUNCTION)
180static bool read_stdin(std::vector<uint8_t>&
data)
182 std::istream::char_type buffer[1024];
183 std::streamsize length;
184 while ((std::cin.read(buffer, 1024), length = std::cin.gcount()) > 0) {
185 data.insert(
data.end(), buffer, buffer + length);
191#if defined(PROVIDE_FUZZ_MAIN_FUNCTION) && !defined(__AFL_LOOP)
192static bool read_file(
fs::path p, std::vector<uint8_t>&
data)
194 uint8_t buffer[1024];
196 if (f ==
nullptr)
return false;
198 const size_t length = fread(buffer,
sizeof(uint8_t),
sizeof(buffer), f);
199 if (ferror(f))
return false;
200 data.insert(
data.end(), buffer, buffer + length);
207#if defined(PROVIDE_FUZZ_MAIN_FUNCTION) && !defined(__AFL_LOOP)
209void signal_handler(
int signal)
211 if (signal == SIGABRT) {
212 std::cerr <<
"Error processing input " << g_input_path << std::endl;
214 std::cerr <<
"Unexpected signal " << signal <<
" received\n";
216 std::_Exit(EXIT_FAILURE);
235#if defined(PROVIDE_FUZZ_MAIN_FUNCTION)
236int main(
int argc,
char** argv)
242 const uint8_t* buffer = __AFL_FUZZ_TESTCASE_BUF;
243 while (__AFL_LOOP(100000)) {
244 size_t buffer_len = __AFL_FUZZ_TESTCASE_LEN;
248 std::vector<uint8_t> buffer;
250 if (!read_stdin(buffer)) {
256 std::signal(SIGABRT, signal_handler);
257 const auto start_time{Now<SteadySeconds>()};
259 for (
int i = 1; i < argc; ++i) {
261 if (fs::is_directory(input_path)) {
262 for (fs::directory_iterator it(input_path); it != fs::directory_iterator(); ++it) {
263 if (!fs::is_regular_file(it->path()))
continue;
264 g_input_path = it->
path();
265 Assert(read_file(it->path(), buffer));
271 g_input_path = input_path;
272 Assert(read_file(input_path, buffer));
278 const auto end_time{Now<SteadySeconds>()};
279 std::cout <<
g_fuzz_target <<
": succeeded against " << tested <<
" files in " <<
count_seconds(end_time - start_time) <<
"s." << std::endl;
int main(int argc, char **argv)
#define Assert(val)
Identity function.
Path class wrapper to block calls to the fs::path(std::string) implicit constructor and the fs::path:...
path(std::filesystem::path path)
void FuzzFrameworkRegisterTarget(std::string_view name, TypeTestOneInput target, FuzzTargetOptions opts)
const std::function< void(const std::string &)> G_TEST_LOG_FUN
This is connected to the logger.
static const TypeTestOneInput * g_test_one_input
static void SetArgs(int argc, char **argv)
const std::function< std::vector< const char * >()> G_TEST_COMMAND_LINE_ARGUMENTS
Retrieve the command line arguments.
int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size)
int LLVMFuzzerInitialize(int *argc, char ***argv)
static std::string_view g_fuzz_target
void ResetCoverageCounters()
static std::vector< const char * > g_args
A copy of the command line arguments that start with --.
const std::function< std::string()> G_TEST_GET_FULL_NAME
Retrieve the unit test name.
void test_one_input(FuzzBufferType buffer)
std::span< const uint8_t > FuzzBufferType
std::function< void(FuzzBufferType)> TypeTestOneInput
FILE * fopen(const fs::path &p, const char *mode)
std::function< std::unique_ptr< Sock >(int, int, int)> CreateSock
Socket factory.
std::vector< CNetAddr > WrappedGetAddrInfo(const std::string &name, bool allow_lookup)
Wrapper for getaddrinfo(3).
const TypeTestOneInput test_one_input
const FuzzTargetOptions opts
void SeedRandomStateForTest(SeedRand seedtype)
Seed the global RNG state for testing and log the seed value.
@ ZEROS
Seed with a compile time constant of zeros.
void SetMockTime(int64_t nMockTimeIn)
DEPRECATED Use SetMockTime with chrono type.
constexpr int64_t count_seconds(std::chrono::seconds t)