-
Notifications
You must be signed in to change notification settings - Fork 4
/
module.c
236 lines (223 loc) · 7.1 KB
/
module.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
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
/*
* module : module.c
* version : 1.20
* date : 10/11/24
*/
#include "globals.h"
/*
* hide stack of private sections. The sections are numbered, while keeping the
* string presentation instead of the number itself.
*/
static int hide_index = -1, hide_count;
static unsigned char inside_hide, been_inside;
/*
* stack of module names. Only one module needs to be considered active for the
* purpose of prefixing the module to the symbol name.
*/
static int module_index = -1;
/*
* savemod saves the global variables, to be restored later with undomod.
*/
void savemod(int *hide, int *modl, int *hcnt)
{
*hide = hide_index;
*modl = module_index;
*hcnt = hide_count;
}
void undomod(int hide, int modl, int hcnt)
{
hide_index = hide;
module_index = modl;
hide_count = hcnt;
}
/*
* initmod registers name as a module name. Modules within modules are
* supported. Up to a certain extent, that is.
*/
void initmod(pEnv env, char *name)
{
if (module_index + 1 == DISPLAYMAX)
execerror(env, "index", "display");
env->module_stack[++module_index].name = name;
env->module_stack[module_index].hide = hide_index;
}
/*
* initpriv starts a new local section. Local sections within local sections
* are supported. Private sections are numbered consecutively. If a module is
* active then the hide_index is registered in the module. The variable
* inside_hide tells whether defined names should be registered as private.
* This is done by the function classify and used in the function enterdef.
* Only register a new private section during the first read. During the
* second read, the number that was installed should be picked up again.
*/
void initpriv(pEnv env)
{
if (hide_index + 1 == DISPLAYMAX)
execerror(env, "index", "display");
env->hide_stack[++hide_index] = ++hide_count;
inside_hide = 1;
}
/*
* stoppriv registers the transition from private to public definitions. Names
* should no longer be prefixed with the name of the private section.
*/
void stoppriv(void)
{
been_inside = 1;
inside_hide = 0;
}
/*
* exitpriv lowers the hide_index after reading the public section.
*/
void exitpriv(void)
{
if (!been_inside)
return;
been_inside = 0;
if (hide_index >= 0)
hide_index--;
}
/*
* exitmod deregisters a module. It also ends an outstanding private section.
*/
void exitmod(void)
{
if (module_index >= 0)
module_index--;
if (module_index < 0)
exitpriv();
}
/*
* classify prepends name with private section or module name, whichever
* comes first. Names are stored in the symbol table together with this prefix,
* allowing the same names to be used in different private sections and modules.
* The symbol table is flat, in the sense that it has no hierarchy. For that
* reason names must be made unique with this prefix. For public names it is
* sufficient that they are preceded by the module name; for private names it
* is ok to have the private number as a prefix.
*
* classify is used when entering the name that has a definition.
*/
char *classify(pEnv env, char *name)
{
size_t leng;
char temp[MAXNUM], *buf = 0, *str;
/*
* if name already has a prefix, there is no need to add another one.
*/
if (strchr(name, '.'))
return name;
/*
* inside a private section, names that are to be entered in the symbol
* table should get the hide number as a prefix.
*/
if (inside_hide) {
snprintf(temp, MAXNUM, "%d", env->hide_stack[hide_index]);
buf = temp;
}
/*
* inside a module, public names that are to be entered in the symbol table
* should get the module name as a prefix. That is also the name used when
* accessing the symbol.
*/
else if (module_index >= 0)
buf = env->module_stack[module_index].name;
/*
* buf, when filled, contains either a module identifier, or a number
* string.
*/
if (buf) {
leng = strlen(buf) + strlen(name) + 2;
str = GC_malloc_atomic(leng);
snprintf(str, leng, "%s.%s", buf, name);
} else
str = name;
/*
* str will contain either the unadorned name, or a classified name.
*/
return str;
}
/*
* qualify does the reverse of classify. A name can be given in full, by
* prefixing the name with a module, but this only needs to be done outside a
* module, and only one module name needs to be given. For private sections it
* is different. If there is a private section active, then the name needs to
* be searched there and if a module is active then the name needs to be
* searched there as well. The return value is the result of lookup, or the
* full name in case the lookup was unsuccessful.
*
* qualify is used when reading a name, as part of a definition.
*/
int qualify(pEnv env, char *name)
{
khint_t key;
size_t leng;
int index, limit;
char temp[MAXNUM], *buf, *str;
/*
* if name has a prefix, it is already a fully qualified name and can be
* searched in the symbol table right away. The prefix can only be a module
* name. If the name is not found, there is an error and a 0 is returned.
*/
if (strchr(name, '.')) {
#ifdef USE_KHASHL
if ((key = symtab_get(env->hash, name)) != kh_end(env->hash))
#else
if ((key = kh_get(Symtab, env->hash, name)) != kh_end(env->hash))
#endif
return kh_val(env->hash, key);
return 0;
}
/*
* the hide stack is searched, trying each of the hidden sections. The
* return value from lookup is returned. In case of a module, only the
* hide stack that is active at module creation is searched, leaving out
* the search through the hide stack of enclosing modules.
*/
if (hide_index >= 0) {
if (module_index >= 0)
limit = env->module_stack[module_index].hide;
else
limit = -1;
for (index = hide_index; index > limit; index--) {
snprintf(temp, MAXNUM, "%d", env->hide_stack[index]);
leng = strlen(temp) + strlen(name) + 2;
str = GC_malloc_atomic(leng);
snprintf(str, leng, "%s.%s", temp, name);
#ifdef USE_KHASHL
if ((key = symtab_get(env->hash, str)) != kh_end(env->hash))
#else
if ((key = kh_get(Symtab, env->hash, str)) != kh_end(env->hash))
#endif
return kh_val(env->hash, key);
}
}
/*
* if the name can not be found in the local tables, it should be searched
* in the currently active module, if there is one.
*/
if (module_index >= 0) {
buf = env->module_stack[module_index].name;
leng = strlen(buf) + strlen(name) + 2;
str = GC_malloc_atomic(leng);
snprintf(str, leng, "%s.%s", buf, name);
#ifdef USE_KHASHL
if ((key = symtab_get(env->hash, str)) != kh_end(env->hash))
#else
if ((key = kh_get(Symtab, env->hash, str)) != kh_end(env->hash))
#endif
return kh_val(env->hash, key);
}
/*
* if the name is not fully classified, cannot be found in the local tables
* and also not in the module, it needs to be searched as is. If not found,
* it is not an error, but simply an undefined name.
*/
#ifdef USE_KHASHL
if ((key = symtab_get(env->hash, name)) != kh_end(env->hash))
#else
if ((key = kh_get(Symtab, env->hash, name)) != kh_end(env->hash))
#endif
return kh_val(env->hash, key);
return 0;
}