diff --git a/fuzzing/CMakeLists.txt b/fuzzing/CMakeLists.txt index 7c5ac4dd..daba61df 100644 --- a/fuzzing/CMakeLists.txt +++ b/fuzzing/CMakeLists.txt @@ -7,11 +7,14 @@ add_library(fuzzhelper NullOStream.h NullOStream.cpp) target_link_libraries(fuzzhelper PUBLIC Catch2::Catch2) +# use C++17 so we can get string_view +target_compile_features(fuzzhelper PUBLIC cxx_std_17) + # This should be possible to set from the outside to be oss-fuzz compatible, # fix later. For now, target libFuzzer only. target_link_options(fuzzhelper PUBLIC "-fsanitize=fuzzer") -foreach(fuzzer TestSpecParser XmlWriter) +foreach(fuzzer TestSpecParser XmlWriter textflow) add_executable(fuzz_${fuzzer} fuzz_${fuzzer}.cpp) target_link_libraries(fuzz_${fuzzer} PRIVATE fuzzhelper) endforeach() diff --git a/fuzzing/NullOStream.h b/fuzzing/NullOStream.h index fc13777b..e1fe15b0 100644 --- a/fuzzing/NullOStream.h +++ b/fuzzing/NullOStream.h @@ -1,6 +1,7 @@ #pragma once -#include +#include +#include // from https://stackoverflow.com/a/8244052 class NullStreambuf : public std::streambuf { @@ -17,4 +18,3 @@ public: virtual void avoidOutOfLineVirtualCompilerWarning(); }; - diff --git a/fuzzing/fuzz_textflow.cpp b/fuzzing/fuzz_textflow.cpp new file mode 100644 index 00000000..eafe79fe --- /dev/null +++ b/fuzzing/fuzz_textflow.cpp @@ -0,0 +1,47 @@ +//License: Boost 1.0 +//By Paul Dreik 2020 + +#include + +#include "NullOStream.h" + +#include +#include + + +template +void split(const char *Data, size_t Size, Callback callback) { + + using namespace std::literals; + constexpr auto sep="\n~~~\n"sv; + + std::string_view remainder(Data,Size); + for (;;) { + auto pos=remainder.find(sep); + if(pos==std::string_view::npos) { + //not found. use the remainder and exit + callback(remainder); + return; + } else { + //found. invoke callback on the first part, then proceed with the rest. + callback(remainder.substr(0,pos)); + remainder=remainder.substr(pos+sep.size()); + } + } +} + +extern "C" int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size) { + + Catch::TextFlow::Columns columns; + + // break the input on separator + split((const char*)Data,Size,[&](std::string_view word) { + columns+=Catch::TextFlow::Column(std::string(word)); + }); + + NullOStream nul; + nul << columns; + + return 0; +} +