]> git.llucax.com Git - software/mutt-debian.git/blob - hash.c
Imported Upstream version 1.5.18
[software/mutt-debian.git] / hash.c
1 /*
2  * Copyright (C) 1996-2000 Michael R. Elkins <me@mutt.org>
3  *
4  *     This program is free software; you can redistribute it and/or modify
5  *     it under the terms of the GNU General Public License as published by
6  *     the Free Software Foundation; either version 2 of the License, or
7  *     (at your option) any later version.
8  * 
9  *     This program is distributed in the hope that it will be useful,
10  *     but WITHOUT ANY WARRANTY; without even the implied warranty of
11  *     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  *     GNU General Public License for more details.
13  * 
14  *     You should have received a copy of the GNU General Public License
15  *     along with this program; if not, write to the Free Software
16  *     Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
17  */ 
18
19 #if HAVE_CONFIG_H
20 # include "config.h"
21 #endif
22
23 #include <stdlib.h>
24 #include <stdio.h>
25 #include <string.h>
26
27 #include "mutt.h"
28
29 #define SOMEPRIME 149711
30
31 int hash_string (const unsigned char *s, int n)
32 {
33   int h = 0;
34
35 #if 0
36   while (*s)
37     h += *s++;
38 #else
39   while (*s)
40     h += (h << 7) + *s++;
41   h = (h * SOMEPRIME) % n;
42   h = (h >= 0) ? h : h + n;
43 #endif
44
45   return (h % n);
46 }
47
48 HASH *hash_create (int nelem)
49 {
50   HASH *table = safe_malloc (sizeof (HASH));
51   if (nelem == 0)
52     nelem = 2;
53   table->nelem = nelem;
54   table->table = safe_calloc (nelem, sizeof (struct hash_elem *));
55   return table;
56 }
57
58 /* table        hash table to update
59  * key          key to hash on
60  * data         data to associate with `key'
61  * allow_dup    if nonzero, duplicate keys are allowed in the table 
62  */
63 int hash_insert (HASH * table, const char *key, void *data, int allow_dup)
64 {
65   struct hash_elem *ptr;
66   int h;
67
68   ptr = (struct hash_elem *) safe_malloc (sizeof (struct hash_elem));
69   h = hash_string ((unsigned char *) key, table->nelem);
70   ptr->key = key;
71   ptr->data = data;
72
73   if (allow_dup)
74   {
75     ptr->next = table->table[h];
76     table->table[h] = ptr;
77   }
78   else
79   {
80     struct hash_elem *tmp, *last;
81     int r;
82
83     for (tmp = table->table[h], last = NULL; tmp; last = tmp, tmp = tmp->next)
84     {
85       r = mutt_strcmp (tmp->key, key);
86       if (r == 0)
87       {
88         FREE (&ptr);
89         return (-1);
90       }
91       if (r > 0)
92         break;
93     }
94     if (last)
95       last->next = ptr;
96     else
97       table->table[h] = ptr;
98     ptr->next = tmp;
99   }
100   return h;
101 }
102
103 void *hash_find_hash (const HASH * table, int hash, const char *key)
104 {
105   struct hash_elem *ptr = table->table[hash];
106   for (; ptr; ptr = ptr->next)
107   {
108     if (mutt_strcmp (key, ptr->key) == 0)
109       return (ptr->data);
110   }
111   return NULL;
112 }
113
114 void hash_delete_hash (HASH * table, int hash, const char *key, const void *data,
115                        void (*destroy) (void *))
116 {
117   struct hash_elem *ptr = table->table[hash];
118   struct hash_elem **last = &table->table[hash];
119
120   while (ptr) 
121   {
122     if ((data == ptr->data || !data)
123         && mutt_strcmp (ptr->key, key) == 0)
124     {
125       *last = ptr->next;
126       if (destroy)
127         destroy (ptr->data);
128       FREE (&ptr);
129       
130       ptr = *last;
131     }
132     else
133     {
134       last = &ptr->next;
135       ptr = ptr->next;
136     }
137   }
138 }
139
140 /* ptr          pointer to the hash table to be freed
141  * destroy()    function to call to free the ->data member (optional) 
142  */
143 void hash_destroy (HASH **ptr, void (*destroy) (void *))
144 {
145   int i;
146   HASH *pptr = *ptr;
147   struct hash_elem *elem, *tmp;
148
149   for (i = 0 ; i < pptr->nelem; i++)
150   {
151     for (elem = pptr->table[i]; elem; )
152     {
153       tmp = elem;
154       elem = elem->next;
155       if (destroy)
156         destroy (tmp->data);
157       FREE (&tmp);
158     }
159   }
160   FREE (&pptr->table);
161   FREE (ptr);           /* __FREE_CHECKED__ */
162 }