partitionmanager/lib/fatlabel/htable.c

235 lines
4.9 KiB
C

/* Copyright 1996,1997,2001,2002,2009 Alain Knaff.
* Copyright 2010 Volker Lanz (vl@fidra.de)
* This file is part of mtools.
*
* Mtools is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Mtools is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Mtools. If not, see <http://www.gnu.org/licenses/>.
*
* hash.c - hash table.
*/
#include "htable.h"
#include "mtools.h"
#include <stdlib.h>
#include <stdio.h>
struct hashtable {
T_HashFunc f1,f2;
T_ComparFunc compar;
int size; /* actual size of the array */
int fill; /* number of deleted or in use slots */
int inuse; /* number of slots in use */
int max; /* maximal number of elements to keep efficient */
T_HashTableEl *entries;
};
static const int sizes[]=
{
5, 11, 23, 47, 97, 197, 397, 797, 1597, 3203, 6421, 12853,
25717, 51437, 102877, 205759, 411527, 823117, 1646237,
3292489, 6584983, 13169977, 26339969, 52679969, 105359939,
210719881, 421439783, 842879579, 1685759167, 0
};
static int deleted=0;
static int unallocated=0;
static int alloc_ht(struct hashtable *H, int size)
{
int i;
for(i = 0; sizes[i]; i++)
if (sizes[i] > size*4)
break;
if (!sizes[i])
for(i = 0; sizes[i]; i++)
if (sizes[i] > size*2 )
break;
if (!sizes[i])
for(i = 0; sizes[i]; i++)
if (sizes[i] > size)
break;
if(!sizes[i])
return -1;
size = sizes[i];
if(size < H->size)
size = H->size; /* never shrink the table */
H->max = size * 4 / 5 - 2;
H->size = size;
H->fill = 0;
H->inuse = 0;
H->entries = NewArray(size, T_HashTableEl);
if (H->entries == NULL)
return -1; /* out of memory error */
for(i = 0; i < size; i++)
H->entries[i] = &unallocated;
return 0;
}
int make_ht(T_HashFunc f1, T_HashFunc f2, T_ComparFunc c, int size, struct hashtable **H)
{
*H = New(struct hashtable);
if (*H == NULL)
return -1; /* out of memory error */
(*H)->f1 = f1;
(*H)->f2 = f2;
(*H)->compar = c;
(*H)->size = 0;
if(alloc_ht(*H,size))
return -1;
return 0;
}
/* add into hash table without checking for repeats */
static int _hash_add(struct hashtable *H,T_HashTableEl *E, int *hint)
{
int pos = H->f1(E) % H->size;
int f2 = -1;
int ctr = 0;
while (H->entries[pos] != &unallocated && H->entries[pos] != &deleted)
{
if (f2 == -1)
f2 = H->f2(E) % (H->size - 1);
pos = (pos + f2 + 1) % H->size;
ctr++;
}
if(H->entries[pos] == &unallocated)
H->fill++; /* only increase fill if the previous element was not yet
* counted, i.e. unallocated */
H->inuse++;
H->entries[pos] = E;
if(hint)
*hint = pos;
return 0;
}
static int rehash(struct hashtable *H)
{
/* resize the table */
int size = H->size;
T_HashTableEl *oldentries = H->entries;
if(alloc_ht(H,((H->inuse+1)*4+H->fill)/5))
return -1;
int i;
for(i = 0; i < size; i++)
{
if (oldentries[i] != &unallocated && oldentries[i] != &deleted)
_hash_add(H, oldentries[i], 0);
}
free(oldentries);
return 0;
}
int hash_add(struct hashtable *H, T_HashTableEl *E, int *hint)
{
if (H->fill >= H->max)
rehash(H);
if (H->fill == H->size)
return -1; /*out of memory error */
return _hash_add(H,E, hint);
}
/* add into hash table without checking for repeats */
static int _hash_lookup(struct hashtable *H,T_HashTableEl *E, T_HashTableEl **E2, int *hint, int isIdentity)
{
int pos = H->f1(E) % H->size;
int ttl = H->size;
int f2 = -1;
int upos = -1;
while(ttl && H->entries[pos] != &unallocated && (H->entries[pos] == &deleted
|| ((isIdentity || H->compar(H->entries[pos], E) != 0) && (!isIdentity || H->entries[pos] != E))))
{
if (f2 == -1)
f2 = H->f2(E) % (H->size - 1);
if (upos == -1 && H->entries[pos] == &deleted)
upos = pos;
pos = (pos + f2 + 1) % H->size;
ttl--;
}
if(H->entries[pos] == &unallocated || !ttl)
return -1;
if (upos != -1)
{
H->entries[upos] = H->entries[pos];
H->entries[pos] = &deleted;
pos = upos;
}
if(hint)
*hint = pos;
*E2 = H->entries[pos];
return 0;
}
int hash_lookup(struct hashtable *H,T_HashTableEl *E, T_HashTableEl **E2, int *hint)
{
return _hash_lookup(H, E, E2, hint, 0);
}
/* add into hash table without checking for repeats */
int hash_remove(struct hashtable *H,T_HashTableEl *E, int hint)
{
T_HashTableEl *E2;
if (hint >=0 && hint < H->size && H->entries[hint] == E)
{
H->inuse--;
H->entries[hint] = &deleted;
return 0;
}
if (_hash_lookup(H, E, &E2, &hint, 1))
{
fprintf(stderr, "Removing non-existent entry\n");
return -1;
}
H->inuse--;
H->entries[hint] = &deleted;
return 0;
}