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