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(const struct sym_entry *s,
			   const struct addr_range *ranges, int entries)
175 176
{
	size_t i;
177
	const 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(const 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 const char * const 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
	static const char * const special_prefixes[] = {
212
		"__crc_",		/* modversions */
213
		"__efistub_",		/* arm64 EFI stub namespace */
214 215
		NULL };

216
	static const char * const special_suffixes[] = {
217
		"_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
static void output_label(const char *label)
L
Linus Torvalds 已提交
309
{
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
}

/* uncompress a compressed symbol. When this function is called, the best table
 * might still be compressed itself, so the function needs to be recursive */
317
static int expand_symbol(const unsigned char *data, int len, char *result)
L
Linus Torvalds 已提交
318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341
{
	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
static int symbol_absolute(const struct sym_entry *s)
343
{
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
		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 */
480
static void learn_symbol(const unsigned char *symbol, int len)
L
Linus Torvalds 已提交
481 482 483 484
{
	int i;

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

/* decrease the count for all the possible tokens in a symbol */
489
static void forget_symbol(const unsigned char *symbol, int len)
L
Linus Torvalds 已提交
490 491 492 493
{
	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
static unsigned char *find_token(unsigned char *str, int len,
507
				 const unsigned char *token)
508 509 510 511 512 513 514 515 516 517
{
	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 已提交
518 519
/* replace a given token in all the valid symbols. Use the sampled symbols
 * to update the counts */
520
static void compress_symbols(const unsigned char *str, int idx)
L
Linus Torvalds 已提交
521
{
522 523
	unsigned int i, len, size;
	unsigned char *p1, *p2;
L
Linus Torvalds 已提交
524

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

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

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

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

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

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

			if (size < 2) break;

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

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

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

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

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

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

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

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

	/* 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 已提交
591
			/* find the token with the best profit value */
L
Linus Torvalds 已提交
592
			best = find_best_token();
593 594
			if (token_profit[best] == 0)
				break;
L
Linus Torvalds 已提交
595 596

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

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

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

612 613 614 615 616
	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 已提交
617 618 619 620 621 622 623 624 625 626 627 628 629
		}
	}
}

static void optimize_token_table(void)
{
	build_initial_tok_table();

	insert_real_symbols_in_table();

	optimize_result();
}

630 631 632
/* guess for "linker script provide" symbol */
static int may_be_linker_script_provide_symbol(const struct sym_entry *se)
{
633
	const char *symbol = sym_name(se);
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
	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;
}

665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685
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;

686 687 688 689 690 691 692
	/* 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 */
693 694
	wa = strspn(sym_name(sa), "_");
	wb = strspn(sym_name(sb), "_");
695 696 697
	if (wa != wb)
		return wa - wb;

698 699 700 701 702 703 704 705
	/* 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 已提交
706

707 708 709 710 711
static void make_percpus_absolute(void)
{
	unsigned int i;

	for (i = 0; i < table_cnt; i++)
712 713 714 715 716 717
		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.
			 */
718
			table[i].sym[0] = 'A';
719 720
			table[i].percpu_absolute = 1;
		}
721 722
}

723 724 725 726 727 728
/* find the minimum non-absolute symbol address */
static void record_relative_base(void)
{
	unsigned int i;

	for (i = 0; i < table_cnt; i++)
729 730 731 732 733
		if (!symbol_absolute(&table[i])) {
			/*
			 * The table is sorted by address.
			 * Take the first non-absolute symbol value.
			 */
734
			relative_base = table[i].addr;
735 736
			return;
		}
737 738
}

739
int main(int argc, char **argv)
L
Linus Torvalds 已提交
740
{
741 742 743 744 745
	if (argc >= 2) {
		int i;
		for (i = 1; i < argc; i++) {
			if(strcmp(argv[i], "--all-symbols") == 0)
				all_symbols = 1;
746 747
			else if (strcmp(argv[i], "--absolute-percpu") == 0)
				absolute_percpu = 1;
748
			else if (strcmp(argv[i], "--base-relative") == 0)
749 750
				base_relative = 1;
			else
751 752 753
				usage();
		}
	} else if (argc != 1)
L
Linus Torvalds 已提交
754 755 756
		usage();

	read_map(stdin);
757
	shrink_table();
758 759
	if (absolute_percpu)
		make_percpus_absolute();
760
	sort_symbols();
761 762
	if (base_relative)
		record_relative_base();
763
	optimize_token_table();
L
Linus Torvalds 已提交
764 765 766 767
	write_src();

	return 0;
}