Coverage Report

Created: 2026-03-07 06:39

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/xpdf-4.06/xpdf/OptionalContent.cc
Line
Count
Source
1
//========================================================================
2
//
3
// OptionalContent.cc
4
//
5
// Copyright 2008-2013 Glyph & Cog, LLC
6
//
7
//========================================================================
8
9
#include <aconf.h>
10
11
#include "gmempp.h"
12
#include "GString.h"
13
#include "GList.h"
14
#include "Error.h"
15
#include "Object.h"
16
#include "PDFDoc.h"
17
#include "TextString.h"
18
#include "OptionalContent.h"
19
20
//------------------------------------------------------------------------
21
22
0
#define ocPolicyAllOn  1
23
0
#define ocPolicyAnyOn  2
24
0
#define ocPolicyAnyOff 3
25
0
#define ocPolicyAllOff 4
26
27
//------------------------------------------------------------------------
28
29
// Max depth of nested visibility expressions.  This is used to catch
30
// infinite loops in the visibility expression object structure.
31
0
#define visibilityExprRecursionLimit 50
32
33
// Max depth of nested display nodes.  This is used to catch infinite
34
// loops in the "Order" object structure.
35
121k
#define displayNodeRecursionLimit 50
36
37
//------------------------------------------------------------------------
38
39
381
OptionalContent::OptionalContent(PDFDoc *doc) {
40
381
  Object *ocProps;
41
381
  Object ocgList, defView, uad, obj1, obj2, obj3, obj4;
42
381
  Ref ref1;
43
381
  OptionalContentGroup *ocg;
44
381
  int i, j;
45
46
381
  xref = doc->getXRef();
47
381
  ocgs = new GList();
48
381
  display = NULL;
49
50
381
  if ((ocProps = doc->getCatalog()->getOCProperties())->isDict()) {
51
87
    if (ocProps->dictLookup("OCGs", &ocgList)->isArray()) {
52
53
      //----- read the OCG list
54
12.1k
      for (i = 0; i < ocgList.arrayGetLength(); ++i) {
55
12.0k
  if (ocgList.arrayGetNF(i, &obj1)->isRef()) {
56
2.64k
    ref1 = obj1.getRef();
57
2.64k
    obj1.fetch(xref, &obj2);
58
2.64k
    if ((ocg = OptionalContentGroup::parse(&ref1, &obj2))) {
59
1.33k
      ocgs->append(ocg);
60
1.33k
    }
61
2.64k
    obj2.free();
62
2.64k
  }
63
12.0k
  obj1.free();
64
12.0k
      }
65
66
      //----- read the default viewing OCCD
67
80
      if (ocProps->dictLookup("D", &defView)->isDict()) {
68
69
  //----- read the usage app dicts
70
52
  if (defView.dictLookup("AS", &obj1)->isArray()) {
71
2.75k
    for (i = 0; i < obj1.arrayGetLength(); ++i) {
72
2.75k
      if (obj1.arrayGet(i, &uad)->isDict()) {
73
119
        if (uad.dictLookup("Event", &obj2)->isName("View")) {
74
1
    if (uad.dictLookup("OCGs", &obj3)->isArray()) {
75
36
      for (j = 0; j < obj3.arrayGetLength(); ++j) {
76
35
        if (obj3.arrayGetNF(j, &obj4)->isRef()) {
77
0
          ref1 = obj4.getRef();
78
0
          if ((ocg = findOCG(&ref1))) {
79
0
      ocg->setInViewUsageAppDict();
80
0
          }
81
0
        }
82
35
        obj4.free();
83
35
      }
84
1
    }
85
1
    obj3.free();
86
1
        }
87
119
        obj2.free();
88
119
      }
89
2.75k
      uad.free();
90
2.75k
    }
91
6
  }
92
52
  obj1.free();
93
94
  //----- initial state from OCCD
95
52
  if (defView.dictLookup("OFF", &obj1)->isArray()) {
96
2.09k
    for (i = 0; i < obj1.arrayGetLength(); ++i) {
97
2.08k
      if (obj1.arrayGetNF(i, &obj2)->isRef()) {
98
1.08k
        ref1 = obj2.getRef();
99
1.08k
        if ((ocg = findOCG(&ref1))) {
100
500
    ocg->setState(gFalse);
101
581
        } else {
102
581
    error(errSyntaxError, -1,
103
581
          "Invalid OCG reference in OFF array in default viewing OCCD");
104
581
        }
105
1.08k
      }
106
2.08k
      obj2.free();
107
2.08k
    }
108
5
  }
109
52
  obj1.free();
110
111
  //----- initial state from OCG usage dict
112
322
  for (i = 0; i < ocgs->getLength(); ++i) {
113
270
    ocg = (OptionalContentGroup *)ocgs->get(i);
114
270
    if (ocg->getInViewUsageAppDict() &&
115
0
        ocg->getViewState() != ocUsageUnset) {
116
0
      ocg->setState(ocg->getViewState() == ocUsageOn);
117
0
    }
118
270
  }
119
120
  //----- display order
121
52
  if (defView.dictLookup("Order", &obj1)->isArray()) {
122
27
    display = OCDisplayNode::parse(&obj1, this, xref);
123
27
  }
124
52
  obj1.free();
125
126
52
      } else {
127
28
  error(errSyntaxError, -1, "Missing or invalid default viewing OCCD");
128
28
      }
129
80
      defView.free();
130
131
80
    }
132
87
    ocgList.free();
133
87
  }
134
135
381
  if (!display) {
136
353
    display = new OCDisplayNode();
137
353
  }
138
381
}
139
140
380
OptionalContent::~OptionalContent() {
141
380
  deleteGList(ocgs, OptionalContentGroup);
142
380
  delete display;
143
380
}
144
145
0
int OptionalContent::getNumOCGs() {
146
0
  return ocgs->getLength();
147
0
}
148
149
0
OptionalContentGroup *OptionalContent::getOCG(int idx) {
150
0
  return (OptionalContentGroup *)ocgs->get(idx);
151
0
}
152
153
21.5k
OptionalContentGroup *OptionalContent::findOCG(Ref *ref) {
154
21.5k
  OptionalContentGroup *ocg;
155
21.5k
  int i;
156
157
36.6k
  for (i = 0; i < ocgs->getLength(); ++i) {
158
15.5k
    ocg = (OptionalContentGroup *)ocgs->get(i);
159
15.5k
    if (ocg->matches(ref)) {
160
508
      return ocg;
161
508
    }
162
15.5k
  }
163
21.0k
  return NULL;
164
21.5k
}
165
166
0
GBool OptionalContent::evalOCObject(Object *obj, GBool *visible) {
167
0
  OptionalContentGroup *ocg;
168
0
  int policy;
169
0
  Ref ref;
170
0
  Object obj2, obj3, obj4, obj5;
171
0
  GBool gotOCG;
172
0
  int i;
173
174
0
  if (obj->isNull()) {
175
0
    return gFalse;
176
0
  }
177
0
  if (obj->isRef()) {
178
0
    ref = obj->getRef();
179
0
    if ((ocg = findOCG(&ref))) {
180
0
      *visible = ocg->getState();
181
0
      return gTrue;
182
0
    }
183
0
  }
184
0
  obj->fetch(xref, &obj2);
185
0
  if (!obj2.isDict("OCMD")) {
186
0
    obj2.free();
187
0
    return gFalse;
188
0
  }
189
0
  if (obj2.dictLookup("VE", &obj3)->isArray()) {
190
0
    *visible = evalOCVisibilityExpr(&obj3, 0);
191
0
    obj3.free();
192
0
  } else {
193
0
    obj3.free();
194
0
    policy = ocPolicyAnyOn;
195
0
    if (obj2.dictLookup("P", &obj3)->isName()) {
196
0
      if (obj3.isName("AllOn")) {
197
0
  policy = ocPolicyAllOn;
198
0
      } else if (obj3.isName("AnyOn")) {
199
0
  policy = ocPolicyAnyOn;
200
0
      } else if (obj3.isName("AnyOff")) {
201
0
  policy = ocPolicyAnyOff;
202
0
      } else if (obj3.isName("AllOff")) {
203
0
  policy = ocPolicyAllOff;
204
0
      }
205
0
    }
206
0
    obj3.free();
207
0
    obj2.dictLookupNF("OCGs", &obj3);
208
0
    ocg = NULL;
209
0
    if (obj3.isRef()) {
210
0
      ref = obj3.getRef();
211
0
      ocg = findOCG(&ref);
212
0
    }
213
0
    if (ocg) {
214
0
      *visible = (policy == ocPolicyAllOn || policy == ocPolicyAnyOn) ?
215
0
             ocg->getState() : !ocg->getState();
216
0
    } else {
217
0
      *visible = policy == ocPolicyAllOn || policy == ocPolicyAllOff;
218
0
      if (!obj3.fetch(xref, &obj4)->isArray()) {
219
0
  obj4.free();
220
0
  obj3.free();
221
0
  obj2.free();
222
0
  return gFalse;
223
0
      }
224
0
      gotOCG = gFalse;
225
0
      for (i = 0; i < obj4.arrayGetLength(); ++i) {
226
0
  obj4.arrayGetNF(i, &obj5);
227
0
  if (obj5.isRef()) {
228
0
    ref = obj5.getRef();
229
0
    if ((ocg = findOCG(&ref))) {
230
0
      gotOCG = gTrue;
231
0
      switch (policy) {
232
0
      case ocPolicyAllOn:
233
0
        *visible = *visible && ocg->getState();
234
0
        break;
235
0
      case ocPolicyAnyOn:
236
0
        *visible = *visible || ocg->getState();
237
0
        break;
238
0
      case ocPolicyAnyOff:
239
0
        *visible = *visible || !ocg->getState();
240
0
        break;
241
0
      case ocPolicyAllOff:
242
0
        *visible = *visible && !ocg->getState();
243
0
        break;
244
0
      }
245
0
    }
246
0
  }
247
0
  obj5.free();
248
0
      }
249
0
      if (!gotOCG) {
250
  // if the "OCGs" array is "empty or contains references only
251
  // to null or deleted objects", this OCMD doesn't have any
252
  // effect
253
0
  obj4.free();
254
0
  obj3.free();
255
0
  obj2.free();
256
0
  return gFalse;
257
0
      }
258
0
      obj4.free();
259
0
    }
260
0
    obj3.free();
261
0
  }
262
0
  obj2.free();
263
0
  return gTrue;
264
0
}
265
266
0
GBool OptionalContent::evalOCVisibilityExpr(Object *expr, int recursion) {
267
0
  OptionalContentGroup *ocg;
268
0
  Object expr2, op, obj;
269
0
  Ref ref;
270
0
  GBool ret;
271
0
  int i;
272
273
0
  if (recursion > visibilityExprRecursionLimit) {
274
0
    error(errSyntaxError, -1,
275
0
    "Loop detected in optional content visibility expression");
276
0
    return gTrue;
277
0
  }
278
0
  if (expr->isRef()) {
279
0
    ref = expr->getRef();
280
0
    if ((ocg = findOCG(&ref))) {
281
0
      return ocg->getState();
282
0
    }
283
0
  }
284
0
  expr->fetch(xref, &expr2);
285
0
  if (!expr2.isArray() || expr2.arrayGetLength() < 1) {
286
0
    error(errSyntaxError, -1,
287
0
    "Invalid optional content visibility expression");
288
0
    expr2.free();
289
0
    return gTrue;
290
0
  }
291
0
  expr2.arrayGet(0, &op);
292
0
  if (op.isName("Not")) {
293
0
    if (expr2.arrayGetLength() == 2) {
294
0
      expr2.arrayGetNF(1, &obj);
295
0
      ret = !evalOCVisibilityExpr(&obj, recursion + 1);
296
0
      obj.free();
297
0
    } else {
298
0
      error(errSyntaxError, -1,
299
0
      "Invalid optional content visibility expression");
300
0
      ret = gTrue;
301
0
    }
302
0
  } else if (op.isName("And")) {
303
0
    ret = gTrue;
304
0
    for (i = 1; i < expr2.arrayGetLength() && ret; ++i) {
305
0
      expr2.arrayGetNF(i, &obj);
306
0
      ret = evalOCVisibilityExpr(&obj, recursion + 1);
307
0
      obj.free();
308
0
    }
309
0
  } else if (op.isName("Or")) {
310
0
    ret = gFalse;
311
0
    for (i = 1; i < expr2.arrayGetLength() && !ret; ++i) {
312
0
      expr2.arrayGetNF(i, &obj);
313
0
      ret = evalOCVisibilityExpr(&obj, recursion + 1);
314
0
      obj.free();
315
0
    }
316
0
  } else {
317
0
    error(errSyntaxError, -1,
318
0
    "Invalid optional content visibility expression");
319
0
    ret = gTrue;
320
0
  }
321
0
  op.free();
322
0
  expr2.free();
323
0
  return ret;
324
0
}
325
326
//------------------------------------------------------------------------
327
328
2.64k
OptionalContentGroup *OptionalContentGroup::parse(Ref *refA, Object *obj) {
329
2.64k
  TextString *nameA;
330
2.64k
  Object obj1, obj2, obj3;
331
2.64k
  OCUsageState viewStateA, printStateA;
332
333
2.64k
  if (!obj->isDict()) {
334
961
    return NULL;
335
961
  }
336
1.68k
  if (!obj->dictLookup("Name", &obj1)->isString()) {
337
349
    error(errSyntaxError, -1, "Missing or invalid Name in OCG");
338
349
    obj1.free();
339
349
    return NULL;
340
349
  }
341
1.33k
  nameA = new TextString(obj1.getString());
342
1.33k
  obj1.free();
343
344
1.33k
  viewStateA = printStateA = ocUsageUnset;
345
1.33k
  if (obj->dictLookup("Usage", &obj1)->isDict()) {
346
85
    if (obj1.dictLookup("View", &obj2)->isDict()) {
347
38
      if (obj2.dictLookup("ViewState", &obj3)->isName()) {
348
12
  if (obj3.isName("ON")) {
349
6
    viewStateA = ocUsageOn;
350
6
  } else {
351
6
    viewStateA = ocUsageOff;
352
6
  }
353
12
      }
354
38
      obj3.free();
355
38
    }
356
85
    obj2.free();
357
85
    if (obj1.dictLookup("Print", &obj2)->isDict()) {
358
38
      if (obj2.dictLookup("PrintState", &obj3)->isName()) {
359
12
  if (obj3.isName("ON")) {
360
6
    printStateA = ocUsageOn;
361
6
  } else {
362
6
    printStateA = ocUsageOff;
363
6
  }
364
12
      }
365
38
      obj3.free();
366
38
    }
367
85
    obj2.free();
368
85
  }
369
1.33k
  obj1.free();
370
371
1.33k
  return new OptionalContentGroup(refA, nameA, viewStateA, printStateA);
372
1.68k
}
373
374
OptionalContentGroup::OptionalContentGroup(Ref *refA, TextString *nameA,
375
             OCUsageState viewStateA,
376
1.33k
             OCUsageState printStateA) {
377
1.33k
  ref = *refA;
378
1.33k
  name = nameA;
379
1.33k
  viewState = viewStateA;
380
1.33k
  printState = printStateA;
381
1.33k
  state = gTrue;
382
1.33k
  inViewUsageAppDict = gFalse;
383
1.33k
}
384
385
1.33k
OptionalContentGroup::~OptionalContentGroup() {
386
1.33k
  delete name;
387
1.33k
}
388
389
15.5k
GBool OptionalContentGroup::matches(Ref *refA) {
390
15.5k
  return refA->num == ref.num && refA->gen == ref.gen;
391
15.5k
}
392
393
0
Unicode *OptionalContentGroup::getName() {
394
0
  return name->getUnicode();
395
0
}
396
397
0
int OptionalContentGroup::getNameLength() {
398
0
  return name->getLength();
399
0
}
400
401
//------------------------------------------------------------------------
402
403
OCDisplayNode *OCDisplayNode::parse(Object *obj, OptionalContent *oc,
404
121k
            XRef *xref, int recursion) {
405
121k
  Object obj2, obj3;
406
121k
  Ref ref;
407
121k
  OptionalContentGroup *ocgA;
408
121k
  OCDisplayNode *node, *child;
409
121k
  int i;
410
411
121k
  if (recursion > displayNodeRecursionLimit) {
412
2.51k
    error(errSyntaxError, -1, "Loop detected in optional content order");
413
2.51k
    return NULL;
414
2.51k
  }
415
119k
  if (obj->isRef()) {
416
20.5k
    ref = obj->getRef();
417
20.5k
    if ((ocgA = oc->findOCG(&ref))) {
418
8
      return new OCDisplayNode(ocgA);
419
8
    }
420
20.5k
  }
421
119k
  obj->fetch(xref, &obj2);
422
119k
  if (!obj2.isArray()) {
423
117k
    obj2.free();
424
117k
    return NULL;
425
117k
  }
426
1.96k
  i = 0;
427
1.96k
  if (obj2.arrayGetLength() >= 1) {
428
1.95k
    if (obj2.arrayGet(0, &obj3)->isString()) {
429
150
      node = new OCDisplayNode(obj3.getString());
430
150
      i = 1;
431
1.80k
    } else {
432
1.80k
      node = new OCDisplayNode();
433
1.80k
    }
434
1.95k
    obj3.free();
435
1.95k
  } else {
436
6
    node = new OCDisplayNode();
437
6
  }
438
123k
  for (; i < obj2.arrayGetLength(); ++i) {
439
121k
    obj2.arrayGetNF(i, &obj3);
440
121k
    if ((child = OCDisplayNode::parse(&obj3, oc, xref, recursion + 1))) {
441
1.94k
      if (!child->ocg && !child->name && node->getNumChildren() > 0) {
442
0
  if (child->getNumChildren() > 0) {
443
0
    node->getChild(node->getNumChildren() - 1)->
444
0
              addChildren(child->takeChildren());
445
0
  }
446
0
  delete child;
447
1.94k
      } else {
448
1.94k
  node->addChild(child);
449
1.94k
      }
450
1.94k
    }
451
121k
    obj3.free();
452
121k
  }
453
1.96k
  obj2.free();
454
1.96k
  return node;
455
119k
}
456
457
2.16k
OCDisplayNode::OCDisplayNode() {
458
2.16k
  name = new TextString();
459
2.16k
  ocg = NULL;
460
2.16k
  parent = NULL;
461
2.16k
  children = NULL;
462
2.16k
}
463
464
150
OCDisplayNode::OCDisplayNode(GString *nameA) {
465
150
  name = new TextString(nameA);
466
150
  ocg = NULL;
467
150
  children = NULL;
468
150
}
469
470
8
OCDisplayNode::OCDisplayNode(OptionalContentGroup *ocgA) {
471
8
  name = new TextString(ocgA->name);
472
8
  ocg = ocgA;
473
8
  children = NULL;
474
8
}
475
476
1.94k
void OCDisplayNode::addChild(OCDisplayNode *child) {
477
1.94k
  if (!children) {
478
634
    children = new GList();
479
634
  }
480
1.94k
  children->append(child);
481
1.94k
  child->parent = this;
482
1.94k
}
483
484
0
void OCDisplayNode::addChildren(GList *childrenA) {
485
0
  int i;
486
487
0
  if (!children) {
488
0
    children = new GList();
489
0
  }
490
0
  children->append(childrenA);
491
0
  for (i = 0; i < childrenA->getLength(); ++i) {
492
0
    ((OCDisplayNode *)childrenA->get(i))->parent = this;
493
0
  }
494
0
  delete childrenA;
495
0
}
496
497
0
GList *OCDisplayNode::takeChildren() {
498
0
  GList *childrenA;
499
0
  int i;
500
501
0
  childrenA = children;
502
0
  children = NULL;
503
0
  for (i = 0; i < childrenA->getLength(); ++i) {
504
0
    ((OCDisplayNode *)childrenA->get(i))->parent = NULL;
505
0
  }
506
0
  return childrenA;
507
0
}
508
509
2.32k
OCDisplayNode::~OCDisplayNode() {
510
2.32k
  delete name;
511
2.32k
  if (children) {
512
634
    deleteGList(children, OCDisplayNode);
513
634
  }
514
2.32k
}
515
516
0
Unicode *OCDisplayNode::getName() {
517
0
  return name->getUnicode();
518
0
}
519
520
0
int OCDisplayNode::getNameLength() {
521
0
  return name->getLength();
522
0
}
523
524
0
int OCDisplayNode::getNumChildren() {
525
0
  if (!children) {
526
0
    return 0;
527
0
  }
528
0
  return children->getLength();
529
0
}
530
531
0
OCDisplayNode *OCDisplayNode::getChild(int idx) {
532
0
  return (OCDisplayNode *)children->get(idx);
533
0
}