mirror of
https://github.com/qpdf/qpdf.git
synced 2024-11-13 16:26:28 +00:00
06001ed25b
Create unresolved objects only for objects in the xref table (except during parsing of the xref table). Do not add indirect nulls into the the object cache as the result of a cache miss during a call to getObject except during parsing or creation/updating from JSON. To support this behaviour, add new private methods getObjectForParser and getObjectForJSON. As a result of this change, dangling references are treated as direct nulls rather than indirect nulls.
517 lines
18 KiB
C++
517 lines
18 KiB
C++
#include <qpdf/QPDFParser.hh>
|
|
|
|
#include <qpdf/QPDF.hh>
|
|
#include <qpdf/QPDFObjGen.hh>
|
|
#include <qpdf/QPDFObjectHandle.hh>
|
|
#include <qpdf/QPDFObject_private.hh>
|
|
#include <qpdf/QPDF_Array.hh>
|
|
#include <qpdf/QPDF_Bool.hh>
|
|
#include <qpdf/QPDF_Dictionary.hh>
|
|
#include <qpdf/QPDF_InlineImage.hh>
|
|
#include <qpdf/QPDF_Integer.hh>
|
|
#include <qpdf/QPDF_Name.hh>
|
|
#include <qpdf/QPDF_Null.hh>
|
|
#include <qpdf/QPDF_Operator.hh>
|
|
#include <qpdf/QPDF_Real.hh>
|
|
#include <qpdf/QPDF_Reserved.hh>
|
|
#include <qpdf/QPDF_Stream.hh>
|
|
#include <qpdf/QPDF_String.hh>
|
|
#include <qpdf/QTC.hh>
|
|
#include <qpdf/QUtil.hh>
|
|
|
|
#include <memory>
|
|
|
|
using ObjectPtr = std::shared_ptr<QPDFObject>;
|
|
|
|
QPDFObjectHandle
|
|
QPDFParser::parse(bool& empty, bool content_stream)
|
|
{
|
|
// This method must take care not to resolve any objects. Don't check the type of any object
|
|
// without first ensuring that it is a direct object. Otherwise, doing so may have the side
|
|
// effect of reading the object and changing the file pointer. If you do this, it will cause a
|
|
// logic error to be thrown from QPDF::inParse().
|
|
|
|
QPDF::ParseGuard pg(context);
|
|
empty = false;
|
|
start = input->tell();
|
|
|
|
if (!tokenizer.nextToken(*input, object_description)) {
|
|
warn(tokenizer.getErrorMessage());
|
|
}
|
|
|
|
switch (tokenizer.getType()) {
|
|
case QPDFTokenizer::tt_eof:
|
|
if (content_stream) {
|
|
// In content stream mode, leave object uninitialized to indicate EOF
|
|
return {};
|
|
}
|
|
QTC::TC("qpdf", "QPDFParser eof in parse");
|
|
warn("unexpected EOF");
|
|
return {QPDF_Null::create()};
|
|
|
|
case QPDFTokenizer::tt_bad:
|
|
QTC::TC("qpdf", "QPDFParser bad token in parse");
|
|
return {QPDF_Null::create()};
|
|
|
|
case QPDFTokenizer::tt_brace_open:
|
|
case QPDFTokenizer::tt_brace_close:
|
|
QTC::TC("qpdf", "QPDFParser bad brace");
|
|
warn("treating unexpected brace token as null");
|
|
return {QPDF_Null::create()};
|
|
|
|
case QPDFTokenizer::tt_array_close:
|
|
QTC::TC("qpdf", "QPDFParser bad array close");
|
|
warn("treating unexpected array close token as null");
|
|
return {QPDF_Null::create()};
|
|
|
|
case QPDFTokenizer::tt_dict_close:
|
|
QTC::TC("qpdf", "QPDFParser bad dictionary close");
|
|
warn("unexpected dictionary close token");
|
|
return {QPDF_Null::create()};
|
|
|
|
case QPDFTokenizer::tt_array_open:
|
|
case QPDFTokenizer::tt_dict_open:
|
|
stack.clear();
|
|
stack.emplace_back(
|
|
input,
|
|
(tokenizer.getType() == QPDFTokenizer::tt_array_open) ? st_array : st_dictionary_key);
|
|
frame = &stack.back();
|
|
return parseRemainder(content_stream);
|
|
|
|
case QPDFTokenizer::tt_bool:
|
|
return withDescription<QPDF_Bool>(tokenizer.getValue() == "true");
|
|
|
|
case QPDFTokenizer::tt_null:
|
|
return {QPDF_Null::create()};
|
|
|
|
case QPDFTokenizer::tt_integer:
|
|
return withDescription<QPDF_Integer>(QUtil::string_to_ll(tokenizer.getValue().c_str()));
|
|
|
|
case QPDFTokenizer::tt_real:
|
|
return withDescription<QPDF_Real>(tokenizer.getValue());
|
|
|
|
case QPDFTokenizer::tt_name:
|
|
return withDescription<QPDF_Name>(tokenizer.getValue());
|
|
|
|
case QPDFTokenizer::tt_word:
|
|
{
|
|
auto const& value = tokenizer.getValue();
|
|
if (content_stream) {
|
|
return withDescription<QPDF_Operator>(value);
|
|
} else if (value == "endobj") {
|
|
// We just saw endobj without having read anything. Treat this as a null and do
|
|
// not move the input source's offset.
|
|
input->seek(input->getLastOffset(), SEEK_SET);
|
|
empty = true;
|
|
return {QPDF_Null::create()};
|
|
} else {
|
|
QTC::TC("qpdf", "QPDFParser treat word as string");
|
|
warn("unknown token while reading object; treating as string");
|
|
return withDescription<QPDF_String>(value);
|
|
}
|
|
}
|
|
|
|
case QPDFTokenizer::tt_string:
|
|
if (decrypter) {
|
|
std::string s{tokenizer.getValue()};
|
|
decrypter->decryptString(s);
|
|
return withDescription<QPDF_String>(s);
|
|
} else {
|
|
return withDescription<QPDF_String>(tokenizer.getValue());
|
|
}
|
|
|
|
default:
|
|
warn("treating unknown token type as null while reading object");
|
|
return {QPDF_Null::create()};
|
|
}
|
|
}
|
|
|
|
QPDFObjectHandle
|
|
QPDFParser::parseRemainder(bool content_stream)
|
|
{
|
|
// This method must take care not to resolve any objects. Don't check the type of any object
|
|
// without first ensuring that it is a direct object. Otherwise, doing so may have the side
|
|
// effect of reading the object and changing the file pointer. If you do this, it will cause a
|
|
// logic error to be thrown from QPDF::inParse().
|
|
|
|
bad_count = 0;
|
|
bool b_contents = false;
|
|
|
|
while (true) {
|
|
if (!tokenizer.nextToken(*input, object_description)) {
|
|
warn(tokenizer.getErrorMessage());
|
|
}
|
|
++good_count; // optimistically
|
|
|
|
if (int_count != 0) {
|
|
// Special handling of indirect references. Treat integer tokens as part of an indirect
|
|
// reference until proven otherwise.
|
|
if (tokenizer.getType() == QPDFTokenizer::tt_integer) {
|
|
if (++int_count > 2) {
|
|
// Process the oldest buffered integer.
|
|
addInt(int_count);
|
|
}
|
|
last_offset_buffer[int_count % 2] = input->getLastOffset();
|
|
int_buffer[int_count % 2] = QUtil::string_to_ll(tokenizer.getValue().c_str());
|
|
continue;
|
|
|
|
} else if (
|
|
int_count >= 2 && tokenizer.getType() == QPDFTokenizer::tt_word &&
|
|
tokenizer.getValue() == "R") {
|
|
if (context == nullptr) {
|
|
QTC::TC("qpdf", "QPDFParser indirect without context");
|
|
throw std::logic_error("QPDFParser::parse called without context on an object "
|
|
"with indirect references");
|
|
}
|
|
auto id = QIntC::to_int(int_buffer[(int_count - 1) % 2]);
|
|
auto gen = QIntC::to_int(int_buffer[(int_count) % 2]);
|
|
if (!(id < 1 || gen < 0 || gen >= 65535)) {
|
|
add(QPDF::ParseGuard::getObject(context, id, gen, parse_pdf));
|
|
} else {
|
|
QTC::TC("qpdf", "QPDFParser invalid objgen");
|
|
addNull();
|
|
}
|
|
int_count = 0;
|
|
continue;
|
|
|
|
} else if (int_count > 0) {
|
|
// Process the buffered integers before processing the current token.
|
|
if (int_count > 1) {
|
|
addInt(int_count - 1);
|
|
}
|
|
addInt(int_count);
|
|
int_count = 0;
|
|
}
|
|
}
|
|
|
|
switch (tokenizer.getType()) {
|
|
case QPDFTokenizer::tt_eof:
|
|
warn("parse error while reading object");
|
|
if (content_stream) {
|
|
// In content stream mode, leave object uninitialized to indicate EOF
|
|
return {};
|
|
}
|
|
QTC::TC("qpdf", "QPDFParser eof in parseRemainder");
|
|
warn("unexpected EOF");
|
|
return {QPDF_Null::create()};
|
|
|
|
case QPDFTokenizer::tt_bad:
|
|
QTC::TC("qpdf", "QPDFParser bad token in parseRemainder");
|
|
if (tooManyBadTokens()) {
|
|
return {QPDF_Null::create()};
|
|
}
|
|
addNull();
|
|
continue;
|
|
|
|
case QPDFTokenizer::tt_brace_open:
|
|
case QPDFTokenizer::tt_brace_close:
|
|
QTC::TC("qpdf", "QPDFParser bad brace in parseRemainder");
|
|
warn("treating unexpected brace token as null");
|
|
if (tooManyBadTokens()) {
|
|
return {QPDF_Null::create()};
|
|
}
|
|
addNull();
|
|
continue;
|
|
|
|
case QPDFTokenizer::tt_array_close:
|
|
if (frame->state == st_array) {
|
|
auto object = QPDF_Array::create(std::move(frame->olist), frame->null_count > 100);
|
|
setDescription(object, frame->offset - 1);
|
|
// The `offset` points to the next of "[". Set the rewind offset to point to the
|
|
// beginning of "[". This has been explicitly tested with whitespace surrounding the
|
|
// array start delimiter. getLastOffset points to the array end token and therefore
|
|
// can't be used here.
|
|
if (stack.size() <= 1) {
|
|
return object;
|
|
}
|
|
stack.pop_back();
|
|
frame = &stack.back();
|
|
add(std::move(object));
|
|
} else {
|
|
QTC::TC("qpdf", "QPDFParser bad array close in parseRemainder");
|
|
warn("treating unexpected array close token as null");
|
|
if (tooManyBadTokens()) {
|
|
return {QPDF_Null::create()};
|
|
}
|
|
addNull();
|
|
}
|
|
continue;
|
|
|
|
case QPDFTokenizer::tt_dict_close:
|
|
if (frame->state <= st_dictionary_value) {
|
|
// Attempt to recover more or less gracefully from invalid dictionaries.
|
|
auto& dict = frame->dict;
|
|
|
|
if (frame->state == st_dictionary_value) {
|
|
QTC::TC("qpdf", "QPDFParser no val for last key");
|
|
warn(
|
|
frame->offset,
|
|
"dictionary ended prematurely; using null as value for last key");
|
|
dict[frame->key] = QPDF_Null::create();
|
|
}
|
|
|
|
if (!frame->olist.empty()) {
|
|
fixMissingKeys();
|
|
}
|
|
|
|
if (!frame->contents_string.empty() && dict.count("/Type") &&
|
|
dict["/Type"].isNameAndEquals("/Sig") && dict.count("/ByteRange") &&
|
|
dict.count("/Contents") && dict["/Contents"].isString()) {
|
|
dict["/Contents"] = QPDFObjectHandle::newString(frame->contents_string);
|
|
dict["/Contents"].setParsedOffset(frame->contents_offset);
|
|
}
|
|
auto object = QPDF_Dictionary::create(std::move(dict));
|
|
setDescription(object, frame->offset - 2);
|
|
// The `offset` points to the next of "<<". Set the rewind offset to point to the
|
|
// beginning of "<<". This has been explicitly tested with whitespace surrounding
|
|
// the dictionary start delimiter. getLastOffset points to the dictionary end token
|
|
// and therefore can't be used here.
|
|
if (stack.size() <= 1) {
|
|
return object;
|
|
}
|
|
stack.pop_back();
|
|
frame = &stack.back();
|
|
add(std::move(object));
|
|
} else {
|
|
QTC::TC("qpdf", "QPDFParser bad dictionary close in parseRemainder");
|
|
warn("unexpected dictionary close token");
|
|
if (tooManyBadTokens()) {
|
|
return {QPDF_Null::create()};
|
|
}
|
|
addNull();
|
|
}
|
|
continue;
|
|
|
|
case QPDFTokenizer::tt_array_open:
|
|
case QPDFTokenizer::tt_dict_open:
|
|
if (stack.size() > 499) {
|
|
QTC::TC("qpdf", "QPDFParser too deep");
|
|
warn("ignoring excessively deeply nested data structure");
|
|
return {QPDF_Null::create()};
|
|
} else {
|
|
b_contents = false;
|
|
stack.emplace_back(
|
|
input,
|
|
(tokenizer.getType() == QPDFTokenizer::tt_array_open) ? st_array
|
|
: st_dictionary_key);
|
|
frame = &stack.back();
|
|
continue;
|
|
}
|
|
|
|
case QPDFTokenizer::tt_bool:
|
|
addScalar<QPDF_Bool>(tokenizer.getValue() == "true");
|
|
continue;
|
|
|
|
case QPDFTokenizer::tt_null:
|
|
addNull();
|
|
continue;
|
|
|
|
case QPDFTokenizer::tt_integer:
|
|
if (!content_stream) {
|
|
// Buffer token in case it is part of an indirect reference.
|
|
last_offset_buffer[1] = input->getLastOffset();
|
|
int_buffer[1] = QUtil::string_to_ll(tokenizer.getValue().c_str());
|
|
int_count = 1;
|
|
} else {
|
|
addScalar<QPDF_Integer>(QUtil::string_to_ll(tokenizer.getValue().c_str()));
|
|
}
|
|
continue;
|
|
|
|
case QPDFTokenizer::tt_real:
|
|
addScalar<QPDF_Real>(tokenizer.getValue());
|
|
continue;
|
|
|
|
case QPDFTokenizer::tt_name:
|
|
if (frame->state == st_dictionary_key) {
|
|
frame->key = tokenizer.getValue();
|
|
frame->state = st_dictionary_value;
|
|
b_contents = decrypter && frame->key == "/Contents";
|
|
continue;
|
|
} else {
|
|
addScalar<QPDF_Name>(tokenizer.getValue());
|
|
}
|
|
continue;
|
|
|
|
case QPDFTokenizer::tt_word:
|
|
if (content_stream) {
|
|
addScalar<QPDF_Operator>(tokenizer.getValue());
|
|
} else {
|
|
QTC::TC("qpdf", "QPDFParser treat word as string in parseRemainder");
|
|
warn("unknown token while reading object; treating as string");
|
|
if (tooManyBadTokens()) {
|
|
return {QPDF_Null::create()};
|
|
}
|
|
addScalar<QPDF_String>(tokenizer.getValue());
|
|
}
|
|
continue;
|
|
|
|
case QPDFTokenizer::tt_string:
|
|
{
|
|
auto const& val = tokenizer.getValue();
|
|
if (decrypter) {
|
|
if (b_contents) {
|
|
frame->contents_string = val;
|
|
frame->contents_offset = input->getLastOffset();
|
|
b_contents = false;
|
|
}
|
|
std::string s{val};
|
|
decrypter->decryptString(s);
|
|
addScalar<QPDF_String>(s);
|
|
} else {
|
|
addScalar<QPDF_String>(val);
|
|
}
|
|
}
|
|
continue;
|
|
|
|
default:
|
|
warn("treating unknown token type as null while reading object");
|
|
if (tooManyBadTokens()) {
|
|
return {QPDF_Null::create()};
|
|
}
|
|
addNull();
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
QPDFParser::add(std::shared_ptr<QPDFObject>&& obj)
|
|
{
|
|
if (frame->state != st_dictionary_value) {
|
|
// If state is st_dictionary_key then there is a missing key. Push onto olist for
|
|
// processing once the tt_dict_close token has been found.
|
|
frame->olist.emplace_back(std::move(obj));
|
|
} else {
|
|
if (auto res = frame->dict.insert_or_assign(frame->key, std::move(obj)); !res.second) {
|
|
warnDuplicateKey();
|
|
}
|
|
frame->state = st_dictionary_key;
|
|
}
|
|
}
|
|
|
|
void
|
|
QPDFParser::addNull()
|
|
{
|
|
const static ObjectPtr null_obj = QPDF_Null::create();
|
|
|
|
if (frame->state != st_dictionary_value) {
|
|
// If state is st_dictionary_key then there is a missing key. Push onto olist for
|
|
// processing once the tt_dict_close token has been found.
|
|
frame->olist.emplace_back(null_obj);
|
|
} else {
|
|
if (auto res = frame->dict.insert_or_assign(frame->key, null_obj); !res.second) {
|
|
warnDuplicateKey();
|
|
}
|
|
frame->state = st_dictionary_key;
|
|
}
|
|
++frame->null_count;
|
|
}
|
|
|
|
void
|
|
QPDFParser::addInt(int count)
|
|
{
|
|
auto obj = QPDF_Integer::create(int_buffer[count % 2]);
|
|
obj->setDescription(context, description, last_offset_buffer[count % 2]);
|
|
add(std::move(obj));
|
|
}
|
|
|
|
template <typename T, typename... Args>
|
|
void
|
|
QPDFParser::addScalar(Args&&... args)
|
|
{
|
|
auto obj = T::create(args...);
|
|
obj->setDescription(context, description, input->getLastOffset());
|
|
add(std::move(obj));
|
|
}
|
|
|
|
template <typename T, typename... Args>
|
|
QPDFObjectHandle
|
|
QPDFParser::withDescription(Args&&... args)
|
|
{
|
|
auto obj = T::create(args...);
|
|
obj->setDescription(context, description, start);
|
|
return {obj};
|
|
}
|
|
|
|
void
|
|
QPDFParser::setDescription(ObjectPtr& obj, qpdf_offset_t parsed_offset)
|
|
{
|
|
if (obj) {
|
|
obj->setDescription(context, description, parsed_offset);
|
|
}
|
|
}
|
|
|
|
void
|
|
QPDFParser::fixMissingKeys()
|
|
{
|
|
std::set<std::string> names;
|
|
for (auto& obj: frame->olist) {
|
|
if (obj->getTypeCode() == ::ot_name) {
|
|
names.insert(obj->getStringValue());
|
|
}
|
|
}
|
|
int next_fake_key = 1;
|
|
for (auto const& item: frame->olist) {
|
|
while (true) {
|
|
const std::string key = "/QPDFFake" + std::to_string(next_fake_key++);
|
|
const bool found_fake = frame->dict.count(key) == 0 && names.count(key) == 0;
|
|
QTC::TC("qpdf", "QPDFParser found fake", (found_fake ? 0 : 1));
|
|
if (found_fake) {
|
|
warn(
|
|
frame->offset,
|
|
"expected dictionary key but found non-name object; inserting key " + key);
|
|
frame->dict[key] = item;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool
|
|
QPDFParser::tooManyBadTokens()
|
|
{
|
|
if (good_count <= 4) {
|
|
if (++bad_count > 5) {
|
|
warn("too many errors; giving up on reading object");
|
|
return true;
|
|
}
|
|
} else {
|
|
bad_count = 1;
|
|
}
|
|
good_count = 0;
|
|
return false;
|
|
}
|
|
|
|
void
|
|
QPDFParser::warn(QPDFExc const& e) const
|
|
{
|
|
// If parsing on behalf of a QPDF object and want to give a warning, we can warn through the
|
|
// object. If parsing for some other reason, such as an explicit creation of an object from a
|
|
// string, then just throw the exception.
|
|
if (context) {
|
|
context->warn(e);
|
|
} else {
|
|
throw e;
|
|
}
|
|
}
|
|
|
|
void
|
|
QPDFParser::warnDuplicateKey()
|
|
{
|
|
QTC::TC("qpdf", "QPDFParser duplicate dict key");
|
|
warn(
|
|
frame->offset,
|
|
"dictionary has duplicated key " + frame->key + "; last occurrence overrides earlier ones");
|
|
}
|
|
|
|
void
|
|
QPDFParser::warn(qpdf_offset_t offset, std::string const& msg) const
|
|
{
|
|
warn(QPDFExc(qpdf_e_damaged_pdf, input->getName(), object_description, offset, msg));
|
|
}
|
|
|
|
void
|
|
QPDFParser::warn(std::string const& msg) const
|
|
{
|
|
warn(input->getLastOffset(), msg);
|
|
}
|