1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-or-later
2*4882a593Smuzhiyun /* Generate kernel symbol version hashes.
3*4882a593Smuzhiyun Copyright 1996, 1997 Linux International.
4*4882a593Smuzhiyun
5*4882a593Smuzhiyun New implementation contributed by Richard Henderson <rth@tamu.edu>
6*4882a593Smuzhiyun Based on original work by Bjorn Ekwall <bj0rn@blox.se>
7*4882a593Smuzhiyun
8*4882a593Smuzhiyun This file was part of the Linux modutils 2.4.22: moved back into the
9*4882a593Smuzhiyun kernel sources by Rusty Russell/Kai Germaschewski.
10*4882a593Smuzhiyun
11*4882a593Smuzhiyun */
12*4882a593Smuzhiyun
13*4882a593Smuzhiyun #include <stdio.h>
14*4882a593Smuzhiyun #include <string.h>
15*4882a593Smuzhiyun #include <stdlib.h>
16*4882a593Smuzhiyun #include <unistd.h>
17*4882a593Smuzhiyun #include <assert.h>
18*4882a593Smuzhiyun #include <stdarg.h>
19*4882a593Smuzhiyun #ifdef __GNU_LIBRARY__
20*4882a593Smuzhiyun #include <getopt.h>
21*4882a593Smuzhiyun #endif /* __GNU_LIBRARY__ */
22*4882a593Smuzhiyun
23*4882a593Smuzhiyun #include "genksyms.h"
24*4882a593Smuzhiyun /*----------------------------------------------------------------------*/
25*4882a593Smuzhiyun
26*4882a593Smuzhiyun #define HASH_BUCKETS 4096
27*4882a593Smuzhiyun
28*4882a593Smuzhiyun static struct symbol *symtab[HASH_BUCKETS];
29*4882a593Smuzhiyun static FILE *debugfile;
30*4882a593Smuzhiyun
31*4882a593Smuzhiyun int cur_line = 1;
32*4882a593Smuzhiyun char *cur_filename, *source_file;
33*4882a593Smuzhiyun int in_source_file;
34*4882a593Smuzhiyun
35*4882a593Smuzhiyun static int flag_debug, flag_dump_defs, flag_reference, flag_dump_types,
36*4882a593Smuzhiyun flag_preserve, flag_warnings, flag_rel_crcs;
37*4882a593Smuzhiyun
38*4882a593Smuzhiyun static int errors;
39*4882a593Smuzhiyun static int nsyms;
40*4882a593Smuzhiyun
41*4882a593Smuzhiyun static struct symbol *expansion_trail;
42*4882a593Smuzhiyun static struct symbol *visited_symbols;
43*4882a593Smuzhiyun
44*4882a593Smuzhiyun static const struct {
45*4882a593Smuzhiyun int n;
46*4882a593Smuzhiyun const char *name;
47*4882a593Smuzhiyun } symbol_types[] = {
48*4882a593Smuzhiyun [SYM_NORMAL] = { 0, NULL},
49*4882a593Smuzhiyun [SYM_TYPEDEF] = {'t', "typedef"},
50*4882a593Smuzhiyun [SYM_ENUM] = {'e', "enum"},
51*4882a593Smuzhiyun [SYM_STRUCT] = {'s', "struct"},
52*4882a593Smuzhiyun [SYM_UNION] = {'u', "union"},
53*4882a593Smuzhiyun [SYM_ENUM_CONST] = {'E', "enum constant"},
54*4882a593Smuzhiyun };
55*4882a593Smuzhiyun
56*4882a593Smuzhiyun static int equal_list(struct string_list *a, struct string_list *b);
57*4882a593Smuzhiyun static void print_list(FILE * f, struct string_list *list);
58*4882a593Smuzhiyun static struct string_list *concat_list(struct string_list *start, ...);
59*4882a593Smuzhiyun static struct string_list *mk_node(const char *string);
60*4882a593Smuzhiyun static void print_location(void);
61*4882a593Smuzhiyun static void print_type_name(enum symbol_type type, const char *name);
62*4882a593Smuzhiyun
63*4882a593Smuzhiyun /*----------------------------------------------------------------------*/
64*4882a593Smuzhiyun
65*4882a593Smuzhiyun static const unsigned int crctab32[] = {
66*4882a593Smuzhiyun 0x00000000U, 0x77073096U, 0xee0e612cU, 0x990951baU, 0x076dc419U,
67*4882a593Smuzhiyun 0x706af48fU, 0xe963a535U, 0x9e6495a3U, 0x0edb8832U, 0x79dcb8a4U,
68*4882a593Smuzhiyun 0xe0d5e91eU, 0x97d2d988U, 0x09b64c2bU, 0x7eb17cbdU, 0xe7b82d07U,
69*4882a593Smuzhiyun 0x90bf1d91U, 0x1db71064U, 0x6ab020f2U, 0xf3b97148U, 0x84be41deU,
70*4882a593Smuzhiyun 0x1adad47dU, 0x6ddde4ebU, 0xf4d4b551U, 0x83d385c7U, 0x136c9856U,
71*4882a593Smuzhiyun 0x646ba8c0U, 0xfd62f97aU, 0x8a65c9ecU, 0x14015c4fU, 0x63066cd9U,
72*4882a593Smuzhiyun 0xfa0f3d63U, 0x8d080df5U, 0x3b6e20c8U, 0x4c69105eU, 0xd56041e4U,
73*4882a593Smuzhiyun 0xa2677172U, 0x3c03e4d1U, 0x4b04d447U, 0xd20d85fdU, 0xa50ab56bU,
74*4882a593Smuzhiyun 0x35b5a8faU, 0x42b2986cU, 0xdbbbc9d6U, 0xacbcf940U, 0x32d86ce3U,
75*4882a593Smuzhiyun 0x45df5c75U, 0xdcd60dcfU, 0xabd13d59U, 0x26d930acU, 0x51de003aU,
76*4882a593Smuzhiyun 0xc8d75180U, 0xbfd06116U, 0x21b4f4b5U, 0x56b3c423U, 0xcfba9599U,
77*4882a593Smuzhiyun 0xb8bda50fU, 0x2802b89eU, 0x5f058808U, 0xc60cd9b2U, 0xb10be924U,
78*4882a593Smuzhiyun 0x2f6f7c87U, 0x58684c11U, 0xc1611dabU, 0xb6662d3dU, 0x76dc4190U,
79*4882a593Smuzhiyun 0x01db7106U, 0x98d220bcU, 0xefd5102aU, 0x71b18589U, 0x06b6b51fU,
80*4882a593Smuzhiyun 0x9fbfe4a5U, 0xe8b8d433U, 0x7807c9a2U, 0x0f00f934U, 0x9609a88eU,
81*4882a593Smuzhiyun 0xe10e9818U, 0x7f6a0dbbU, 0x086d3d2dU, 0x91646c97U, 0xe6635c01U,
82*4882a593Smuzhiyun 0x6b6b51f4U, 0x1c6c6162U, 0x856530d8U, 0xf262004eU, 0x6c0695edU,
83*4882a593Smuzhiyun 0x1b01a57bU, 0x8208f4c1U, 0xf50fc457U, 0x65b0d9c6U, 0x12b7e950U,
84*4882a593Smuzhiyun 0x8bbeb8eaU, 0xfcb9887cU, 0x62dd1ddfU, 0x15da2d49U, 0x8cd37cf3U,
85*4882a593Smuzhiyun 0xfbd44c65U, 0x4db26158U, 0x3ab551ceU, 0xa3bc0074U, 0xd4bb30e2U,
86*4882a593Smuzhiyun 0x4adfa541U, 0x3dd895d7U, 0xa4d1c46dU, 0xd3d6f4fbU, 0x4369e96aU,
87*4882a593Smuzhiyun 0x346ed9fcU, 0xad678846U, 0xda60b8d0U, 0x44042d73U, 0x33031de5U,
88*4882a593Smuzhiyun 0xaa0a4c5fU, 0xdd0d7cc9U, 0x5005713cU, 0x270241aaU, 0xbe0b1010U,
89*4882a593Smuzhiyun 0xc90c2086U, 0x5768b525U, 0x206f85b3U, 0xb966d409U, 0xce61e49fU,
90*4882a593Smuzhiyun 0x5edef90eU, 0x29d9c998U, 0xb0d09822U, 0xc7d7a8b4U, 0x59b33d17U,
91*4882a593Smuzhiyun 0x2eb40d81U, 0xb7bd5c3bU, 0xc0ba6cadU, 0xedb88320U, 0x9abfb3b6U,
92*4882a593Smuzhiyun 0x03b6e20cU, 0x74b1d29aU, 0xead54739U, 0x9dd277afU, 0x04db2615U,
93*4882a593Smuzhiyun 0x73dc1683U, 0xe3630b12U, 0x94643b84U, 0x0d6d6a3eU, 0x7a6a5aa8U,
94*4882a593Smuzhiyun 0xe40ecf0bU, 0x9309ff9dU, 0x0a00ae27U, 0x7d079eb1U, 0xf00f9344U,
95*4882a593Smuzhiyun 0x8708a3d2U, 0x1e01f268U, 0x6906c2feU, 0xf762575dU, 0x806567cbU,
96*4882a593Smuzhiyun 0x196c3671U, 0x6e6b06e7U, 0xfed41b76U, 0x89d32be0U, 0x10da7a5aU,
97*4882a593Smuzhiyun 0x67dd4accU, 0xf9b9df6fU, 0x8ebeeff9U, 0x17b7be43U, 0x60b08ed5U,
98*4882a593Smuzhiyun 0xd6d6a3e8U, 0xa1d1937eU, 0x38d8c2c4U, 0x4fdff252U, 0xd1bb67f1U,
99*4882a593Smuzhiyun 0xa6bc5767U, 0x3fb506ddU, 0x48b2364bU, 0xd80d2bdaU, 0xaf0a1b4cU,
100*4882a593Smuzhiyun 0x36034af6U, 0x41047a60U, 0xdf60efc3U, 0xa867df55U, 0x316e8eefU,
101*4882a593Smuzhiyun 0x4669be79U, 0xcb61b38cU, 0xbc66831aU, 0x256fd2a0U, 0x5268e236U,
102*4882a593Smuzhiyun 0xcc0c7795U, 0xbb0b4703U, 0x220216b9U, 0x5505262fU, 0xc5ba3bbeU,
103*4882a593Smuzhiyun 0xb2bd0b28U, 0x2bb45a92U, 0x5cb36a04U, 0xc2d7ffa7U, 0xb5d0cf31U,
104*4882a593Smuzhiyun 0x2cd99e8bU, 0x5bdeae1dU, 0x9b64c2b0U, 0xec63f226U, 0x756aa39cU,
105*4882a593Smuzhiyun 0x026d930aU, 0x9c0906a9U, 0xeb0e363fU, 0x72076785U, 0x05005713U,
106*4882a593Smuzhiyun 0x95bf4a82U, 0xe2b87a14U, 0x7bb12baeU, 0x0cb61b38U, 0x92d28e9bU,
107*4882a593Smuzhiyun 0xe5d5be0dU, 0x7cdcefb7U, 0x0bdbdf21U, 0x86d3d2d4U, 0xf1d4e242U,
108*4882a593Smuzhiyun 0x68ddb3f8U, 0x1fda836eU, 0x81be16cdU, 0xf6b9265bU, 0x6fb077e1U,
109*4882a593Smuzhiyun 0x18b74777U, 0x88085ae6U, 0xff0f6a70U, 0x66063bcaU, 0x11010b5cU,
110*4882a593Smuzhiyun 0x8f659effU, 0xf862ae69U, 0x616bffd3U, 0x166ccf45U, 0xa00ae278U,
111*4882a593Smuzhiyun 0xd70dd2eeU, 0x4e048354U, 0x3903b3c2U, 0xa7672661U, 0xd06016f7U,
112*4882a593Smuzhiyun 0x4969474dU, 0x3e6e77dbU, 0xaed16a4aU, 0xd9d65adcU, 0x40df0b66U,
113*4882a593Smuzhiyun 0x37d83bf0U, 0xa9bcae53U, 0xdebb9ec5U, 0x47b2cf7fU, 0x30b5ffe9U,
114*4882a593Smuzhiyun 0xbdbdf21cU, 0xcabac28aU, 0x53b39330U, 0x24b4a3a6U, 0xbad03605U,
115*4882a593Smuzhiyun 0xcdd70693U, 0x54de5729U, 0x23d967bfU, 0xb3667a2eU, 0xc4614ab8U,
116*4882a593Smuzhiyun 0x5d681b02U, 0x2a6f2b94U, 0xb40bbe37U, 0xc30c8ea1U, 0x5a05df1bU,
117*4882a593Smuzhiyun 0x2d02ef8dU
118*4882a593Smuzhiyun };
119*4882a593Smuzhiyun
partial_crc32_one(unsigned char c,unsigned long crc)120*4882a593Smuzhiyun static unsigned long partial_crc32_one(unsigned char c, unsigned long crc)
121*4882a593Smuzhiyun {
122*4882a593Smuzhiyun return crctab32[(crc ^ c) & 0xff] ^ (crc >> 8);
123*4882a593Smuzhiyun }
124*4882a593Smuzhiyun
partial_crc32(const char * s,unsigned long crc)125*4882a593Smuzhiyun static unsigned long partial_crc32(const char *s, unsigned long crc)
126*4882a593Smuzhiyun {
127*4882a593Smuzhiyun while (*s)
128*4882a593Smuzhiyun crc = partial_crc32_one(*s++, crc);
129*4882a593Smuzhiyun return crc;
130*4882a593Smuzhiyun }
131*4882a593Smuzhiyun
crc32(const char * s)132*4882a593Smuzhiyun static unsigned long crc32(const char *s)
133*4882a593Smuzhiyun {
134*4882a593Smuzhiyun return partial_crc32(s, 0xffffffff) ^ 0xffffffff;
135*4882a593Smuzhiyun }
136*4882a593Smuzhiyun
137*4882a593Smuzhiyun /*----------------------------------------------------------------------*/
138*4882a593Smuzhiyun
map_to_ns(enum symbol_type t)139*4882a593Smuzhiyun static enum symbol_type map_to_ns(enum symbol_type t)
140*4882a593Smuzhiyun {
141*4882a593Smuzhiyun switch (t) {
142*4882a593Smuzhiyun case SYM_ENUM_CONST:
143*4882a593Smuzhiyun case SYM_NORMAL:
144*4882a593Smuzhiyun case SYM_TYPEDEF:
145*4882a593Smuzhiyun return SYM_NORMAL;
146*4882a593Smuzhiyun case SYM_ENUM:
147*4882a593Smuzhiyun case SYM_STRUCT:
148*4882a593Smuzhiyun case SYM_UNION:
149*4882a593Smuzhiyun return SYM_STRUCT;
150*4882a593Smuzhiyun }
151*4882a593Smuzhiyun return t;
152*4882a593Smuzhiyun }
153*4882a593Smuzhiyun
find_symbol(const char * name,enum symbol_type ns,int exact)154*4882a593Smuzhiyun struct symbol *find_symbol(const char *name, enum symbol_type ns, int exact)
155*4882a593Smuzhiyun {
156*4882a593Smuzhiyun unsigned long h = crc32(name) % HASH_BUCKETS;
157*4882a593Smuzhiyun struct symbol *sym;
158*4882a593Smuzhiyun
159*4882a593Smuzhiyun for (sym = symtab[h]; sym; sym = sym->hash_next)
160*4882a593Smuzhiyun if (map_to_ns(sym->type) == map_to_ns(ns) &&
161*4882a593Smuzhiyun strcmp(name, sym->name) == 0 &&
162*4882a593Smuzhiyun sym->is_declared)
163*4882a593Smuzhiyun break;
164*4882a593Smuzhiyun
165*4882a593Smuzhiyun if (exact && sym && sym->type != ns)
166*4882a593Smuzhiyun return NULL;
167*4882a593Smuzhiyun return sym;
168*4882a593Smuzhiyun }
169*4882a593Smuzhiyun
is_unknown_symbol(struct symbol * sym)170*4882a593Smuzhiyun static int is_unknown_symbol(struct symbol *sym)
171*4882a593Smuzhiyun {
172*4882a593Smuzhiyun struct string_list *defn;
173*4882a593Smuzhiyun
174*4882a593Smuzhiyun return ((sym->type == SYM_STRUCT ||
175*4882a593Smuzhiyun sym->type == SYM_UNION ||
176*4882a593Smuzhiyun sym->type == SYM_ENUM) &&
177*4882a593Smuzhiyun (defn = sym->defn) && defn->tag == SYM_NORMAL &&
178*4882a593Smuzhiyun strcmp(defn->string, "}") == 0 &&
179*4882a593Smuzhiyun (defn = defn->next) && defn->tag == SYM_NORMAL &&
180*4882a593Smuzhiyun strcmp(defn->string, "UNKNOWN") == 0 &&
181*4882a593Smuzhiyun (defn = defn->next) && defn->tag == SYM_NORMAL &&
182*4882a593Smuzhiyun strcmp(defn->string, "{") == 0);
183*4882a593Smuzhiyun }
184*4882a593Smuzhiyun
__add_symbol(const char * name,enum symbol_type type,struct string_list * defn,int is_extern,int is_reference)185*4882a593Smuzhiyun static struct symbol *__add_symbol(const char *name, enum symbol_type type,
186*4882a593Smuzhiyun struct string_list *defn, int is_extern,
187*4882a593Smuzhiyun int is_reference)
188*4882a593Smuzhiyun {
189*4882a593Smuzhiyun unsigned long h;
190*4882a593Smuzhiyun struct symbol *sym;
191*4882a593Smuzhiyun enum symbol_status status = STATUS_UNCHANGED;
192*4882a593Smuzhiyun /* The parser adds symbols in the order their declaration completes,
193*4882a593Smuzhiyun * so it is safe to store the value of the previous enum constant in
194*4882a593Smuzhiyun * a static variable.
195*4882a593Smuzhiyun */
196*4882a593Smuzhiyun static int enum_counter;
197*4882a593Smuzhiyun static struct string_list *last_enum_expr;
198*4882a593Smuzhiyun
199*4882a593Smuzhiyun if (type == SYM_ENUM_CONST) {
200*4882a593Smuzhiyun if (defn) {
201*4882a593Smuzhiyun free_list(last_enum_expr, NULL);
202*4882a593Smuzhiyun last_enum_expr = copy_list_range(defn, NULL);
203*4882a593Smuzhiyun enum_counter = 1;
204*4882a593Smuzhiyun } else {
205*4882a593Smuzhiyun struct string_list *expr;
206*4882a593Smuzhiyun char buf[20];
207*4882a593Smuzhiyun
208*4882a593Smuzhiyun snprintf(buf, sizeof(buf), "%d", enum_counter++);
209*4882a593Smuzhiyun if (last_enum_expr) {
210*4882a593Smuzhiyun expr = copy_list_range(last_enum_expr, NULL);
211*4882a593Smuzhiyun defn = concat_list(mk_node("("),
212*4882a593Smuzhiyun expr,
213*4882a593Smuzhiyun mk_node(")"),
214*4882a593Smuzhiyun mk_node("+"),
215*4882a593Smuzhiyun mk_node(buf), NULL);
216*4882a593Smuzhiyun } else {
217*4882a593Smuzhiyun defn = mk_node(buf);
218*4882a593Smuzhiyun }
219*4882a593Smuzhiyun }
220*4882a593Smuzhiyun } else if (type == SYM_ENUM) {
221*4882a593Smuzhiyun free_list(last_enum_expr, NULL);
222*4882a593Smuzhiyun last_enum_expr = NULL;
223*4882a593Smuzhiyun enum_counter = 0;
224*4882a593Smuzhiyun if (!name)
225*4882a593Smuzhiyun /* Anonymous enum definition, nothing more to do */
226*4882a593Smuzhiyun return NULL;
227*4882a593Smuzhiyun }
228*4882a593Smuzhiyun
229*4882a593Smuzhiyun h = crc32(name) % HASH_BUCKETS;
230*4882a593Smuzhiyun for (sym = symtab[h]; sym; sym = sym->hash_next) {
231*4882a593Smuzhiyun if (map_to_ns(sym->type) == map_to_ns(type) &&
232*4882a593Smuzhiyun strcmp(name, sym->name) == 0) {
233*4882a593Smuzhiyun if (is_reference)
234*4882a593Smuzhiyun /* fall through */ ;
235*4882a593Smuzhiyun else if (sym->type == type &&
236*4882a593Smuzhiyun equal_list(sym->defn, defn)) {
237*4882a593Smuzhiyun if (!sym->is_declared && sym->is_override) {
238*4882a593Smuzhiyun print_location();
239*4882a593Smuzhiyun print_type_name(type, name);
240*4882a593Smuzhiyun fprintf(stderr, " modversion is "
241*4882a593Smuzhiyun "unchanged\n");
242*4882a593Smuzhiyun }
243*4882a593Smuzhiyun sym->is_declared = 1;
244*4882a593Smuzhiyun return sym;
245*4882a593Smuzhiyun } else if (!sym->is_declared) {
246*4882a593Smuzhiyun if (sym->is_override && flag_preserve) {
247*4882a593Smuzhiyun print_location();
248*4882a593Smuzhiyun fprintf(stderr, "ignoring ");
249*4882a593Smuzhiyun print_type_name(type, name);
250*4882a593Smuzhiyun fprintf(stderr, " modversion change\n");
251*4882a593Smuzhiyun sym->is_declared = 1;
252*4882a593Smuzhiyun return sym;
253*4882a593Smuzhiyun } else {
254*4882a593Smuzhiyun status = is_unknown_symbol(sym) ?
255*4882a593Smuzhiyun STATUS_DEFINED : STATUS_MODIFIED;
256*4882a593Smuzhiyun }
257*4882a593Smuzhiyun } else {
258*4882a593Smuzhiyun error_with_pos("redefinition of %s", name);
259*4882a593Smuzhiyun return sym;
260*4882a593Smuzhiyun }
261*4882a593Smuzhiyun break;
262*4882a593Smuzhiyun }
263*4882a593Smuzhiyun }
264*4882a593Smuzhiyun
265*4882a593Smuzhiyun if (sym) {
266*4882a593Smuzhiyun struct symbol **psym;
267*4882a593Smuzhiyun
268*4882a593Smuzhiyun for (psym = &symtab[h]; *psym; psym = &(*psym)->hash_next) {
269*4882a593Smuzhiyun if (*psym == sym) {
270*4882a593Smuzhiyun *psym = sym->hash_next;
271*4882a593Smuzhiyun break;
272*4882a593Smuzhiyun }
273*4882a593Smuzhiyun }
274*4882a593Smuzhiyun --nsyms;
275*4882a593Smuzhiyun }
276*4882a593Smuzhiyun
277*4882a593Smuzhiyun sym = xmalloc(sizeof(*sym));
278*4882a593Smuzhiyun sym->name = name;
279*4882a593Smuzhiyun sym->type = type;
280*4882a593Smuzhiyun sym->defn = defn;
281*4882a593Smuzhiyun sym->expansion_trail = NULL;
282*4882a593Smuzhiyun sym->visited = NULL;
283*4882a593Smuzhiyun sym->is_extern = is_extern;
284*4882a593Smuzhiyun
285*4882a593Smuzhiyun sym->hash_next = symtab[h];
286*4882a593Smuzhiyun symtab[h] = sym;
287*4882a593Smuzhiyun
288*4882a593Smuzhiyun sym->is_declared = !is_reference;
289*4882a593Smuzhiyun sym->status = status;
290*4882a593Smuzhiyun sym->is_override = 0;
291*4882a593Smuzhiyun
292*4882a593Smuzhiyun if (flag_debug) {
293*4882a593Smuzhiyun if (symbol_types[type].name)
294*4882a593Smuzhiyun fprintf(debugfile, "Defn for %s %s == <",
295*4882a593Smuzhiyun symbol_types[type].name, name);
296*4882a593Smuzhiyun else
297*4882a593Smuzhiyun fprintf(debugfile, "Defn for type%d %s == <",
298*4882a593Smuzhiyun type, name);
299*4882a593Smuzhiyun if (is_extern)
300*4882a593Smuzhiyun fputs("extern ", debugfile);
301*4882a593Smuzhiyun print_list(debugfile, defn);
302*4882a593Smuzhiyun fputs(">\n", debugfile);
303*4882a593Smuzhiyun }
304*4882a593Smuzhiyun
305*4882a593Smuzhiyun ++nsyms;
306*4882a593Smuzhiyun return sym;
307*4882a593Smuzhiyun }
308*4882a593Smuzhiyun
add_symbol(const char * name,enum symbol_type type,struct string_list * defn,int is_extern)309*4882a593Smuzhiyun struct symbol *add_symbol(const char *name, enum symbol_type type,
310*4882a593Smuzhiyun struct string_list *defn, int is_extern)
311*4882a593Smuzhiyun {
312*4882a593Smuzhiyun return __add_symbol(name, type, defn, is_extern, 0);
313*4882a593Smuzhiyun }
314*4882a593Smuzhiyun
add_reference_symbol(const char * name,enum symbol_type type,struct string_list * defn,int is_extern)315*4882a593Smuzhiyun static struct symbol *add_reference_symbol(const char *name, enum symbol_type type,
316*4882a593Smuzhiyun struct string_list *defn, int is_extern)
317*4882a593Smuzhiyun {
318*4882a593Smuzhiyun return __add_symbol(name, type, defn, is_extern, 1);
319*4882a593Smuzhiyun }
320*4882a593Smuzhiyun
321*4882a593Smuzhiyun /*----------------------------------------------------------------------*/
322*4882a593Smuzhiyun
free_node(struct string_list * node)323*4882a593Smuzhiyun void free_node(struct string_list *node)
324*4882a593Smuzhiyun {
325*4882a593Smuzhiyun free(node->string);
326*4882a593Smuzhiyun free(node);
327*4882a593Smuzhiyun }
328*4882a593Smuzhiyun
free_list(struct string_list * s,struct string_list * e)329*4882a593Smuzhiyun void free_list(struct string_list *s, struct string_list *e)
330*4882a593Smuzhiyun {
331*4882a593Smuzhiyun while (s != e) {
332*4882a593Smuzhiyun struct string_list *next = s->next;
333*4882a593Smuzhiyun free_node(s);
334*4882a593Smuzhiyun s = next;
335*4882a593Smuzhiyun }
336*4882a593Smuzhiyun }
337*4882a593Smuzhiyun
mk_node(const char * string)338*4882a593Smuzhiyun static struct string_list *mk_node(const char *string)
339*4882a593Smuzhiyun {
340*4882a593Smuzhiyun struct string_list *newnode;
341*4882a593Smuzhiyun
342*4882a593Smuzhiyun newnode = xmalloc(sizeof(*newnode));
343*4882a593Smuzhiyun newnode->string = xstrdup(string);
344*4882a593Smuzhiyun newnode->tag = SYM_NORMAL;
345*4882a593Smuzhiyun newnode->next = NULL;
346*4882a593Smuzhiyun
347*4882a593Smuzhiyun return newnode;
348*4882a593Smuzhiyun }
349*4882a593Smuzhiyun
concat_list(struct string_list * start,...)350*4882a593Smuzhiyun static struct string_list *concat_list(struct string_list *start, ...)
351*4882a593Smuzhiyun {
352*4882a593Smuzhiyun va_list ap;
353*4882a593Smuzhiyun struct string_list *n, *n2;
354*4882a593Smuzhiyun
355*4882a593Smuzhiyun if (!start)
356*4882a593Smuzhiyun return NULL;
357*4882a593Smuzhiyun for (va_start(ap, start); (n = va_arg(ap, struct string_list *));) {
358*4882a593Smuzhiyun for (n2 = n; n2->next; n2 = n2->next)
359*4882a593Smuzhiyun ;
360*4882a593Smuzhiyun n2->next = start;
361*4882a593Smuzhiyun start = n;
362*4882a593Smuzhiyun }
363*4882a593Smuzhiyun va_end(ap);
364*4882a593Smuzhiyun return start;
365*4882a593Smuzhiyun }
366*4882a593Smuzhiyun
copy_node(struct string_list * node)367*4882a593Smuzhiyun struct string_list *copy_node(struct string_list *node)
368*4882a593Smuzhiyun {
369*4882a593Smuzhiyun struct string_list *newnode;
370*4882a593Smuzhiyun
371*4882a593Smuzhiyun newnode = xmalloc(sizeof(*newnode));
372*4882a593Smuzhiyun newnode->string = xstrdup(node->string);
373*4882a593Smuzhiyun newnode->tag = node->tag;
374*4882a593Smuzhiyun
375*4882a593Smuzhiyun return newnode;
376*4882a593Smuzhiyun }
377*4882a593Smuzhiyun
copy_list_range(struct string_list * start,struct string_list * end)378*4882a593Smuzhiyun struct string_list *copy_list_range(struct string_list *start,
379*4882a593Smuzhiyun struct string_list *end)
380*4882a593Smuzhiyun {
381*4882a593Smuzhiyun struct string_list *res, *n;
382*4882a593Smuzhiyun
383*4882a593Smuzhiyun if (start == end)
384*4882a593Smuzhiyun return NULL;
385*4882a593Smuzhiyun n = res = copy_node(start);
386*4882a593Smuzhiyun for (start = start->next; start != end; start = start->next) {
387*4882a593Smuzhiyun n->next = copy_node(start);
388*4882a593Smuzhiyun n = n->next;
389*4882a593Smuzhiyun }
390*4882a593Smuzhiyun n->next = NULL;
391*4882a593Smuzhiyun return res;
392*4882a593Smuzhiyun }
393*4882a593Smuzhiyun
equal_list(struct string_list * a,struct string_list * b)394*4882a593Smuzhiyun static int equal_list(struct string_list *a, struct string_list *b)
395*4882a593Smuzhiyun {
396*4882a593Smuzhiyun while (a && b) {
397*4882a593Smuzhiyun if (a->tag != b->tag || strcmp(a->string, b->string))
398*4882a593Smuzhiyun return 0;
399*4882a593Smuzhiyun a = a->next;
400*4882a593Smuzhiyun b = b->next;
401*4882a593Smuzhiyun }
402*4882a593Smuzhiyun
403*4882a593Smuzhiyun return !a && !b;
404*4882a593Smuzhiyun }
405*4882a593Smuzhiyun
406*4882a593Smuzhiyun #define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]))
407*4882a593Smuzhiyun
read_node(FILE * f)408*4882a593Smuzhiyun static struct string_list *read_node(FILE *f)
409*4882a593Smuzhiyun {
410*4882a593Smuzhiyun char buffer[256];
411*4882a593Smuzhiyun struct string_list node = {
412*4882a593Smuzhiyun .string = buffer,
413*4882a593Smuzhiyun .tag = SYM_NORMAL };
414*4882a593Smuzhiyun int c, in_string = 0;
415*4882a593Smuzhiyun
416*4882a593Smuzhiyun while ((c = fgetc(f)) != EOF) {
417*4882a593Smuzhiyun if (!in_string && c == ' ') {
418*4882a593Smuzhiyun if (node.string == buffer)
419*4882a593Smuzhiyun continue;
420*4882a593Smuzhiyun break;
421*4882a593Smuzhiyun } else if (c == '"') {
422*4882a593Smuzhiyun in_string = !in_string;
423*4882a593Smuzhiyun } else if (c == '\n') {
424*4882a593Smuzhiyun if (node.string == buffer)
425*4882a593Smuzhiyun return NULL;
426*4882a593Smuzhiyun ungetc(c, f);
427*4882a593Smuzhiyun break;
428*4882a593Smuzhiyun }
429*4882a593Smuzhiyun if (node.string >= buffer + sizeof(buffer) - 1) {
430*4882a593Smuzhiyun fprintf(stderr, "Token too long\n");
431*4882a593Smuzhiyun exit(1);
432*4882a593Smuzhiyun }
433*4882a593Smuzhiyun *node.string++ = c;
434*4882a593Smuzhiyun }
435*4882a593Smuzhiyun if (node.string == buffer)
436*4882a593Smuzhiyun return NULL;
437*4882a593Smuzhiyun *node.string = 0;
438*4882a593Smuzhiyun node.string = buffer;
439*4882a593Smuzhiyun
440*4882a593Smuzhiyun if (node.string[1] == '#') {
441*4882a593Smuzhiyun size_t n;
442*4882a593Smuzhiyun
443*4882a593Smuzhiyun for (n = 0; n < ARRAY_SIZE(symbol_types); n++) {
444*4882a593Smuzhiyun if (node.string[0] == symbol_types[n].n) {
445*4882a593Smuzhiyun node.tag = n;
446*4882a593Smuzhiyun node.string += 2;
447*4882a593Smuzhiyun return copy_node(&node);
448*4882a593Smuzhiyun }
449*4882a593Smuzhiyun }
450*4882a593Smuzhiyun fprintf(stderr, "Unknown type %c\n", node.string[0]);
451*4882a593Smuzhiyun exit(1);
452*4882a593Smuzhiyun }
453*4882a593Smuzhiyun return copy_node(&node);
454*4882a593Smuzhiyun }
455*4882a593Smuzhiyun
read_reference(FILE * f)456*4882a593Smuzhiyun static void read_reference(FILE *f)
457*4882a593Smuzhiyun {
458*4882a593Smuzhiyun while (!feof(f)) {
459*4882a593Smuzhiyun struct string_list *defn = NULL;
460*4882a593Smuzhiyun struct string_list *sym, *def;
461*4882a593Smuzhiyun int is_extern = 0, is_override = 0;
462*4882a593Smuzhiyun struct symbol *subsym;
463*4882a593Smuzhiyun
464*4882a593Smuzhiyun sym = read_node(f);
465*4882a593Smuzhiyun if (sym && sym->tag == SYM_NORMAL &&
466*4882a593Smuzhiyun !strcmp(sym->string, "override")) {
467*4882a593Smuzhiyun is_override = 1;
468*4882a593Smuzhiyun free_node(sym);
469*4882a593Smuzhiyun sym = read_node(f);
470*4882a593Smuzhiyun }
471*4882a593Smuzhiyun if (!sym)
472*4882a593Smuzhiyun continue;
473*4882a593Smuzhiyun def = read_node(f);
474*4882a593Smuzhiyun if (def && def->tag == SYM_NORMAL &&
475*4882a593Smuzhiyun !strcmp(def->string, "extern")) {
476*4882a593Smuzhiyun is_extern = 1;
477*4882a593Smuzhiyun free_node(def);
478*4882a593Smuzhiyun def = read_node(f);
479*4882a593Smuzhiyun }
480*4882a593Smuzhiyun while (def) {
481*4882a593Smuzhiyun def->next = defn;
482*4882a593Smuzhiyun defn = def;
483*4882a593Smuzhiyun def = read_node(f);
484*4882a593Smuzhiyun }
485*4882a593Smuzhiyun subsym = add_reference_symbol(xstrdup(sym->string), sym->tag,
486*4882a593Smuzhiyun defn, is_extern);
487*4882a593Smuzhiyun subsym->is_override = is_override;
488*4882a593Smuzhiyun free_node(sym);
489*4882a593Smuzhiyun }
490*4882a593Smuzhiyun }
491*4882a593Smuzhiyun
print_node(FILE * f,struct string_list * list)492*4882a593Smuzhiyun static void print_node(FILE * f, struct string_list *list)
493*4882a593Smuzhiyun {
494*4882a593Smuzhiyun if (symbol_types[list->tag].n) {
495*4882a593Smuzhiyun putc(symbol_types[list->tag].n, f);
496*4882a593Smuzhiyun putc('#', f);
497*4882a593Smuzhiyun }
498*4882a593Smuzhiyun fputs(list->string, f);
499*4882a593Smuzhiyun }
500*4882a593Smuzhiyun
print_list(FILE * f,struct string_list * list)501*4882a593Smuzhiyun static void print_list(FILE * f, struct string_list *list)
502*4882a593Smuzhiyun {
503*4882a593Smuzhiyun struct string_list **e, **b;
504*4882a593Smuzhiyun struct string_list *tmp, **tmp2;
505*4882a593Smuzhiyun int elem = 1;
506*4882a593Smuzhiyun
507*4882a593Smuzhiyun if (list == NULL) {
508*4882a593Smuzhiyun fputs("(nil)", f);
509*4882a593Smuzhiyun return;
510*4882a593Smuzhiyun }
511*4882a593Smuzhiyun
512*4882a593Smuzhiyun tmp = list;
513*4882a593Smuzhiyun while ((tmp = tmp->next) != NULL)
514*4882a593Smuzhiyun elem++;
515*4882a593Smuzhiyun
516*4882a593Smuzhiyun b = alloca(elem * sizeof(*e));
517*4882a593Smuzhiyun e = b + elem;
518*4882a593Smuzhiyun tmp2 = e - 1;
519*4882a593Smuzhiyun
520*4882a593Smuzhiyun (*tmp2--) = list;
521*4882a593Smuzhiyun while ((list = list->next) != NULL)
522*4882a593Smuzhiyun *(tmp2--) = list;
523*4882a593Smuzhiyun
524*4882a593Smuzhiyun while (b != e) {
525*4882a593Smuzhiyun print_node(f, *b++);
526*4882a593Smuzhiyun putc(' ', f);
527*4882a593Smuzhiyun }
528*4882a593Smuzhiyun }
529*4882a593Smuzhiyun
expand_and_crc_sym(struct symbol * sym,unsigned long crc)530*4882a593Smuzhiyun static unsigned long expand_and_crc_sym(struct symbol *sym, unsigned long crc)
531*4882a593Smuzhiyun {
532*4882a593Smuzhiyun struct string_list *list = sym->defn;
533*4882a593Smuzhiyun struct string_list **e, **b;
534*4882a593Smuzhiyun struct string_list *tmp, **tmp2;
535*4882a593Smuzhiyun int elem = 1;
536*4882a593Smuzhiyun
537*4882a593Smuzhiyun if (!list)
538*4882a593Smuzhiyun return crc;
539*4882a593Smuzhiyun
540*4882a593Smuzhiyun tmp = list;
541*4882a593Smuzhiyun while ((tmp = tmp->next) != NULL)
542*4882a593Smuzhiyun elem++;
543*4882a593Smuzhiyun
544*4882a593Smuzhiyun b = alloca(elem * sizeof(*e));
545*4882a593Smuzhiyun e = b + elem;
546*4882a593Smuzhiyun tmp2 = e - 1;
547*4882a593Smuzhiyun
548*4882a593Smuzhiyun *(tmp2--) = list;
549*4882a593Smuzhiyun while ((list = list->next) != NULL)
550*4882a593Smuzhiyun *(tmp2--) = list;
551*4882a593Smuzhiyun
552*4882a593Smuzhiyun while (b != e) {
553*4882a593Smuzhiyun struct string_list *cur;
554*4882a593Smuzhiyun struct symbol *subsym;
555*4882a593Smuzhiyun
556*4882a593Smuzhiyun cur = *(b++);
557*4882a593Smuzhiyun switch (cur->tag) {
558*4882a593Smuzhiyun case SYM_NORMAL:
559*4882a593Smuzhiyun if (flag_dump_defs)
560*4882a593Smuzhiyun fprintf(debugfile, "%s ", cur->string);
561*4882a593Smuzhiyun crc = partial_crc32(cur->string, crc);
562*4882a593Smuzhiyun crc = partial_crc32_one(' ', crc);
563*4882a593Smuzhiyun break;
564*4882a593Smuzhiyun
565*4882a593Smuzhiyun case SYM_ENUM_CONST:
566*4882a593Smuzhiyun case SYM_TYPEDEF:
567*4882a593Smuzhiyun subsym = find_symbol(cur->string, cur->tag, 0);
568*4882a593Smuzhiyun /* FIXME: Bad reference files can segfault here. */
569*4882a593Smuzhiyun if (subsym->expansion_trail) {
570*4882a593Smuzhiyun if (flag_dump_defs)
571*4882a593Smuzhiyun fprintf(debugfile, "%s ", cur->string);
572*4882a593Smuzhiyun crc = partial_crc32(cur->string, crc);
573*4882a593Smuzhiyun crc = partial_crc32_one(' ', crc);
574*4882a593Smuzhiyun } else {
575*4882a593Smuzhiyun subsym->expansion_trail = expansion_trail;
576*4882a593Smuzhiyun expansion_trail = subsym;
577*4882a593Smuzhiyun crc = expand_and_crc_sym(subsym, crc);
578*4882a593Smuzhiyun }
579*4882a593Smuzhiyun break;
580*4882a593Smuzhiyun
581*4882a593Smuzhiyun case SYM_STRUCT:
582*4882a593Smuzhiyun case SYM_UNION:
583*4882a593Smuzhiyun case SYM_ENUM:
584*4882a593Smuzhiyun subsym = find_symbol(cur->string, cur->tag, 0);
585*4882a593Smuzhiyun if (!subsym) {
586*4882a593Smuzhiyun struct string_list *n;
587*4882a593Smuzhiyun
588*4882a593Smuzhiyun error_with_pos("expand undefined %s %s",
589*4882a593Smuzhiyun symbol_types[cur->tag].name,
590*4882a593Smuzhiyun cur->string);
591*4882a593Smuzhiyun n = concat_list(mk_node
592*4882a593Smuzhiyun (symbol_types[cur->tag].name),
593*4882a593Smuzhiyun mk_node(cur->string),
594*4882a593Smuzhiyun mk_node("{"),
595*4882a593Smuzhiyun mk_node("UNKNOWN"),
596*4882a593Smuzhiyun mk_node("}"), NULL);
597*4882a593Smuzhiyun subsym =
598*4882a593Smuzhiyun add_symbol(cur->string, cur->tag, n, 0);
599*4882a593Smuzhiyun }
600*4882a593Smuzhiyun if (subsym->expansion_trail) {
601*4882a593Smuzhiyun if (flag_dump_defs) {
602*4882a593Smuzhiyun fprintf(debugfile, "%s %s ",
603*4882a593Smuzhiyun symbol_types[cur->tag].name,
604*4882a593Smuzhiyun cur->string);
605*4882a593Smuzhiyun }
606*4882a593Smuzhiyun
607*4882a593Smuzhiyun crc = partial_crc32(symbol_types[cur->tag].name,
608*4882a593Smuzhiyun crc);
609*4882a593Smuzhiyun crc = partial_crc32_one(' ', crc);
610*4882a593Smuzhiyun crc = partial_crc32(cur->string, crc);
611*4882a593Smuzhiyun crc = partial_crc32_one(' ', crc);
612*4882a593Smuzhiyun } else {
613*4882a593Smuzhiyun subsym->expansion_trail = expansion_trail;
614*4882a593Smuzhiyun expansion_trail = subsym;
615*4882a593Smuzhiyun crc = expand_and_crc_sym(subsym, crc);
616*4882a593Smuzhiyun }
617*4882a593Smuzhiyun break;
618*4882a593Smuzhiyun }
619*4882a593Smuzhiyun }
620*4882a593Smuzhiyun
621*4882a593Smuzhiyun {
622*4882a593Smuzhiyun static struct symbol **end = &visited_symbols;
623*4882a593Smuzhiyun
624*4882a593Smuzhiyun if (!sym->visited) {
625*4882a593Smuzhiyun *end = sym;
626*4882a593Smuzhiyun end = &sym->visited;
627*4882a593Smuzhiyun sym->visited = (struct symbol *)-1L;
628*4882a593Smuzhiyun }
629*4882a593Smuzhiyun }
630*4882a593Smuzhiyun
631*4882a593Smuzhiyun return crc;
632*4882a593Smuzhiyun }
633*4882a593Smuzhiyun
export_symbol(const char * name)634*4882a593Smuzhiyun void export_symbol(const char *name)
635*4882a593Smuzhiyun {
636*4882a593Smuzhiyun struct symbol *sym;
637*4882a593Smuzhiyun
638*4882a593Smuzhiyun sym = find_symbol(name, SYM_NORMAL, 0);
639*4882a593Smuzhiyun if (!sym)
640*4882a593Smuzhiyun error_with_pos("export undefined symbol %s", name);
641*4882a593Smuzhiyun else {
642*4882a593Smuzhiyun unsigned long crc;
643*4882a593Smuzhiyun int has_changed = 0;
644*4882a593Smuzhiyun
645*4882a593Smuzhiyun if (flag_dump_defs)
646*4882a593Smuzhiyun fprintf(debugfile, "Export %s == <", name);
647*4882a593Smuzhiyun
648*4882a593Smuzhiyun expansion_trail = (struct symbol *)-1L;
649*4882a593Smuzhiyun
650*4882a593Smuzhiyun sym->expansion_trail = expansion_trail;
651*4882a593Smuzhiyun expansion_trail = sym;
652*4882a593Smuzhiyun crc = expand_and_crc_sym(sym, 0xffffffff) ^ 0xffffffff;
653*4882a593Smuzhiyun
654*4882a593Smuzhiyun sym = expansion_trail;
655*4882a593Smuzhiyun while (sym != (struct symbol *)-1L) {
656*4882a593Smuzhiyun struct symbol *n = sym->expansion_trail;
657*4882a593Smuzhiyun
658*4882a593Smuzhiyun if (sym->status != STATUS_UNCHANGED) {
659*4882a593Smuzhiyun if (!has_changed) {
660*4882a593Smuzhiyun print_location();
661*4882a593Smuzhiyun fprintf(stderr, "%s: %s: modversion "
662*4882a593Smuzhiyun "changed because of changes "
663*4882a593Smuzhiyun "in ", flag_preserve ? "error" :
664*4882a593Smuzhiyun "warning", name);
665*4882a593Smuzhiyun } else
666*4882a593Smuzhiyun fprintf(stderr, ", ");
667*4882a593Smuzhiyun print_type_name(sym->type, sym->name);
668*4882a593Smuzhiyun if (sym->status == STATUS_DEFINED)
669*4882a593Smuzhiyun fprintf(stderr, " (became defined)");
670*4882a593Smuzhiyun has_changed = 1;
671*4882a593Smuzhiyun if (flag_preserve)
672*4882a593Smuzhiyun errors++;
673*4882a593Smuzhiyun }
674*4882a593Smuzhiyun sym->expansion_trail = 0;
675*4882a593Smuzhiyun sym = n;
676*4882a593Smuzhiyun }
677*4882a593Smuzhiyun if (has_changed)
678*4882a593Smuzhiyun fprintf(stderr, "\n");
679*4882a593Smuzhiyun
680*4882a593Smuzhiyun if (flag_dump_defs)
681*4882a593Smuzhiyun fputs(">\n", debugfile);
682*4882a593Smuzhiyun
683*4882a593Smuzhiyun /* Used as a linker script. */
684*4882a593Smuzhiyun printf(!flag_rel_crcs ? "__crc_%s = 0x%08lx;\n" :
685*4882a593Smuzhiyun "SECTIONS { .rodata : ALIGN(4) { "
686*4882a593Smuzhiyun "__crc_%s = .; LONG(0x%08lx); } }\n",
687*4882a593Smuzhiyun name, crc);
688*4882a593Smuzhiyun }
689*4882a593Smuzhiyun }
690*4882a593Smuzhiyun
691*4882a593Smuzhiyun /*----------------------------------------------------------------------*/
692*4882a593Smuzhiyun
print_location(void)693*4882a593Smuzhiyun static void print_location(void)
694*4882a593Smuzhiyun {
695*4882a593Smuzhiyun fprintf(stderr, "%s:%d: ", cur_filename ? : "<stdin>", cur_line);
696*4882a593Smuzhiyun }
697*4882a593Smuzhiyun
print_type_name(enum symbol_type type,const char * name)698*4882a593Smuzhiyun static void print_type_name(enum symbol_type type, const char *name)
699*4882a593Smuzhiyun {
700*4882a593Smuzhiyun if (symbol_types[type].name)
701*4882a593Smuzhiyun fprintf(stderr, "%s %s", symbol_types[type].name, name);
702*4882a593Smuzhiyun else
703*4882a593Smuzhiyun fprintf(stderr, "%s", name);
704*4882a593Smuzhiyun }
705*4882a593Smuzhiyun
error_with_pos(const char * fmt,...)706*4882a593Smuzhiyun void error_with_pos(const char *fmt, ...)
707*4882a593Smuzhiyun {
708*4882a593Smuzhiyun va_list args;
709*4882a593Smuzhiyun
710*4882a593Smuzhiyun if (flag_warnings) {
711*4882a593Smuzhiyun print_location();
712*4882a593Smuzhiyun
713*4882a593Smuzhiyun va_start(args, fmt);
714*4882a593Smuzhiyun vfprintf(stderr, fmt, args);
715*4882a593Smuzhiyun va_end(args);
716*4882a593Smuzhiyun putc('\n', stderr);
717*4882a593Smuzhiyun
718*4882a593Smuzhiyun errors++;
719*4882a593Smuzhiyun }
720*4882a593Smuzhiyun }
721*4882a593Smuzhiyun
genksyms_usage(void)722*4882a593Smuzhiyun static void genksyms_usage(void)
723*4882a593Smuzhiyun {
724*4882a593Smuzhiyun fputs("Usage:\n" "genksyms [-adDTwqhVR] > /path/to/.tmp_obj.ver\n" "\n"
725*4882a593Smuzhiyun #ifdef __GNU_LIBRARY__
726*4882a593Smuzhiyun " -s, --symbol-prefix Select symbol prefix\n"
727*4882a593Smuzhiyun " -d, --debug Increment the debug level (repeatable)\n"
728*4882a593Smuzhiyun " -D, --dump Dump expanded symbol defs (for debugging only)\n"
729*4882a593Smuzhiyun " -r, --reference file Read reference symbols from a file\n"
730*4882a593Smuzhiyun " -T, --dump-types file Dump expanded types into file\n"
731*4882a593Smuzhiyun " -p, --preserve Preserve reference modversions or fail\n"
732*4882a593Smuzhiyun " -w, --warnings Enable warnings\n"
733*4882a593Smuzhiyun " -q, --quiet Disable warnings (default)\n"
734*4882a593Smuzhiyun " -h, --help Print this message\n"
735*4882a593Smuzhiyun " -V, --version Print the release version\n"
736*4882a593Smuzhiyun " -R, --relative-crc Emit section relative symbol CRCs\n"
737*4882a593Smuzhiyun #else /* __GNU_LIBRARY__ */
738*4882a593Smuzhiyun " -s Select symbol prefix\n"
739*4882a593Smuzhiyun " -d Increment the debug level (repeatable)\n"
740*4882a593Smuzhiyun " -D Dump expanded symbol defs (for debugging only)\n"
741*4882a593Smuzhiyun " -r file Read reference symbols from a file\n"
742*4882a593Smuzhiyun " -T file Dump expanded types into file\n"
743*4882a593Smuzhiyun " -p Preserve reference modversions or fail\n"
744*4882a593Smuzhiyun " -w Enable warnings\n"
745*4882a593Smuzhiyun " -q Disable warnings (default)\n"
746*4882a593Smuzhiyun " -h Print this message\n"
747*4882a593Smuzhiyun " -V Print the release version\n"
748*4882a593Smuzhiyun " -R Emit section relative symbol CRCs\n"
749*4882a593Smuzhiyun #endif /* __GNU_LIBRARY__ */
750*4882a593Smuzhiyun , stderr);
751*4882a593Smuzhiyun }
752*4882a593Smuzhiyun
main(int argc,char ** argv)753*4882a593Smuzhiyun int main(int argc, char **argv)
754*4882a593Smuzhiyun {
755*4882a593Smuzhiyun FILE *dumpfile = NULL, *ref_file = NULL;
756*4882a593Smuzhiyun int o;
757*4882a593Smuzhiyun
758*4882a593Smuzhiyun #ifdef __GNU_LIBRARY__
759*4882a593Smuzhiyun struct option long_opts[] = {
760*4882a593Smuzhiyun {"debug", 0, 0, 'd'},
761*4882a593Smuzhiyun {"warnings", 0, 0, 'w'},
762*4882a593Smuzhiyun {"quiet", 0, 0, 'q'},
763*4882a593Smuzhiyun {"dump", 0, 0, 'D'},
764*4882a593Smuzhiyun {"reference", 1, 0, 'r'},
765*4882a593Smuzhiyun {"dump-types", 1, 0, 'T'},
766*4882a593Smuzhiyun {"preserve", 0, 0, 'p'},
767*4882a593Smuzhiyun {"version", 0, 0, 'V'},
768*4882a593Smuzhiyun {"help", 0, 0, 'h'},
769*4882a593Smuzhiyun {"relative-crc", 0, 0, 'R'},
770*4882a593Smuzhiyun {0, 0, 0, 0}
771*4882a593Smuzhiyun };
772*4882a593Smuzhiyun
773*4882a593Smuzhiyun while ((o = getopt_long(argc, argv, "s:dwqVDr:T:phR",
774*4882a593Smuzhiyun &long_opts[0], NULL)) != EOF)
775*4882a593Smuzhiyun #else /* __GNU_LIBRARY__ */
776*4882a593Smuzhiyun while ((o = getopt(argc, argv, "s:dwqVDr:T:phR")) != EOF)
777*4882a593Smuzhiyun #endif /* __GNU_LIBRARY__ */
778*4882a593Smuzhiyun switch (o) {
779*4882a593Smuzhiyun case 'd':
780*4882a593Smuzhiyun flag_debug++;
781*4882a593Smuzhiyun break;
782*4882a593Smuzhiyun case 'w':
783*4882a593Smuzhiyun flag_warnings = 1;
784*4882a593Smuzhiyun break;
785*4882a593Smuzhiyun case 'q':
786*4882a593Smuzhiyun flag_warnings = 0;
787*4882a593Smuzhiyun break;
788*4882a593Smuzhiyun case 'V':
789*4882a593Smuzhiyun fputs("genksyms version 2.5.60\n", stderr);
790*4882a593Smuzhiyun break;
791*4882a593Smuzhiyun case 'D':
792*4882a593Smuzhiyun flag_dump_defs = 1;
793*4882a593Smuzhiyun break;
794*4882a593Smuzhiyun case 'r':
795*4882a593Smuzhiyun flag_reference = 1;
796*4882a593Smuzhiyun ref_file = fopen(optarg, "r");
797*4882a593Smuzhiyun if (!ref_file) {
798*4882a593Smuzhiyun perror(optarg);
799*4882a593Smuzhiyun return 1;
800*4882a593Smuzhiyun }
801*4882a593Smuzhiyun break;
802*4882a593Smuzhiyun case 'T':
803*4882a593Smuzhiyun flag_dump_types = 1;
804*4882a593Smuzhiyun dumpfile = fopen(optarg, "w");
805*4882a593Smuzhiyun if (!dumpfile) {
806*4882a593Smuzhiyun perror(optarg);
807*4882a593Smuzhiyun return 1;
808*4882a593Smuzhiyun }
809*4882a593Smuzhiyun break;
810*4882a593Smuzhiyun case 'p':
811*4882a593Smuzhiyun flag_preserve = 1;
812*4882a593Smuzhiyun break;
813*4882a593Smuzhiyun case 'h':
814*4882a593Smuzhiyun genksyms_usage();
815*4882a593Smuzhiyun return 0;
816*4882a593Smuzhiyun case 'R':
817*4882a593Smuzhiyun flag_rel_crcs = 1;
818*4882a593Smuzhiyun break;
819*4882a593Smuzhiyun default:
820*4882a593Smuzhiyun genksyms_usage();
821*4882a593Smuzhiyun return 1;
822*4882a593Smuzhiyun }
823*4882a593Smuzhiyun {
824*4882a593Smuzhiyun extern int yydebug;
825*4882a593Smuzhiyun extern int yy_flex_debug;
826*4882a593Smuzhiyun
827*4882a593Smuzhiyun yydebug = (flag_debug > 1);
828*4882a593Smuzhiyun yy_flex_debug = (flag_debug > 2);
829*4882a593Smuzhiyun
830*4882a593Smuzhiyun debugfile = stderr;
831*4882a593Smuzhiyun /* setlinebuf(debugfile); */
832*4882a593Smuzhiyun }
833*4882a593Smuzhiyun
834*4882a593Smuzhiyun if (flag_reference) {
835*4882a593Smuzhiyun read_reference(ref_file);
836*4882a593Smuzhiyun fclose(ref_file);
837*4882a593Smuzhiyun }
838*4882a593Smuzhiyun
839*4882a593Smuzhiyun yyparse();
840*4882a593Smuzhiyun
841*4882a593Smuzhiyun if (flag_dump_types && visited_symbols) {
842*4882a593Smuzhiyun while (visited_symbols != (struct symbol *)-1L) {
843*4882a593Smuzhiyun struct symbol *sym = visited_symbols;
844*4882a593Smuzhiyun
845*4882a593Smuzhiyun if (sym->is_override)
846*4882a593Smuzhiyun fputs("override ", dumpfile);
847*4882a593Smuzhiyun if (symbol_types[sym->type].n) {
848*4882a593Smuzhiyun putc(symbol_types[sym->type].n, dumpfile);
849*4882a593Smuzhiyun putc('#', dumpfile);
850*4882a593Smuzhiyun }
851*4882a593Smuzhiyun fputs(sym->name, dumpfile);
852*4882a593Smuzhiyun putc(' ', dumpfile);
853*4882a593Smuzhiyun if (sym->is_extern)
854*4882a593Smuzhiyun fputs("extern ", dumpfile);
855*4882a593Smuzhiyun print_list(dumpfile, sym->defn);
856*4882a593Smuzhiyun putc('\n', dumpfile);
857*4882a593Smuzhiyun
858*4882a593Smuzhiyun visited_symbols = sym->visited;
859*4882a593Smuzhiyun sym->visited = NULL;
860*4882a593Smuzhiyun }
861*4882a593Smuzhiyun }
862*4882a593Smuzhiyun
863*4882a593Smuzhiyun if (flag_debug) {
864*4882a593Smuzhiyun fprintf(debugfile, "Hash table occupancy %d/%d = %g\n",
865*4882a593Smuzhiyun nsyms, HASH_BUCKETS,
866*4882a593Smuzhiyun (double)nsyms / (double)HASH_BUCKETS);
867*4882a593Smuzhiyun }
868*4882a593Smuzhiyun
869*4882a593Smuzhiyun if (dumpfile)
870*4882a593Smuzhiyun fclose(dumpfile);
871*4882a593Smuzhiyun
872*4882a593Smuzhiyun return errors != 0;
873*4882a593Smuzhiyun }
874