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