1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70
|
#include <cassert>
#include <cmath>
#include <cstddef>
#include <cstdint>
#include <cstring>
#include <glaze/cbor.hpp>
#include <glaze/glaze.hpp>
#include <vector>
// must be outside test() to work in gcc<14
template <typename T>
struct Value
{
T value{};
};
template <typename T>
void test(const uint8_t* Data, size_t Size)
{
using S = Value<T>;
S s{};
if (Size >= sizeof(T)) {
std::memcpy(&s.value, Data, sizeof(T));
}
else {
return;
}
// Only test finite values for equality (NaN != NaN by definition)
if (std::isfinite(s.value)) {
std::string buffer{};
auto write_ec = glz::write_cbor(s, buffer);
if (write_ec) {
return;
}
auto restored = glz::read_cbor<S>(buffer);
assert(restored);
assert(restored.value().value == s.value);
}
else {
// For non-finite values, just test that encoding/decoding doesn't crash
std::string buffer{};
auto write_ec = glz::write_cbor(s, buffer);
if (write_ec) {
return;
}
[[maybe_unused]] auto restored = glz::read_cbor<S>(buffer);
}
}
extern "C" int LLVMFuzzerTestOneInput(const uint8_t* Data, size_t Size)
{
if (Size < 1 + sizeof(float)) {
return 0;
}
const auto action = Data[0];
++Data;
--Size;
switch (action & 0b1) {
case 0:
test<float>(Data, Size);
break;
case 1:
test<double>(Data, Size);
break;
}
return 0;
}
|