2
1
mirror of https://github.com/qpdf/qpdf.git synced 2024-11-08 14:21:06 +00:00

Add DCT decompression config methods in favor of compile-time changes

As a rule, we should avoid conditional compilation is it always causes
code paths that are sometimes not even seen lexically by the compiler.
Also, we want the actual code being fuzzed to be as close as possible
to the real code. Conditional compilation is suitable to handle
underlying system differences.

Instead, favor configuration using callbacks or other methods that can
be triggered in the places where they need to be exercised.
This commit is contained in:
Jay Berkenbilt 2024-07-02 09:04:53 -04:00 committed by m-holger
parent b45e3420d6
commit 65bd8bc57d
8 changed files with 79 additions and 101 deletions

View File

@ -131,10 +131,6 @@ if(FUTURE)
add_compile_definitions(QPDF_FUTURE=1) add_compile_definitions(QPDF_FUTURE=1)
endif() endif()
if(OSS_FUZZ)
add_compile_definitions(QPDF_OSS_FUZZ=1)
endif()
enable_testing() enable_testing()
set(RUN_QTEST perl ${qpdf_SOURCE_DIR}/run-qtest ${ENABLE_QTC_ARG}) set(RUN_QTEST perl ${qpdf_SOURCE_DIR}/run-qtest ${ENABLE_QTC_ARG})

View File

@ -26,8 +26,18 @@ FuzzHelper::FuzzHelper(unsigned char const* data, size_t size) :
void void
FuzzHelper::doChecks() FuzzHelper::doChecks()
{ {
// Limit the memory used to decompress JPEG files during fuzzing. Excessive memory use during
// fuzzing is due to corrupt JPEG data which sometimes cannot be detected before
// jpeg_start_decompress is called. During normal use of qpdf very large JPEGs can occasionally
// occur legitimately and therefore must be allowed during normal operations.
Pl_DCT::setMemoryLimit(1'000'000'000);
// Do not decompress corrupt data. This may cause extended runtime within jpeglib without
// exercising additional code paths in qpdf.
Pl_DCT::setThrowOnCorruptData(true);
Pl_Discard discard; Pl_Discard discard;
Pl_DCT p("decode", &discard, 20'000'000); Pl_DCT p("decode", &discard);
p.write(const_cast<unsigned char*>(data), size); p.write(const_cast<unsigned char*>(data), size);
p.finish(); p.finish();
} }

View File

@ -1,5 +1,6 @@
#include <qpdf/Buffer.hh> #include <qpdf/Buffer.hh>
#include <qpdf/BufferInputSource.hh> #include <qpdf/BufferInputSource.hh>
#include <qpdf/Pl_DCT.hh>
#include <qpdf/Pl_Discard.hh> #include <qpdf/Pl_Discard.hh>
#include <qpdf/QPDF.hh> #include <qpdf/QPDF.hh>
#include <qpdf/QPDFAcroFormDocumentHelper.hh> #include <qpdf/QPDFAcroFormDocumentHelper.hh>
@ -171,6 +172,16 @@ FuzzHelper::testOutlines()
void void
FuzzHelper::doChecks() FuzzHelper::doChecks()
{ {
// Limit the memory used to decompress JPEG files during fuzzing. Excessive memory use during
// fuzzing is due to corrupt JPEG data which sometimes cannot be detected before
// jpeg_start_decompress is called. During normal use of qpdf very large JPEGs can occasionally
// occur legitimately and therefore must be allowed during normal operations.
Pl_DCT::setMemoryLimit(1'000'000'000);
// Do not decompress corrupt data. This may cause extended runtime within jpeglib without
// exercising additional code paths in qpdf, and potentially causing counterproductive timeouts.
Pl_DCT::setThrowOnCorruptData(true);
// Get as much coverage as possible in parts of the library that // Get as much coverage as possible in parts of the library that
// might benefit from fuzzing. // might benefit from fuzzing.
std::cerr << "\ninfo: starting testWrite\n"; std::cerr << "\ninfo: starting testWrite\n";

View File

@ -34,20 +34,15 @@ class QPDF_DLL_CLASS Pl_DCT: public Pipeline
QPDF_DLL QPDF_DLL
Pl_DCT(char const* identifier, Pipeline* next); Pl_DCT(char const* identifier, Pipeline* next);
// Constructor for decompressing image data. If corrupt_data_limit is non-zero and the data is // Limit the memory used by jpeglib when decompressing data.
// corrupt, only attempt to uncompress if the uncompressed size is less than corrupt_data_limit. // NB This is a static option affecting all Pl_DCT instances.
QPDF_DLL QPDF_DLL
Pl_DCT(char const* identifier, Pipeline* next, size_t corrupt_data_limit); static void setMemoryLimit(long limit);
class QPDF_DLL_CLASS CompressConfig // Treat corrupt data as a runtime error rather than attempting to decompress regardless.
{ // NB This is a static option affecting all Pl_DCT instances.
public:
QPDF_DLL QPDF_DLL
CompressConfig() = default; static void setThrowOnCorruptData(bool treat_as_error);
QPDF_DLL
virtual ~CompressConfig() = default;
virtual void apply(jpeg_compress_struct*) = 0;
};
// Constructor for compressing image data // Constructor for compressing image data
QPDF_DLL QPDF_DLL
@ -57,8 +52,7 @@ class QPDF_DLL_CLASS Pl_DCT: public Pipeline
JDIMENSION image_width, JDIMENSION image_width,
JDIMENSION image_height, JDIMENSION image_height,
int components, int components,
J_COLOR_SPACE color_space, J_COLOR_SPACE color_space);
CompressConfig* config_callback = nullptr);
QPDF_DLL QPDF_DLL
~Pl_DCT() override; ~Pl_DCT() override;
@ -83,6 +77,7 @@ class QPDF_DLL_CLASS Pl_DCT: public Pipeline
public: public:
QPDF_DLL QPDF_DLL
~Members() = default; ~Members() = default;
Members(Members const&) = delete;
private: private:
// For compression // For compression
@ -90,25 +85,18 @@ class QPDF_DLL_CLASS Pl_DCT: public Pipeline
JDIMENSION image_width, JDIMENSION image_width,
JDIMENSION image_height, JDIMENSION image_height,
int components, int components,
J_COLOR_SPACE color_space, J_COLOR_SPACE color_space);
CompressConfig* config_callback);
// For decompression // For decompression
Members(size_t corrupt_data_limit); Members();
Members(Members const&) = delete;
action_e action; action_e action;
Pl_Buffer buf; Pl_Buffer buf;
// Used for decompression
size_t corrupt_data_limit{0};
// Used for compression // Used for compression
JDIMENSION image_width{0}; JDIMENSION image_width{0};
JDIMENSION image_height{0}; JDIMENSION image_height{0};
int components{1}; int components{1};
J_COLOR_SPACE color_space{JCS_GRAYSCALE}; J_COLOR_SPACE color_space{JCS_GRAYSCALE};
CompressConfig* config_callback{nullptr};
}; };
std::shared_ptr<Members> m; std::shared_ptr<Members> m;

View File

@ -1,5 +1,5 @@
# Generated by generate_auto_job # Generated by generate_auto_job
CMakeLists.txt 456938b9debc4997f142ccfb13f3baf2517ae5855e1fe9b2ada1a0b8f7e4facf CMakeLists.txt 47752f33b17fa526d46fc608a25ad6b8c61feba9deb1bd659fddf93e6e08b102
generate_auto_job f64733b79dcee5a0e3e8ccc6976448e8ddf0e8b6529987a66a7d3ab2ebc10a86 generate_auto_job f64733b79dcee5a0e3e8ccc6976448e8ddf0e8b6529987a66a7d3ab2ebc10a86
include/qpdf/auto_job_c_att.hh 4c2b171ea00531db54720bf49a43f8b34481586ae7fb6cbf225099ee42bc5bb4 include/qpdf/auto_job_c_att.hh 4c2b171ea00531db54720bf49a43f8b34481586ae7fb6cbf225099ee42bc5bb4
include/qpdf/auto_job_c_copy_att.hh 50609012bff14fd82f0649185940d617d05d530cdc522185c7f3920a561ccb42 include/qpdf/auto_job_c_copy_att.hh 50609012bff14fd82f0649185940d617d05d530cdc522185c7f3920a561ccb42

View File

@ -20,6 +20,9 @@ namespace
jmp_buf jmpbuf; jmp_buf jmpbuf;
std::string msg; std::string msg;
}; };
long memory_limit{0};
bool throw_on_corrupt_data{false};
} // namespace } // namespace
static void static void
@ -32,38 +35,39 @@ error_handler(j_common_ptr cinfo)
longjmp(jerr->jmpbuf, 1); longjmp(jerr->jmpbuf, 1);
} }
Pl_DCT::Members::Members(size_t corrupt_data_limit) : Pl_DCT::Members::Members() :
action(a_decompress), action(a_decompress),
buf("DCT compressed image"), buf("DCT compressed image")
corrupt_data_limit(corrupt_data_limit)
{ {
} }
Pl_DCT::Members::Members( Pl_DCT::Members::Members(
JDIMENSION image_width, JDIMENSION image_width, JDIMENSION image_height, int components, J_COLOR_SPACE color_space) :
JDIMENSION image_height,
int components,
J_COLOR_SPACE color_space,
CompressConfig* config_callback) :
action(a_compress), action(a_compress),
buf("DCT uncompressed image"), buf("DCT uncompressed image"),
image_width(image_width), image_width(image_width),
image_height(image_height), image_height(image_height),
components(components), components(components),
color_space(color_space), color_space(color_space)
config_callback(config_callback)
{ {
} }
Pl_DCT::Pl_DCT(char const* identifier, Pipeline* next) : Pl_DCT::Pl_DCT(char const* identifier, Pipeline* next) :
Pl_DCT(identifier, next, 0) Pipeline(identifier, next),
m(new Members())
{ {
} }
Pl_DCT::Pl_DCT(char const* identifier, Pipeline* next, size_t corrupt_data_limit) : void
Pipeline(identifier, next), Pl_DCT::setMemoryLimit(long limit)
m(new Members(corrupt_data_limit))
{ {
memory_limit = limit;
}
void
Pl_DCT::setThrowOnCorruptData(bool treat_as_error)
{
throw_on_corrupt_data = treat_as_error;
} }
Pl_DCT::Pl_DCT( Pl_DCT::Pl_DCT(
@ -72,10 +76,9 @@ Pl_DCT::Pl_DCT(
JDIMENSION image_width, JDIMENSION image_width,
JDIMENSION image_height, JDIMENSION image_height,
int components, int components,
J_COLOR_SPACE color_space, J_COLOR_SPACE color_space) :
CompressConfig* config_callback) :
Pipeline(identifier, next), Pipeline(identifier, next),
m(new Members(image_width, image_height, components, color_space, config_callback)) m(new Members(image_width, image_height, components, color_space))
{ {
} }
@ -273,9 +276,6 @@ Pl_DCT::compress(void* cinfo_p, Buffer* b)
cinfo->input_components = m->components; cinfo->input_components = m->components;
cinfo->in_color_space = m->color_space; cinfo->in_color_space = m->color_space;
jpeg_set_defaults(cinfo); jpeg_set_defaults(cinfo);
if (m->config_callback) {
m->config_callback->apply(cinfo);
}
jpeg_start_compress(cinfo, TRUE); jpeg_start_compress(cinfo, TRUE);
@ -312,36 +312,32 @@ Pl_DCT::decompress(void* cinfo_p, Buffer* b)
# pragma GCC diagnostic pop # pragma GCC diagnostic pop
#endif #endif
#ifdef QPDF_OSS_FUZZ if (memory_limit > 0) {
// Limit the memory used to decompress JPEG files during fuzzing. Excessive memory use during cinfo->mem->max_memory_to_use = memory_limit;
// fuzzing is due to corrupt JPEG data which sometimes cannot be detected before }
// jpeg_start_decompress is called. During normal use of qpdf very large JPEGs can occasionally
// occur legitimately and therefore must be allowed during normal operations.
cinfo->mem->max_memory_to_use = 1'000'000'000;
// For some corrupt files the memory used internally by libjpeg stays within the above limits
// even though the size written to the next pipeline is significantly larger.
m->corrupt_data_limit = 10'000'000;
#endif
jpeg_buffer_src(cinfo, b); jpeg_buffer_src(cinfo, b);
(void)jpeg_read_header(cinfo, TRUE); (void)jpeg_read_header(cinfo, TRUE);
if (throw_on_corrupt_data && cinfo->err->num_warnings > 0) {
throw std::runtime_error("Pl_DCT::decompress: JPEG data is corrupt");
}
(void)jpeg_calc_output_dimensions(cinfo); (void)jpeg_calc_output_dimensions(cinfo);
unsigned int width = cinfo->output_width * QIntC::to_uint(cinfo->output_components); unsigned int width = cinfo->output_width * QIntC::to_uint(cinfo->output_components);
if (cinfo->err->num_warnings == 0 || m->corrupt_data_limit == 0 ||
(width * QIntC::to_uint(cinfo->output_height)) < m->corrupt_data_limit) {
// err->num_warnings is the number of corrupt data warnings emitted. // err->num_warnings is the number of corrupt data warnings emitted.
// err->msg_code could also be the code of an informational message. // err->msg_code could also be the code of an informational message.
JSAMPARRAY buffer = (*cinfo->mem->alloc_sarray)( JSAMPARRAY buffer =
reinterpret_cast<j_common_ptr>(cinfo), JPOOL_IMAGE, width, 1); (*cinfo->mem->alloc_sarray)(reinterpret_cast<j_common_ptr>(cinfo), JPOOL_IMAGE, width, 1);
(void)jpeg_start_decompress(cinfo); (void)jpeg_start_decompress(cinfo);
while (cinfo->output_scanline < cinfo->output_height) { while (cinfo->output_scanline < cinfo->output_height &&
(!throw_on_corrupt_data || cinfo->err->num_warnings == 0)) {
(void)jpeg_read_scanlines(cinfo, buffer, 1); (void)jpeg_read_scanlines(cinfo, buffer, 1);
getNext()->write(buffer[0], width * sizeof(buffer[0][0])); getNext()->write(buffer[0], width * sizeof(buffer[0][0]));
} }
(void)jpeg_finish_decompress(cinfo); (void)jpeg_finish_decompress(cinfo);
} else { if (throw_on_corrupt_data && cinfo->err->num_warnings > 0) {
*QPDFLogger::defaultLogger()->getError() << "corrupt JPEG data ignored" << "\n"; throw std::runtime_error("Pl_DCT::decompress: JPEG data is corrupt");
} }
getNext()->finish(); getNext()->finish();
} }

View File

@ -500,6 +500,7 @@ QPDF::warn(QPDFExc const& e)
{ {
m->warnings.push_back(e); m->warnings.push_back(e);
if (!m->suppress_warnings) { if (!m->suppress_warnings) {
// QXXXQ
#ifdef QPDF_OSS_FUZZ #ifdef QPDF_OSS_FUZZ
if (m->warnings.size() > 20) { if (m->warnings.size() > 20) {
*m->log->getWarn() << "WARNING: too many warnings - additional warnings surpressed\n"; *m->log->getWarn() << "WARNING: too many warnings - additional warnings surpressed\n";

View File

@ -14,21 +14,6 @@ usage()
exit(2); exit(2);
} }
class Callback: public Pl_DCT::CompressConfig
{
public:
Callback() = default;
~Callback() override = default;
void apply(jpeg_compress_struct*) override;
bool called{false};
};
void
Callback::apply(jpeg_compress_struct*)
{
this->called = true;
}
int int
main(int argc, char* argv[]) main(int argc, char* argv[])
{ {
@ -66,21 +51,12 @@ main(int argc, char* argv[])
FILE* outfile = QUtil::safe_fopen(outfilename, "wb"); FILE* outfile = QUtil::safe_fopen(outfilename, "wb");
Pl_StdioFile out("stdout", outfile); Pl_StdioFile out("stdout", outfile);
unsigned char buf[100]; unsigned char buf[100];
bool done = false; Pl_DCT dct("dct", &out, width, height, components, cs);
Callback callback; while (size_t len = fread(buf, 1, sizeof(buf), infile)) {
Pl_DCT dct("dct", &out, width, height, components, cs, &callback);
while (!done) {
size_t len = fread(buf, 1, sizeof(buf), infile);
if (len <= 0) {
done = true;
} else {
dct.write(buf, len); dct.write(buf, len);
} }
}
dct.finish(); dct.finish();
if (!callback.called) {
std::cout << "Callback was not called" << std::endl;
}
fclose(infile); fclose(infile);
fclose(outfile); fclose(outfile);
return 0; return 0;