/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 | } |