diff --git a/.circleci/config.yml b/.circleci/config.yml
index fff6e6b67..1dc01451d 100644
--- a/.circleci/config.yml
+++ b/.circleci/config.yml
@@ -90,7 +90,9 @@ defaults:
paths:
- test/tools/ossfuzz/abiv2_proto_ossfuzz
- test/tools/ossfuzz/const_opt_ossfuzz
+ - test/tools/ossfuzz/solc_noopt_mutator_ossfuzz
- test/tools/ossfuzz/solc_noopt_ossfuzz
+ - test/tools/ossfuzz/solc_opt_mutator_ossfuzz
- test/tools/ossfuzz/solc_opt_ossfuzz
- test/tools/ossfuzz/strictasm_assembly_ossfuzz
- test/tools/ossfuzz/strictasm_diff_ossfuzz
diff --git a/test/tools/ossfuzz/CMakeLists.txt b/test/tools/ossfuzz/CMakeLists.txt
index beab9e79c..a6d4addc1 100644
--- a/test/tools/ossfuzz/CMakeLists.txt
+++ b/test/tools/ossfuzz/CMakeLists.txt
@@ -1,12 +1,14 @@
add_custom_target(ossfuzz)
add_dependencies(ossfuzz
solc_opt_ossfuzz
+ solc_opt_mutator_ossfuzz
solc_noopt_ossfuzz
+ solc_noopt_mutator_ossfuzz
const_opt_ossfuzz
strictasm_diff_ossfuzz
strictasm_opt_ossfuzz
strictasm_assembly_ossfuzz
- )
+)
if (OSSFUZZ)
@@ -31,6 +33,16 @@ if (OSSFUZZ)
target_link_libraries(solc_opt_ossfuzz PRIVATE libsolc evmasm)
set_target_properties(solc_opt_ossfuzz PROPERTIES LINK_FLAGS ${LIB_FUZZING_ENGINE})
+ add_executable(solc_opt_mutator_ossfuzz
+ solc_opt_ossfuzz.cpp
+ ../fuzzer_common.cpp
+ ../../TestCaseReader.cpp
+ SolidityGenerator.cpp
+ SolidityCustomMutatorInterface.cpp
+ )
+ target_link_libraries(solc_opt_mutator_ossfuzz PRIVATE libsolc evmasm)
+ set_target_properties(solc_opt_mutator_ossfuzz PROPERTIES LINK_FLAGS ${LIB_FUZZING_ENGINE})
+
add_executable(solc_noopt_ossfuzz
solc_noopt_ossfuzz.cpp
../fuzzer_common.cpp
@@ -39,6 +51,16 @@ if (OSSFUZZ)
target_link_libraries(solc_noopt_ossfuzz PRIVATE libsolc evmasm)
set_target_properties(solc_noopt_ossfuzz PROPERTIES LINK_FLAGS ${LIB_FUZZING_ENGINE})
+ add_executable(solc_noopt_mutator_ossfuzz
+ solc_noopt_ossfuzz.cpp
+ ../fuzzer_common.cpp
+ ../../TestCaseReader.cpp
+ SolidityGenerator.cpp
+ SolidityCustomMutatorInterface.cpp
+ )
+ target_link_libraries(solc_noopt_mutator_ossfuzz PRIVATE libsolc evmasm)
+ set_target_properties(solc_noopt_mutator_ossfuzz PROPERTIES LINK_FLAGS ${LIB_FUZZING_ENGINE})
+
add_executable(const_opt_ossfuzz const_opt_ossfuzz.cpp ../fuzzer_common.cpp)
target_link_libraries(const_opt_ossfuzz PRIVATE libsolc evmasm)
set_target_properties(const_opt_ossfuzz PROPERTIES LINK_FLAGS ${LIB_FUZZING_ENGINE})
@@ -82,14 +104,14 @@ if (OSSFUZZ)
protoToYul.cpp
yulProto.pb.cc
protomutators/YulProtoMutator.cpp
-)
+ )
target_include_directories(yul_proto_diff_custom_mutate_ossfuzz PRIVATE /usr/include/libprotobuf-mutator)
target_link_libraries(yul_proto_diff_custom_mutate_ossfuzz PRIVATE yul
yulInterpreter
protobuf-mutator-libfuzzer.a
protobuf-mutator.a
protobuf.a
-)
+ )
set_target_properties(yul_proto_diff_custom_mutate_ossfuzz PROPERTIES LINK_FLAGS ${LIB_FUZZING_ENGINE})
target_compile_options(yul_proto_diff_custom_mutate_ossfuzz PUBLIC ${COMPILE_OPTIONS} -Wno-sign-conversion)
@@ -139,12 +161,24 @@ else()
)
target_link_libraries(solc_opt_ossfuzz PRIVATE libsolc evmasm)
+ add_library(solc_opt_mutator_ossfuzz
+ solc_opt_ossfuzz.cpp
+ ../fuzzer_common.cpp
+ )
+ target_link_libraries(solc_opt_mutator_ossfuzz PRIVATE libsolc evmasm)
+
add_library(solc_noopt_ossfuzz
solc_noopt_ossfuzz.cpp
../fuzzer_common.cpp
)
target_link_libraries(solc_noopt_ossfuzz PRIVATE libsolc evmasm)
+ add_library(solc_noopt_mutator_ossfuzz
+ solc_noopt_ossfuzz.cpp
+ ../fuzzer_common.cpp
+ )
+ target_link_libraries(solc_noopt_mutator_ossfuzz PRIVATE libsolc evmasm)
+
add_library(const_opt_ossfuzz
const_opt_ossfuzz.cpp
../fuzzer_common.cpp)
diff --git a/test/tools/ossfuzz/SolidityCustomMutatorInterface.cpp b/test/tools/ossfuzz/SolidityCustomMutatorInterface.cpp
new file mode 100644
index 000000000..ac56860b9
--- /dev/null
+++ b/test/tools/ossfuzz/SolidityCustomMutatorInterface.cpp
@@ -0,0 +1,69 @@
+/*
+ This file is part of solidity.
+
+ solidity is free software: you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation, either version 3 of the License, or
+ (at your option) any later version.
+
+ solidity is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with solidity. If not, see .
+*/
+// SPDX-License-Identifier: GPL-3.0
+
+#include
+#include
+
+#include
+
+using namespace std;
+using namespace solidity::test::fuzzer;
+
+namespace
+{
+/// Forward declare libFuzzer's default mutator definition
+extern "C" size_t LLVMFuzzerMutate(uint8_t* _data, size_t _size, size_t _maxSize);
+
+/// Define Solidity's custom mutator by implementing libFuzzer's
+/// custom mutator external interface.
+extern "C" size_t LLVMFuzzerCustomMutator(
+ uint8_t* _data,
+ size_t _size,
+ size_t _maxSize,
+ unsigned int _seed
+)
+{
+ if (_maxSize <= _size || _size == 0)
+ return LLVMFuzzerMutate(_data, _size, _maxSize);
+ return SolidityCustomMutatorInterface{_data, _size, _maxSize, _seed}.generate();
+}
+}
+
+SolidityCustomMutatorInterface::SolidityCustomMutatorInterface(
+ uint8_t* _data,
+ size_t _size,
+ size_t _maxSize,
+ unsigned int _seed
+ ):
+ data(_data),
+ size(_size),
+ maxMutantSize(_maxSize),
+ generator(make_shared(_seed))
+{}
+
+size_t SolidityCustomMutatorInterface::generate()
+{
+ string testCase = generator->generateTestProgram();
+ solAssert(
+ !testCase.empty() && data,
+ "Solc custom mutator: Invalid mutant or memory pointer"
+ );
+ size_t mutantSize = min(testCase.size(), maxMutantSize - 1);
+ mempcpy(data, testCase.data(), mutantSize);
+ return mutantSize;
+}
diff --git a/test/tools/ossfuzz/SolidityCustomMutatorInterface.h b/test/tools/ossfuzz/SolidityCustomMutatorInterface.h
new file mode 100644
index 000000000..2b7be6cc5
--- /dev/null
+++ b/test/tools/ossfuzz/SolidityCustomMutatorInterface.h
@@ -0,0 +1,46 @@
+/*
+ This file is part of solidity.
+
+ solidity is free software: you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation, either version 3 of the License, or
+ (at your option) any later version.
+
+ solidity is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with solidity. If not, see .
+*/
+// SPDX-License-Identifier: GPL-3.0
+/**
+ * Implements libFuzzer's custom mutator interface.
+ */
+
+#pragma once
+
+#include
+
+#include
+
+namespace solidity::test::fuzzer
+{
+struct SolidityCustomMutatorInterface
+{
+ SolidityCustomMutatorInterface(uint8_t* _data, size_t _size, size_t _maxSize, unsigned _seed);
+ /// Generates Solidity test program, copies it into buffer
+ /// provided by libFuzzer and @returns size of the test program.
+ size_t generate();
+
+ /// Raw pointer to libFuzzer provided input
+ uint8_t* data;
+ /// Size of libFuzzer provided input
+ size_t size;
+ /// Maximum length of mutant specified by libFuzzer
+ size_t maxMutantSize;
+ /// Solidity generator handle
+ std::shared_ptr generator;
+};
+}
diff --git a/test/tools/ossfuzz/SolidityGenerator.cpp b/test/tools/ossfuzz/SolidityGenerator.cpp
new file mode 100644
index 000000000..f9c43ef66
--- /dev/null
+++ b/test/tools/ossfuzz/SolidityGenerator.cpp
@@ -0,0 +1,35 @@
+/*
+ This file is part of solidity.
+
+ solidity is free software: you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation, either version 3 of the License, or
+ (at your option) any later version.
+
+ solidity is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with solidity. If not, see .
+*/
+// SPDX-License-Identifier: GPL-3.0
+
+#include
+
+#include
+
+using namespace solidity::test::fuzzer;
+using namespace solidity::util;
+using namespace std;
+
+string SolidityGenerator::generateTestProgram()
+{
+ // TODO: Add generators for grammar elements of
+ // Solidity antlr4 grammar. Currently, the generated
+ // test program consists of a version pragma only.
+ return Whiskers(R"(pragma ;)")
+ ("directive", "solidity >= 0.0.0")
+ .render();
+}
\ No newline at end of file
diff --git a/test/tools/ossfuzz/SolidityGenerator.h b/test/tools/ossfuzz/SolidityGenerator.h
new file mode 100644
index 000000000..5b7abce04
--- /dev/null
+++ b/test/tools/ossfuzz/SolidityGenerator.h
@@ -0,0 +1,42 @@
+/*
+ This file is part of solidity.
+
+ solidity is free software: you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation, either version 3 of the License, or
+ (at your option) any later version.
+
+ solidity is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with solidity. If not, see .
+*/
+// SPDX-License-Identifier: GPL-3.0
+/**
+ * Implements generators for synthesizing mostly syntactically valid
+ * Solidity test programs.
+ */
+
+#pragma once
+
+#include
+
+namespace solidity::test::fuzzer
+{
+using RandomEngine = std::mt19937_64;
+
+class SolidityGenerator
+{
+public:
+ SolidityGenerator(uint64_t _seed): m_rand(_seed)
+ {}
+ /// @returns a pseudo randomly generated test program
+ std::string generateTestProgram();
+private:
+ /// Random number generator
+ RandomEngine const m_rand;
+};
+}
\ No newline at end of file