diff options
Diffstat (limited to 'scripts/kallsyms.c')
-rw-r--r-- | scripts/kallsyms.c | 76 |
1 files changed, 61 insertions, 15 deletions
diff --git a/scripts/kallsyms.c b/scripts/kallsyms.c index 6654cbed965b..3cb57895c9ea 100644 --- a/scripts/kallsyms.c +++ b/scripts/kallsyms.c | |||
@@ -23,6 +23,10 @@ | |||
23 | #include <string.h> | 23 | #include <string.h> |
24 | #include <ctype.h> | 24 | #include <ctype.h> |
25 | 25 | ||
26 | #ifndef ARRAY_SIZE | ||
27 | #define ARRAY_SIZE(arr) (sizeof(arr) / sizeof(arr[0])) | ||
28 | #endif | ||
29 | |||
26 | #define KSYM_NAME_LEN 128 | 30 | #define KSYM_NAME_LEN 128 |
27 | 31 | ||
28 | struct sym_entry { | 32 | struct sym_entry { |
@@ -32,9 +36,23 @@ struct sym_entry { | |||
32 | unsigned char *sym; | 36 | unsigned char *sym; |
33 | }; | 37 | }; |
34 | 38 | ||
39 | struct text_range { | ||
40 | const char *stext, *etext; | ||
41 | unsigned long long start, end; | ||
42 | }; | ||
43 | |||
44 | static unsigned long long _text; | ||
45 | static struct text_range text_ranges[] = { | ||
46 | { "_stext", "_etext" }, | ||
47 | { "_sinittext", "_einittext" }, | ||
48 | { "_stext_l1", "_etext_l1" }, /* Blackfin on-chip L1 inst SRAM */ | ||
49 | { "_stext_l2", "_etext_l2" }, /* Blackfin on-chip L2 SRAM */ | ||
50 | }; | ||
51 | #define text_range_text (&text_ranges[0]) | ||
52 | #define text_range_inittext (&text_ranges[1]) | ||
53 | |||
35 | static struct sym_entry *table; | 54 | static struct sym_entry *table; |
36 | static unsigned int table_size, table_cnt; | 55 | static unsigned int table_size, table_cnt; |
37 | static unsigned long long _text, _stext, _etext, _sinittext, _einittext; | ||
38 | static int all_symbols = 0; | 56 | static int all_symbols = 0; |
39 | static char symbol_prefix_char = '\0'; | 57 | static char symbol_prefix_char = '\0'; |
40 | 58 | ||
@@ -61,6 +79,26 @@ static inline int is_arm_mapping_symbol(const char *str) | |||
61 | && (str[2] == '\0' || str[2] == '.'); | 79 | && (str[2] == '\0' || str[2] == '.'); |
62 | } | 80 | } |
63 | 81 | ||
82 | static int read_symbol_tr(const char *sym, unsigned long long addr) | ||
83 | { | ||
84 | size_t i; | ||
85 | struct text_range *tr; | ||
86 | |||
87 | for (i = 0; i < ARRAY_SIZE(text_ranges); ++i) { | ||
88 | tr = &text_ranges[i]; | ||
89 | |||
90 | if (strcmp(sym, tr->stext) == 0) { | ||
91 | tr->start = addr; | ||
92 | return 0; | ||
93 | } else if (strcmp(sym, tr->etext) == 0) { | ||
94 | tr->end = addr; | ||
95 | return 0; | ||
96 | } | ||
97 | } | ||
98 | |||
99 | return 1; | ||
100 | } | ||
101 | |||
64 | static int read_symbol(FILE *in, struct sym_entry *s) | 102 | static int read_symbol(FILE *in, struct sym_entry *s) |
65 | { | 103 | { |
66 | char str[500]; | 104 | char str[500]; |
@@ -84,14 +122,8 @@ static int read_symbol(FILE *in, struct sym_entry *s) | |||
84 | /* Ignore most absolute/undefined (?) symbols. */ | 122 | /* Ignore most absolute/undefined (?) symbols. */ |
85 | if (strcmp(sym, "_text") == 0) | 123 | if (strcmp(sym, "_text") == 0) |
86 | _text = s->addr; | 124 | _text = s->addr; |
87 | else if (strcmp(sym, "_stext") == 0) | 125 | else if (read_symbol_tr(sym, s->addr) == 0) |
88 | _stext = s->addr; | 126 | /* nothing to do */; |
89 | else if (strcmp(sym, "_etext") == 0) | ||
90 | _etext = s->addr; | ||
91 | else if (strcmp(sym, "_sinittext") == 0) | ||
92 | _sinittext = s->addr; | ||
93 | else if (strcmp(sym, "_einittext") == 0) | ||
94 | _einittext = s->addr; | ||
95 | else if (toupper(stype) == 'A') | 127 | else if (toupper(stype) == 'A') |
96 | { | 128 | { |
97 | /* Keep these useful absolute symbols */ | 129 | /* Keep these useful absolute symbols */ |
@@ -127,6 +159,21 @@ static int read_symbol(FILE *in, struct sym_entry *s) | |||
127 | return 0; | 159 | return 0; |
128 | } | 160 | } |
129 | 161 | ||
162 | static int symbol_valid_tr(struct sym_entry *s) | ||
163 | { | ||
164 | size_t i; | ||
165 | struct text_range *tr; | ||
166 | |||
167 | for (i = 0; i < ARRAY_SIZE(text_ranges); ++i) { | ||
168 | tr = &text_ranges[i]; | ||
169 | |||
170 | if (s->addr >= tr->start && s->addr < tr->end) | ||
171 | return 0; | ||
172 | } | ||
173 | |||
174 | return 1; | ||
175 | } | ||
176 | |||
130 | static int symbol_valid(struct sym_entry *s) | 177 | static int symbol_valid(struct sym_entry *s) |
131 | { | 178 | { |
132 | /* Symbols which vary between passes. Passes 1 and 2 must have | 179 | /* Symbols which vary between passes. Passes 1 and 2 must have |
@@ -156,8 +203,7 @@ static int symbol_valid(struct sym_entry *s) | |||
156 | /* if --all-symbols is not specified, then symbols outside the text | 203 | /* if --all-symbols is not specified, then symbols outside the text |
157 | * and inittext sections are discarded */ | 204 | * and inittext sections are discarded */ |
158 | if (!all_symbols) { | 205 | if (!all_symbols) { |
159 | if ((s->addr < _stext || s->addr > _etext) | 206 | if (symbol_valid_tr(s) == 0) |
160 | && (s->addr < _sinittext || s->addr > _einittext)) | ||
161 | return 0; | 207 | return 0; |
162 | /* Corner case. Discard any symbols with the same value as | 208 | /* Corner case. Discard any symbols with the same value as |
163 | * _etext _einittext; they can move between pass 1 and 2 when | 209 | * _etext _einittext; they can move between pass 1 and 2 when |
@@ -165,10 +211,10 @@ static int symbol_valid(struct sym_entry *s) | |||
165 | * they may get dropped in pass 2, which breaks the kallsyms | 211 | * they may get dropped in pass 2, which breaks the kallsyms |
166 | * rules. | 212 | * rules. |
167 | */ | 213 | */ |
168 | if ((s->addr == _etext && | 214 | if ((s->addr == text_range_text->end && |
169 | strcmp((char *)s->sym + offset, "_etext")) || | 215 | strcmp((char *)s->sym + offset, text_range_text->etext)) || |
170 | (s->addr == _einittext && | 216 | (s->addr == text_range_inittext->end && |
171 | strcmp((char *)s->sym + offset, "_einittext"))) | 217 | strcmp((char *)s->sym + offset, text_range_inittext->etext))) |
172 | return 0; | 218 | return 0; |
173 | } | 219 | } |
174 | 220 | ||