xref: /OK3568_Linux_fs/kernel/scripts/genksyms/genksyms.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
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