-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathhash.c
169 lines (138 loc) · 3.21 KB
/
hash.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
/*
* CDDL HEADER START
*
* The contents of this file are subject to the terms of the
* Common Development and Distribution License, Version 1.0 only
* (the "License"). You may not use this file except in compliance
* with the License.
*
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
* or http://www.opensolaris.org/os/licensing.
* See the License for the specific language governing permissions
* and limitations under the License.
*
* When distributing Covered Code, include this CDDL HEADER in each
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
* If applicable, add the following below this CDDL HEADER, with the
* fields enclosed by brackets "[]" replaced with your own identifying
* information: Portions Copyright [yyyy] [name of copyright owner]
*
* CDDL HEADER END
*/
/* Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */
/* All Rights Reserved */
/*
* Portions Copyright (c) 2005 Gunnar Ritter, Freiburg i. Br., Germany
*
* Sccsid @(#)hash.c 1.6 (gritter) 6/26/05
*/
/* from OpenSolaris "hash.c 1.8 05/06/08 SMI" SVr4.0 1.3.2.1 */
#include "hash.h"
#include "defs.h"
#define STRCMP(A, B) (cf(A, B) != 0)
#define FACTOR 035761254233 /* Magic multiplication factor */
#define TABLENGTH 64 /* must be multiple of 2 */
#define LOG2LEN 6 /* log2 of TABLENGTH */
/*
NOTE: The following algorithm only works on machines where
the results of multiplying two integers is the least
significant part of the double word integer required to hold
the result. It is adapted from Knuth, Volume 3, section 6.4.
*/
#define hash(str) (int)(((unsigned)(crunch(str) * FACTOR)) >> shift)
struct node
{
ENTRY item;
struct node *next;
};
static struct node **last;
static struct node *next;
static struct node **table;
static unsigned int bitsper; /* Bits per byte */
static unsigned int shift;
static unsigned int crunch(unsigned char *);
void
hcreate(void)
{
unsigned char c = (unsigned char)~0; /* A byte full of 1's */
int j;
table = (struct node **)alloc(TABLENGTH * sizeof(struct node *));
for (j=0; j < TABLENGTH; ++j)
{
table[j] = 0;
}
bitsper = 0;
while (c)
{
c = (unsigned int)c >> 1;
bitsper++;
}
shift = (bitsper * sizeof(int)) - LOG2LEN;
}
void
hscan(void (*uscan)(ENTRY *))
{
struct node *p, *nxt;
int j;
for (j=0; j < TABLENGTH; ++j)
{
p = table[j];
while (p)
{
nxt = p->next;
(*uscan)(&p->item);
p = nxt;
}
}
}
ENTRY *
hfind(unsigned char *str)
{
struct node *p;
struct node **q;
unsigned int i;
int res = 0;
i = hash(str);
if(table[i] == 0)
{
last = &table[i];
next = 0;
return(0);
}
else
{
q = &table[i];
p = table[i];
while (p != 0 && (res = STRCMP(str, p->item.key)))
{
q = &(p->next);
p = p->next;
}
if (p != 0 && res == 0)
return(&(p->item));
else
{
last = q;
next = p;
return(0);
}
}
}
ENTRY *
henter(ENTRY item)
{
struct node *p = (struct node *)alloc(sizeof(struct node));
p->item = item;
*last = p;
p->next = next;
return(&(p->item));
}
static unsigned int
crunch(unsigned char *key)
{
unsigned int sum = 0;
int s;
for (s = 0; *key; s++) /* Simply add up the bytes */
sum += *key++;
return(sum + s);
}