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