atom.c revision c7b4381a
1a96d7823Smrg/*
2a96d7823Smrg
3a96d7823Smrg  Copyright 1990, 1994, 1998  The Open Group
4a96d7823Smrg
5a96d7823Smrg  Permission to use, copy, modify, distribute, and sell this software and its
6a96d7823Smrg  documentation for any purpose is hereby granted without fee, provided that
7a96d7823Smrg  the above copyright notice appear in all copies and that both that
8a96d7823Smrg  copyright notice and this permission notice appear in supporting
9a96d7823Smrg  documentation.
10a96d7823Smrg
11a96d7823Smrg  The above copyright notice and this permission notice shall be included in
12a96d7823Smrg  all copies or substantial portions of the Software.
13a96d7823Smrg
14a96d7823Smrg  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15a96d7823Smrg  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16a96d7823Smrg  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
17a96d7823Smrg  OPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
18a96d7823Smrg  AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
19a96d7823Smrg  CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
20a96d7823Smrg
21a96d7823Smrg  Except as contained in this notice, the name of The Open Group shall not be
22a96d7823Smrg  used in advertising or otherwise to promote the sale, use or other dealings
23a96d7823Smrg  in this Software without prior written authorization from The Open Group.
24a96d7823Smrg
25a96d7823Smrg*/
26a96d7823Smrg
27a96d7823Smrg/*
28a96d7823Smrg * Author:  Keith Packard, MIT X Consortium
29a96d7823Smrg */
30a96d7823Smrg
31a96d7823Smrg/* lame atom replacement routines for font applications */
32a96d7823Smrg
33a96d7823Smrg#ifdef HAVE_CONFIG_H
34a96d7823Smrg#include <config.h>
35a96d7823Smrg#endif
36a96d7823Smrg#include "libxfontint.h"
37c7b4381aSmrg#include "src/util/replace.h"
38a96d7823Smrg#include <X11/fonts/fontmisc.h>
39a96d7823Smrg
40a96d7823Smrgtypedef struct _AtomList {
41a96d7823Smrg    char *name;
42a96d7823Smrg    int len;
43a96d7823Smrg    int hash;
44a96d7823Smrg    Atom atom;
45a96d7823Smrg} AtomListRec, *AtomListPtr;
46a96d7823Smrg
47a96d7823Smrgstatic AtomListPtr *hashTable;
48a96d7823Smrg
49a96d7823Smrgstatic int hashSize, hashUsed;
50a96d7823Smrgstatic int hashMask;
51a96d7823Smrgstatic int rehash;
52a96d7823Smrg
53a96d7823Smrgstatic AtomListPtr *reverseMap;
54a96d7823Smrgstatic int reverseMapSize;
55a96d7823Smrgstatic Atom lastAtom;
56a96d7823Smrg
57a96d7823Smrgstatic int
58a96d7823SmrgHash(const char *string, int len)
59a96d7823Smrg{
60a96d7823Smrg    int h;
61a96d7823Smrg
62a96d7823Smrg    h = 0;
63a96d7823Smrg    while (len--)
64a96d7823Smrg        h = (h << 3) ^ *string++;
65a96d7823Smrg    if (h < 0)
66a96d7823Smrg        return -h;
67a96d7823Smrg    return h;
68a96d7823Smrg}
69a96d7823Smrg
70a96d7823Smrgstatic int
71a96d7823SmrgResizeHashTable(void)
72a96d7823Smrg{
73a96d7823Smrg    int newHashSize;
74a96d7823Smrg    int newHashMask;
75a96d7823Smrg    AtomListPtr *newHashTable;
76a96d7823Smrg    int i;
77a96d7823Smrg    int h;
78a96d7823Smrg    int newRehash;
79a96d7823Smrg    int r;
80a96d7823Smrg
81a96d7823Smrg    if (hashSize == 0)
82a96d7823Smrg        newHashSize = 1024;
83a96d7823Smrg    else
84a96d7823Smrg        newHashSize = hashSize * 2;
85a96d7823Smrg    newHashTable = calloc(newHashSize, sizeof(AtomListPtr));
86a96d7823Smrg    if (!newHashTable) {
87a96d7823Smrg        fprintf(stderr, "ResizeHashTable(): Error: Couldn't allocate"
88a96d7823Smrg                " newHashTable (%ld)\n",
89a96d7823Smrg                newHashSize * (unsigned long) sizeof(AtomListPtr));
90a96d7823Smrg        return FALSE;
91a96d7823Smrg    }
92a96d7823Smrg    newHashMask = newHashSize - 1;
93a96d7823Smrg    newRehash = (newHashMask - 2);
94a96d7823Smrg    for (i = 0; i < hashSize; i++) {
95a96d7823Smrg        if (hashTable[i]) {
96a96d7823Smrg            h = (hashTable[i]->hash) & newHashMask;
97a96d7823Smrg            if (newHashTable[h]) {
98a96d7823Smrg                r = hashTable[i]->hash % newRehash | 1;
99a96d7823Smrg                do {
100a96d7823Smrg                    h += r;
101a96d7823Smrg                    if (h >= newHashSize)
102a96d7823Smrg                        h -= newHashSize;
103a96d7823Smrg                } while (newHashTable[h]);
104a96d7823Smrg            }
105a96d7823Smrg            newHashTable[h] = hashTable[i];
106a96d7823Smrg        }
107a96d7823Smrg    }
108a96d7823Smrg    free(hashTable);
109a96d7823Smrg    hashTable = newHashTable;
110a96d7823Smrg    hashSize = newHashSize;
111a96d7823Smrg    hashMask = newHashMask;
112a96d7823Smrg    rehash = newRehash;
113a96d7823Smrg    return TRUE;
114a96d7823Smrg}
115a96d7823Smrg
116a96d7823Smrgstatic int
117a96d7823SmrgResizeReverseMap(void)
118a96d7823Smrg{
119a96d7823Smrg    AtomListPtr *newMap;
120a96d7823Smrg    int newMapSize;
121a96d7823Smrg
122a96d7823Smrg    if (reverseMapSize == 0)
123a96d7823Smrg        newMapSize = 1000;
124a96d7823Smrg    else
125a96d7823Smrg        newMapSize = reverseMapSize * 2;
126c7b4381aSmrg    newMap = reallocarray(reverseMap, newMapSize, sizeof(AtomListPtr));
127a96d7823Smrg    if (newMap == NULL) {
128a96d7823Smrg        fprintf(stderr, "ResizeReverseMap(): Error: Couldn't reallocate"
129a96d7823Smrg                " reverseMap (%ld)\n",
130a96d7823Smrg                newMapSize * (unsigned long) sizeof(AtomListPtr));
131a96d7823Smrg        return FALSE;
132a96d7823Smrg    }
133a96d7823Smrg    reverseMap = newMap;
134a96d7823Smrg    reverseMapSize = newMapSize;
135a96d7823Smrg    return TRUE;
136a96d7823Smrg}
137a96d7823Smrg
138a96d7823Smrgstatic int
139a96d7823SmrgNameEqual(const char *a, const char *b, int l)
140a96d7823Smrg{
141a96d7823Smrg    while (l--)
142a96d7823Smrg        if (*a++ != *b++)
143a96d7823Smrg            return FALSE;
144a96d7823Smrg    return TRUE;
145a96d7823Smrg}
146a96d7823Smrg
147a96d7823SmrgAtom
148a96d7823Smrg__libxfont_internal__MakeAtom(const char *string, unsigned len, int makeit)
149a96d7823Smrg{
150a96d7823Smrg    AtomListPtr a;
151a96d7823Smrg    int hash;
152a96d7823Smrg    int h = 0;
153a96d7823Smrg    int r;
154a96d7823Smrg
155a96d7823Smrg    hash = Hash(string, len);
156a96d7823Smrg    if (hashTable) {
157a96d7823Smrg        h = hash & hashMask;
158a96d7823Smrg        if (hashTable[h]) {
159a96d7823Smrg            if (hashTable[h]->hash == hash && hashTable[h]->len == len &&
160a96d7823Smrg                NameEqual(hashTable[h]->name, string, len)) {
161a96d7823Smrg                return hashTable[h]->atom;
162a96d7823Smrg            }
163a96d7823Smrg            r = (hash % rehash) | 1;
164a96d7823Smrg            for (;;) {
165a96d7823Smrg                h += r;
166a96d7823Smrg                if (h >= hashSize)
167a96d7823Smrg                    h -= hashSize;
168a96d7823Smrg                if (!hashTable[h])
169a96d7823Smrg                    break;
170a96d7823Smrg                if (hashTable[h]->hash == hash && hashTable[h]->len == len &&
171a96d7823Smrg                    NameEqual(hashTable[h]->name, string, len)) {
172a96d7823Smrg                    return hashTable[h]->atom;
173a96d7823Smrg                }
174a96d7823Smrg            }
175a96d7823Smrg        }
176a96d7823Smrg    }
177a96d7823Smrg    if (!makeit)
178a96d7823Smrg        return None;
179a96d7823Smrg    a = malloc(sizeof(AtomListRec) + len + 1);
180a96d7823Smrg    if (a == NULL) {
181a96d7823Smrg        fprintf(stderr, "MakeAtom(): Error: Couldn't allocate AtomListRec"
182a96d7823Smrg                " (%ld)\n", (unsigned long) sizeof(AtomListRec) + len + 1);
183a96d7823Smrg        return None;
184a96d7823Smrg    }
185a96d7823Smrg    a->name = (char *) (a + 1);
186a96d7823Smrg    a->len = len;
187a96d7823Smrg    strncpy(a->name, string, len);
188a96d7823Smrg    a->name[len] = '\0';
189a96d7823Smrg    a->atom = ++lastAtom;
190a96d7823Smrg    a->hash = hash;
191a96d7823Smrg    if (hashUsed >= hashSize / 2) {
192c7b4381aSmrg        if ((ResizeHashTable() == FALSE) &&
193c7b4381aSmrg	    ((hashTable == NULL) || (hashUsed == hashSize)))
194c7b4381aSmrg	    return None;
195a96d7823Smrg        h = hash & hashMask;
196a96d7823Smrg        if (hashTable[h]) {
197a96d7823Smrg            r = (hash % rehash) | 1;
198a96d7823Smrg            do {
199a96d7823Smrg                h += r;
200a96d7823Smrg                if (h >= hashSize)
201a96d7823Smrg                    h -= hashSize;
202a96d7823Smrg            } while (hashTable[h]);
203a96d7823Smrg        }
204a96d7823Smrg    }
205a96d7823Smrg    hashTable[h] = a;
206a96d7823Smrg    hashUsed++;
207a96d7823Smrg    if (reverseMapSize <= a->atom) {
208a96d7823Smrg        if (!ResizeReverseMap())
209a96d7823Smrg            return None;
210a96d7823Smrg    }
211a96d7823Smrg    reverseMap[a->atom] = a;
212a96d7823Smrg    return a->atom;
213a96d7823Smrg}
214a96d7823Smrg
215a96d7823Smrgint
216a96d7823Smrg__libxfont_internal__ValidAtom(Atom atom)
217a96d7823Smrg{
218a96d7823Smrg    return (atom != None) && (atom <= lastAtom);
219a96d7823Smrg}
220a96d7823Smrg
221a96d7823Smrgconst char *
222a96d7823Smrg__libxfont_internal__NameForAtom(Atom atom)
223a96d7823Smrg{
224a96d7823Smrg    if (atom != None && atom <= lastAtom)
225a96d7823Smrg        return reverseMap[atom]->name;
226a96d7823Smrg    return NULL;
227a96d7823Smrg}
228