aboutsummaryrefslogtreecommitdiff
path: root/fuzz/fuzz_serialization.cpp
diff options
context:
space:
mode:
authorIvan Nardi <12729895+IvanNardi@users.noreply.github.com>2023-01-27 07:09:18 +0100
committerGitHub <noreply@github.com>2023-01-27 07:09:18 +0100
commit6d00a9e0bd4cdb2be26577f53b75b3078cefd4c3 (patch)
tree327e7d1bc8e89a990e99ea8cfb4a748c80d12f32 /fuzz/fuzz_serialization.cpp
parenteea99564308fffca675918506a6e12f7a4f71a6f (diff)
fuzz: add a new fuzzer to test serialization/deserialization code (#1876)
Autodetecting the needed buffer size is quite complex (especially with float/double values) so it is mandatory to properly check for `ndpi_snprintf` truncation. These issues have been undetected so far probably because the default buffer is big enough for all common cases. Add an example of usage of `ndpi_deserialize_clone_all()` (taken from `ntopng`)
Diffstat (limited to 'fuzz/fuzz_serialization.cpp')
-rw-r--r--fuzz/fuzz_serialization.cpp139
1 files changed, 139 insertions, 0 deletions
diff --git a/fuzz/fuzz_serialization.cpp b/fuzz/fuzz_serialization.cpp
new file mode 100644
index 000000000..45596e742
--- /dev/null
+++ b/fuzz/fuzz_serialization.cpp
@@ -0,0 +1,139 @@
+#include "ndpi_api.h"
+#include "fuzz_common_code.h"
+
+#include <stdint.h>
+#include <stdio.h>
+#include <assert.h>
+#include "fuzzer/FuzzedDataProvider.h"
+
+extern "C" int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
+ FuzzedDataProvider fuzzed_data(data, size);
+ u_int16_t i, num_iteration;
+ ndpi_serializer serializer, serializer_cloned, deserializer;
+ ndpi_serialization_format fmt;
+ int rc;
+ std::vector<char>d;
+ char kbuf[32];
+ u_int32_t buffer_len;
+
+ /* To allow memory allocation failures */
+ fuzz_set_alloc_callbacks_and_seed(size);
+
+ fmt = static_cast<ndpi_serialization_format>(fuzzed_data.ConsumeIntegralInRange(1, 3));
+
+ if (fuzzed_data.ConsumeBool())
+ rc = ndpi_init_serializer(&serializer, fmt);
+ else
+ rc = ndpi_init_serializer_ll(&serializer, fmt, fuzzed_data.ConsumeIntegralInRange(0, 64));
+
+ if (rc != 0)
+ return 0;
+
+ if (fmt == ndpi_serialization_format_csv)
+ ndpi_serializer_set_csv_separator(&serializer, ',');
+
+ num_iteration = fuzzed_data.ConsumeIntegralInRange(0, 16);
+ for (i = 0; i < num_iteration; i++) {
+ memset(kbuf, '\0', sizeof(kbuf)); /* It is also used as binary key */
+ snprintf(kbuf, sizeof(kbuf), "Key %d", i);
+
+ ndpi_serialize_uint32_uint32(&serializer, i, fuzzed_data.ConsumeIntegral<u_int32_t>());
+ ndpi_serialize_uint32_int32(&serializer, i, fuzzed_data.ConsumeIntegral<int32_t>());
+ ndpi_serialize_uint32_uint64(&serializer, i, fuzzed_data.ConsumeIntegral<u_int64_t>());
+ ndpi_serialize_uint32_int64(&serializer, i, fuzzed_data.ConsumeIntegral<int64_t>());
+ ndpi_serialize_uint32_float(&serializer, i, fuzzed_data.ConsumeFloatingPoint<float>(), "%f");
+ if (fmt != ndpi_serialization_format_tlv)
+ ndpi_serialize_uint32_double(&serializer, i, fuzzed_data.ConsumeFloatingPoint<double>(), "%lf");
+ d = fuzzed_data.ConsumeBytes<char>(16);
+ ndpi_serialize_uint32_binary(&serializer, i, d.data(), d.size());
+ ndpi_serialize_uint32_string(&serializer, i, fuzzed_data.ConsumeBytesAsString(8).c_str());
+ ndpi_serialize_uint32_boolean(&serializer, i, fuzzed_data.ConsumeIntegral<int8_t>());
+
+ ndpi_serialize_string_uint32(&serializer, kbuf, fuzzed_data.ConsumeIntegral<u_int32_t>());
+ ndpi_serialize_string_int32(&serializer, kbuf, fuzzed_data.ConsumeIntegral<int32_t>());
+ ndpi_serialize_string_uint64(&serializer, kbuf, fuzzed_data.ConsumeIntegral<u_int64_t>());
+ ndpi_serialize_string_int64(&serializer, kbuf, fuzzed_data.ConsumeIntegral<int64_t>());
+ ndpi_serialize_string_float(&serializer, kbuf, fuzzed_data.ConsumeFloatingPoint<float>(), "%f");
+ if (fmt != ndpi_serialization_format_tlv)
+ ndpi_serialize_string_double(&serializer, kbuf, fuzzed_data.ConsumeFloatingPoint<double>(), "%lf");
+ ndpi_serialize_string_string(&serializer, kbuf, fuzzed_data.ConsumeBytesAsString(8).c_str());
+ ndpi_serialize_string_boolean(&serializer, kbuf, fuzzed_data.ConsumeIntegral<int8_t>());
+
+ ndpi_serialize_binary_uint32(&serializer, kbuf, sizeof(kbuf), fuzzed_data.ConsumeIntegral<u_int32_t>());
+ ndpi_serialize_binary_int32(&serializer, kbuf, sizeof(kbuf), fuzzed_data.ConsumeIntegral<int32_t>());
+ ndpi_serialize_binary_uint64(&serializer, kbuf, sizeof(kbuf), fuzzed_data.ConsumeIntegral<u_int64_t>());
+ ndpi_serialize_binary_int64(&serializer, kbuf, sizeof(kbuf), fuzzed_data.ConsumeIntegral<int64_t>());
+ ndpi_serialize_binary_float(&serializer, kbuf, sizeof(kbuf), fuzzed_data.ConsumeFloatingPoint<float>(), "%f");
+ if (fmt != ndpi_serialization_format_tlv)
+ ndpi_serialize_binary_double(&serializer, kbuf, sizeof(kbuf), fuzzed_data.ConsumeFloatingPoint<double>(), "%lf");
+ ndpi_serialize_binary_boolean(&serializer, kbuf, sizeof(kbuf), fuzzed_data.ConsumeIntegral<int8_t>());
+ d = fuzzed_data.ConsumeBytes<char>(16);
+ ndpi_serialize_binary_binary(&serializer, kbuf, sizeof(kbuf), d.data(), d.size());
+
+ if ((i & 0x3) == 0x3)
+ ndpi_serialize_end_of_record(&serializer);
+ }
+
+ ndpi_serializer_create_snapshot(&serializer);
+
+ if (fuzzed_data.ConsumeBool()) {
+ ndpi_serialize_start_of_block(&serializer, "Block");
+ memset(kbuf, '\0', sizeof(kbuf)); /* It is also used as binary key */
+ snprintf(kbuf, sizeof(kbuf), "K-Ignored");
+ ndpi_serialize_uint32_uint32(&serializer, i, fuzzed_data.ConsumeIntegral<u_int32_t>());
+ ndpi_serialize_string_string(&serializer, kbuf, fuzzed_data.ConsumeBytesAsString(8).c_str());
+ ndpi_serialize_string_float(&serializer, kbuf, fuzzed_data.ConsumeFloatingPoint<float>(), "%f");
+ ndpi_serialize_binary_boolean(&serializer, kbuf, sizeof(kbuf), fuzzed_data.ConsumeIntegral<int8_t>());
+ ndpi_serialize_end_of_block(&serializer);
+ }
+
+ if (fuzzed_data.ConsumeBool()) {
+ ndpi_serialize_start_of_block_uint32(&serializer, 0);
+ memset(kbuf, '\0', sizeof(kbuf)); /* It is also used as binary key */
+ snprintf(kbuf, sizeof(kbuf), "K32-Ignored");
+ ndpi_serialize_uint32_uint32(&serializer, i, fuzzed_data.ConsumeIntegral<u_int32_t>());
+ ndpi_serialize_string_string(&serializer, kbuf, fuzzed_data.ConsumeBytesAsString(8).c_str());
+ ndpi_serialize_string_float(&serializer, kbuf, fuzzed_data.ConsumeFloatingPoint<float>(), "%f");
+ ndpi_serialize_binary_boolean(&serializer, kbuf, sizeof(kbuf), fuzzed_data.ConsumeIntegral<int8_t>());
+ ndpi_serialize_end_of_block(&serializer);
+ }
+
+ if (fuzzed_data.ConsumeBool())
+ ndpi_serializer_rollback_snapshot(&serializer);
+
+ if (fmt == ndpi_serialization_format_json) {
+
+ ndpi_serialize_start_of_list(&serializer, "List");
+
+ num_iteration = fuzzed_data.ConsumeIntegralInRange(0, 8);
+ for (i = 0; i < num_iteration; i++) {
+ memset(kbuf, '\0', sizeof(kbuf)); /* It is also used as binary key */
+ snprintf(kbuf, sizeof(kbuf), "Ignored");
+ ndpi_serialize_uint32_uint32(&serializer, i, fuzzed_data.ConsumeIntegral<u_int32_t>());
+ ndpi_serialize_string_string(&serializer, kbuf, fuzzed_data.ConsumeBytesAsString(8).c_str());
+ ndpi_serialize_string_float(&serializer, kbuf, fuzzed_data.ConsumeFloatingPoint<float>(), "%f");
+ ndpi_serialize_binary_boolean(&serializer, kbuf, sizeof(kbuf), fuzzed_data.ConsumeIntegral<int8_t>());
+ }
+
+ ndpi_serialize_end_of_list(&serializer);
+ ndpi_serialize_string_string(&serializer, "Last", "Ok");
+ } else if (fmt == ndpi_serialization_format_csv) {
+ ndpi_serializer_get_header(&serializer, &buffer_len);
+ ndpi_serializer_get_buffer(&serializer, &buffer_len);
+ } else {
+ /* Conversion from tlv to json */
+ rc = ndpi_init_deserializer(&deserializer, &serializer);
+ if (rc == 0) {
+ rc = ndpi_init_serializer_ll(&serializer_cloned, ndpi_serialization_format_json, fuzzed_data.ConsumeIntegralInRange(0, 2048));
+ if (rc == 0) {
+ ndpi_deserialize_clone_all(&deserializer, &serializer_cloned);
+ ndpi_serializer_get_buffer(&serializer_cloned, &buffer_len);
+ ndpi_term_serializer(&serializer_cloned);
+ }
+ }
+ }
+
+ ndpi_term_serializer(&serializer);
+
+ return 0;
+}