Files
fmt/test/fuzzing/chrono-duration.cc
T

137 lines
3.2 KiB
C++
Raw Normal View History

2019-06-30 09:11:13 +02:00
// Copyright (c) 2019, Paul Dreik
// For the license information refer to format.h.
2019-06-30 09:11:13 +02:00
#include <fmt/chrono.h>
2021-08-22 20:18:11 +02:00
#include <cstdint>
2020-10-11 08:30:14 -07:00
#include "fuzzer-common.h"
2019-06-30 09:11:13 +02:00
2020-10-13 08:08:48 -07:00
template <typename Period, typename Rep>
void invoke_inner(fmt::string_view format_str, Rep rep) {
auto value = std::chrono::duration<Rep, Period>(rep);
2019-06-30 09:11:13 +02:00
try {
#if FMT_FUZZ_FORMAT_TO_STRING
std::string message = fmt::format(format_str, value);
2019-06-30 09:11:13 +02:00
#else
2021-07-13 07:56:24 -07:00
auto buf = fmt::memory_buffer();
fmt::format_to(std::back_inserter(buf), format_str, value);
2019-06-30 09:11:13 +02:00
#endif
} catch (std::exception&) {
2019-06-30 09:11:13 +02:00
}
}
2020-10-13 08:08:48 -07:00
// Rep is a duration's representation type.
template <typename Rep>
void invoke_outer(const uint8_t* data, size_t size, int period) {
// Always use a fixed location of the data.
2020-10-13 08:08:48 -07:00
static_assert(sizeof(Rep) <= fixed_size, "fixed size is too small");
if (size <= fixed_size + 1) return;
2019-06-30 09:11:13 +02:00
2020-10-13 08:08:48 -07:00
const Rep rep = assign_from_buf<Rep>(data);
data += fixed_size;
size -= fixed_size;
2019-06-30 09:11:13 +02:00
2021-08-22 20:18:11 +02:00
// data is already allocated separately in libFuzzer so reading past the end
// will most likely be detected anyway.
2020-10-13 08:08:48 -07:00
const auto format_str = fmt::string_view(as_chars(data), size);
2019-06-30 09:11:13 +02:00
// yocto, zepto, zetta and yotta are not handled.
2020-10-13 08:08:48 -07:00
switch (period) {
2019-06-30 09:11:13 +02:00
case 1:
2020-10-13 08:08:48 -07:00
invoke_inner<std::atto>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 2:
2020-10-13 08:08:48 -07:00
invoke_inner<std::femto>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 3:
2020-10-13 08:08:48 -07:00
invoke_inner<std::pico>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 4:
2020-10-13 08:08:48 -07:00
invoke_inner<std::nano>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 5:
2020-10-13 08:08:48 -07:00
invoke_inner<std::micro>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 6:
2020-10-13 08:08:48 -07:00
invoke_inner<std::milli>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 7:
2020-10-13 08:08:48 -07:00
invoke_inner<std::centi>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 8:
2020-10-13 08:08:48 -07:00
invoke_inner<std::deci>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 9:
2020-10-13 08:08:48 -07:00
invoke_inner<std::deca>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 10:
2020-10-13 08:08:48 -07:00
invoke_inner<std::kilo>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 11:
2020-10-13 08:08:48 -07:00
invoke_inner<std::mega>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 12:
2020-10-13 08:08:48 -07:00
invoke_inner<std::giga>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 13:
2020-10-13 08:08:48 -07:00
invoke_inner<std::tera>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 14:
2020-10-13 08:08:48 -07:00
invoke_inner<std::peta>(format_str, rep);
2019-06-30 09:11:13 +02:00
break;
case 15:
2020-10-13 08:08:48 -07:00
invoke_inner<std::exa>(format_str, rep);
2020-10-14 07:13:37 -07:00
break;
2019-06-30 09:11:13 +02:00
}
}
extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) {
2021-08-22 20:18:11 +02:00
if (size <= 4) return 0;
2019-06-30 09:11:13 +02:00
const auto representation = data[0];
2020-10-13 08:08:48 -07:00
const auto period = data[1];
data += 2;
size -= 2;
2019-06-30 09:11:13 +02:00
switch (representation) {
case 1:
2020-10-13 08:08:48 -07:00
invoke_outer<char>(data, size, period);
2019-06-30 09:11:13 +02:00
break;
case 2:
2020-10-13 08:08:48 -07:00
invoke_outer<signed char>(data, size, period);
2019-06-30 09:11:13 +02:00
break;
case 3:
2020-10-13 08:08:48 -07:00
invoke_outer<unsigned char>(data, size, period);
2019-06-30 09:11:13 +02:00
break;
case 4:
2020-10-13 08:08:48 -07:00
invoke_outer<short>(data, size, period);
2019-06-30 09:11:13 +02:00
break;
case 5:
2020-10-13 08:08:48 -07:00
invoke_outer<unsigned short>(data, size, period);
2019-06-30 09:11:13 +02:00
break;
case 6:
2020-10-13 08:08:48 -07:00
invoke_outer<int>(data, size, period);
2019-06-30 09:11:13 +02:00
break;
case 7:
2020-10-13 08:08:48 -07:00
invoke_outer<unsigned int>(data, size, period);
2019-06-30 09:11:13 +02:00
break;
case 8:
2020-10-13 08:08:48 -07:00
invoke_outer<long>(data, size, period);
2019-06-30 09:11:13 +02:00
break;
case 9:
2020-10-13 08:08:48 -07:00
invoke_outer<unsigned long>(data, size, period);
2019-06-30 09:11:13 +02:00
break;
case 10:
2020-10-13 08:08:48 -07:00
invoke_outer<float>(data, size, period);
2019-06-30 09:11:13 +02:00
break;
case 11:
2020-10-13 08:08:48 -07:00
invoke_outer<double>(data, size, period);
2019-06-30 09:11:13 +02:00
break;
case 12:
2020-10-13 08:08:48 -07:00
invoke_outer<long double>(data, size, period);
2019-06-30 09:11:13 +02:00
break;
}
return 0;
}