-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsymtab.c
98 lines (81 loc) · 2.76 KB
/
symtab.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
/*
This file is part of "Filter Foundry", a filter plugin for Adobe Photoshop
Copyright (C) 2003-2009 Toby Thain, toby@telegraphics.net
Copyright (C) 2018-2024 Daniel Marschall, ViaThinkSoft
This program 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 2 of the License, or
(at your option) any later version.
This program 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 this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#include <stdio.h>
#include <string.h>
#include "symtab.h"
unsigned long djb2(const char *str);
/* following constant (need not) be prime. for a list of prime numbers,
see https://primes.utm.edu/lists/small/1000.txt */
#define TABLE_SIZE 128 // if you're anticipating many symbols, increase this value!
#define HASH(s) (djb2(s) % TABLE_SIZE)
struct sym_rec *hash_table[TABLE_SIZE];
extern struct sym_rec predefs[];
/**
hash function recommended by Ozan Yigit
http://www.cs.yorku.ca/~oz/hash.html
"this algorithm (k=33) was first reported by dan bernstein"
*/
unsigned long djb2(const char *str){
unsigned long hash = 5381;
int c;
while( (c = *str++) )
hash = ((hash << 5) + hash) + c; /* hash * 33 + c */
return hash;
}
void init_symtab(struct sym_rec *pre){
struct sym_rec *p;
int i;
for(i=TABLE_SIZE;i--;)
hash_table[i] = 0;
for(p=pre;p->name;p++){
if(lookup(p->name))
printf("!!!! duplicate predefined symbol: %s\n",p->name);
insert(p);
}
}
void dump_symbols(void){
struct sym_rec *p;
int i,occ,maxchain,chain;
puts("\nsymbol table:");
for(i=occ=maxchain=0;i<TABLE_SIZE;i++)
if(hash_table[i]){
++occ;
for(p=hash_table[i],chain=0;p;p=p->next){
puts(p->name);
++chain;
}
if(chain>maxchain)
maxchain = chain;
}
printf("# hash stats: occupancy=%d/%d (%.1f%%) longest chain=%d\n",
occ,TABLE_SIZE,(100.*occ)/TABLE_SIZE,maxchain);
}
struct sym_rec *lookup(const char *s){
struct sym_rec *p;
int idx = HASH(s);
// printf("# lookup \"%s\" hash=%5d\n",s,idx);
for(p=hash_table[idx];p;p=p->next)
if(!strcmp(s,p->name))
return p;
return 0; /* not found */
}
void insert(struct sym_rec *p){
int idx = HASH(p->name);
p->next = hash_table[idx];
hash_table[idx] = p;
/* DPRINTF("# insert symbol [%5d] \"%s\"\n",idx,p->name);*/
}