2
1
mirror of https://github.com/qpdf/qpdf.git synced 2025-01-05 08:02:11 +00:00

Merge pull request #1242 from m-holger/fuzz

Tighten page tree checks
This commit is contained in:
m-holger 2024-07-17 00:59:56 +01:00 committed by GitHub
commit bcf81a1423
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
9 changed files with 32 additions and 12 deletions

View File

@ -127,7 +127,9 @@ set(CORPUS_OTHER
69977b.fuzz 69977b.fuzz
69977c.fuzz 69977c.fuzz
70055.fuzz 70055.fuzz
4599089157701632.fuzz 70245.fuzz
70306.fuzz
4826608268017664.fuzz
) )
set(CORPUS_DIR ${CMAKE_CURRENT_BINARY_DIR}/qpdf_corpus) set(CORPUS_DIR ${CMAKE_CURRENT_BINARY_DIR}/qpdf_corpus)

View File

@ -33,6 +33,7 @@ FuzzHelper::doChecks()
std::cerr << "runtime_error parsing json: " << e.what() << std::endl; std::cerr << "runtime_error parsing json: " << e.what() << std::endl;
} }
QPDF q; QPDF q;
q.setMaxWarnings(1000);
Buffer buf(const_cast<unsigned char*>(data), size); Buffer buf(const_cast<unsigned char*>(data), size);
auto is = std::make_shared<BufferInputSource>("json", &buf); auto is = std::make_shared<BufferInputSource>("json", &buf);
q.createFromJSON(is); q.createFromJSON(is);

Binary file not shown.

View File

View File

@ -21,7 +21,7 @@ my @fuzzers = (
['pngpredictor' => 1], ['pngpredictor' => 1],
['runlength' => 6], ['runlength' => 6],
['tiffpredictor' => 2], ['tiffpredictor' => 2],
['qpdf' => 70], # increment when adding new files ['qpdf' => 72], # increment when adding new files
); );
my $n_tests = 0; my $n_tests = 0;

View File

@ -1515,6 +1515,7 @@ class QPDF
std::set<QPDFObjGen> resolving; std::set<QPDFObjGen> resolving;
QPDFObjectHandle trailer; QPDFObjectHandle trailer;
std::vector<QPDFObjectHandle> all_pages; std::vector<QPDFObjectHandle> all_pages;
bool invalid_page_found{false};
std::map<QPDFObjGen, int> pageobj_to_pages_pos; std::map<QPDFObjGen, int> pageobj_to_pages_pos;
bool pushed_inherited_attributes_to_pages{false}; bool pushed_inherited_attributes_to_pages{false};
bool ever_pushed_inherited_attributes_to_pages{false}; bool ever_pushed_inherited_attributes_to_pages{false};

View File

@ -233,12 +233,13 @@ provide_data(std::shared_ptr<InputSource> is, qpdf_offset_t start, qpdf_offset_t
class QPDF::JSONReactor: public JSON::Reactor class QPDF::JSONReactor: public JSON::Reactor
{ {
public: public:
JSONReactor(QPDF& pdf, std::shared_ptr<InputSource> is, bool must_be_complete) : JSONReactor(QPDF& pdf, std::shared_ptr<InputSource> is, bool must_be_complete, int max_warnings) :
pdf(pdf), pdf(pdf),
is(is), is(is),
must_be_complete(must_be_complete), must_be_complete(must_be_complete),
descr(std::make_shared<QPDFValue::Description>( descr(std::make_shared<QPDFValue::Description>(
QPDFValue::JSON_Descr(std::make_shared<std::string>(is->getName()), ""))) QPDFValue::JSON_Descr(std::make_shared<std::string>(is->getName()), ""))),
max_warnings(max_warnings)
{ {
for (auto& oc: pdf.m->obj_cache) { for (auto& oc: pdf.m->obj_cache) {
if (oc.second.object->getTypeCode() == ::ot_reserved) { if (oc.second.object->getTypeCode() == ::ot_reserved) {
@ -291,7 +292,8 @@ class QPDF::JSONReactor: public JSON::Reactor
std::shared_ptr<InputSource> is; std::shared_ptr<InputSource> is;
bool must_be_complete{true}; bool must_be_complete{true};
std::shared_ptr<QPDFValue::Description> descr; std::shared_ptr<QPDFValue::Description> descr;
bool errors{false}; int errors{0};
int max_warnings{0};
bool saw_qpdf{false}; bool saw_qpdf{false};
bool saw_qpdf_meta{false}; bool saw_qpdf_meta{false};
bool saw_objects{false}; bool saw_objects{false};
@ -314,18 +316,21 @@ class QPDF::JSONReactor: public JSON::Reactor
void void
QPDF::JSONReactor::error(qpdf_offset_t offset, std::string const& msg) QPDF::JSONReactor::error(qpdf_offset_t offset, std::string const& msg)
{ {
this->errors = true; ++errors;
std::string object = this->cur_object; std::string object = this->cur_object;
if (is->getName() != pdf.getFilename()) { if (is->getName() != pdf.getFilename()) {
object += " from " + is->getName(); object += " from " + is->getName();
} }
this->pdf.warn(qpdf_e_json, object, offset, msg); this->pdf.warn(qpdf_e_json, object, offset, msg);
if (max_warnings > 0 && errors >= max_warnings) {
throw std::runtime_error("errors found in JSON");
}
} }
bool bool
QPDF::JSONReactor::anyErrors() const QPDF::JSONReactor::anyErrors() const
{ {
return this->errors; return errors > 0;
} }
void void
@ -820,7 +825,7 @@ QPDF::updateFromJSON(std::shared_ptr<InputSource> is)
void void
QPDF::importJSON(std::shared_ptr<InputSource> is, bool must_be_complete) QPDF::importJSON(std::shared_ptr<InputSource> is, bool must_be_complete)
{ {
JSONReactor reactor(*this, is, must_be_complete); JSONReactor reactor(*this, is, must_be_complete, m->max_warnings);
try { try {
JSON::parse(*is, &reactor); JSON::parse(*is, &reactor);
} catch (std::runtime_error& e) { } catch (std::runtime_error& e) {

View File

@ -40,7 +40,7 @@ std::vector<QPDFObjectHandle> const&
QPDF::getAllPages() QPDF::getAllPages()
{ {
// Note that pushInheritedAttributesToPage may also be used to initialize m->all_pages. // Note that pushInheritedAttributesToPage may also be used to initialize m->all_pages.
if (m->all_pages.empty()) { if (m->all_pages.empty() && !m->invalid_page_found) {
m->ever_called_get_all_pages = true; m->ever_called_get_all_pages = true;
QPDFObjGen::set visited; QPDFObjGen::set visited;
QPDFObjGen::set seen; QPDFObjGen::set seen;
@ -66,9 +66,15 @@ QPDF::getAllPages()
getRoot().replaceKey("/Pages", pages); getRoot().replaceKey("/Pages", pages);
} }
seen.clear(); seen.clear();
if (pages.hasKey("/Kids")) { if (!pages.hasKey("/Kids")) {
// Ensure we actually found a /Pages object. // Ensure we actually found a /Pages object.
getAllPagesInternal(pages, visited, seen, false); throw QPDFExc(
qpdf_e_pages, m->file->getName(), "", 0, "root of pages tree has no /Kids array");
}
getAllPagesInternal(pages, visited, seen, false);
if (m->invalid_page_found) {
flattenPagesTree();
m->invalid_page_found = false;
} }
} }
return m->all_pages; return m->all_pages;
@ -100,6 +106,7 @@ QPDF::getAllPagesInternal(
auto kid = kids.getArrayItem(i); auto kid = kids.getArrayItem(i);
if (!kid.isDictionary()) { if (!kid.isDictionary()) {
kid.warnIfPossible("Pages tree includes non-dictionary object; ignoring"); kid.warnIfPossible("Pages tree includes non-dictionary object; ignoring");
m->invalid_page_found = true;
continue; continue;
} }
if (kid.hasKey("/Kids")) { if (kid.hasKey("/Kids")) {
@ -181,7 +188,11 @@ QPDF::flattenPagesTree()
pages.replaceKey("/Kids", QPDFObjectHandle::newArray(m->all_pages)); pages.replaceKey("/Kids", QPDFObjectHandle::newArray(m->all_pages));
// /Count has not changed // /Count has not changed
if (pages.getKey("/Count").getUIntValue() != len) { if (pages.getKey("/Count").getUIntValue() != len) {
throw std::runtime_error("/Count is wrong after flattening pages tree"); if (m->invalid_page_found && pages.getKey("/Count").getUIntValue() > len) {
pages.replaceKey("/Count", QPDFObjectHandle::newInteger(toI(len)));
} else {
throw std::runtime_error("/Count is wrong after flattening pages tree");
}
} }
} }