• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 #include <cstdint>
2 #include <filesystem>
3 #include <type_traits>
4 
5 #include "64bit/test_64bit_bfbs_generated.h"
6 #include "64bit/test_64bit_generated.h"
7 #include "flatbuffers/base.h"
8 #include "flatbuffers/flatbuffer_builder.h"
9 #include "flatbuffers/flatbuffers.h"
10 #include "flatbuffers/reflection.h"
11 #include "flatbuffers/verifier.h"
12 #include "test_assert.h"
13 #include "test_init.h"
14 
15 OneTimeTestInit OneTimeTestInit::one_time_init_;
16 
17 static RootTableBinarySchema schema;
18 
19 static constexpr uint8_t flags_sized_prefixed = 0b00000001;
20 
21 static const uint64_t kFnvPrime = 0x00000100000001b3ULL;
22 static const uint64_t kOffsetBasis = 0xcbf29ce484222645ULL;
23 
24 namespace flatbuffers {
25 
26 template<typename T, typename = std::enable_if_t<std::is_integral_v<T>>>
Hash(T value,uint64_t hash)27 uint64_t Hash(T value, uint64_t hash) {
28   return (hash * kFnvPrime) ^ value;
29 }
30 
Hash(double value,uint64_t hash)31 uint64_t Hash(double value, uint64_t hash) {
32   static_assert(sizeof(double) == sizeof(uint64_t));
33   return (hash * kFnvPrime) ^ static_cast<uint64_t>(value);
34 }
35 
Hash(const flatbuffers::String * value,uint64_t hash)36 uint64_t Hash(const flatbuffers::String *value, uint64_t hash) {
37   if (value == nullptr) { return hash * kFnvPrime; }
38   for (auto &c : value->str()) { hash = Hash(static_cast<uint8_t>(c), hash); }
39   return hash;
40 }
41 
Hash(const LeafStruct * value,uint64_t hash)42 uint64_t Hash(const LeafStruct *value, uint64_t hash) {
43   if (value == nullptr) { return hash * kFnvPrime; }
44   hash = Hash(value->a(), hash);
45   hash = Hash(value->b(), hash);
46   return hash;
47 }
48 
Hash(const Vector<T> * value,uint64_t hash)49 template<typename T> uint64_t Hash(const Vector<T> *value, uint64_t hash) {
50   if (value == nullptr) { return hash * kFnvPrime; }
51   for (const T c : *value) { hash = Hash(c, hash); }
52   return hash;
53 }
54 
Hash(const Vector64<T> * value,uint64_t hash)55 template<typename T> uint64_t Hash(const Vector64<T> *value, uint64_t hash) {
56   if (value == nullptr) { return hash * kFnvPrime; }
57   for (const T c : *value) { hash = Hash(c, hash); }
58   return hash;
59 }
60 
Hash(const RootTable * value,uint64_t hash)61 uint64_t Hash(const RootTable *value, uint64_t hash) {
62   if (value == nullptr) { return hash * kFnvPrime; }
63   // Hash all the fields so we can exercise all parts of the code.
64   hash = Hash(value->far_vector(), hash);
65   hash = Hash(value->a(), hash);
66   hash = Hash(value->far_string(), hash);
67   hash = Hash(value->big_vector(), hash);
68   hash = Hash(value->near_string(), hash);
69   hash = Hash(value->nested_root(), hash);
70   hash = Hash(value->far_struct_vector(), hash);
71   hash = Hash(value->big_struct_vector(), hash);
72   return hash;
73 }
74 
AccessBuffer(const uint8_t * data,size_t size,bool is_size_prefixed)75 static int AccessBuffer(const uint8_t *data, size_t size,
76                         bool is_size_prefixed) {
77   const RootTable *root_table =
78       is_size_prefixed ? GetSizePrefixedRootTable(data) : GetRootTable(data);
79   TEST_NOTNULL(root_table);
80 
81   uint64_t hash = kOffsetBasis;
82   hash = Hash(root_table, hash);
83   hash = Hash(root_table->nested_root_nested_root(), hash);
84 
85   return 0;
86 }
87 
LLVMFuzzerInitialize(int *,char *** argv)88 extern "C" int LLVMFuzzerInitialize(int *, char ***argv) {
89   Verifier verifier(schema.begin(), schema.size());
90   TEST_EQ(true, reflection::VerifySchemaBuffer(verifier));
91 
92   return 0;
93 }
94 
LLVMFuzzerTestOneInput(const uint8_t * data,size_t size)95 extern "C" int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
96   if (size < FLATBUFFERS_MIN_BUFFER_SIZE) { return 0; }
97 
98   // Take the first bit of data as a flag to control things.
99   const uint8_t flags = data[0];
100   data++;
101   size--;
102 
103   Verifier::Options options;
104   options.assert = true;
105   options.check_alignment = true;
106   options.check_nested_flatbuffers = true;
107 
108   Verifier verifier(data, size, options);
109 
110   const bool is_size_prefixed = flags & flags_sized_prefixed;
111 
112   // Filter out data that isn't valid.
113   if ((is_size_prefixed && !VerifySizePrefixedRootTableBuffer(verifier)) ||
114       !VerifyRootTableBuffer(verifier)) {
115     return 0;
116   }
117 
118   return AccessBuffer(data, size, is_size_prefixed);
119 }
120 
121 }  // namespace flatbuffers