Blame view

kernel/kallsyms.c 11.9 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
  /*
   * 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>
  #include <linux/err.h>
  #include <linux/proc_fs.h>
4e57b6817   Tim Schmielau   [PATCH] fix missi...
21
  #include <linux/sched.h>	/* for cond_resched */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
22
  #include <linux/mm.h>
07354a009   Adam B. Jerome   [PATCH] /proc/kal...
23
  #include <linux/ctype.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
24
25
26
27
28
29
30
31
32
33
  
  #include <asm/sections.h>
  
  #ifdef CONFIG_KALLSYMS_ALL
  #define all_var 1
  #else
  #define all_var 0
  #endif
  
  /* These will be re-linked against their real values during the second link stage */
aad094701   Jan Beulich   [PATCH] move kall...
34
35
36
  extern const unsigned long kallsyms_addresses[] __attribute__((weak));
  extern const unsigned long kallsyms_num_syms __attribute__((weak));
  extern const u8 kallsyms_names[] __attribute__((weak));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
37

aad094701   Jan Beulich   [PATCH] move kall...
38
39
  extern const u8 kallsyms_token_table[] __attribute__((weak));
  extern const u16 kallsyms_token_index[] __attribute__((weak));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
40

aad094701   Jan Beulich   [PATCH] move kall...
41
  extern const unsigned long kallsyms_markers[] __attribute__((weak));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
42
43
44
45
46
47
48
49
  
  static inline int is_kernel_inittext(unsigned long addr)
  {
  	if (addr >= (unsigned long)_sinittext
  	    && addr <= (unsigned long)_einittext)
  		return 1;
  	return 0;
  }
075d6eb16   David Woodhouse   [PATCH] ppc32: pl...
50
51
52
53
54
55
56
  static inline int is_kernel_extratext(unsigned long addr)
  {
  	if (addr >= (unsigned long)_sextratext
  	    && addr <= (unsigned long)_eextratext)
  		return 1;
  	return 0;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
57
58
59
60
61
62
63
64
65
66
67
68
69
  static inline int is_kernel_text(unsigned long addr)
  {
  	if (addr >= (unsigned long)_stext && addr <= (unsigned long)_etext)
  		return 1;
  	return in_gate_area_no_task(addr);
  }
  
  static inline int is_kernel(unsigned long addr)
  {
  	if (addr >= (unsigned long)_stext && addr <= (unsigned long)_end)
  		return 1;
  	return in_gate_area_no_task(addr);
  }
ffc508919   Franck Bui-Huu   [PATCH] Create ka...
70
71
72
73
74
75
76
77
  static int is_ksym_addr(unsigned long addr)
  {
  	if (all_var)
  		return is_kernel(addr);
  
  	return is_kernel_text(addr) || is_kernel_inittext(addr) ||
  		is_kernel_extratext(addr);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
78
79
80
81
82
  /* expand a compressed symbol data into the resulting uncompressed string,
     given the offset to where the symbol is in the compressed stream */
  static unsigned int kallsyms_expand_symbol(unsigned int off, char *result)
  {
  	int len, skipped_first = 0;
aad094701   Jan Beulich   [PATCH] move kall...
83
  	const u8 *tptr, *data;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
  
  	/* get the compressed symbol length from the first symbol byte */
  	data = &kallsyms_names[off];
  	len = *data;
  	data++;
  
  	/* update the offset to return the offset for the next symbol on
  	 * the compressed stream */
  	off += len + 1;
  
  	/* 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] ];
  		data++;
  		len--;
  
  		while (*tptr) {
  			if(skipped_first) {
  				*result = *tptr;
  				result++;
  			} else
  				skipped_first = 1;
  			tptr++;
  		}
  	}
  
  	*result = '\0';
  
  	/* return to offset to the next symbol */
  	return off;
  }
  
  /* get symbol type information. This is encoded as a single char at the
   * begining of the symbol name */
  static char kallsyms_get_symbol_type(unsigned int off)
  {
  	/* 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] ] ];
  }
  
  
  /* find the offset on the compressed stream given and index in the
   * kallsyms array */
  static unsigned int get_symbol_offset(unsigned long pos)
  {
aad094701   Jan Beulich   [PATCH] move kall...
131
  	const u8 *name;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
  	int i;
  
  	/* 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] ];
  
  	/* 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++)
  		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)
  {
9281acea6   Tejun Heo   kallsyms: make KS...
151
  	char namebuf[KSYM_NAME_LEN];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
152
153
154
155
156
157
158
159
160
161
162
  	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);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
163

ffc508919   Franck Bui-Huu   [PATCH] Create ka...
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
  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;
  
  	/* This kernel should never had been booted. */
  	BUG_ON(!kallsyms_addresses);
  
  	/* do a binary search on the sorted kallsyms_addresses array */
  	low = 0;
  	high = kallsyms_num_syms;
  
  	while (high - low > 1) {
  		mid = (low + high) / 2;
  		if (kallsyms_addresses[mid] <= addr)
  			low = mid;
  		else
  			high = mid;
  	}
  
  	/*
  	 * search for the first aliased symbol. Aliased
  	 * symbols are symbols with the same address
  	 */
  	while (low && kallsyms_addresses[low-1] == kallsyms_addresses[low])
  		--low;
  
  	symbol_start = kallsyms_addresses[low];
  
  	/* Search for next non-aliased symbol */
  	for (i = low + 1; i < kallsyms_num_syms; i++) {
  		if (kallsyms_addresses[i] > symbol_start) {
  			symbol_end = kallsyms_addresses[i];
  			break;
  		}
  	}
  
  	/* if we found no next symbol, we use the end of the section */
  	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;
  	}
ffb451227   Alexey Dobriyan   Simplify kallsyms...
212
213
214
215
  	if (symbolsize)
  		*symbolsize = symbol_end - symbol_start;
  	if (offset)
  		*offset = addr - symbol_start;
ffc508919   Franck Bui-Huu   [PATCH] Create ka...
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
  
  	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)
  {
  	if (is_ksym_addr(addr))
  		return !!get_symbol_pos(addr, symbolsize, offset);
  
  	return !!module_address_lookup(addr, symbolsize, offset, NULL);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
231
232
233
234
235
236
237
238
239
240
241
242
  /*
   * Lookup an address
   * - 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
   */
  const char *kallsyms_lookup(unsigned long addr,
  			    unsigned long *symbolsize,
  			    unsigned long *offset,
  			    char **modname, char *namebuf)
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
243
  	const char *msym;
9281acea6   Tejun Heo   kallsyms: make KS...
244
  	namebuf[KSYM_NAME_LEN - 1] = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
245
  	namebuf[0] = 0;
ffc508919   Franck Bui-Huu   [PATCH] Create ka...
246
247
  	if (is_ksym_addr(addr)) {
  		unsigned long pos;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
248

ffc508919   Franck Bui-Huu   [PATCH] Create ka...
249
  		pos = get_symbol_pos(addr, symbolsize, offset);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
250
  		/* Grab name */
ffc508919   Franck Bui-Huu   [PATCH] Create ka...
251
  		kallsyms_expand_symbol(get_symbol_offset(pos), namebuf);
7a74fc492   Kyle McMartin   fix possible null...
252
253
  		if (modname)
  			*modname = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
254
255
256
257
258
259
  		return namebuf;
  	}
  
  	/* see if it's in a module */
  	msym = module_address_lookup(addr, symbolsize, offset, modname);
  	if (msym)
9281acea6   Tejun Heo   kallsyms: make KS...
260
  		return strncpy(namebuf, msym, KSYM_NAME_LEN - 1);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
261
262
263
  
  	return NULL;
  }
9d65cb4a1   Alexey Dobriyan   Fix race between ...
264
265
266
  int lookup_symbol_name(unsigned long addr, char *symname)
  {
  	symname[0] = '\0';
9281acea6   Tejun Heo   kallsyms: make KS...
267
  	symname[KSYM_NAME_LEN - 1] = '\0';
9d65cb4a1   Alexey Dobriyan   Fix race between ...
268
269
270
271
272
273
274
275
276
277
278
279
  
  	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;
  	}
  	/* see if it's in a module */
  	return lookup_module_symbol_name(addr, symname);
  }
a5c43dae7   Alexey Dobriyan   Fix race between ...
280
281
282
283
  int lookup_symbol_attrs(unsigned long addr, unsigned long *size,
  			unsigned long *offset, char *modname, char *name)
  {
  	name[0] = '\0';
9281acea6   Tejun Heo   kallsyms: make KS...
284
  	name[KSYM_NAME_LEN - 1] = '\0';
a5c43dae7   Alexey Dobriyan   Fix race between ...
285
286
287
288
289
290
291
292
293
294
295
296
297
  
  	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;
  	}
  	/* see if it's in a module */
  	return lookup_module_symbol_attrs(addr, size, offset, modname, name);
  }
42e380832   Robert Peterson   Extend print_symb...
298
299
  /* Look up a kernel symbol and return it in a text buffer. */
  int sprint_symbol(char *buffer, unsigned long address)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
300
301
302
303
  {
  	char *modname;
  	const char *name;
  	unsigned long offset, size;
9281acea6   Tejun Heo   kallsyms: make KS...
304
  	char namebuf[KSYM_NAME_LEN];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
305
306
  
  	name = kallsyms_lookup(address, &size, &offset, &modname, namebuf);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
307
  	if (!name)
42e380832   Robert Peterson   Extend print_symb...
308
  		return sprintf(buffer, "0x%lx", address);
19769b762   Andrew Morton   sprint_symbol() c...
309
310
311
  
  	if (modname)
  		return sprintf(buffer, "%s+%#lx/%#lx [%s]", name, offset,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
312
  				size, modname);
19769b762   Andrew Morton   sprint_symbol() c...
313
314
  	else
  		return sprintf(buffer, "%s+%#lx/%#lx", name, offset, size);
42e380832   Robert Peterson   Extend print_symb...
315
316
317
318
319
320
321
322
  }
  
  /* 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);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
323
324
325
326
327
328
329
  	printk(fmt, buffer);
  }
  
  /* To avoid using get_symbol_offset for every symbol, we carry prefix along. */
  struct kallsym_iter
  {
  	loff_t pos;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
330
331
332
  	unsigned long value;
  	unsigned int nameoff; /* If iterating in core kernel symbols */
  	char type;
9281acea6   Tejun Heo   kallsyms: make KS...
333
334
  	char name[KSYM_NAME_LEN];
  	char module_name[MODULE_NAME_LEN];
ea07890a6   Alexey Dobriyan   Fix race between ...
335
  	int exported;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
336
  };
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
337
338
  static int get_ksymbol_mod(struct kallsym_iter *iter)
  {
ea07890a6   Alexey Dobriyan   Fix race between ...
339
340
341
  	if (module_get_kallsym(iter->pos - kallsyms_num_syms, &iter->value,
  				&iter->type, iter->name, iter->module_name,
  				&iter->exported) < 0)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
342
  		return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
343
344
345
346
347
348
349
  	return 1;
  }
  
  /* Returns space to next name. */
  static unsigned long get_ksymbol_core(struct kallsym_iter *iter)
  {
  	unsigned off = iter->nameoff;
ea07890a6   Alexey Dobriyan   Fix race between ...
350
  	iter->module_name[0] = '\0';
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
  	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);
  	}
  	
  	/* 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;
  
  	/* Some debugging symbols have no name.  Ignore them. */ 
  	if (!iter->name[0])
  		return 0;
ea07890a6   Alexey Dobriyan   Fix race between ...
413
414
415
416
417
418
419
  	if (iter->module_name[0]) {
  		char type;
  
  		/* Label it "global" if it is exported,
  		 * "local" if not exported. */
  		type = iter->exported ? toupper(iter->type) :
  					tolower(iter->type);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
420
421
422
  		seq_printf(m, "%0*lx %c %s\t[%s]
  ",
  			   (int)(2*sizeof(void*)),
ea07890a6   Alexey Dobriyan   Fix race between ...
423
424
  			   iter->value, type, iter->name, iter->module_name);
  	} else
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
425
426
427
428
429
430
  		seq_printf(m, "%0*lx %c %s
  ",
  			   (int)(2*sizeof(void*)),
  			   iter->value, iter->type, iter->name);
  	return 0;
  }
15ad7cdcf   Helge Deller   [PATCH] struct se...
431
  static const struct seq_operations kallsyms_op = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
  	.start = s_start,
  	.next = s_next,
  	.stop = s_stop,
  	.show = s_show
  };
  
  static int kallsyms_open(struct inode *inode, struct file *file)
  {
  	/* We keep iterator in m->private, since normal case is to
  	 * s_start from where we left off, so we avoid doing
  	 * using get_symbol_offset for every symbol */
  	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;
  }
15ad7cdcf   Helge Deller   [PATCH] struct se...
458
  static const struct file_operations kallsyms_operations = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
459
460
461
  	.open = kallsyms_open,
  	.read = seq_read,
  	.llseek = seq_lseek,
5a0c6a0d1   Martin Peschke   kallsyms: cleanup...
462
  	.release = seq_release_private,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
463
464
465
466
467
468
469
470
471
472
473
474
475
476
  };
  
  static int __init kallsyms_init(void)
  {
  	struct proc_dir_entry *entry;
  
  	entry = create_proc_entry("kallsyms", 0444, NULL);
  	if (entry)
  		entry->proc_fops = &kallsyms_operations;
  	return 0;
  }
  __initcall(kallsyms_init);
  
  EXPORT_SYMBOL(__print_symbol);
42e380832   Robert Peterson   Extend print_symb...
477
  EXPORT_SYMBOL_GPL(sprint_symbol);