Coverage Report

Created: 2025-08-08 06:36

/src/xpdf-4.05/xpdf/Dict.cc
Line
Count
Source (jump to first uncovered line)
1
//========================================================================
2
//
3
// Dict.cc
4
//
5
// Copyright 1996-2003 Glyph & Cog, LLC
6
//
7
//========================================================================
8
9
#include <aconf.h>
10
11
#include <stddef.h>
12
#include <string.h>
13
#include "gmem.h"
14
#include "gmempp.h"
15
#include "Object.h"
16
#include "XRef.h"
17
#include "Dict.h"
18
19
//------------------------------------------------------------------------
20
21
struct DictEntry {
22
  char *key;
23
  Object val;
24
  DictEntry *next;
25
};
26
27
//------------------------------------------------------------------------
28
// Dict
29
//------------------------------------------------------------------------
30
31
824k
Dict::Dict(XRef *xrefA) {
32
824k
  xref = xrefA;
33
824k
  size = 8;
34
824k
  length = 0;
35
824k
  entries = (DictEntry *)gmallocn(size, sizeof(DictEntry));
36
824k
  hashTab = (DictEntry **)gmallocn(2 * size - 1, sizeof(DictEntry *));
37
824k
  memset(hashTab, 0, (2 * size - 1) * sizeof(DictEntry *));
38
824k
  ref = 1;
39
824k
}
40
41
677k
Dict::~Dict() {
42
677k
  int i;
43
44
3.02M
  for (i = 0; i < length; ++i) {
45
2.35M
    gfree(entries[i].key);
46
2.35M
    entries[i].val.free();
47
2.35M
  }
48
677k
  gfree(entries);
49
677k
  gfree(hashTab);
50
677k
}
51
52
3.46M
void Dict::add(char *key, Object *val) {
53
3.46M
  DictEntry *e;
54
3.46M
  int h;
55
56
3.46M
  if ((e = find(key))) {
57
509k
    e->val.free();
58
509k
    e->val = *val;
59
509k
    gfree(key);
60
2.95M
  } else {
61
2.95M
    if (length == size) {
62
97.0k
      expand();
63
97.0k
    }
64
2.95M
    h = hash(key);
65
2.95M
    entries[length].key = key;
66
2.95M
    entries[length].val = *val;
67
2.95M
    entries[length].next = hashTab[h];
68
2.95M
    hashTab[h] = &entries[length];
69
2.95M
    ++length;
70
2.95M
  }
71
3.46M
}
72
73
97.0k
void Dict::expand() {
74
97.0k
  int h, i;
75
76
97.0k
  size *= 2;
77
97.0k
  entries = (DictEntry *)greallocn(entries, size, sizeof(DictEntry));
78
97.0k
  hashTab = (DictEntry **)greallocn(hashTab, 2 * size - 1,
79
97.0k
            sizeof(DictEntry *));
80
97.0k
  memset(hashTab, 0, (2 * size - 1) * sizeof(DictEntry *));
81
934k
  for (i = 0; i < length; ++i) {
82
837k
    h = hash(entries[i].key);
83
837k
    entries[i].next = hashTab[h];
84
837k
    hashTab[h] = &entries[i];
85
837k
  }
86
97.0k
}
87
88
5.62M
inline DictEntry *Dict::find(const char *key) {
89
5.62M
  DictEntry *e;
90
5.62M
  int h;
91
92
5.62M
  h = hash(key);
93
7.06M
  for (e = hashTab[h]; e; e = e->next) {
94
2.63M
    if (!strcmp(key, e->key)) {
95
1.19M
      return e;
96
1.19M
    }
97
2.63M
  }
98
4.43M
  return NULL;
99
5.62M
}
100
101
9.42M
int Dict::hash(const char *key) {
102
9.42M
  const char *p;
103
9.42M
  unsigned int h;
104
105
9.42M
  h = 0;
106
64.1M
  for (p = key; *p; ++p) {
107
54.7M
    h = 17 * h + (int)(*p & 0xff);
108
54.7M
  }
109
9.42M
  return (int)(h % (2 * size - 1));
110
9.42M
}
111
112
0
GBool Dict::is(const char *type) {
113
0
  DictEntry *e;
114
115
0
  return (e = find("Type")) && e->val.isName(type);
116
0
}
117
118
1.96M
Object *Dict::lookup(const char *key, Object *obj, int recursion) {
119
1.96M
  DictEntry *e;
120
121
1.96M
  return (e = find(key)) ? e->val.fetch(xref, obj, recursion)
122
1.96M
                         : obj->initNull();
123
1.96M
}
124
125
197k
Object *Dict::lookupNF(const char *key, Object *obj) {
126
197k
  DictEntry *e;
127
128
197k
  return (e = find(key)) ? e->val.copy(obj) : obj->initNull();
129
197k
}
130
131
12.2k
char *Dict::getKey(int i) {
132
12.2k
  return entries[i].key;
133
12.2k
}
134
135
6.55k
Object *Dict::getVal(int i, Object *obj) {
136
6.55k
  return entries[i].val.fetch(xref, obj);
137
6.55k
}
138
139
5.67k
Object *Dict::getValNF(int i, Object *obj) {
140
5.67k
  return entries[i].val.copy(obj);
141
5.67k
}