forked from mirrors/linux
		
	kallsyms: generalize text region handling
Signed-off-by: Mike Frysinger <vapier@gentoo.org> Signed-off-by: Sam Ravnborg <sam@ravnborg.org>
This commit is contained in:
		
							parent
							
								
									028f042613
								
							
						
					
					
						commit
						17b1f0de79
					
				
					 1 changed files with 61 additions and 26 deletions
				
			
		| 
						 | 
					@ -23,6 +23,10 @@
 | 
				
			||||||
#include <string.h>
 | 
					#include <string.h>
 | 
				
			||||||
#include <ctype.h>
 | 
					#include <ctype.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifndef ARRAY_SIZE
 | 
				
			||||||
 | 
					#define ARRAY_SIZE(arr) (sizeof(arr) / sizeof(arr[0]))
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define KSYM_NAME_LEN		128
 | 
					#define KSYM_NAME_LEN		128
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct sym_entry {
 | 
					struct sym_entry {
 | 
				
			||||||
| 
						 | 
					@ -32,10 +36,23 @@ struct sym_entry {
 | 
				
			||||||
	unsigned char *sym;
 | 
						unsigned char *sym;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct text_range {
 | 
				
			||||||
 | 
						const char *stext, *etext;
 | 
				
			||||||
 | 
						unsigned long long start, end;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static unsigned long long _text;
 | 
				
			||||||
 | 
					static struct text_range text_ranges[] = {
 | 
				
			||||||
 | 
						{ "_stext",     "_etext"     },
 | 
				
			||||||
 | 
						{ "_sinittext", "_einittext" },
 | 
				
			||||||
 | 
						{ "_stext_l1",  "_etext_l1"  },	/* Blackfin on-chip L1 inst SRAM */
 | 
				
			||||||
 | 
						{ "_stext_l2",  "_etext_l2"  },	/* Blackfin on-chip L2 SRAM */
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					#define text_range_text     (&text_ranges[0])
 | 
				
			||||||
 | 
					#define text_range_inittext (&text_ranges[1])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static struct sym_entry *table;
 | 
					static struct sym_entry *table;
 | 
				
			||||||
static unsigned int table_size, table_cnt;
 | 
					static unsigned int table_size, table_cnt;
 | 
				
			||||||
static unsigned long long _text, _stext, _etext, _sinittext, _einittext;
 | 
					 | 
				
			||||||
static unsigned long long _stext_l1, _etext_l1, _stext_l2, _etext_l2;
 | 
					 | 
				
			||||||
static int all_symbols = 0;
 | 
					static int all_symbols = 0;
 | 
				
			||||||
static char symbol_prefix_char = '\0';
 | 
					static char symbol_prefix_char = '\0';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -62,6 +79,26 @@ static inline int is_arm_mapping_symbol(const char *str)
 | 
				
			||||||
	       && (str[2] == '\0' || str[2] == '.');
 | 
						       && (str[2] == '\0' || str[2] == '.');
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int read_symbol_tr(const char *sym, unsigned long long addr)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						size_t i;
 | 
				
			||||||
 | 
						struct text_range *tr;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for (i = 0; i < ARRAY_SIZE(text_ranges); ++i) {
 | 
				
			||||||
 | 
							tr = &text_ranges[i];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (strcmp(sym, tr->stext) == 0) {
 | 
				
			||||||
 | 
								tr->start = addr;
 | 
				
			||||||
 | 
								return 0;
 | 
				
			||||||
 | 
							} else if (strcmp(sym, tr->etext) == 0) {
 | 
				
			||||||
 | 
								tr->end = addr;
 | 
				
			||||||
 | 
								return 0;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return 1;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int read_symbol(FILE *in, struct sym_entry *s)
 | 
					static int read_symbol(FILE *in, struct sym_entry *s)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	char str[500];
 | 
						char str[500];
 | 
				
			||||||
| 
						 | 
					@ -85,22 +122,8 @@ static int read_symbol(FILE *in, struct sym_entry *s)
 | 
				
			||||||
	/* Ignore most absolute/undefined (?) symbols. */
 | 
						/* Ignore most absolute/undefined (?) symbols. */
 | 
				
			||||||
	if (strcmp(sym, "_text") == 0)
 | 
						if (strcmp(sym, "_text") == 0)
 | 
				
			||||||
		_text = s->addr;
 | 
							_text = s->addr;
 | 
				
			||||||
	else if (strcmp(sym, "_stext") == 0)
 | 
						else if (read_symbol_tr(sym, s->addr) == 0)
 | 
				
			||||||
		_stext = s->addr;
 | 
							/* nothing to do */;
 | 
				
			||||||
	else if (strcmp(sym, "_etext") == 0)
 | 
					 | 
				
			||||||
		_etext = s->addr;
 | 
					 | 
				
			||||||
	else if (strcmp(sym, "_sinittext") == 0)
 | 
					 | 
				
			||||||
		_sinittext = s->addr;
 | 
					 | 
				
			||||||
	else if (strcmp(sym, "_einittext") == 0)
 | 
					 | 
				
			||||||
		_einittext = s->addr;
 | 
					 | 
				
			||||||
	else if (strcmp(sym, "_stext_l1") == 0)
 | 
					 | 
				
			||||||
		_stext_l1 = s->addr;
 | 
					 | 
				
			||||||
	else if (strcmp(sym, "_etext_l1") == 0)
 | 
					 | 
				
			||||||
		_etext_l1 = s->addr;
 | 
					 | 
				
			||||||
	else if (strcmp(sym, "_stext_l2") == 0)
 | 
					 | 
				
			||||||
		_stext_l2 = s->addr;
 | 
					 | 
				
			||||||
	else if (strcmp(sym, "_etext_l2") == 0)
 | 
					 | 
				
			||||||
		_etext_l2 = s->addr;
 | 
					 | 
				
			||||||
	else if (toupper(stype) == 'A')
 | 
						else if (toupper(stype) == 'A')
 | 
				
			||||||
	{
 | 
						{
 | 
				
			||||||
		/* Keep these useful absolute symbols */
 | 
							/* Keep these useful absolute symbols */
 | 
				
			||||||
| 
						 | 
					@ -136,6 +159,21 @@ static int read_symbol(FILE *in, struct sym_entry *s)
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int symbol_valid_tr(struct sym_entry *s)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						size_t i;
 | 
				
			||||||
 | 
						struct text_range *tr;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for (i = 0; i < ARRAY_SIZE(text_ranges); ++i) {
 | 
				
			||||||
 | 
							tr = &text_ranges[i];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (s->addr >= tr->start && s->addr < tr->end)
 | 
				
			||||||
 | 
								return 0;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return 1;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int symbol_valid(struct sym_entry *s)
 | 
					static int symbol_valid(struct sym_entry *s)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	/* Symbols which vary between passes.  Passes 1 and 2 must have
 | 
						/* Symbols which vary between passes.  Passes 1 and 2 must have
 | 
				
			||||||
| 
						 | 
					@ -165,10 +203,7 @@ static int symbol_valid(struct sym_entry *s)
 | 
				
			||||||
	/* if --all-symbols is not specified, then symbols outside the text
 | 
						/* if --all-symbols is not specified, then symbols outside the text
 | 
				
			||||||
	 * and inittext sections are discarded */
 | 
						 * and inittext sections are discarded */
 | 
				
			||||||
	if (!all_symbols) {
 | 
						if (!all_symbols) {
 | 
				
			||||||
		if ((s->addr < _stext || s->addr > _etext)
 | 
							if (symbol_valid_tr(s) == 0)
 | 
				
			||||||
		    && (s->addr < _sinittext || s->addr > _einittext)
 | 
					 | 
				
			||||||
		    && (s->addr < _stext_l1 || s->addr > _etext_l1)
 | 
					 | 
				
			||||||
		    && (s->addr < _stext_l2 || s->addr > _etext_l2))
 | 
					 | 
				
			||||||
			return 0;
 | 
								return 0;
 | 
				
			||||||
		/* Corner case.  Discard any symbols with the same value as
 | 
							/* Corner case.  Discard any symbols with the same value as
 | 
				
			||||||
		 * _etext _einittext; they can move between pass 1 and 2 when
 | 
							 * _etext _einittext; they can move between pass 1 and 2 when
 | 
				
			||||||
| 
						 | 
					@ -176,10 +211,10 @@ static int symbol_valid(struct sym_entry *s)
 | 
				
			||||||
		 * they may get dropped in pass 2, which breaks the kallsyms
 | 
							 * they may get dropped in pass 2, which breaks the kallsyms
 | 
				
			||||||
		 * rules.
 | 
							 * rules.
 | 
				
			||||||
		 */
 | 
							 */
 | 
				
			||||||
		if ((s->addr == _etext &&
 | 
							if ((s->addr == text_range_text->end &&
 | 
				
			||||||
				strcmp((char *)s->sym + offset, "_etext")) ||
 | 
									strcmp((char *)s->sym + offset, text_range_text->etext)) ||
 | 
				
			||||||
		    (s->addr == _einittext &&
 | 
							    (s->addr == text_range_inittext->end &&
 | 
				
			||||||
				strcmp((char *)s->sym + offset, "_einittext")))
 | 
									strcmp((char *)s->sym + offset, text_range_inittext->etext)))
 | 
				
			||||||
			return 0;
 | 
								return 0;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue