Coverage Report

Created: 2026-02-04 06:14

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/xpdf-4.06/xpdf/Dict.cc
Line
Count
Source
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
678k
Dict::Dict(XRef *xrefA) {
32
678k
  xref = xrefA;
33
678k
  size = 8;
34
678k
  length = 0;
35
678k
  entries = (DictEntry *)gmallocn(size, sizeof(DictEntry));
36
678k
  hashTab = (DictEntry **)gmallocn(2 * size - 1, sizeof(DictEntry *));
37
678k
  memset(hashTab, 0, (2 * size - 1) * sizeof(DictEntry *));
38
678k
  ref = 1;
39
678k
}
40
41
677k
Dict::~Dict() {
42
677k
  int i;
43
44
3.65M
  for (i = 0; i < length; ++i) {
45
2.97M
    gfree(entries[i].key);
46
2.97M
    entries[i].val.free();
47
2.97M
  }
48
677k
  gfree(entries);
49
677k
  gfree(hashTab);
50
677k
}
51
52
3.25M
void Dict::add(char *key, Object *val) {
53
3.25M
  DictEntry *e;
54
3.25M
  int h;
55
56
3.25M
  if ((e = find(key))) {
57
272k
    e->val.free();
58
272k
    e->val = *val;
59
272k
    gfree(key);
60
2.98M
  } else {
61
2.98M
    if (length == size) {
62
73.8k
      expand();
63
73.8k
    }
64
2.98M
    h = hash(key);
65
2.98M
    entries[length].key = key;
66
2.98M
    entries[length].val = *val;
67
2.98M
    entries[length].next = hashTab[h];
68
2.98M
    hashTab[h] = &entries[length];
69
2.98M
    ++length;
70
2.98M
  }
71
3.25M
}
72
73
73.8k
void Dict::expand() {
74
73.8k
  int h, i;
75
76
73.8k
  size *= 2;
77
73.8k
  entries = (DictEntry *)greallocn(entries, size, sizeof(DictEntry));
78
73.8k
  hashTab = (DictEntry **)greallocn(hashTab, 2 * size - 1,
79
73.8k
            sizeof(DictEntry *));
80
73.8k
  memset(hashTab, 0, (2 * size - 1) * sizeof(DictEntry *));
81
976k
  for (i = 0; i < length; ++i) {
82
903k
    h = hash(entries[i].key);
83
903k
    entries[i].next = hashTab[h];
84
903k
    hashTab[h] = &entries[i];
85
903k
  }
86
73.8k
}
87
88
15.4M
inline DictEntry *Dict::find(const char *key) {
89
15.4M
  DictEntry *e;
90
15.4M
  int h;
91
92
15.4M
  h = hash(key);
93
19.5M
  for (e = hashTab[h]; e; e = e->next) {
94
6.92M
    if (!strcmp(key, e->key)) {
95
2.89M
      return e;
96
2.89M
    }
97
6.92M
  }
98
12.5M
  return NULL;
99
15.4M
}
100
101
19.3M
int Dict::hash(const char *key) {
102
19.3M
  const char *p;
103
19.3M
  unsigned int h;
104
105
19.3M
  h = 0;
106
135M
  for (p = key; *p; ++p) {
107
115M
    h = 17 * h + (int)(*p & 0xff);
108
115M
  }
109
19.3M
  return (int)(h % (2 * size - 1));
110
19.3M
}
111
112
2.63k
GBool Dict::is(const char *type) {
113
2.63k
  DictEntry *e;
114
115
2.63k
  return (e = find("Type")) && e->val.isName(type);
116
2.63k
}
117
118
10.0M
Object *Dict::lookup(const char *key, Object *obj, int recursion) {
119
10.0M
  DictEntry *e;
120
121
10.0M
  return (e = find(key)) ? e->val.fetch(xref, obj, recursion)
122
10.0M
                         : obj->initNull();
123
10.0M
}
124
125
2.18M
Object *Dict::lookupNF(const char *key, Object *obj) {
126
2.18M
  DictEntry *e;
127
128
2.18M
  return (e = find(key)) ? e->val.copy(obj) : obj->initNull();
129
2.18M
}
130
131
525k
char *Dict::getKey(int i) {
132
525k
  return entries[i].key;
133
525k
}
134
135
10.0k
Object *Dict::getVal(int i, Object *obj) {
136
10.0k
  return entries[i].val.fetch(xref, obj);
137
10.0k
}
138
139
519k
Object *Dict::getValNF(int i, Object *obj) {
140
519k
  return entries[i].val.copy(obj);
141
519k
}