kallsyms.c 15.0 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
/*
 * kallsyms.c: in-kernel printing of symbolic oopses and stack traces.
 *
 * Rewritten and vastly simplified by Rusty Russell for in-kernel
 * module loader:
 *   Copyright 2002 Rusty Russell <rusty@rustcorp.com.au> IBM Corporation
 *
 * ChangeLog:
 *
 * (25/Aug/2004) Paulo Marques <pmarques@grupopie.com>
 *      Changed the compression method from stem compression to "table lookup"
 *      compression (see scripts/kallsyms.c for a more complete description)
 */
#include <linux/kallsyms.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/seq_file.h>
#include <linux/fs.h>
19
#include <linux/kdb.h>
L
Linus Torvalds 已提交
20 21
#include <linux/err.h>
#include <linux/proc_fs.h>
T
Tim Schmielau 已提交
22
#include <linux/sched.h>	/* for cond_resched */
L
Linus Torvalds 已提交
23
#include <linux/mm.h>
24
#include <linux/ctype.h>
25
#include <linux/slab.h>
26
#include <linux/compiler.h>
L
Linus Torvalds 已提交
27 28 29 30 31 32 33 34 35

#include <asm/sections.h>

#ifdef CONFIG_KALLSYMS_ALL
#define all_var 1
#else
#define all_var 0
#endif

36 37 38 39
/*
 * These will be re-linked against their real values
 * during the second link stage.
 */
40 41
extern const unsigned long kallsyms_addresses[] __weak;
extern const u8 kallsyms_names[] __weak;
L
Linus Torvalds 已提交
42

43 44 45
/*
 * Tell the compiler that the count isn't in the small data section if the arch
 * has one (eg: FRV).
46 47
 */
extern const unsigned long kallsyms_num_syms
48
__attribute__((weak, section(".rodata")));
49

50 51
extern const u8 kallsyms_token_table[] __weak;
extern const u16 kallsyms_token_index[] __weak;
L
Linus Torvalds 已提交
52

53
extern const unsigned long kallsyms_markers[] __weak;
L
Linus Torvalds 已提交
54 55 56 57 58 59 60 61 62 63 64

static inline int is_kernel_inittext(unsigned long addr)
{
	if (addr >= (unsigned long)_sinittext
	    && addr <= (unsigned long)_einittext)
		return 1;
	return 0;
}

static inline int is_kernel_text(unsigned long addr)
{
65 66
	if ((addr >= (unsigned long)_stext && addr <= (unsigned long)_etext) ||
	    arch_is_kernel_text(addr))
L
Linus Torvalds 已提交
67
		return 1;
68
	return in_gate_area_no_mm(addr);
L
Linus Torvalds 已提交
69 70 71 72 73 74
}

static inline int is_kernel(unsigned long addr)
{
	if (addr >= (unsigned long)_stext && addr <= (unsigned long)_end)
		return 1;
75
	return in_gate_area_no_mm(addr);
L
Linus Torvalds 已提交
76 77
}

78 79 80 81 82
static int is_ksym_addr(unsigned long addr)
{
	if (all_var)
		return is_kernel(addr);

83
	return is_kernel_text(addr) || is_kernel_inittext(addr);
84 85
}

86 87
/*
 * Expand a compressed symbol data into the resulting uncompressed string,
88
 * if uncompressed string is too long (>= maxlen), it will be truncated,
89 90
 * given the offset to where the symbol is in the compressed stream.
 */
91 92
static unsigned int kallsyms_expand_symbol(unsigned int off,
					   char *result, size_t maxlen)
L
Linus Torvalds 已提交
93 94
{
	int len, skipped_first = 0;
95
	const u8 *tptr, *data;
L
Linus Torvalds 已提交
96

97
	/* Get the compressed symbol length from the first symbol byte. */
L
Linus Torvalds 已提交
98 99 100 101
	data = &kallsyms_names[off];
	len = *data;
	data++;

102 103 104 105
	/*
	 * Update the offset to return the offset for the next symbol on
	 * the compressed stream.
	 */
L
Linus Torvalds 已提交
106 107
	off += len + 1;

108 109 110 111 112 113
	/*
	 * For every byte on the compressed symbol data, copy the table
	 * entry for that byte.
	 */
	while (len) {
		tptr = &kallsyms_token_table[kallsyms_token_index[*data]];
L
Linus Torvalds 已提交
114 115 116 117
		data++;
		len--;

		while (*tptr) {
118
			if (skipped_first) {
119 120
				if (maxlen <= 1)
					goto tail;
L
Linus Torvalds 已提交
121 122
				*result = *tptr;
				result++;
123
				maxlen--;
L
Linus Torvalds 已提交
124 125 126 127 128 129
			} else
				skipped_first = 1;
			tptr++;
		}
	}

130 131 132
tail:
	if (maxlen)
		*result = '\0';
L
Linus Torvalds 已提交
133

134
	/* Return to offset to the next symbol. */
L
Linus Torvalds 已提交
135 136 137
	return off;
}

138 139 140 141
/*
 * Get symbol type information. This is encoded as a single char at the
 * beginning of the symbol name.
 */
L
Linus Torvalds 已提交
142 143
static char kallsyms_get_symbol_type(unsigned int off)
{
144 145 146 147 148
	/*
	 * Get just the first code, look it up in the token table,
	 * and return the first char from this token.
	 */
	return kallsyms_token_table[kallsyms_token_index[kallsyms_names[off + 1]]];
L
Linus Torvalds 已提交
149 150 151
}


152 153 154 155
/*
 * Find the offset on the compressed stream given and index in the
 * kallsyms array.
 */
L
Linus Torvalds 已提交
156 157
static unsigned int get_symbol_offset(unsigned long pos)
{
158
	const u8 *name;
L
Linus Torvalds 已提交
159 160
	int i;

161 162 163 164 165
	/*
	 * Use the closest marker we have. We have markers every 256 positions,
	 * so that should be close enough.
	 */
	name = &kallsyms_names[kallsyms_markers[pos >> 8]];
L
Linus Torvalds 已提交
166

167 168 169 170 171 172 173
	/*
	 * Sequentially scan all the symbols up to the point we're searching
	 * for. Every symbol is stored in a [<len>][<len> bytes of data] format,
	 * so we just need to add the len to the current pointer for every
	 * symbol we wish to skip.
	 */
	for (i = 0; i < (pos & 0xFF); i++)
L
Linus Torvalds 已提交
174 175 176 177 178 179 180 181
		name = name + (*name) + 1;

	return name - kallsyms_names;
}

/* Lookup the address for this symbol. Returns 0 if not found. */
unsigned long kallsyms_lookup_name(const char *name)
{
182
	char namebuf[KSYM_NAME_LEN];
L
Linus Torvalds 已提交
183 184 185 186
	unsigned long i;
	unsigned int off;

	for (i = 0, off = 0; i < kallsyms_num_syms; i++) {
187
		off = kallsyms_expand_symbol(off, namebuf, ARRAY_SIZE(namebuf));
L
Linus Torvalds 已提交
188 189 190 191 192 193

		if (strcmp(namebuf, name) == 0)
			return kallsyms_addresses[i];
	}
	return module_kallsyms_lookup_name(name);
}
194
EXPORT_SYMBOL_GPL(kallsyms_lookup_name);
L
Linus Torvalds 已提交
195

196 197 198 199 200 201 202 203 204 205
int kallsyms_on_each_symbol(int (*fn)(void *, const char *, struct module *,
				      unsigned long),
			    void *data)
{
	char namebuf[KSYM_NAME_LEN];
	unsigned long i;
	unsigned int off;
	int ret;

	for (i = 0, off = 0; i < kallsyms_num_syms; i++) {
206
		off = kallsyms_expand_symbol(off, namebuf, ARRAY_SIZE(namebuf));
207 208 209 210 211 212 213 214
		ret = fn(data, namebuf, NULL, kallsyms_addresses[i]);
		if (ret != 0)
			return ret;
	}
	return module_kallsyms_on_each_symbol(fn, data);
}
EXPORT_SYMBOL_GPL(kallsyms_on_each_symbol);

215 216 217 218 219 220 221
static unsigned long get_symbol_pos(unsigned long addr,
				    unsigned long *symbolsize,
				    unsigned long *offset)
{
	unsigned long symbol_start = 0, symbol_end = 0;
	unsigned long i, low, high, mid;

222 223 224
	/* This kernel should never had been booted. */
	BUG_ON(!kallsyms_addresses);

225
	/* Do a binary search on the sorted kallsyms_addresses array. */
226 227 228 229
	low = 0;
	high = kallsyms_num_syms;

	while (high - low > 1) {
230
		mid = low + (high - low) / 2;
231 232 233 234 235 236 237
		if (kallsyms_addresses[mid] <= addr)
			low = mid;
		else
			high = mid;
	}

	/*
238 239
	 * Search for the first aliased symbol. Aliased
	 * symbols are symbols with the same address.
240 241 242 243 244 245
	 */
	while (low && kallsyms_addresses[low-1] == kallsyms_addresses[low])
		--low;

	symbol_start = kallsyms_addresses[low];

246
	/* Search for next non-aliased symbol. */
247 248 249 250 251 252 253
	for (i = low + 1; i < kallsyms_num_syms; i++) {
		if (kallsyms_addresses[i] > symbol_start) {
			symbol_end = kallsyms_addresses[i];
			break;
		}
	}

254
	/* If we found no next symbol, we use the end of the section. */
255 256 257 258 259 260 261 262 263
	if (!symbol_end) {
		if (is_kernel_inittext(addr))
			symbol_end = (unsigned long)_einittext;
		else if (all_var)
			symbol_end = (unsigned long)_end;
		else
			symbol_end = (unsigned long)_etext;
	}

A
Alexey Dobriyan 已提交
264 265 266 267
	if (symbolsize)
		*symbolsize = symbol_end - symbol_start;
	if (offset)
		*offset = addr - symbol_start;
268 269 270 271 272 273 274 275 276 277

	return low;
}

/*
 * Lookup an address but don't bother to find any names.
 */
int kallsyms_lookup_size_offset(unsigned long addr, unsigned long *symbolsize,
				unsigned long *offset)
{
278
	char namebuf[KSYM_NAME_LEN];
279 280 281
	if (is_ksym_addr(addr))
		return !!get_symbol_pos(addr, symbolsize, offset);

282
	return !!module_address_lookup(addr, symbolsize, offset, NULL, namebuf);
283 284
}

L
Linus Torvalds 已提交
285 286
/*
 * Lookup an address
287 288 289 290
 * - modname is set to NULL if it's in the kernel.
 * - We guarantee that the returned name is valid until we reschedule even if.
 *   It resides in a module.
 * - We also guarantee that modname will be valid until rescheduled.
L
Linus Torvalds 已提交
291 292 293 294 295 296
 */
const char *kallsyms_lookup(unsigned long addr,
			    unsigned long *symbolsize,
			    unsigned long *offset,
			    char **modname, char *namebuf)
{
297
	namebuf[KSYM_NAME_LEN - 1] = 0;
L
Linus Torvalds 已提交
298 299
	namebuf[0] = 0;

300 301
	if (is_ksym_addr(addr)) {
		unsigned long pos;
L
Linus Torvalds 已提交
302

303
		pos = get_symbol_pos(addr, symbolsize, offset);
L
Linus Torvalds 已提交
304
		/* Grab name */
305 306
		kallsyms_expand_symbol(get_symbol_offset(pos),
				       namebuf, KSYM_NAME_LEN);
307 308
		if (modname)
			*modname = NULL;
L
Linus Torvalds 已提交
309 310 311
		return namebuf;
	}

312
	/* See if it's in a module. */
313 314
	return module_address_lookup(addr, symbolsize, offset, modname,
				     namebuf);
L
Linus Torvalds 已提交
315 316
}

317 318 319
int lookup_symbol_name(unsigned long addr, char *symname)
{
	symname[0] = '\0';
320
	symname[KSYM_NAME_LEN - 1] = '\0';
321 322 323 324 325 326

	if (is_ksym_addr(addr)) {
		unsigned long pos;

		pos = get_symbol_pos(addr, NULL, NULL);
		/* Grab name */
327 328
		kallsyms_expand_symbol(get_symbol_offset(pos),
				       symname, KSYM_NAME_LEN);
329 330
		return 0;
	}
331
	/* See if it's in a module. */
332 333 334
	return lookup_module_symbol_name(addr, symname);
}

335 336 337 338
int lookup_symbol_attrs(unsigned long addr, unsigned long *size,
			unsigned long *offset, char *modname, char *name)
{
	name[0] = '\0';
339
	name[KSYM_NAME_LEN - 1] = '\0';
340 341 342 343 344 345

	if (is_ksym_addr(addr)) {
		unsigned long pos;

		pos = get_symbol_pos(addr, size, offset);
		/* Grab name */
346 347
		kallsyms_expand_symbol(get_symbol_offset(pos),
				       name, KSYM_NAME_LEN);
348 349 350
		modname[0] = '\0';
		return 0;
	}
351
	/* See if it's in a module. */
352 353 354
	return lookup_module_symbol_attrs(addr, size, offset, modname, name);
}

R
Robert Peterson 已提交
355
/* Look up a kernel symbol and return it in a text buffer. */
356
static int __sprint_symbol(char *buffer, unsigned long address,
357
			   int symbol_offset, int add_offset)
L
Linus Torvalds 已提交
358 359 360 361
{
	char *modname;
	const char *name;
	unsigned long offset, size;
H
Hugh Dickins 已提交
362
	int len;
L
Linus Torvalds 已提交
363

364
	address += symbol_offset;
H
Hugh Dickins 已提交
365
	name = kallsyms_lookup(address, &size, &offset, &modname, buffer);
L
Linus Torvalds 已提交
366
	if (!name)
R
Robert Peterson 已提交
367
		return sprintf(buffer, "0x%lx", address);
A
Andrew Morton 已提交
368

H
Hugh Dickins 已提交
369 370 371
	if (name != buffer)
		strcpy(buffer, name);
	len = strlen(buffer);
372
	offset -= symbol_offset;
H
Hugh Dickins 已提交
373

374 375 376
	if (add_offset)
		len += sprintf(buffer + len, "+%#lx/%#lx", offset, size);

A
Andrew Morton 已提交
377
	if (modname)
378
		len += sprintf(buffer + len, " [%s]", modname);
H
Hugh Dickins 已提交
379 380

	return len;
R
Robert Peterson 已提交
381
}
382 383 384 385 386 387 388 389 390 391 392 393 394 395

/**
 * sprint_symbol - Look up a kernel symbol and return it in a text buffer
 * @buffer: buffer to be stored
 * @address: address to lookup
 *
 * This function looks up a kernel symbol with @address and stores its name,
 * offset, size and module name to @buffer if possible. If no symbol was found,
 * just saves its @address as is.
 *
 * This function returns the number of bytes stored in @buffer.
 */
int sprint_symbol(char *buffer, unsigned long address)
{
396
	return __sprint_symbol(buffer, address, 0, 1);
397
}
398
EXPORT_SYMBOL_GPL(sprint_symbol);
R
Robert Peterson 已提交
399

400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416
/**
 * sprint_symbol_no_offset - Look up a kernel symbol and return it in a text buffer
 * @buffer: buffer to be stored
 * @address: address to lookup
 *
 * This function looks up a kernel symbol with @address and stores its name
 * and module name to @buffer if possible. If no symbol was found, just saves
 * its @address as is.
 *
 * This function returns the number of bytes stored in @buffer.
 */
int sprint_symbol_no_offset(char *buffer, unsigned long address)
{
	return __sprint_symbol(buffer, address, 0, 0);
}
EXPORT_SYMBOL_GPL(sprint_symbol_no_offset);

417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432
/**
 * sprint_backtrace - Look up a backtrace symbol and return it in a text buffer
 * @buffer: buffer to be stored
 * @address: address to lookup
 *
 * This function is for stack backtrace and does the same thing as
 * sprint_symbol() but with modified/decreased @address. If there is a
 * tail-call to the function marked "noreturn", gcc optimized out code after
 * the call so that the stack-saved return address could point outside of the
 * caller. This function ensures that kallsyms will find the original caller
 * by decreasing @address.
 *
 * This function returns the number of bytes stored in @buffer.
 */
int sprint_backtrace(char *buffer, unsigned long address)
{
433
	return __sprint_symbol(buffer, address, -1, 1);
434 435
}

R
Robert Peterson 已提交
436 437 438 439 440 441 442
/* Look up a kernel symbol and print it to the kernel messages. */
void __print_symbol(const char *fmt, unsigned long address)
{
	char buffer[KSYM_SYMBOL_LEN];

	sprint_symbol(buffer, address);

L
Linus Torvalds 已提交
443 444
	printk(fmt, buffer);
}
445
EXPORT_SYMBOL(__print_symbol);
L
Linus Torvalds 已提交
446 447

/* To avoid using get_symbol_offset for every symbol, we carry prefix along. */
448
struct kallsym_iter {
L
Linus Torvalds 已提交
449 450
	loff_t pos;
	unsigned long value;
451
	unsigned int nameoff; /* If iterating in core kernel symbols. */
L
Linus Torvalds 已提交
452
	char type;
453 454
	char name[KSYM_NAME_LEN];
	char module_name[MODULE_NAME_LEN];
455
	int exported;
L
Linus Torvalds 已提交
456 457 458 459
};

static int get_ksymbol_mod(struct kallsym_iter *iter)
{
460 461 462
	if (module_get_kallsym(iter->pos - kallsyms_num_syms, &iter->value,
				&iter->type, iter->name, iter->module_name,
				&iter->exported) < 0)
L
Linus Torvalds 已提交
463 464 465 466 467 468 469 470 471
		return 0;
	return 1;
}

/* Returns space to next name. */
static unsigned long get_ksymbol_core(struct kallsym_iter *iter)
{
	unsigned off = iter->nameoff;

472
	iter->module_name[0] = '\0';
L
Linus Torvalds 已提交
473 474 475 476
	iter->value = kallsyms_addresses[iter->pos];

	iter->type = kallsyms_get_symbol_type(off);

477
	off = kallsyms_expand_symbol(off, iter->name, ARRAY_SIZE(iter->name));
L
Linus Torvalds 已提交
478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496

	return off - iter->nameoff;
}

static void reset_iter(struct kallsym_iter *iter, loff_t new_pos)
{
	iter->name[0] = '\0';
	iter->nameoff = get_symbol_offset(new_pos);
	iter->pos = new_pos;
}

/* Returns false if pos at or past end of file. */
static int update_iter(struct kallsym_iter *iter, loff_t pos)
{
	/* Module symbols can be accessed randomly. */
	if (pos >= kallsyms_num_syms) {
		iter->pos = pos;
		return get_ksymbol_mod(iter);
	}
497

L
Linus Torvalds 已提交
498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531
	/* If we're not on the desired position, reset to new position. */
	if (pos != iter->pos)
		reset_iter(iter, pos);

	iter->nameoff += get_ksymbol_core(iter);
	iter->pos++;

	return 1;
}

static void *s_next(struct seq_file *m, void *p, loff_t *pos)
{
	(*pos)++;

	if (!update_iter(m->private, *pos))
		return NULL;
	return p;
}

static void *s_start(struct seq_file *m, loff_t *pos)
{
	if (!update_iter(m->private, *pos))
		return NULL;
	return m->private;
}

static void s_stop(struct seq_file *m, void *p)
{
}

static int s_show(struct seq_file *m, void *p)
{
	struct kallsym_iter *iter = m->private;

532
	/* Some debugging symbols have no name.  Ignore them. */
L
Linus Torvalds 已提交
533 534 535
	if (!iter->name[0])
		return 0;

536 537 538
	if (iter->module_name[0]) {
		char type;

539 540 541 542
		/*
		 * Label it "global" if it is exported,
		 * "local" if not exported.
		 */
543 544
		type = iter->exported ? toupper(iter->type) :
					tolower(iter->type);
545 546
		seq_printf(m, "%pK %c %s\t[%s]\n", (void *)iter->value,
			   type, iter->name, iter->module_name);
547
	} else
548 549
		seq_printf(m, "%pK %c %s\n", (void *)iter->value,
			   iter->type, iter->name);
L
Linus Torvalds 已提交
550 551 552
	return 0;
}

553
static const struct seq_operations kallsyms_op = {
L
Linus Torvalds 已提交
554 555 556 557 558 559 560 561
	.start = s_start,
	.next = s_next,
	.stop = s_stop,
	.show = s_show
};

static int kallsyms_open(struct inode *inode, struct file *file)
{
562 563
	/*
	 * We keep iterator in m->private, since normal case is to
L
Linus Torvalds 已提交
564
	 * s_start from where we left off, so we avoid doing
565 566
	 * using get_symbol_offset for every symbol.
	 */
L
Linus Torvalds 已提交
567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582
	struct kallsym_iter *iter;
	int ret;

	iter = kmalloc(sizeof(*iter), GFP_KERNEL);
	if (!iter)
		return -ENOMEM;
	reset_iter(iter, 0);

	ret = seq_open(file, &kallsyms_op);
	if (ret == 0)
		((struct seq_file *)file->private_data)->private = iter;
	else
		kfree(iter);
	return ret;
}

583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602
#ifdef	CONFIG_KGDB_KDB
const char *kdb_walk_kallsyms(loff_t *pos)
{
	static struct kallsym_iter kdb_walk_kallsyms_iter;
	if (*pos == 0) {
		memset(&kdb_walk_kallsyms_iter, 0,
		       sizeof(kdb_walk_kallsyms_iter));
		reset_iter(&kdb_walk_kallsyms_iter, 0);
	}
	while (1) {
		if (!update_iter(&kdb_walk_kallsyms_iter, *pos))
			return NULL;
		++*pos;
		/* Some debugging symbols have no name.  Ignore them. */
		if (kdb_walk_kallsyms_iter.name[0])
			return kdb_walk_kallsyms_iter.name;
	}
}
#endif	/* CONFIG_KGDB_KDB */

603
static const struct file_operations kallsyms_operations = {
L
Linus Torvalds 已提交
604 605 606
	.open = kallsyms_open,
	.read = seq_read,
	.llseek = seq_lseek,
607
	.release = seq_release_private,
L
Linus Torvalds 已提交
608 609 610 611
};

static int __init kallsyms_init(void)
{
612
	proc_create("kallsyms", 0444, NULL, &kallsyms_operations);
L
Linus Torvalds 已提交
613 614
	return 0;
}
615
device_initcall(kallsyms_init);