Commit 4b2e72c4cd7dc9bc17ad78ca983ff884c1e1ee69
1 parent
3f3dbe22
Test for direct, rather than resolved nulls in parser
Just because we know an indirect reference is null, doesn't mean we shouldn't keep it indirect.
Showing
8 changed files
with
259 additions
and
10 deletions
ChangeLog
| 1 | 1 | 2019-08-22 Jay Berkenbilt <ejb@ql.org> |
| 2 | 2 | |
| 3 | + * Add QPDFObjectHandle::isDirectNull() -- a const method that | |
| 4 | + allows determining whether an object is a literal null without | |
| 5 | + attempting to resolve it. | |
| 6 | + | |
| 3 | 7 | * Stop replacing indirect references to null with literal null in |
| 4 | 8 | arrays when writing output with QPDFWriter. |
| 5 | 9 | ... | ... |
include/qpdf/QPDFObjectHandle.hh
| ... | ... | @@ -281,13 +281,12 @@ class QPDFObjectHandle |
| 281 | 281 | QPDF_DLL |
| 282 | 282 | bool isReserved(); |
| 283 | 283 | |
| 284 | - // True for objects that are direct nulls or have previously been | |
| 285 | - // resolved to be nulls. Does not attempt to resolve objects. This | |
| 286 | - // is intended for internal use, but it can be used as an | |
| 287 | - // efficient way to check for nulls if you don't mind unresolved | |
| 288 | - // indirect nulls being false negatives. | |
| 284 | + // True for objects that are direct nulls. Does not attempt to | |
| 285 | + // resolve objects. This is intended for internal use, but it can | |
| 286 | + // be used as an efficient way to check for nulls that are not | |
| 287 | + // indirect objects. | |
| 289 | 288 | QPDF_DLL |
| 290 | - bool isResolvedNull() const; | |
| 289 | + bool isDirectNull() const; | |
| 291 | 290 | |
| 292 | 291 | // This returns true in addition to the query for the specific |
| 293 | 292 | // type for indirect objects. | ... | ... |
libqpdf/QPDFObjectHandle.cc
| ... | ... | @@ -263,9 +263,10 @@ QPDFObjectHandle::isBool() |
| 263 | 263 | } |
| 264 | 264 | |
| 265 | 265 | bool |
| 266 | -QPDFObjectHandle::isResolvedNull() const | |
| 266 | +QPDFObjectHandle::isDirectNull() const | |
| 267 | 267 | { |
| 268 | - return QPDFObjectTypeAccessor<QPDF_Null>::check(m->obj.getPointer()); | |
| 268 | + return (this->m->initialized && (this->m->objid == 0) && | |
| 269 | + QPDFObjectTypeAccessor<QPDF_Null>::check(m->obj.getPointer())); | |
| 269 | 270 | } |
| 270 | 271 | |
| 271 | 272 | bool | ... | ... |
libqpdf/SparseOHArray.cc
| ... | ... | @@ -15,7 +15,7 @@ SparseOHArray::size() const |
| 15 | 15 | void |
| 16 | 16 | SparseOHArray::append(QPDFObjectHandle oh) |
| 17 | 17 | { |
| 18 | - if (! oh.isResolvedNull()) | |
| 18 | + if (! oh.isDirectNull()) | |
| 19 | 19 | { |
| 20 | 20 | this->elements[this->n_elements] = oh; |
| 21 | 21 | } |
| ... | ... | @@ -73,7 +73,7 @@ SparseOHArray::setAt(size_t idx, QPDFObjectHandle oh) |
| 73 | 73 | { |
| 74 | 74 | throw std::logic_error("bounds error setting item in SparseOHArray"); |
| 75 | 75 | } |
| 76 | - if (oh.isResolvedNull()) | |
| 76 | + if (oh.isDirectNull()) | |
| 77 | 77 | { |
| 78 | 78 | this->elements.erase(idx); |
| 79 | 79 | } | ... | ... |
qpdf/qtest/qpdf.test
| ... | ... | @@ -2456,6 +2456,7 @@ my @goodfiles = ("implicit null", # 1 |
| 2456 | 2456 | "hybrid xref old mode", # 18 |
| 2457 | 2457 | "xref with prev", # 19 |
| 2458 | 2458 | "lots of compressible objects", # 20 |
| 2459 | + "array with indirect nulls", # 21 | |
| 2459 | 2460 | ); |
| 2460 | 2461 | |
| 2461 | 2462 | $n_tests += (3 * @goodfiles) + 6; | ... | ... |
qpdf/qtest/qpdf/good21.out
0 → 100644
| 1 | +/QTest is indirect and has type array (8) | |
| 2 | +/QTest is an array with 6 items | |
| 3 | + item 0 is direct | |
| 4 | + item 1 is direct | |
| 5 | + item 2 is direct | |
| 6 | + item 3 is indirect | |
| 7 | + item 4 is direct | |
| 8 | + item 5 is indirect | |
| 9 | +unparse: 9 0 R | |
| 10 | +unparseResolved: [ /literal null /indirect 8 0 R /undefined 10 0 R ] | |
| 11 | +test 1 done | ... | ... |
qpdf/qtest/qpdf/good21.pdf
0 → 100644
| 1 | +%PDF-1.3 | |
| 2 | +%¿÷¢þ | |
| 3 | +%QDF-1.0 | |
| 4 | + | |
| 5 | +1 0 obj | |
| 6 | +<< | |
| 7 | + /Pages 2 0 R | |
| 8 | + /Type /Catalog | |
| 9 | +>> | |
| 10 | +endobj | |
| 11 | + | |
| 12 | +2 0 obj | |
| 13 | +<< | |
| 14 | + /Count 1 | |
| 15 | + /Kids [ | |
| 16 | + 3 0 R | |
| 17 | + ] | |
| 18 | + /Type /Pages | |
| 19 | +>> | |
| 20 | +endobj | |
| 21 | + | |
| 22 | +%% Page 1 | |
| 23 | +3 0 obj | |
| 24 | +<< | |
| 25 | + /Contents 4 0 R | |
| 26 | + /MediaBox [ | |
| 27 | + 0 | |
| 28 | + 0 | |
| 29 | + 612 | |
| 30 | + 792 | |
| 31 | + ] | |
| 32 | + /Parent 2 0 R | |
| 33 | + /Resources << | |
| 34 | + /Font << | |
| 35 | + /F1 6 0 R | |
| 36 | + >> | |
| 37 | + /ProcSet 7 0 R | |
| 38 | + >> | |
| 39 | + /Type /Page | |
| 40 | +>> | |
| 41 | +endobj | |
| 42 | + | |
| 43 | +%% Contents for page 1 | |
| 44 | +4 0 obj | |
| 45 | +<< | |
| 46 | + /Length 5 0 R | |
| 47 | +>> | |
| 48 | +stream | |
| 49 | +BT | |
| 50 | + /F1 24 Tf | |
| 51 | + 72 720 Td | |
| 52 | + (Potato) Tj | |
| 53 | +ET | |
| 54 | +endstream | |
| 55 | +endobj | |
| 56 | + | |
| 57 | +5 0 obj | |
| 58 | +44 | |
| 59 | +endobj | |
| 60 | + | |
| 61 | +6 0 obj | |
| 62 | +<< | |
| 63 | + /BaseFont /Helvetica | |
| 64 | + /Encoding /WinAnsiEncoding | |
| 65 | + /Name /F1 | |
| 66 | + /Subtype /Type1 | |
| 67 | + /Type /Font | |
| 68 | +>> | |
| 69 | +endobj | |
| 70 | + | |
| 71 | +7 0 obj | |
| 72 | +[ | |
| 73 | ||
| 74 | + /Text | |
| 75 | +] | |
| 76 | +endobj | |
| 77 | + | |
| 78 | +8 0 obj | |
| 79 | +null | |
| 80 | +endobj | |
| 81 | + | |
| 82 | +9 0 obj | |
| 83 | +[ /literal null /indirect 8 0 R /undefined 10 0 R ] | |
| 84 | +endobj | |
| 85 | + | |
| 86 | +xref | |
| 87 | +0 10 | |
| 88 | +0000000000 65535 f | |
| 89 | +0000000025 00000 n | |
| 90 | +0000000079 00000 n | |
| 91 | +0000000161 00000 n | |
| 92 | +0000000376 00000 n | |
| 93 | +0000000475 00000 n | |
| 94 | +0000000494 00000 n | |
| 95 | +0000000612 00000 n | |
| 96 | +0000000647 00000 n | |
| 97 | +0000000668 00000 n | |
| 98 | +trailer << | |
| 99 | + /Root 1 0 R | |
| 100 | + /Size 10 | |
| 101 | + /QTest 9 0 R | |
| 102 | + /ID [<06c2c8fc54c5f9cc9246898e1e1a7146><06c2c8fc54c5f9cc9246898e1e1a7146>] | |
| 103 | +>> | |
| 104 | +startxref | |
| 105 | +736 | |
| 106 | +%%EOF | ... | ... |
qpdf/qtest/qpdf/good21.qdf
0 → 100644
| 1 | +%PDF-1.3 | |
| 2 | +%¿÷¢þ | |
| 3 | +%QDF-1.0 | |
| 4 | + | |
| 5 | +%% Original object ID: 1 0 | |
| 6 | +1 0 obj | |
| 7 | +<< | |
| 8 | + /Pages 3 0 R | |
| 9 | + /Type /Catalog | |
| 10 | +>> | |
| 11 | +endobj | |
| 12 | + | |
| 13 | +%% Original object ID: 9 0 | |
| 14 | +2 0 obj | |
| 15 | +[ | |
| 16 | + /literal | |
| 17 | + null | |
| 18 | + /indirect | |
| 19 | + 4 0 R | |
| 20 | + /undefined | |
| 21 | + 5 0 R | |
| 22 | +] | |
| 23 | +endobj | |
| 24 | + | |
| 25 | +%% Original object ID: 2 0 | |
| 26 | +3 0 obj | |
| 27 | +<< | |
| 28 | + /Count 1 | |
| 29 | + /Kids [ | |
| 30 | + 6 0 R | |
| 31 | + ] | |
| 32 | + /Type /Pages | |
| 33 | +>> | |
| 34 | +endobj | |
| 35 | + | |
| 36 | +%% Original object ID: 8 0 | |
| 37 | +4 0 obj | |
| 38 | +null | |
| 39 | +endobj | |
| 40 | + | |
| 41 | +%% Original object ID: 10 0 | |
| 42 | +5 0 obj | |
| 43 | +null | |
| 44 | +endobj | |
| 45 | + | |
| 46 | +%% Page 1 | |
| 47 | +%% Original object ID: 3 0 | |
| 48 | +6 0 obj | |
| 49 | +<< | |
| 50 | + /Contents 7 0 R | |
| 51 | + /MediaBox [ | |
| 52 | + 0 | |
| 53 | + 0 | |
| 54 | + 612 | |
| 55 | + 792 | |
| 56 | + ] | |
| 57 | + /Parent 3 0 R | |
| 58 | + /Resources << | |
| 59 | + /Font << | |
| 60 | + /F1 9 0 R | |
| 61 | + >> | |
| 62 | + /ProcSet 10 0 R | |
| 63 | + >> | |
| 64 | + /Type /Page | |
| 65 | +>> | |
| 66 | +endobj | |
| 67 | + | |
| 68 | +%% Contents for page 1 | |
| 69 | +%% Original object ID: 4 0 | |
| 70 | +7 0 obj | |
| 71 | +<< | |
| 72 | + /Length 8 0 R | |
| 73 | +>> | |
| 74 | +stream | |
| 75 | +BT | |
| 76 | + /F1 24 Tf | |
| 77 | + 72 720 Td | |
| 78 | + (Potato) Tj | |
| 79 | +ET | |
| 80 | +endstream | |
| 81 | +endobj | |
| 82 | + | |
| 83 | +8 0 obj | |
| 84 | +44 | |
| 85 | +endobj | |
| 86 | + | |
| 87 | +%% Original object ID: 6 0 | |
| 88 | +9 0 obj | |
| 89 | +<< | |
| 90 | + /BaseFont /Helvetica | |
| 91 | + /Encoding /WinAnsiEncoding | |
| 92 | + /Name /F1 | |
| 93 | + /Subtype /Type1 | |
| 94 | + /Type /Font | |
| 95 | +>> | |
| 96 | +endobj | |
| 97 | + | |
| 98 | +%% Original object ID: 7 0 | |
| 99 | +10 0 obj | |
| 100 | +[ | |
| 101 | ||
| 102 | + /Text | |
| 103 | +] | |
| 104 | +endobj | |
| 105 | + | |
| 106 | +xref | |
| 107 | +0 11 | |
| 108 | +0000000000 65535 f | |
| 109 | +0000000052 00000 n | |
| 110 | +0000000133 00000 n | |
| 111 | +0000000239 00000 n | |
| 112 | +0000000338 00000 n | |
| 113 | +0000000387 00000 n | |
| 114 | +0000000445 00000 n | |
| 115 | +0000000688 00000 n | |
| 116 | +0000000787 00000 n | |
| 117 | +0000000833 00000 n | |
| 118 | +0000000978 00000 n | |
| 119 | +trailer << | |
| 120 | + /QTest 2 0 R | |
| 121 | + /Root 1 0 R | |
| 122 | + /Size 11 | |
| 123 | + /ID [<06c2c8fc54c5f9cc9246898e1e1a7146><31415926535897932384626433832795>] | |
| 124 | +>> | |
| 125 | +startxref | |
| 126 | +1014 | |
| 127 | +%%EOF | ... | ... |