kallsyms.c 17.9 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
/* Generate assembler source containing symbol information
 *
 * Copyright 2002       by Kai Germaschewski
 *
 * This software may be used and distributed according to the terms
 * of the GNU General Public License, incorporated herein by reference.
 *
 * Usage: nm -n vmlinux | scripts/kallsyms [--all-symbols] > symbols.S
 *
 *      Table compression uses all the unused char codes on the symbols and
 *  maps these to the most used substrings (tokens). For instance, it might
 *  map char code 0xF7 to represent "write_" and then in every symbol where
 *  "write_" appears it can be replaced by 0xF7, saving 5 bytes.
 *      The used codes themselves are also placed in the table so that the
 *  decompresion can work without "special cases".
 *      Applied to kernel symbols, this usually produces a compression ratio
 *  of about 50%.
 *
 */

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
25
#include <limits.h>
L
Linus Torvalds 已提交
26

27 28
#define ARRAY_SIZE(arr) (sizeof(arr) / sizeof(arr[0]))

29
#define KSYM_NAME_LEN		128
L
Linus Torvalds 已提交
30 31 32

struct sym_entry {
	unsigned long long addr;
33
	unsigned int len;
34
	unsigned int start_pos;
L
Linus Torvalds 已提交
35
	unsigned char *sym;
36
	unsigned int percpu_absolute;
L
Linus Torvalds 已提交
37 38
};

39 40
struct addr_range {
	const char *start_sym, *end_sym;
41 42 43 44
	unsigned long long start, end;
};

static unsigned long long _text;
45
static unsigned long long relative_base;
46
static struct addr_range text_ranges[] = {
47 48 49 50 51 52
	{ "_stext",     "_etext"     },
	{ "_sinittext", "_einittext" },
};
#define text_range_text     (&text_ranges[0])
#define text_range_inittext (&text_ranges[1])

53 54 55 56
static struct addr_range percpu_range = {
	"__per_cpu_start", "__per_cpu_end", -1ULL, 0
};

L
Linus Torvalds 已提交
57
static struct sym_entry *table;
58
static unsigned int table_size, table_cnt;
L
Linus Torvalds 已提交
59
static int all_symbols = 0;
60
static int absolute_percpu = 0;
61
static int base_relative = 0;
L
Linus Torvalds 已提交
62

63
static int token_profit[0x10000];
L
Linus Torvalds 已提交
64 65

/* the table that holds the result of the compression */
66 67
static unsigned char best_table[256][2];
static unsigned char best_table_len[256];
L
Linus Torvalds 已提交
68 69


70
static void usage(void)
L
Linus Torvalds 已提交
71
{
72
	fprintf(stderr, "Usage: kallsyms [--all-symbols] "
73
			"[--base-relative] < in.map > out.S\n");
L
Linus Torvalds 已提交
74 75 76
	exit(1);
}

77 78 79 80 81
static char *sym_name(const struct sym_entry *s)
{
	return (char *)s->sym + 1;
}

82 83
static int check_symbol_range(const char *sym, unsigned long long addr,
			      struct addr_range *ranges, int entries)
84 85
{
	size_t i;
86
	struct addr_range *ar;
87

88 89
	for (i = 0; i < entries; ++i) {
		ar = &ranges[i];
90

91 92
		if (strcmp(sym, ar->start_sym) == 0) {
			ar->start = addr;
93
			return 0;
94 95
		} else if (strcmp(sym, ar->end_sym) == 0) {
			ar->end = addr;
96 97 98 99 100 101 102
			return 0;
		}
	}

	return 1;
}

103
static int read_symbol(FILE *in, struct sym_entry *s)
L
Linus Torvalds 已提交
104
{
105
	char sym[500], stype;
L
Linus Torvalds 已提交
106 107
	int rc;

108
	rc = fscanf(in, "%llx %c %499s\n", &s->addr, &stype, sym);
L
Linus Torvalds 已提交
109
	if (rc != 3) {
110
		if (rc != EOF && fgets(sym, 500, in) == NULL)
111
			fprintf(stderr, "Read error or end of file.\n");
L
Linus Torvalds 已提交
112 113
		return -1;
	}
114 115
	if (strlen(sym) >= KSYM_NAME_LEN) {
		fprintf(stderr, "Symbol %s too long for kallsyms (%zu >= %d).\n"
116
				"Please increase KSYM_NAME_LEN both in kernel and kallsyms.c\n",
117
			sym, strlen(sym), KSYM_NAME_LEN);
118 119
		return -1;
	}
L
Linus Torvalds 已提交
120 121

	/* Ignore most absolute/undefined (?) symbols. */
122 123
	if (strcmp(sym, "_text") == 0)
		_text = s->addr;
124 125
	else if (check_symbol_range(sym, s->addr, text_ranges,
				    ARRAY_SIZE(text_ranges)) == 0)
126
		/* nothing to do */;
127
	else if (toupper(stype) == 'A')
L
Linus Torvalds 已提交
128 129
	{
		/* Keep these useful absolute symbols */
130 131 132 133
		if (strcmp(sym, "__kernel_syscall_via_break") &&
		    strcmp(sym, "__kernel_syscall_via_epc") &&
		    strcmp(sym, "__kernel_sigtramp") &&
		    strcmp(sym, "__gp"))
L
Linus Torvalds 已提交
134 135 136
			return -1;

	}
137
	else if (toupper(stype) == 'U')
L
Linus Torvalds 已提交
138
		return -1;
139 140 141 142 143
	/*
	 * Ignore generated symbols such as:
	 *  - mapping symbols in ARM ELF files ($a, $t, and $d)
	 *  - MIPS ELF local symbols ($L123 instead of .L123)
	 */
144
	else if (sym[0] == '$')
145
		return -1;
146
	/* exclude debugging symbols */
147
	else if (stype == 'N' || stype == 'n')
148
		return -1;
149 150 151
	/* exclude s390 kasan local symbols */
	else if (!strncmp(sym, ".LASANPC", 8))
		return -1;
L
Linus Torvalds 已提交
152 153 154

	/* include the type field in the symbol name, so that it gets
	 * compressed together */
155
	s->len = strlen(sym) + 1;
156
	s->sym = malloc(s->len + 1);
157 158 159 160 161
	if (!s->sym) {
		fprintf(stderr, "kallsyms failure: "
			"unable to allocate required amount of memory\n");
		exit(EXIT_FAILURE);
	}
162
	strcpy(sym_name(s), sym);
163
	s->sym[0] = stype;
L
Linus Torvalds 已提交
164

165 166
	s->percpu_absolute = 0;

167 168 169
	/* Record if we've found __per_cpu_start/end. */
	check_symbol_range(sym, s->addr, &percpu_range, 1);

L
Linus Torvalds 已提交
170 171 172
	return 0;
}

173 174
static int symbol_in_range(struct sym_entry *s, struct addr_range *ranges,
			   int entries)
175 176
{
	size_t i;
177
	struct addr_range *ar;
178

179 180
	for (i = 0; i < entries; ++i) {
		ar = &ranges[i];
181

182
		if (s->addr >= ar->start && s->addr <= ar->end)
183
			return 1;
184 185
	}

186
	return 0;
187 188
}

189
static int symbol_valid(struct sym_entry *s)
L
Linus Torvalds 已提交
190 191
{
	/* Symbols which vary between passes.  Passes 1 and 2 must have
192 193 194
	 * identical symbol lists.  The kallsyms_* symbols below are only added
	 * after pass 1, they would be included in pass 2 when --all-symbols is
	 * specified so exclude them to get a stable symbol list.
L
Linus Torvalds 已提交
195 196
	 */
	static char *special_symbols[] = {
197
		"kallsyms_addresses",
198 199
		"kallsyms_offsets",
		"kallsyms_relative_base",
200 201 202 203 204 205
		"kallsyms_num_syms",
		"kallsyms_names",
		"kallsyms_markers",
		"kallsyms_token_table",
		"kallsyms_token_index",

L
Linus Torvalds 已提交
206 207 208 209
	/* Exclude linker generated symbols which vary between passes */
		"_SDA_BASE_",		/* ppc */
		"_SDA2_BASE_",		/* ppc */
		NULL };
210

211 212
	static char *special_prefixes[] = {
		"__crc_",		/* modversions */
213
		"__efistub_",		/* arm64 EFI stub namespace */
214 215
		NULL };

216 217
	static char *special_suffixes[] = {
		"_veneer",		/* arm */
218 219
		"_from_arm",		/* arm */
		"_from_thumb",		/* arm */
220 221
		NULL };

L
Linus Torvalds 已提交
222
	int i;
223
	const char *name = sym_name(s);
224

L
Linus Torvalds 已提交
225 226 227
	/* if --all-symbols is not specified, then symbols outside the text
	 * and inittext sections are discarded */
	if (!all_symbols) {
228 229
		if (symbol_in_range(s, text_ranges,
				    ARRAY_SIZE(text_ranges)) == 0)
L
Linus Torvalds 已提交
230 231
			return 0;
		/* Corner case.  Discard any symbols with the same value as
232 233 234 235
		 * _etext _einittext; they can move between pass 1 and 2 when
		 * the kallsyms data are added.  If these symbols move then
		 * they may get dropped in pass 2, which breaks the kallsyms
		 * rules.
L
Linus Torvalds 已提交
236
		 */
237
		if ((s->addr == text_range_text->end &&
238
		     strcmp(name, text_range_text->end_sym)) ||
239
		    (s->addr == text_range_inittext->end &&
240
		     strcmp(name, text_range_inittext->end_sym)))
L
Linus Torvalds 已提交
241 242 243 244 245
			return 0;
	}

	/* Exclude symbols which vary between passes. */
	for (i = 0; special_symbols[i]; i++)
246
		if (strcmp(name, special_symbols[i]) == 0)
L
Linus Torvalds 已提交
247 248
			return 0;

249 250 251
	for (i = 0; special_prefixes[i]; i++) {
		int l = strlen(special_prefixes[i]);

252
		if (strncmp(name, special_prefixes[i], l) == 0)
253 254 255
			return 0;
	}

256
	for (i = 0; special_suffixes[i]; i++) {
257
		int l = strlen(name) - strlen(special_suffixes[i]);
258

259
		if (l >= 0 && strcmp(name + l, special_suffixes[i]) == 0)
260 261 262
			return 0;
	}

L
Linus Torvalds 已提交
263 264 265
	return 1;
}

266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289
/* remove all the invalid symbols from the table */
static void shrink_table(void)
{
	unsigned int i, pos;

	pos = 0;
	for (i = 0; i < table_cnt; i++) {
		if (symbol_valid(&table[i])) {
			if (pos != i)
				table[pos] = table[i];
			pos++;
		} else {
			free(table[i].sym);
		}
	}
	table_cnt = pos;

	/* When valid symbol is not registered, exit to error */
	if (!table_cnt) {
		fprintf(stderr, "No valid symbol.\n");
		exit(1);
	}
}

290
static void read_map(FILE *in)
L
Linus Torvalds 已提交
291 292
{
	while (!feof(in)) {
293 294 295
		if (table_cnt >= table_size) {
			table_size += 10000;
			table = realloc(table, sizeof(*table) * table_size);
L
Linus Torvalds 已提交
296 297 298 299 300
			if (!table) {
				fprintf(stderr, "out of memory\n");
				exit (1);
			}
		}
301 302
		if (read_symbol(in, &table[table_cnt]) == 0) {
			table[table_cnt].start_pos = table_cnt;
303
			table_cnt++;
304
		}
L
Linus Torvalds 已提交
305 306 307 308 309
	}
}

static void output_label(char *label)
{
310
	printf(".globl %s\n", label);
L
Linus Torvalds 已提交
311
	printf("\tALGN\n");
312
	printf("%s:\n", label);
L
Linus Torvalds 已提交
313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341
}

/* uncompress a compressed symbol. When this function is called, the best table
 * might still be compressed itself, so the function needs to be recursive */
static int expand_symbol(unsigned char *data, int len, char *result)
{
	int c, rlen, total=0;

	while (len) {
		c = *data;
		/* if the table holds a single char that is the same as the one
		 * we are looking for, then end the search */
		if (best_table[c][0]==c && best_table_len[c]==1) {
			*result++ = c;
			total++;
		} else {
			/* if not, recurse and expand */
			rlen = expand_symbol(best_table[c], best_table_len[c], result);
			total += rlen;
			result += rlen;
		}
		data++;
		len--;
	}
	*result=0;

	return total;
}

342 343
static int symbol_absolute(struct sym_entry *s)
{
344
	return s->percpu_absolute;
345 346
}

347
static void write_src(void)
L
Linus Torvalds 已提交
348
{
349
	unsigned int i, k, off;
L
Linus Torvalds 已提交
350 351
	unsigned int best_idx[256];
	unsigned int *markers;
352
	char buf[KSYM_NAME_LEN];
L
Linus Torvalds 已提交
353

354
	printf("#include <asm/bitsperlong.h>\n");
L
Linus Torvalds 已提交
355 356
	printf("#if BITS_PER_LONG == 64\n");
	printf("#define PTR .quad\n");
357
	printf("#define ALGN .balign 8\n");
L
Linus Torvalds 已提交
358 359
	printf("#else\n");
	printf("#define PTR .long\n");
360
	printf("#define ALGN .balign 4\n");
L
Linus Torvalds 已提交
361 362
	printf("#endif\n");

363
	printf("\t.section .rodata, \"a\"\n");
L
Linus Torvalds 已提交
364

365 366 367 368 369 370 371 372 373 374 375 376
	/* Provide proper symbols relocatability by their relativeness
	 * to a fixed anchor point in the runtime image, either '_text'
	 * for absolute address tables, in which case the linker will
	 * emit the final addresses at build time. Otherwise, use the
	 * offset relative to the lowest value encountered of all relative
	 * symbols, and emit non-relocatable fixed offsets that will be fixed
	 * up at runtime.
	 *
	 * The symbol names cannot be used to construct normal symbol
	 * references as the list of symbols contains symbols that are
	 * declared static and are private to their .o files.  This prevents
	 * .tmp_kallsyms.o or any other object from referencing them.
377
	 */
378 379 380 381 382
	if (!base_relative)
		output_label("kallsyms_addresses");
	else
		output_label("kallsyms_offsets");

383
	for (i = 0; i < table_cnt; i++) {
384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406
		if (base_relative) {
			long long offset;
			int overflow;

			if (!absolute_percpu) {
				offset = table[i].addr - relative_base;
				overflow = (offset < 0 || offset > UINT_MAX);
			} else if (symbol_absolute(&table[i])) {
				offset = table[i].addr;
				overflow = (offset < 0 || offset > INT_MAX);
			} else {
				offset = relative_base - table[i].addr - 1;
				overflow = (offset < INT_MIN || offset >= 0);
			}
			if (overflow) {
				fprintf(stderr, "kallsyms failure: "
					"%s symbol value %#llx out of range in relative mode\n",
					symbol_absolute(&table[i]) ? "absolute" : "relative",
					table[i].addr);
				exit(EXIT_FAILURE);
			}
			printf("\t.long\t%#x\n", (int)offset);
		} else if (!symbol_absolute(&table[i])) {
407 408 409 410
			if (_text <= table[i].addr)
				printf("\tPTR\t_text + %#llx\n",
					table[i].addr - _text);
			else
411 412
				printf("\tPTR\t_text - %#llx\n",
					_text - table[i].addr);
413 414 415
		} else {
			printf("\tPTR\t%#llx\n", table[i].addr);
		}
L
Linus Torvalds 已提交
416 417 418
	}
	printf("\n");

419 420 421 422 423 424
	if (base_relative) {
		output_label("kallsyms_relative_base");
		printf("\tPTR\t_text - %#llx\n", _text - relative_base);
		printf("\n");
	}

L
Linus Torvalds 已提交
425
	output_label("kallsyms_num_syms");
426
	printf("\t.long\t%u\n", table_cnt);
L
Linus Torvalds 已提交
427 428 429 430
	printf("\n");

	/* table of offset markers, that give the offset in the compressed stream
	 * every 256 symbols */
431 432 433 434 435 436
	markers = malloc(sizeof(unsigned int) * ((table_cnt + 255) / 256));
	if (!markers) {
		fprintf(stderr, "kallsyms failure: "
			"unable to allocate required memory\n");
		exit(EXIT_FAILURE);
	}
L
Linus Torvalds 已提交
437 438 439

	output_label("kallsyms_names");
	off = 0;
440 441 442
	for (i = 0; i < table_cnt; i++) {
		if ((i & 0xFF) == 0)
			markers[i >> 8] = off;
L
Linus Torvalds 已提交
443 444 445 446 447 448 449 450 451 452 453

		printf("\t.byte 0x%02x", table[i].len);
		for (k = 0; k < table[i].len; k++)
			printf(", 0x%02x", table[i].sym[k]);
		printf("\n");

		off += table[i].len + 1;
	}
	printf("\n");

	output_label("kallsyms_markers");
454
	for (i = 0; i < ((table_cnt + 255) >> 8); i++)
455
		printf("\t.long\t%u\n", markers[i]);
L
Linus Torvalds 已提交
456 457 458 459 460 461 462 463
	printf("\n");

	free(markers);

	output_label("kallsyms_token_table");
	off = 0;
	for (i = 0; i < 256; i++) {
		best_idx[i] = off;
464
		expand_symbol(best_table[i], best_table_len[i], buf);
L
Linus Torvalds 已提交
465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484
		printf("\t.asciz\t\"%s\"\n", buf);
		off += strlen(buf) + 1;
	}
	printf("\n");

	output_label("kallsyms_token_index");
	for (i = 0; i < 256; i++)
		printf("\t.short\t%d\n", best_idx[i]);
	printf("\n");
}


/* table lookup compression functions */

/* count all the possible tokens in a symbol */
static void learn_symbol(unsigned char *symbol, int len)
{
	int i;

	for (i = 0; i < len - 1; i++)
485
		token_profit[ symbol[i] + (symbol[i + 1] << 8) ]++;
L
Linus Torvalds 已提交
486 487 488 489 490 491 492 493
}

/* decrease the count for all the possible tokens in a symbol */
static void forget_symbol(unsigned char *symbol, int len)
{
	int i;

	for (i = 0; i < len - 1; i++)
494
		token_profit[ symbol[i] + (symbol[i + 1] << 8) ]--;
L
Linus Torvalds 已提交
495 496
}

497
/* do the initial token count */
L
Linus Torvalds 已提交
498 499
static void build_initial_tok_table(void)
{
500
	unsigned int i;
L
Linus Torvalds 已提交
501

502 503
	for (i = 0; i < table_cnt; i++)
		learn_symbol(table[i].sym, table[i].len);
L
Linus Torvalds 已提交
504 505
}

506 507 508 509 510 511 512 513 514 515 516
static void *find_token(unsigned char *str, int len, unsigned char *token)
{
	int i;

	for (i = 0; i < len - 1; i++) {
		if (str[i] == token[0] && str[i+1] == token[1])
			return &str[i];
	}
	return NULL;
}

L
Linus Torvalds 已提交
517 518
/* replace a given token in all the valid symbols. Use the sampled symbols
 * to update the counts */
519
static void compress_symbols(unsigned char *str, int idx)
L
Linus Torvalds 已提交
520
{
521 522
	unsigned int i, len, size;
	unsigned char *p1, *p2;
L
Linus Torvalds 已提交
523

524
	for (i = 0; i < table_cnt; i++) {
L
Linus Torvalds 已提交
525 526

		len = table[i].len;
527 528 529
		p1 = table[i].sym;

		/* find the token on the symbol */
530
		p2 = find_token(p1, len, str);
531 532 533 534 535 536
		if (!p2) continue;

		/* decrease the counts for this symbol's tokens */
		forget_symbol(table[i].sym, len);

		size = len;
L
Linus Torvalds 已提交
537 538

		do {
539 540 541 542 543 544 545 546 547
			*p2 = idx;
			p2++;
			size -= (p2 - p1);
			memmove(p2, p2 + 1, size);
			p1 = p2;
			len--;

			if (size < 2) break;

L
Linus Torvalds 已提交
548
			/* find the token on the symbol */
549
			p2 = find_token(p1, size, str);
L
Linus Torvalds 已提交
550

551
		} while (p2);
L
Linus Torvalds 已提交
552

553
		table[i].len = len;
L
Linus Torvalds 已提交
554

555 556
		/* increase the counts for this symbol's new tokens */
		learn_symbol(table[i].sym, len);
L
Linus Torvalds 已提交
557 558 559 560
	}
}

/* search the token with the maximum profit */
561
static int find_best_token(void)
L
Linus Torvalds 已提交
562
{
563
	int i, best, bestprofit;
L
Linus Torvalds 已提交
564 565

	bestprofit=-10000;
566
	best = 0;
L
Linus Torvalds 已提交
567

568 569 570 571
	for (i = 0; i < 0x10000; i++) {
		if (token_profit[i] > bestprofit) {
			best = i;
			bestprofit = token_profit[i];
L
Linus Torvalds 已提交
572 573 574 575 576 577 578 579
		}
	}
	return best;
}

/* this is the core of the algorithm: calculate the "best" table */
static void optimize_result(void)
{
580
	int i, best;
L
Linus Torvalds 已提交
581 582 583 584 585 586 587 588 589

	/* using the '\0' symbol last allows compress_symbols to use standard
	 * fast string functions */
	for (i = 255; i >= 0; i--) {

		/* if this table slot is empty (it is not used by an actual
		 * original char code */
		if (!best_table_len[i]) {

C
Cao jin 已提交
590
			/* find the token with the best profit value */
L
Linus Torvalds 已提交
591
			best = find_best_token();
592 593
			if (token_profit[best] == 0)
				break;
L
Linus Torvalds 已提交
594 595

			/* place it in the "best" table */
596 597 598
			best_table_len[i] = 2;
			best_table[i][0] = best & 0xFF;
			best_table[i][1] = (best >> 8) & 0xFF;
L
Linus Torvalds 已提交
599 600

			/* replace this token in all the valid symbols */
601
			compress_symbols(best_table[i], i);
L
Linus Torvalds 已提交
602 603 604 605 606 607 608
		}
	}
}

/* start by placing the symbols that are actually used on the table */
static void insert_real_symbols_in_table(void)
{
609
	unsigned int i, j, c;
L
Linus Torvalds 已提交
610

611 612 613 614 615
	for (i = 0; i < table_cnt; i++) {
		for (j = 0; j < table[i].len; j++) {
			c = table[i].sym[j];
			best_table[c][0]=c;
			best_table_len[c]=1;
L
Linus Torvalds 已提交
616 617 618 619 620 621 622 623 624 625 626 627 628
		}
	}
}

static void optimize_token_table(void)
{
	build_initial_tok_table();

	insert_real_symbols_in_table();

	optimize_result();
}

629 630 631
/* guess for "linker script provide" symbol */
static int may_be_linker_script_provide_symbol(const struct sym_entry *se)
{
632
	const char *symbol = sym_name(se);
633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667
	int len = se->len - 1;

	if (len < 8)
		return 0;

	if (symbol[0] != '_' || symbol[1] != '_')
		return 0;

	/* __start_XXXXX */
	if (!memcmp(symbol + 2, "start_", 6))
		return 1;

	/* __stop_XXXXX */
	if (!memcmp(symbol + 2, "stop_", 5))
		return 1;

	/* __end_XXXXX */
	if (!memcmp(symbol + 2, "end_", 4))
		return 1;

	/* __XXXXX_start */
	if (!memcmp(symbol + len - 6, "_start", 6))
		return 1;

	/* __XXXXX_end */
	if (!memcmp(symbol + len - 4, "_end", 4))
		return 1;

	return 0;
}

static int prefix_underscores_count(const char *str)
{
	const char *tail = str;

668
	while (*tail == '_')
669 670 671 672 673
		tail++;

	return tail - str;
}

674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694
static int compare_symbols(const void *a, const void *b)
{
	const struct sym_entry *sa;
	const struct sym_entry *sb;
	int wa, wb;

	sa = a;
	sb = b;

	/* sort by address first */
	if (sa->addr > sb->addr)
		return 1;
	if (sa->addr < sb->addr)
		return -1;

	/* sort by "weakness" type */
	wa = (sa->sym[0] == 'w') || (sa->sym[0] == 'W');
	wb = (sb->sym[0] == 'w') || (sb->sym[0] == 'W');
	if (wa != wb)
		return wa - wb;

695 696 697 698 699 700 701
	/* sort by "linker script provide" type */
	wa = may_be_linker_script_provide_symbol(sa);
	wb = may_be_linker_script_provide_symbol(sb);
	if (wa != wb)
		return wa - wb;

	/* sort by the number of prefix underscores */
702 703
	wa = prefix_underscores_count(sym_name(sa));
	wb = prefix_underscores_count(sym_name(sb));
704 705 706
	if (wa != wb)
		return wa - wb;

707 708 709 710 711 712 713 714
	/* sort by initial order, so that other symbols are left undisturbed */
	return sa->start_pos - sb->start_pos;
}

static void sort_symbols(void)
{
	qsort(table, table_cnt, sizeof(struct sym_entry), compare_symbols);
}
L
Linus Torvalds 已提交
715

716 717 718 719 720
static void make_percpus_absolute(void)
{
	unsigned int i;

	for (i = 0; i < table_cnt; i++)
721 722 723 724 725 726
		if (symbol_in_range(&table[i], &percpu_range, 1)) {
			/*
			 * Keep the 'A' override for percpu symbols to
			 * ensure consistent behavior compared to older
			 * versions of this tool.
			 */
727
			table[i].sym[0] = 'A';
728 729
			table[i].percpu_absolute = 1;
		}
730 731
}

732 733 734 735 736 737
/* find the minimum non-absolute symbol address */
static void record_relative_base(void)
{
	unsigned int i;

	for (i = 0; i < table_cnt; i++)
738 739 740 741 742
		if (!symbol_absolute(&table[i])) {
			/*
			 * The table is sorted by address.
			 * Take the first non-absolute symbol value.
			 */
743
			relative_base = table[i].addr;
744 745
			return;
		}
746 747
}

748
int main(int argc, char **argv)
L
Linus Torvalds 已提交
749
{
750 751 752 753 754
	if (argc >= 2) {
		int i;
		for (i = 1; i < argc; i++) {
			if(strcmp(argv[i], "--all-symbols") == 0)
				all_symbols = 1;
755 756
			else if (strcmp(argv[i], "--absolute-percpu") == 0)
				absolute_percpu = 1;
757
			else if (strcmp(argv[i], "--base-relative") == 0)
758 759
				base_relative = 1;
			else
760 761 762
				usage();
		}
	} else if (argc != 1)
L
Linus Torvalds 已提交
763 764 765
		usage();

	read_map(stdin);
766
	shrink_table();
767 768
	if (absolute_percpu)
		make_percpus_absolute();
769
	sort_symbols();
770 771
	if (base_relative)
		record_relative_base();
772
	optimize_token_table();
L
Linus Torvalds 已提交
773 774 775 776
	write_src();

	return 0;
}