Coverage Report

Created: 2025-07-11 07:47

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