mirror of
https://github.com/jmcnamara/libxlsxwriter
synced 2025-03-28 21:13:14 +00:00
OSSFuzz integration
This commit is contained in:
parent
db5c0a6705
commit
965ea12517
@ -41,6 +41,10 @@
|
||||
# Build example files (default off). To build the examples,
|
||||
# pass `-DBUILD_EXAMPLES=ON` during configuration.
|
||||
#
|
||||
# BUILD_FUZZERS
|
||||
# Build fuzz harnesses (default off). To build the harnesses,
|
||||
# pass `-DBUILD_FUZZERS=ON` during configuration.
|
||||
#
|
||||
# USE_STANDARD_TMPFILE
|
||||
# Use the standard tmpfile() function (default off). To enable
|
||||
# the standard tmpfile, pass `-DUSE_STANDARD_TMPFILE=ON`
|
||||
@ -127,6 +131,7 @@ SET(ZLIB_ROOT "" CACHE STRING "Optional root for the ZLIB installation")
|
||||
|
||||
option(BUILD_TESTS "Build libxlsxwriter tests" OFF)
|
||||
option(BUILD_EXAMPLES "Build libxlsxwriter examples" OFF)
|
||||
option(BUILD_FUZZERS "Build harness(es) for fuzzing" OFF)
|
||||
option(USE_SYSTEM_MINIZIP "Use system minizip installation" OFF)
|
||||
option(USE_STANDARD_TMPFILE "Use the C standard library's tmpfile()" OFF)
|
||||
option(USE_NO_MD5 "Build libxlsxwriter without third party MD5 lib" OFF)
|
||||
@ -283,6 +288,7 @@ endif()
|
||||
set(LXW_PROJECT_DIR "${CMAKE_CURRENT_SOURCE_DIR}")
|
||||
set(LXW_LIB_DIR "${LXW_PROJECT_DIR}/lib")
|
||||
add_library(${PROJECT_NAME} "")
|
||||
|
||||
set_target_properties(${PROJECT_NAME} PROPERTIES SOVERSION ${SOVERSION})
|
||||
target_sources(${PROJECT_NAME}
|
||||
PRIVATE ${LXW_SOURCES}
|
||||
@ -421,6 +427,12 @@ if(BUILD_EXAMPLES)
|
||||
endforeach(source)
|
||||
endif()
|
||||
|
||||
# FUZZING
|
||||
# -------
|
||||
if (BUILD_FUZZERS AND DEFINED ENV{LIB_FUZZING_ENGINE})
|
||||
add_subdirectory(dev/fuzzing)
|
||||
endif()
|
||||
|
||||
# INSTALL
|
||||
# -------
|
||||
include(GNUInstallDirs)
|
||||
|
23
dev/fuzzing/CMakeLists.txt
Normal file
23
dev/fuzzing/CMakeLists.txt
Normal file
@ -0,0 +1,23 @@
|
||||
# Utilized by OSSFuzz to build the harness(es) for continuous fuzz-testing
|
||||
# OSSFuzz defines the following environment variables, that this target relies upon:
|
||||
# CXX, CFLAGS, LIB_FUZZING_ENGINE, OUT
|
||||
|
||||
add_definitions(-DNDEBUG) # Do not want assertions
|
||||
|
||||
if (DEFINED ENV{CFLAGS})
|
||||
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} $ENV{CFLAGS}")
|
||||
endif()
|
||||
if (DEFINED ENV{CXXFLAGS})
|
||||
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} $ENV{CXXFLAGS}")
|
||||
endif()
|
||||
|
||||
|
||||
add_executable(xlsx_fuzzer xlsx_fuzzer.cpp)
|
||||
target_link_libraries(xlsx_fuzzer PRIVATE ${PROJECT_NAME} $ENV{LIB_FUZZING_ENGINE})
|
||||
target_compile_features(xlsx_fuzzer PRIVATE cxx_std_17)
|
||||
|
||||
if (DEFINED ENV{OUT})
|
||||
install(TARGETS xlsx_fuzzer DESTINATION $ENV{OUT})
|
||||
else ()
|
||||
message(WARNING "Cannot install if $OUT is not defined!")
|
||||
endif ()
|
11
dev/fuzzing/build.sh
Executable file
11
dev/fuzzing/build.sh
Executable file
@ -0,0 +1,11 @@
|
||||
cd $SRC/libxlsxwriter
|
||||
|
||||
printenv
|
||||
|
||||
mkdir -p build
|
||||
cmake -S . -B build -DBUILD_FUZZERS=ON && cmake --build build --target install
|
||||
|
||||
# Build the corpus using the existing xlsx files in the source
|
||||
mkdir -p corpus
|
||||
find $SRC/libxlsxwriter -name "*.xlsx" -exec cp {} corpus \;
|
||||
zip -q $OUT/xlsx_fuzzer_seed_corpus.zip corpus/*
|
84
dev/fuzzing/xlsx_fuzzer.cpp
Normal file
84
dev/fuzzing/xlsx_fuzzer.cpp
Normal file
@ -0,0 +1,84 @@
|
||||
#include <cstdint>
|
||||
#include <unistd.h>
|
||||
#include <fuzzer/FuzzedDataProvider.h>
|
||||
|
||||
#include "xlsxwriter.h"
|
||||
|
||||
const std::string mem_dir{"/dev/shm"};
|
||||
const std::string file_template = "/fuzzXXXXXX";
|
||||
char temp_file_dir[FILENAME_MAX] = {0};
|
||||
|
||||
/**
|
||||
* \brief: Performs all prep-work needed for continuous fuzzing
|
||||
* \return: Whether initialization was successful
|
||||
*/
|
||||
int init_for_fuzzing()
|
||||
{
|
||||
// Initialize the temporary file directory, based off what is available on the system
|
||||
|
||||
if (0 == access(mem_dir.c_str(), W_OK | R_OK))
|
||||
{
|
||||
// We can read and write to the in-memory directory
|
||||
memcpy(temp_file_dir, mem_dir.c_str(), strnlen(mem_dir.c_str(), FILENAME_MAX));
|
||||
}
|
||||
else
|
||||
{
|
||||
// Default to a temporary directory
|
||||
const char* tmp_prefix = getenv("TMPDIR");
|
||||
if (nullptr == tmp_prefix)
|
||||
{
|
||||
tmp_prefix = "/tmp";
|
||||
}
|
||||
memcpy((void*) temp_file_dir, tmp_prefix, strnlen(tmp_prefix, FILENAME_MAX));
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, const size_t size)
|
||||
{
|
||||
static bool init_fuzzing = init_for_fuzzing();
|
||||
|
||||
char fuzz_file[FILENAME_MAX + 1] = {0};
|
||||
int fuzz_fd = 0;
|
||||
int ret = -1;
|
||||
ssize_t wc = 0;
|
||||
size_t byte_len;
|
||||
lxw_workbook *workbook = nullptr;
|
||||
lxw_worksheet *worksheet = nullptr;
|
||||
FuzzedDataProvider fdp{data, size};
|
||||
std::vector<std::uint8_t> file_bytes{};
|
||||
|
||||
strncpy(fuzz_file, temp_file_dir, strlen(temp_file_dir));
|
||||
strncat(fuzz_file, file_template.c_str(), file_template.length());
|
||||
|
||||
if ((fuzz_fd = mkstemp(fuzz_file)) < 0)
|
||||
{
|
||||
goto fail;
|
||||
}
|
||||
|
||||
byte_len = fdp.ConsumeIntegralInRange<size_t>(0, fdp.remaining_bytes());
|
||||
file_bytes = fdp.ConsumeBytes<uint8_t>(byte_len);
|
||||
write(fuzz_fd, file_bytes.data(), std::min(file_bytes.size(), byte_len));
|
||||
|
||||
workbook = workbook_new(fuzz_file);
|
||||
worksheet = workbook_add_worksheet(workbook, nullptr);
|
||||
|
||||
for (int row = 0; row < fdp.ConsumeIntegralInRange(0, 25); ++row)
|
||||
{
|
||||
for (int col = 0; col < fdp.ConsumeIntegralInRange(0, 25); ++col)
|
||||
{
|
||||
worksheet_write_string(worksheet, row, col, fdp.ConsumeRandomLengthString().c_str(), nullptr);
|
||||
}
|
||||
}
|
||||
|
||||
ret = 0;
|
||||
|
||||
fail:
|
||||
if (nullptr != workbook)
|
||||
{
|
||||
workbook_close(workbook);
|
||||
}
|
||||
close(fuzz_fd);
|
||||
unlink(fuzz_file);
|
||||
return ret;
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user