kallsyms.c 14.7 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>
L
Linus Torvalds 已提交
26 27 28 29 30 31 32 33 34

#include <asm/sections.h>

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

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

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

49 50
extern const u8 kallsyms_token_table[] __attribute__((weak));
extern const u16 kallsyms_token_index[] __attribute__((weak));
L
Linus Torvalds 已提交
51

52
extern const unsigned long kallsyms_markers[] __attribute__((weak));
L
Linus Torvalds 已提交
53 54 55 56 57 58 59 60 61 62 63

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)
{
64 65
	if ((addr >= (unsigned long)_stext && addr <= (unsigned long)_etext) ||
	    arch_is_kernel_text(addr))
L
Linus Torvalds 已提交
66
		return 1;
67
	return in_gate_area_no_mm(addr);
L
Linus Torvalds 已提交
68 69 70 71 72 73
}

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

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

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

85 86 87 88
/*
 * Expand a compressed symbol data into the resulting uncompressed string,
 * given the offset to where the symbol is in the compressed stream.
 */
L
Linus Torvalds 已提交
89 90 91
static unsigned int kallsyms_expand_symbol(unsigned int off, char *result)
{
	int len, skipped_first = 0;
92
	const u8 *tptr, *data;
L
Linus Torvalds 已提交
93

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

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

105 106 107 108 109 110
	/*
	 * 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 已提交
111 112 113 114
		data++;
		len--;

		while (*tptr) {
115
			if (skipped_first) {
L
Linus Torvalds 已提交
116 117 118 119 120 121 122 123 124 125
				*result = *tptr;
				result++;
			} else
				skipped_first = 1;
			tptr++;
		}
	}

	*result = '\0';

126
	/* Return to offset to the next symbol. */
L
Linus Torvalds 已提交
127 128 129
	return off;
}

130 131 132 133
/*
 * Get symbol type information. This is encoded as a single char at the
 * beginning of the symbol name.
 */
L
Linus Torvalds 已提交
134 135
static char kallsyms_get_symbol_type(unsigned int off)
{
136 137 138 139 140
	/*
	 * 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 已提交
141 142 143
}


144 145 146 147
/*
 * Find the offset on the compressed stream given and index in the
 * kallsyms array.
 */
L
Linus Torvalds 已提交
148 149
static unsigned int get_symbol_offset(unsigned long pos)
{
150
	const u8 *name;
L
Linus Torvalds 已提交
151 152
	int i;

153 154 155 156 157
	/*
	 * 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 已提交
158

159 160 161 162 163 164 165
	/*
	 * 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 已提交
166 167 168 169 170 171 172 173
		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)
{
174
	char namebuf[KSYM_NAME_LEN];
L
Linus Torvalds 已提交
175 176 177 178 179 180 181 182 183 184 185
	unsigned long i;
	unsigned int off;

	for (i = 0, off = 0; i < kallsyms_num_syms; i++) {
		off = kallsyms_expand_symbol(off, namebuf);

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

188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206
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++) {
		off = kallsyms_expand_symbol(off, namebuf);
		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);

207 208 209 210 211 212 213
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;

214 215 216
	/* This kernel should never had been booted. */
	BUG_ON(!kallsyms_addresses);

217
	/* Do a binary search on the sorted kallsyms_addresses array. */
218 219 220 221
	low = 0;
	high = kallsyms_num_syms;

	while (high - low > 1) {
222
		mid = low + (high - low) / 2;
223 224 225 226 227 228 229
		if (kallsyms_addresses[mid] <= addr)
			low = mid;
		else
			high = mid;
	}

	/*
230 231
	 * Search for the first aliased symbol. Aliased
	 * symbols are symbols with the same address.
232 233 234 235 236 237
	 */
	while (low && kallsyms_addresses[low-1] == kallsyms_addresses[low])
		--low;

	symbol_start = kallsyms_addresses[low];

238
	/* Search for next non-aliased symbol. */
239 240 241 242 243 244 245
	for (i = low + 1; i < kallsyms_num_syms; i++) {
		if (kallsyms_addresses[i] > symbol_start) {
			symbol_end = kallsyms_addresses[i];
			break;
		}
	}

246
	/* If we found no next symbol, we use the end of the section. */
247 248 249 250 251 252 253 254 255
	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 已提交
256 257 258 259
	if (symbolsize)
		*symbolsize = symbol_end - symbol_start;
	if (offset)
		*offset = addr - symbol_start;
260 261 262 263 264 265 266 267 268 269

	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)
{
270
	char namebuf[KSYM_NAME_LEN];
271 272 273
	if (is_ksym_addr(addr))
		return !!get_symbol_pos(addr, symbolsize, offset);

274
	return !!module_address_lookup(addr, symbolsize, offset, NULL, namebuf);
275 276
}

L
Linus Torvalds 已提交
277 278
/*
 * Lookup an address
279 280 281 282
 * - 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 已提交
283 284 285 286 287 288
 */
const char *kallsyms_lookup(unsigned long addr,
			    unsigned long *symbolsize,
			    unsigned long *offset,
			    char **modname, char *namebuf)
{
289
	namebuf[KSYM_NAME_LEN - 1] = 0;
L
Linus Torvalds 已提交
290 291
	namebuf[0] = 0;

292 293
	if (is_ksym_addr(addr)) {
		unsigned long pos;
L
Linus Torvalds 已提交
294

295
		pos = get_symbol_pos(addr, symbolsize, offset);
L
Linus Torvalds 已提交
296
		/* Grab name */
297
		kallsyms_expand_symbol(get_symbol_offset(pos), namebuf);
298 299
		if (modname)
			*modname = NULL;
L
Linus Torvalds 已提交
300 301 302
		return namebuf;
	}

303
	/* See if it's in a module. */
304 305
	return module_address_lookup(addr, symbolsize, offset, modname,
				     namebuf);
L
Linus Torvalds 已提交
306 307
}

308 309 310
int lookup_symbol_name(unsigned long addr, char *symname)
{
	symname[0] = '\0';
311
	symname[KSYM_NAME_LEN - 1] = '\0';
312 313 314 315 316 317 318 319 320

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

		pos = get_symbol_pos(addr, NULL, NULL);
		/* Grab name */
		kallsyms_expand_symbol(get_symbol_offset(pos), symname);
		return 0;
	}
321
	/* See if it's in a module. */
322 323 324
	return lookup_module_symbol_name(addr, symname);
}

325 326 327 328
int lookup_symbol_attrs(unsigned long addr, unsigned long *size,
			unsigned long *offset, char *modname, char *name)
{
	name[0] = '\0';
329
	name[KSYM_NAME_LEN - 1] = '\0';
330 331 332 333 334 335 336 337 338 339

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

		pos = get_symbol_pos(addr, size, offset);
		/* Grab name */
		kallsyms_expand_symbol(get_symbol_offset(pos), name);
		modname[0] = '\0';
		return 0;
	}
340
	/* See if it's in a module. */
341 342 343
	return lookup_module_symbol_attrs(addr, size, offset, modname, name);
}

R
Robert Peterson 已提交
344
/* Look up a kernel symbol and return it in a text buffer. */
345
static int __sprint_symbol(char *buffer, unsigned long address,
346
			   int symbol_offset, int add_offset)
L
Linus Torvalds 已提交
347 348 349 350
{
	char *modname;
	const char *name;
	unsigned long offset, size;
H
Hugh Dickins 已提交
351
	int len;
L
Linus Torvalds 已提交
352

353
	address += symbol_offset;
H
Hugh Dickins 已提交
354
	name = kallsyms_lookup(address, &size, &offset, &modname, buffer);
L
Linus Torvalds 已提交
355
	if (!name)
R
Robert Peterson 已提交
356
		return sprintf(buffer, "0x%lx", address);
A
Andrew Morton 已提交
357

H
Hugh Dickins 已提交
358 359 360
	if (name != buffer)
		strcpy(buffer, name);
	len = strlen(buffer);
361
	offset -= symbol_offset;
H
Hugh Dickins 已提交
362

363 364 365
	if (add_offset)
		len += sprintf(buffer + len, "+%#lx/%#lx", offset, size);

A
Andrew Morton 已提交
366
	if (modname)
367
		len += sprintf(buffer + len, " [%s]", modname);
H
Hugh Dickins 已提交
368 369

	return len;
R
Robert Peterson 已提交
370
}
371 372 373 374 375 376 377 378 379 380 381 382 383 384

/**
 * 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)
{
385
	return __sprint_symbol(buffer, address, 0, 1);
386
}
387
EXPORT_SYMBOL_GPL(sprint_symbol);
R
Robert Peterson 已提交
388

389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405
/**
 * 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);

406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421
/**
 * 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)
{
422
	return __sprint_symbol(buffer, address, -1, 1);
423 424
}

R
Robert Peterson 已提交
425 426 427 428 429 430 431
/* 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 已提交
432 433
	printk(fmt, buffer);
}
434
EXPORT_SYMBOL(__print_symbol);
L
Linus Torvalds 已提交
435 436

/* To avoid using get_symbol_offset for every symbol, we carry prefix along. */
437
struct kallsym_iter {
L
Linus Torvalds 已提交
438 439
	loff_t pos;
	unsigned long value;
440
	unsigned int nameoff; /* If iterating in core kernel symbols. */
L
Linus Torvalds 已提交
441
	char type;
442 443
	char name[KSYM_NAME_LEN];
	char module_name[MODULE_NAME_LEN];
444
	int exported;
L
Linus Torvalds 已提交
445 446 447 448
};

static int get_ksymbol_mod(struct kallsym_iter *iter)
{
449 450 451
	if (module_get_kallsym(iter->pos - kallsyms_num_syms, &iter->value,
				&iter->type, iter->name, iter->module_name,
				&iter->exported) < 0)
L
Linus Torvalds 已提交
452 453 454 455 456 457 458 459 460
		return 0;
	return 1;
}

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

461
	iter->module_name[0] = '\0';
L
Linus Torvalds 已提交
462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485
	iter->value = kallsyms_addresses[iter->pos];

	iter->type = kallsyms_get_symbol_type(off);

	off = kallsyms_expand_symbol(off, iter->name);

	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);
	}
486

L
Linus Torvalds 已提交
487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520
	/* 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;

521
	/* Some debugging symbols have no name.  Ignore them. */
L
Linus Torvalds 已提交
522 523 524
	if (!iter->name[0])
		return 0;

525 526 527
	if (iter->module_name[0]) {
		char type;

528 529 530 531
		/*
		 * Label it "global" if it is exported,
		 * "local" if not exported.
		 */
532 533
		type = iter->exported ? toupper(iter->type) :
					tolower(iter->type);
534 535
		seq_printf(m, "%pK %c %s\t[%s]\n", (void *)iter->value,
			   type, iter->name, iter->module_name);
536
	} else
537 538
		seq_printf(m, "%pK %c %s\n", (void *)iter->value,
			   iter->type, iter->name);
L
Linus Torvalds 已提交
539 540 541
	return 0;
}

542
static const struct seq_operations kallsyms_op = {
L
Linus Torvalds 已提交
543 544 545 546 547 548 549 550
	.start = s_start,
	.next = s_next,
	.stop = s_stop,
	.show = s_show
};

static int kallsyms_open(struct inode *inode, struct file *file)
{
551 552
	/*
	 * We keep iterator in m->private, since normal case is to
L
Linus Torvalds 已提交
553
	 * s_start from where we left off, so we avoid doing
554 555
	 * using get_symbol_offset for every symbol.
	 */
L
Linus Torvalds 已提交
556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571
	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;
}

572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591
#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 */

592
static const struct file_operations kallsyms_operations = {
L
Linus Torvalds 已提交
593 594 595
	.open = kallsyms_open,
	.read = seq_read,
	.llseek = seq_lseek,
596
	.release = seq_release_private,
L
Linus Torvalds 已提交
597 598 599 600
};

static int __init kallsyms_init(void)
{
601
	proc_create("kallsyms", 0444, NULL, &kallsyms_operations);
L
Linus Torvalds 已提交
602 603
	return 0;
}
604
device_initcall(kallsyms_init);