Coverage Report

Created: 2025-08-26 06:28

/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
931k
Dict::Dict(XRef *xrefA) {
32
931k
  xref = xrefA;
33
931k
  size = 8;
34
931k
  length = 0;
35
931k
  entries = (DictEntry *)gmallocn(size, sizeof(DictEntry));
36
931k
  hashTab = (DictEntry **)gmallocn(2 * size - 1, sizeof(DictEntry *));
37
931k
  memset(hashTab, 0, (2 * size - 1) * sizeof(DictEntry *));
38
931k
  ref = 1;
39
931k
}
40
41
747k
Dict::~Dict() {
42
747k
  int i;
43
44
3.39M
  for (i = 0; i < length; ++i) {
45
2.65M
    gfree(entries[i].key);
46
2.65M
    entries[i].val.free();
47
2.65M
  }
48
747k
  gfree(entries);
49
747k
  gfree(hashTab);
50
747k
}
51
52
3.96M
void Dict::add(char *key, Object *val) {
53
3.96M
  DictEntry *e;
54
3.96M
  int h;
55
56
3.96M
  if ((e = find(key))) {
57
523k
    e->val.free();
58
523k
    e->val = *val;
59
523k
    gfree(key);
60
3.44M
  } else {
61
3.44M
    if (length == size) {
62
129k
      expand();
63
129k
    }
64
3.44M
    h = hash(key);
65
3.44M
    entries[length].key = key;
66
3.44M
    entries[length].val = *val;
67
3.44M
    entries[length].next = hashTab[h];
68
3.44M
    hashTab[h] = &entries[length];
69
3.44M
    ++length;
70
3.44M
  }
71
3.96M
}
72
73
129k
void Dict::expand() {
74
129k
  int h, i;
75
76
129k
  size *= 2;
77
129k
  entries = (DictEntry *)greallocn(entries, size, sizeof(DictEntry));
78
129k
  hashTab = (DictEntry **)greallocn(hashTab, 2 * size - 1,
79
129k
            sizeof(DictEntry *));
80
129k
  memset(hashTab, 0, (2 * size - 1) * sizeof(DictEntry *));
81
1.22M
  for (i = 0; i < length; ++i) {
82
1.09M
    h = hash(entries[i].key);
83
1.09M
    entries[i].next = hashTab[h];
84
1.09M
    hashTab[h] = &entries[i];
85
1.09M
  }
86
129k
}
87
88
4.98M
inline DictEntry *Dict::find(const char *key) {
89
4.98M
  DictEntry *e;
90
4.98M
  int h;
91
92
4.98M
  h = hash(key);
93
5.92M
  for (e = hashTab[h]; e; e = e->next) {
94
1.99M
    if (!strcmp(key, e->key)) {
95
1.06M
      return e;
96
1.06M
    }
97
1.99M
  }
98
3.92M
  return NULL;
99
4.98M
}
100
101
9.52M
int Dict::hash(const char *key) {
102
9.52M
  const char *p;
103
9.52M
  unsigned int h;
104
105
9.52M
  h = 0;
106
56.6M
  for (p = key; *p; ++p) {
107
47.0M
    h = 17 * h + (int)(*p & 0xff);
108
47.0M
  }
109
9.52M
  return (int)(h % (2 * size - 1));
110
9.52M
}
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
995k
Object *Dict::lookup(const char *key, Object *obj, int recursion) {
119
995k
  DictEntry *e;
120
121
995k
  return (e = find(key)) ? e->val.fetch(xref, obj, recursion)
122
995k
                         : obj->initNull();
123
995k
}
124
125
29.4k
Object *Dict::lookupNF(const char *key, Object *obj) {
126
29.4k
  DictEntry *e;
127
128
29.4k
  return (e = find(key)) ? e->val.copy(obj) : obj->initNull();
129
29.4k
}
130
131
1.58k
char *Dict::getKey(int i) {
132
1.58k
  return entries[i].key;
133
1.58k
}
134
135
836
Object *Dict::getVal(int i, Object *obj) {
136
836
  return entries[i].val.fetch(xref, obj);
137
836
}
138
139
746
Object *Dict::getValNF(int i, Object *obj) {
140
746
  return entries[i].val.copy(obj);
141
746
}