]> git.llucax.com Git - software/mutt-debian.git/blob - hash.c
fix index weirdness if mailbox is emptied while inside the pager (Closes: 375530)
[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 unsigned int hash_string (const unsigned char *s, unsigned int n)
32 {
33   unsigned int h = 0;
34
35   while (*s)
36     h += (h << 7) + *s++;
37   h = (h * SOMEPRIME) % n;
38
39   return h;
40 }
41
42 HASH *hash_create (int nelem)
43 {
44   HASH *table = safe_malloc (sizeof (HASH));
45   if (nelem == 0)
46     nelem = 2;
47   table->nelem = nelem;
48   table->table = safe_calloc (nelem, sizeof (struct hash_elem *));
49   return table;
50 }
51
52 /* table        hash table to update
53  * key          key to hash on
54  * data         data to associate with `key'
55  * allow_dup    if nonzero, duplicate keys are allowed in the table 
56  */
57 int hash_insert (HASH * table, const char *key, void *data, int allow_dup)
58 {
59   struct hash_elem *ptr;
60   int h;
61
62   ptr = (struct hash_elem *) safe_malloc (sizeof (struct hash_elem));
63   h = hash_string ((unsigned char *) key, table->nelem);
64   ptr->key = key;
65   ptr->data = data;
66
67   if (allow_dup)
68   {
69     ptr->next = table->table[h];
70     table->table[h] = ptr;
71   }
72   else
73   {
74     struct hash_elem *tmp, *last;
75     int r;
76
77     for (tmp = table->table[h], last = NULL; tmp; last = tmp, tmp = tmp->next)
78     {
79       r = mutt_strcmp (tmp->key, key);
80       if (r == 0)
81       {
82         FREE (&ptr);
83         return (-1);
84       }
85       if (r > 0)
86         break;
87     }
88     if (last)
89       last->next = ptr;
90     else
91       table->table[h] = ptr;
92     ptr->next = tmp;
93   }
94   return h;
95 }
96
97 void *hash_find_hash (const HASH * table, int hash, const char *key)
98 {
99   struct hash_elem *ptr = table->table[hash];
100   for (; ptr; ptr = ptr->next)
101   {
102     if (mutt_strcmp (key, ptr->key) == 0)
103       return (ptr->data);
104   }
105   return NULL;
106 }
107
108 void hash_delete_hash (HASH * table, int hash, const char *key, const void *data,
109                        void (*destroy) (void *))
110 {
111   struct hash_elem *ptr = table->table[hash];
112   struct hash_elem **last = &table->table[hash];
113
114   while (ptr) 
115   {
116     if ((data == ptr->data || !data)
117         && mutt_strcmp (ptr->key, key) == 0)
118     {
119       *last = ptr->next;
120       if (destroy)
121         destroy (ptr->data);
122       FREE (&ptr);
123       
124       ptr = *last;
125     }
126     else
127     {
128       last = &ptr->next;
129       ptr = ptr->next;
130     }
131   }
132 }
133
134 /* ptr          pointer to the hash table to be freed
135  * destroy()    function to call to free the ->data member (optional) 
136  */
137 void hash_destroy (HASH **ptr, void (*destroy) (void *))
138 {
139   int i;
140   HASH *pptr = *ptr;
141   struct hash_elem *elem, *tmp;
142
143   for (i = 0 ; i < pptr->nelem; i++)
144   {
145     for (elem = pptr->table[i]; elem; )
146     {
147       tmp = elem;
148       elem = elem->next;
149       if (destroy)
150         destroy (tmp->data);
151       FREE (&tmp);
152     }
153   }
154   FREE (&pptr->table);
155   FREE (ptr);           /* __FREE_CHECKED__ */
156 }