Coverage Report

Created: 2026-06-22 07:09

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
6.12k
#define ocPolicyAllOn  1
23
4.56k
#define ocPolicyAnyOn  2
24
0
#define ocPolicyAnyOff 3
25
6.07k
#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
648
#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
29.3k
#define displayNodeRecursionLimit 50
36
37
//------------------------------------------------------------------------
38
39
10.2k
OptionalContent::OptionalContent(PDFDoc *doc) {
40
10.2k
  Object *ocProps;
41
10.2k
  Object ocgList, defView, uad, obj1, obj2, obj3, obj4;
42
10.2k
  Ref ref1;
43
10.2k
  OptionalContentGroup *ocg;
44
10.2k
  int i, j;
45
46
10.2k
  xref = doc->getXRef();
47
10.2k
  ocgs = new GList();
48
10.2k
  display = NULL;
49
50
10.2k
  if ((ocProps = doc->getCatalog()->getOCProperties())->isDict()) {
51
194
    if (ocProps->dictLookup("OCGs", &ocgList)->isArray()) {
52
53
      //----- read the OCG list
54
16.4k
      for (i = 0; i < ocgList.arrayGetLength(); ++i) {
55
16.3k
  if (ocgList.arrayGetNF(i, &obj1)->isRef()) {
56
4.40k
    ref1 = obj1.getRef();
57
4.40k
    obj1.fetch(xref, &obj2);
58
4.40k
    if ((ocg = OptionalContentGroup::parse(&ref1, &obj2))) {
59
2.61k
      ocgs->append(ocg);
60
2.61k
    }
61
4.40k
    obj2.free();
62
4.40k
  }
63
16.3k
  obj1.free();
64
16.3k
      }
65
66
      //----- read the default viewing OCCD
67
187
      if (ocProps->dictLookup("D", &defView)->isDict()) {
68
69
  //----- read the usage app dicts
70
186
  if (defView.dictLookup("AS", &obj1)->isArray()) {
71
80
    for (i = 0; i < obj1.arrayGetLength(); ++i) {
72
65
      if (obj1.arrayGet(i, &uad)->isDict()) {
73
23
        if (uad.dictLookup("Event", &obj2)->isName("View")) {
74
5
    if (uad.dictLookup("OCGs", &obj3)->isArray()) {
75
14
      for (j = 0; j < obj3.arrayGetLength(); ++j) {
76
10
        if (obj3.arrayGetNF(j, &obj4)->isRef()) {
77
2
          ref1 = obj4.getRef();
78
2
          if ((ocg = findOCG(&ref1))) {
79
1
      ocg->setInViewUsageAppDict();
80
1
          }
81
2
        }
82
10
        obj4.free();
83
10
      }
84
4
    }
85
5
    obj3.free();
86
5
        }
87
23
        obj2.free();
88
23
      }
89
65
      uad.free();
90
65
    }
91
15
  }
92
186
  obj1.free();
93
94
  //----- initial state from OCCD
95
186
  if (defView.dictLookup("OFF", &obj1)->isArray()) {
96
19.7k
    for (i = 0; i < obj1.arrayGetLength(); ++i) {
97
19.6k
      if (obj1.arrayGetNF(i, &obj2)->isRef()) {
98
17.0k
        ref1 = obj2.getRef();
99
17.0k
        if ((ocg = findOCG(&ref1))) {
100
6.13k
    ocg->setState(gFalse);
101
10.8k
        } else {
102
10.8k
    error(errSyntaxError, -1,
103
10.8k
          "Invalid OCG reference in OFF array in default viewing OCCD");
104
10.8k
        }
105
17.0k
      }
106
19.6k
      obj2.free();
107
19.6k
    }
108
90
  }
109
186
  obj1.free();
110
111
  //----- initial state from OCG usage dict
112
2.80k
  for (i = 0; i < ocgs->getLength(); ++i) {
113
2.61k
    ocg = (OptionalContentGroup *)ocgs->get(i);
114
2.61k
    if (ocg->getInViewUsageAppDict() &&
115
1
        ocg->getViewState() != ocUsageUnset) {
116
1
      ocg->setState(ocg->getViewState() == ocUsageOn);
117
1
    }
118
2.61k
  }
119
120
  //----- display order
121
186
  if (defView.dictLookup("Order", &obj1)->isArray()) {
122
139
    display = OCDisplayNode::parse(&obj1, this, xref);
123
139
  }
124
186
  obj1.free();
125
126
186
      } else {
127
1
  error(errSyntaxError, -1, "Missing or invalid default viewing OCCD");
128
1
      }
129
187
      defView.free();
130
131
187
    }
132
194
    ocgList.free();
133
194
  }
134
135
10.2k
  if (!display) {
136
10.1k
    display = new OCDisplayNode();
137
10.1k
  }
138
10.2k
}
139
140
10.2k
OptionalContent::~OptionalContent() {
141
10.2k
  deleteGList(ocgs, OptionalContentGroup);
142
10.2k
  delete display;
143
10.2k
}
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
30.3k
OptionalContentGroup *OptionalContent::findOCG(Ref *ref) {
154
30.3k
  OptionalContentGroup *ocg;
155
30.3k
  int i;
156
157
1.03M
  for (i = 0; i < ocgs->getLength(); ++i) {
158
1.01M
    ocg = (OptionalContentGroup *)ocgs->get(i);
159
1.01M
    if (ocg->matches(ref)) {
160
9.58k
      return ocg;
161
9.58k
    }
162
1.01M
  }
163
20.7k
  return NULL;
164
30.3k
}
165
166
65.5k
GBool OptionalContent::evalOCObject(Object *obj, GBool *visible) {
167
65.5k
  OptionalContentGroup *ocg;
168
65.5k
  int policy;
169
65.5k
  Ref ref;
170
65.5k
  Object obj2, obj3, obj4, obj5;
171
65.5k
  GBool gotOCG;
172
65.5k
  int i;
173
174
65.5k
  if (obj->isNull()) {
175
58.9k
    return gFalse;
176
58.9k
  }
177
6.57k
  if (obj->isRef()) {
178
6.36k
    ref = obj->getRef();
179
6.36k
    if ((ocg = findOCG(&ref))) {
180
51
      *visible = ocg->getState();
181
51
      return gTrue;
182
51
    }
183
6.36k
  }
184
6.52k
  obj->fetch(xref, &obj2);
185
6.52k
  if (!obj2.isDict("OCMD")) {
186
3.37k
    obj2.free();
187
3.37k
    return gFalse;
188
3.37k
  }
189
3.14k
  if (obj2.dictLookup("VE", &obj3)->isArray()) {
190
84
    *visible = evalOCVisibilityExpr(&obj3, 0);
191
84
    obj3.free();
192
3.06k
  } else {
193
3.06k
    obj3.free();
194
3.06k
    policy = ocPolicyAnyOn;
195
3.06k
    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
3.06k
    obj3.free();
207
3.06k
    obj2.dictLookupNF("OCGs", &obj3);
208
3.06k
    ocg = NULL;
209
3.06k
    if (obj3.isRef()) {
210
43
      ref = obj3.getRef();
211
43
      ocg = findOCG(&ref);
212
43
    }
213
3.06k
    if (ocg) {
214
28
      *visible = (policy == ocPolicyAllOn || policy == ocPolicyAnyOn) ?
215
28
             ocg->getState() : !ocg->getState();
216
3.03k
    } else {
217
3.03k
      *visible = policy == ocPolicyAllOn || policy == ocPolicyAllOff;
218
3.03k
      if (!obj3.fetch(xref, &obj4)->isArray()) {
219
896
  obj4.free();
220
896
  obj3.free();
221
896
  obj2.free();
222
896
  return gFalse;
223
896
      }
224
2.13k
      gotOCG = gFalse;
225
13.5k
      for (i = 0; i < obj4.arrayGetLength(); ++i) {
226
11.3k
  obj4.arrayGetNF(i, &obj5);
227
11.3k
  if (obj5.isRef()) {
228
2.49k
    ref = obj5.getRef();
229
2.49k
    if ((ocg = findOCG(&ref))) {
230
1.47k
      gotOCG = gTrue;
231
1.47k
      switch (policy) {
232
0
      case ocPolicyAllOn:
233
0
        *visible = *visible && ocg->getState();
234
0
        break;
235
1.47k
      case ocPolicyAnyOn:
236
1.47k
        *visible = *visible || ocg->getState();
237
1.47k
        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
1.47k
      }
245
1.47k
    }
246
2.49k
  }
247
11.3k
  obj5.free();
248
11.3k
      }
249
2.13k
      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
948
  obj4.free();
254
948
  obj3.free();
255
948
  obj2.free();
256
948
  return gFalse;
257
948
      }
258
1.19k
      obj4.free();
259
1.19k
    }
260
1.21k
    obj3.free();
261
1.21k
  }
262
1.30k
  obj2.free();
263
1.30k
  return gTrue;
264
3.14k
}
265
266
648
GBool OptionalContent::evalOCVisibilityExpr(Object *expr, int recursion) {
267
648
  OptionalContentGroup *ocg;
268
648
  Object expr2, op, obj;
269
648
  Ref ref;
270
648
  GBool ret;
271
648
  int i;
272
273
648
  if (recursion > visibilityExprRecursionLimit) {
274
0
    error(errSyntaxError, -1,
275
0
    "Loop detected in optional content visibility expression");
276
0
    return gTrue;
277
0
  }
278
648
  if (expr->isRef()) {
279
89
    ref = expr->getRef();
280
89
    if ((ocg = findOCG(&ref))) {
281
18
      return ocg->getState();
282
18
    }
283
89
  }
284
630
  expr->fetch(xref, &expr2);
285
630
  if (!expr2.isArray() || expr2.arrayGetLength() < 1) {
286
514
    error(errSyntaxError, -1,
287
514
    "Invalid optional content visibility expression");
288
514
    expr2.free();
289
514
    return gTrue;
290
514
  }
291
116
  expr2.arrayGet(0, &op);
292
116
  if (op.isName("Not")) {
293
35
    if (expr2.arrayGetLength() == 2) {
294
24
      expr2.arrayGetNF(1, &obj);
295
24
      ret = !evalOCVisibilityExpr(&obj, recursion + 1);
296
24
      obj.free();
297
24
    } else {
298
11
      error(errSyntaxError, -1,
299
11
      "Invalid optional content visibility expression");
300
11
      ret = gTrue;
301
11
    }
302
81
  } else if (op.isName("And")) {
303
54
    ret = gTrue;
304
590
    for (i = 1; i < expr2.arrayGetLength() && ret; ++i) {
305
536
      expr2.arrayGetNF(i, &obj);
306
536
      ret = evalOCVisibilityExpr(&obj, recursion + 1);
307
536
      obj.free();
308
536
    }
309
54
  } else if (op.isName("Or")) {
310
4
    ret = gFalse;
311
8
    for (i = 1; i < expr2.arrayGetLength() && !ret; ++i) {
312
4
      expr2.arrayGetNF(i, &obj);
313
4
      ret = evalOCVisibilityExpr(&obj, recursion + 1);
314
4
      obj.free();
315
4
    }
316
23
  } else {
317
23
    error(errSyntaxError, -1,
318
23
    "Invalid optional content visibility expression");
319
23
    ret = gTrue;
320
23
  }
321
116
  op.free();
322
116
  expr2.free();
323
116
  return ret;
324
630
}
325
326
//------------------------------------------------------------------------
327
328
4.40k
OptionalContentGroup *OptionalContentGroup::parse(Ref *refA, Object *obj) {
329
4.40k
  TextString *nameA;
330
4.40k
  Object obj1, obj2, obj3;
331
4.40k
  OCUsageState viewStateA, printStateA;
332
333
4.40k
  if (!obj->isDict()) {
334
1.62k
    return NULL;
335
1.62k
  }
336
2.78k
  if (!obj->dictLookup("Name", &obj1)->isString()) {
337
165
    error(errSyntaxError, -1, "Missing or invalid Name in OCG");
338
165
    obj1.free();
339
165
    return NULL;
340
165
  }
341
2.61k
  nameA = new TextString(obj1.getString());
342
2.61k
  obj1.free();
343
344
2.61k
  viewStateA = printStateA = ocUsageUnset;
345
2.61k
  if (obj->dictLookup("Usage", &obj1)->isDict()) {
346
44
    if (obj1.dictLookup("View", &obj2)->isDict()) {
347
5
      if (obj2.dictLookup("ViewState", &obj3)->isName()) {
348
4
  if (obj3.isName("ON")) {
349
2
    viewStateA = ocUsageOn;
350
2
  } else {
351
2
    viewStateA = ocUsageOff;
352
2
  }
353
4
      }
354
5
      obj3.free();
355
5
    }
356
44
    obj2.free();
357
44
    if (obj1.dictLookup("Print", &obj2)->isDict()) {
358
5
      if (obj2.dictLookup("PrintState", &obj3)->isName()) {
359
5
  if (obj3.isName("ON")) {
360
2
    printStateA = ocUsageOn;
361
3
  } else {
362
3
    printStateA = ocUsageOff;
363
3
  }
364
5
      }
365
5
      obj3.free();
366
5
    }
367
44
    obj2.free();
368
44
  }
369
2.61k
  obj1.free();
370
371
2.61k
  return new OptionalContentGroup(refA, nameA, viewStateA, printStateA);
372
2.78k
}
373
374
OptionalContentGroup::OptionalContentGroup(Ref *refA, TextString *nameA,
375
             OCUsageState viewStateA,
376
2.61k
             OCUsageState printStateA) {
377
2.61k
  ref = *refA;
378
2.61k
  name = nameA;
379
2.61k
  viewState = viewStateA;
380
2.61k
  printState = printStateA;
381
2.61k
  state = gTrue;
382
2.61k
  inViewUsageAppDict = gFalse;
383
2.61k
}
384
385
2.61k
OptionalContentGroup::~OptionalContentGroup() {
386
2.61k
  delete name;
387
2.61k
}
388
389
1.01M
GBool OptionalContentGroup::matches(Ref *refA) {
390
1.01M
  return refA->num == ref.num && refA->gen == ref.gen;
391
1.01M
}
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
29.3k
            XRef *xref, int recursion) {
405
29.3k
  Object obj2, obj3;
406
29.3k
  Ref ref;
407
29.3k
  OptionalContentGroup *ocgA;
408
29.3k
  OCDisplayNode *node, *child;
409
29.3k
  int i;
410
411
29.3k
  if (recursion > displayNodeRecursionLimit) {
412
705
    error(errSyntaxError, -1, "Loop detected in optional content order");
413
705
    return NULL;
414
705
  }
415
28.6k
  if (obj->isRef()) {
416
4.32k
    ref = obj->getRef();
417
4.32k
    if ((ocgA = oc->findOCG(&ref))) {
418
1.87k
      return new OCDisplayNode(ocgA);
419
1.87k
    }
420
4.32k
  }
421
26.8k
  obj->fetch(xref, &obj2);
422
26.8k
  if (!obj2.isArray()) {
423
25.6k
    obj2.free();
424
25.6k
    return NULL;
425
25.6k
  }
426
1.15k
  i = 0;
427
1.15k
  if (obj2.arrayGetLength() >= 1) {
428
1.14k
    if (obj2.arrayGet(0, &obj3)->isString()) {
429
5
      node = new OCDisplayNode(obj3.getString());
430
5
      i = 1;
431
1.13k
    } else {
432
1.13k
      node = new OCDisplayNode();
433
1.13k
    }
434
1.14k
    obj3.free();
435
1.14k
  } else {
436
15
    node = new OCDisplayNode();
437
15
  }
438
30.4k
  for (; i < obj2.arrayGetLength(); ++i) {
439
29.2k
    obj2.arrayGetNF(i, &obj3);
440
29.2k
    if ((child = OCDisplayNode::parse(&obj3, oc, xref, recursion + 1))) {
441
2.89k
      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
2.89k
      } else {
448
2.89k
  node->addChild(child);
449
2.89k
      }
450
2.89k
    }
451
29.2k
    obj3.free();
452
29.2k
  }
453
1.15k
  obj2.free();
454
1.15k
  return node;
455
26.8k
}
456
457
11.2k
OCDisplayNode::OCDisplayNode() {
458
11.2k
  name = new TextString();
459
11.2k
  ocg = NULL;
460
11.2k
  parent = NULL;
461
11.2k
  children = NULL;
462
11.2k
}
463
464
5
OCDisplayNode::OCDisplayNode(GString *nameA) {
465
5
  name = new TextString(nameA);
466
5
  ocg = NULL;
467
5
  children = NULL;
468
5
}
469
470
1.87k
OCDisplayNode::OCDisplayNode(OptionalContentGroup *ocgA) {
471
1.87k
  name = new TextString(ocgA->name);
472
1.87k
  ocg = ocgA;
473
1.87k
  children = NULL;
474
1.87k
}
475
476
2.89k
void OCDisplayNode::addChild(OCDisplayNode *child) {
477
2.89k
  if (!children) {
478
981
    children = new GList();
479
981
  }
480
2.89k
  children->append(child);
481
2.89k
  child->parent = this;
482
2.89k
}
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
13.1k
OCDisplayNode::~OCDisplayNode() {
510
13.1k
  delete name;
511
13.1k
  if (children) {
512
981
    deleteGList(children, OCDisplayNode);
513
981
  }
514
13.1k
}
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
}