mirror of
https://github.com/qpdf/qpdf.git
synced 2024-12-22 02:49:00 +00:00
Move QPDF::insertXrefEntry etc to QPDF::Xref_table
This commit is contained in:
parent
0ac37bc956
commit
1e072e223a
@ -777,9 +777,6 @@ class QPDF
|
||||
QPDFObjectHandle& dict,
|
||||
int max_num_entries,
|
||||
std::function<QPDFExc(std::string_view)> damaged);
|
||||
void insertXrefEntry(int obj, int f0, qpdf_offset_t f1, int f2);
|
||||
void insertFreeXrefEntry(QPDFObjGen);
|
||||
void insertReconstructedXrefEntry(int obj, qpdf_offset_t f1, int f2);
|
||||
void setLastObjectDescription(std::string const& description, QPDFObjGen const& og);
|
||||
QPDFObjectHandle readTrailer();
|
||||
QPDFObjectHandle readObject(std::string const& description, QPDFObjGen og);
|
||||
|
@ -196,15 +196,16 @@ QPDF::EncryptionParameters::EncryptionParameters() :
|
||||
{
|
||||
}
|
||||
|
||||
QPDF::Members::Members() :
|
||||
QPDF::Members::Members(QPDF& qpdf) :
|
||||
log(QPDFLogger::defaultLogger()),
|
||||
file(new InvalidInputSource()),
|
||||
encp(new EncryptionParameters)
|
||||
encp(new EncryptionParameters),
|
||||
xref_table(qpdf)
|
||||
{
|
||||
}
|
||||
|
||||
QPDF::QPDF() :
|
||||
m(new Members())
|
||||
m(new Members(*this))
|
||||
{
|
||||
m->tokenizer.allowEOF();
|
||||
// Generate a unique ID. It just has to be unique among all QPDF objects allocated throughout
|
||||
@ -585,7 +586,7 @@ QPDF::reconstruct_xref(QPDFExc& e)
|
||||
int obj = QUtil::string_to_int(t1.getValue().c_str());
|
||||
int gen = QUtil::string_to_int(t2.getValue().c_str());
|
||||
if (obj <= m->xref_table.max_id) {
|
||||
insertReconstructedXrefEntry(obj, token_start, gen);
|
||||
m->xref_table.insert_reconstructed(obj, token_start, gen);
|
||||
} else {
|
||||
warn(damagedPDF(
|
||||
"", 0, "ignoring object with impossibly large id " + std::to_string(obj)));
|
||||
@ -981,9 +982,9 @@ QPDF::read_xrefTable(qpdf_offset_t xref_offset)
|
||||
"xref table", "invalid xref entry (obj=" + std::to_string(i) + ")");
|
||||
}
|
||||
if (type == 'f') {
|
||||
insertFreeXrefEntry(QPDFObjGen(toI(i), f2));
|
||||
m->xref_table.insert_free(QPDFObjGen(toI(i), f2));
|
||||
} else {
|
||||
insertXrefEntry(toI(i), 1, f1, f2);
|
||||
m->xref_table.insert(toI(i), 1, f1, f2);
|
||||
}
|
||||
}
|
||||
qpdf_offset_t pos = m->file->tell();
|
||||
@ -1251,9 +1252,9 @@ QPDF::processXRefStream(qpdf_offset_t xref_offset, QPDFObjectHandle& xref_obj)
|
||||
// Ignore fields[2], which we don't care about in this case. This works around the
|
||||
// issue of some PDF files that put invalid values, like -1, here for deleted
|
||||
// objects.
|
||||
insertFreeXrefEntry(QPDFObjGen(obj, 0));
|
||||
m->xref_table.insert_free(QPDFObjGen(obj, 0));
|
||||
} else {
|
||||
insertXrefEntry(obj, toI(fields[0]), fields[1], toI(fields[2]));
|
||||
m->xref_table.insert(obj, toI(fields[0]), fields[1], toI(fields[2]));
|
||||
}
|
||||
++obj;
|
||||
}
|
||||
@ -1276,7 +1277,7 @@ QPDF::processXRefStream(qpdf_offset_t xref_offset, QPDFObjectHandle& xref_obj)
|
||||
}
|
||||
|
||||
void
|
||||
QPDF::insertXrefEntry(int obj, int f0, qpdf_offset_t f1, int f2)
|
||||
QPDF::Xref_table::insert(int obj, int f0, qpdf_offset_t f1, int f2)
|
||||
{
|
||||
// Populate the xref table in such a way that the first reference to an object that we see,
|
||||
// which is the one in the latest xref table in which it appears, is the one that gets stored.
|
||||
@ -1285,22 +1286,23 @@ QPDF::insertXrefEntry(int obj, int f0, qpdf_offset_t f1, int f2)
|
||||
// If there is already an entry for this object and generation in the table, it means that a
|
||||
// later xref table has registered this object. Disregard this one.
|
||||
|
||||
if (obj > m->xref_table.max_id) {
|
||||
if (obj > max_id) {
|
||||
// ignore impossibly large object ids or object ids > Size.
|
||||
return;
|
||||
}
|
||||
|
||||
if (m->xref_table.deleted_objects.count(obj)) {
|
||||
if (deleted_objects.count(obj)) {
|
||||
QTC::TC("qpdf", "QPDF xref deleted object");
|
||||
return;
|
||||
}
|
||||
|
||||
if (f0 == 2 && static_cast<int>(f1) == obj) {
|
||||
warn(damagedPDF("xref stream", "self-referential object stream " + std::to_string(obj)));
|
||||
qpdf.warn(qpdf.damagedPDF(
|
||||
"xref stream", "self-referential object stream " + std::to_string(obj)));
|
||||
return;
|
||||
}
|
||||
|
||||
auto [iter, created] = m->xref_table.try_emplace(QPDFObjGen(obj, (f0 == 2 ? 0 : f2)));
|
||||
auto [iter, created] = try_emplace(QPDFObjGen(obj, (f0 == 2 ? 0 : f2)));
|
||||
if (!created) {
|
||||
QTC::TC("qpdf", "QPDF xref reused object");
|
||||
return;
|
||||
@ -1318,35 +1320,36 @@ QPDF::insertXrefEntry(int obj, int f0, qpdf_offset_t f1, int f2)
|
||||
break;
|
||||
|
||||
default:
|
||||
throw damagedPDF("xref stream", "unknown xref stream entry type " + std::to_string(f0));
|
||||
throw qpdf.damagedPDF(
|
||||
"xref stream", "unknown xref stream entry type " + std::to_string(f0));
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
QPDF::insertFreeXrefEntry(QPDFObjGen og)
|
||||
QPDF::Xref_table::insert_free(QPDFObjGen og)
|
||||
{
|
||||
if (!m->xref_table.count(og)) {
|
||||
m->xref_table.deleted_objects.insert(og.getObj());
|
||||
if (!count(og)) {
|
||||
deleted_objects.insert(og.getObj());
|
||||
}
|
||||
}
|
||||
|
||||
// Replace uncompressed object. This is used in xref recovery mode, which reads the file from
|
||||
// beginning to end.
|
||||
void
|
||||
QPDF::insertReconstructedXrefEntry(int obj, qpdf_offset_t f1, int f2)
|
||||
QPDF::Xref_table::insert_reconstructed(int obj, qpdf_offset_t f1, int f2)
|
||||
{
|
||||
if (!(obj > 0 && obj <= m->xref_table.max_id && 0 <= f2 && f2 < 65535)) {
|
||||
if (!(obj > 0 && obj <= max_id && 0 <= f2 && f2 < 65535)) {
|
||||
QTC::TC("qpdf", "QPDF xref overwrite invalid objgen");
|
||||
return;
|
||||
}
|
||||
|
||||
QPDFObjGen og(obj, f2);
|
||||
if (!m->xref_table.deleted_objects.count(obj)) {
|
||||
if (!deleted_objects.count(obj)) {
|
||||
// deleted_objects stores the uncompressed objects removed from the xref table at the start
|
||||
// of recovery.
|
||||
QTC::TC("qpdf", "QPDF xref overwrite object");
|
||||
m->xref_table[QPDFObjGen(obj, f2)] = QPDFXRefEntry(f1);
|
||||
insert_or_assign(QPDFObjGen(obj, f2), QPDFXRefEntry(f1));
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -7,6 +7,15 @@
|
||||
class QPDF::Xref_table: public std::map<QPDFObjGen, QPDFXRefEntry>
|
||||
{
|
||||
public:
|
||||
Xref_table(QPDF& qpdf) :
|
||||
qpdf(qpdf)
|
||||
{
|
||||
}
|
||||
|
||||
void insert_reconstructed(int obj, qpdf_offset_t f1, int f2);
|
||||
void insert(int obj, int f0, qpdf_offset_t f1, int f2);
|
||||
void insert_free(QPDFObjGen);
|
||||
|
||||
QPDFObjectHandle trailer;
|
||||
bool reconstructed{false};
|
||||
// Various tables are indexed by object id, with potential size id + 1
|
||||
@ -20,6 +29,9 @@ class QPDF::Xref_table: public std::map<QPDFObjGen, QPDFXRefEntry>
|
||||
// Linearization data
|
||||
bool uncompressed_after_compressed{false};
|
||||
qpdf_offset_t first_item_offset{0}; // actual value from file
|
||||
|
||||
private:
|
||||
QPDF& qpdf;
|
||||
};
|
||||
|
||||
// Writer class is restricted to QPDFWriter so that only it can call certain methods.
|
||||
@ -469,7 +481,7 @@ class QPDF::Members
|
||||
~Members() = default;
|
||||
|
||||
private:
|
||||
Members();
|
||||
Members(QPDF& qpdf);
|
||||
Members(Members const&) = delete;
|
||||
|
||||
std::shared_ptr<QPDFLogger> log;
|
||||
|
Loading…
Reference in New Issue
Block a user