11da177e4SLinus Torvalds /* Postprocess module symbol versions 21da177e4SLinus Torvalds * 31da177e4SLinus Torvalds * Copyright 2003 Kai Germaschewski 41da177e4SLinus Torvalds * Copyright 2002-2004 Rusty Russell, IBM Corporation 5df578e7dSSam Ravnborg * Copyright 2006-2008 Sam Ravnborg 61da177e4SLinus Torvalds * Based in part on module-init-tools/depmod.c,file2alias 71da177e4SLinus Torvalds * 81da177e4SLinus Torvalds * This software may be used and distributed according to the terms 91da177e4SLinus Torvalds * of the GNU General Public License, incorporated herein by reference. 101da177e4SLinus Torvalds * 111da177e4SLinus Torvalds * Usage: modpost vmlinux module1.o module2.o ... 121da177e4SLinus Torvalds */ 131da177e4SLinus Torvalds 14b2e3e658SMathieu Desnoyers #define _GNU_SOURCE 15b2e3e658SMathieu Desnoyers #include <stdio.h> 161da177e4SLinus Torvalds #include <ctype.h> 171da177e4SLinus Torvalds #include "modpost.h" 185a865c06SLinus Torvalds #include "../../include/generated/autoconf.h" 19b817f6feSSam Ravnborg #include "../../include/linux/license.h" 201da177e4SLinus Torvalds 219e1b9b80SAlan Jenkins /* Some toolchains use a `_' prefix for all user symbols. */ 229e1b9b80SAlan Jenkins #ifdef CONFIG_SYMBOL_PREFIX 239e1b9b80SAlan Jenkins #define MODULE_SYMBOL_PREFIX CONFIG_SYMBOL_PREFIX 249e1b9b80SAlan Jenkins #else 259e1b9b80SAlan Jenkins #define MODULE_SYMBOL_PREFIX "" 269e1b9b80SAlan Jenkins #endif 279e1b9b80SAlan Jenkins 289e1b9b80SAlan Jenkins 291da177e4SLinus Torvalds /* Are we using CONFIG_MODVERSIONS? */ 301da177e4SLinus Torvalds int modversions = 0; 311da177e4SLinus Torvalds /* Warn about undefined symbols? (do so if we have vmlinux) */ 321da177e4SLinus Torvalds int have_vmlinux = 0; 331da177e4SLinus Torvalds /* Is CONFIG_MODULE_SRCVERSION_ALL set? */ 341da177e4SLinus Torvalds static int all_versions = 0; 35040fcc81SSam Ravnborg /* If we are modposting external module set to 1 */ 36040fcc81SSam Ravnborg static int external_module = 0; 378d8d8289SSam Ravnborg /* Warn about section mismatch in vmlinux if set to 1 */ 388d8d8289SSam Ravnborg static int vmlinux_section_warnings = 1; 39c53ddacdSKirill Korotaev /* Only warn about unresolved symbols */ 40c53ddacdSKirill Korotaev static int warn_unresolved = 0; 41bd5cbcedSRam Pai /* How a symbol is exported */ 42588ccd73SSam Ravnborg static int sec_mismatch_count = 0; 43588ccd73SSam Ravnborg static int sec_mismatch_verbose = 1; 44588ccd73SSam Ravnborg 45c96fca21SSam Ravnborg enum export { 46c96fca21SSam Ravnborg export_plain, export_unused, export_gpl, 47c96fca21SSam Ravnborg export_unused_gpl, export_gpl_future, export_unknown 48c96fca21SSam Ravnborg }; 491da177e4SLinus Torvalds 506d9a89eaSAndi Kleen #define PRINTF __attribute__ ((format (printf, 1, 2))) 516d9a89eaSAndi Kleen 526d9a89eaSAndi Kleen PRINTF void fatal(const char *fmt, ...) 531da177e4SLinus Torvalds { 541da177e4SLinus Torvalds va_list arglist; 551da177e4SLinus Torvalds 561da177e4SLinus Torvalds fprintf(stderr, "FATAL: "); 571da177e4SLinus Torvalds 581da177e4SLinus Torvalds va_start(arglist, fmt); 591da177e4SLinus Torvalds vfprintf(stderr, fmt, arglist); 601da177e4SLinus Torvalds va_end(arglist); 611da177e4SLinus Torvalds 621da177e4SLinus Torvalds exit(1); 631da177e4SLinus Torvalds } 641da177e4SLinus Torvalds 656d9a89eaSAndi Kleen PRINTF void warn(const char *fmt, ...) 661da177e4SLinus Torvalds { 671da177e4SLinus Torvalds va_list arglist; 681da177e4SLinus Torvalds 691da177e4SLinus Torvalds fprintf(stderr, "WARNING: "); 701da177e4SLinus Torvalds 711da177e4SLinus Torvalds va_start(arglist, fmt); 721da177e4SLinus Torvalds vfprintf(stderr, fmt, arglist); 731da177e4SLinus Torvalds va_end(arglist); 741da177e4SLinus Torvalds } 751da177e4SLinus Torvalds 766d9a89eaSAndi Kleen PRINTF void merror(const char *fmt, ...) 772a116659SMatthew Wilcox { 782a116659SMatthew Wilcox va_list arglist; 792a116659SMatthew Wilcox 802a116659SMatthew Wilcox fprintf(stderr, "ERROR: "); 812a116659SMatthew Wilcox 822a116659SMatthew Wilcox va_start(arglist, fmt); 832a116659SMatthew Wilcox vfprintf(stderr, fmt, arglist); 842a116659SMatthew Wilcox va_end(arglist); 852a116659SMatthew Wilcox } 862a116659SMatthew Wilcox 87040fcc81SSam Ravnborg static int is_vmlinux(const char *modname) 88040fcc81SSam Ravnborg { 89040fcc81SSam Ravnborg const char *myname; 90040fcc81SSam Ravnborg 91df578e7dSSam Ravnborg myname = strrchr(modname, '/'); 92df578e7dSSam Ravnborg if (myname) 93040fcc81SSam Ravnborg myname++; 94040fcc81SSam Ravnborg else 95040fcc81SSam Ravnborg myname = modname; 96040fcc81SSam Ravnborg 97741f98feSSam Ravnborg return (strcmp(myname, "vmlinux") == 0) || 98741f98feSSam Ravnborg (strcmp(myname, "vmlinux.o") == 0); 99040fcc81SSam Ravnborg } 100040fcc81SSam Ravnborg 1011da177e4SLinus Torvalds void *do_nofail(void *ptr, const char *expr) 1021da177e4SLinus Torvalds { 103df578e7dSSam Ravnborg if (!ptr) 1041da177e4SLinus Torvalds fatal("modpost: Memory allocation failure: %s.\n", expr); 105df578e7dSSam Ravnborg 1061da177e4SLinus Torvalds return ptr; 1071da177e4SLinus Torvalds } 1081da177e4SLinus Torvalds 1091da177e4SLinus Torvalds /* A list of all modules we processed */ 1101da177e4SLinus Torvalds static struct module *modules; 1111da177e4SLinus Torvalds 1125c3ead8cSSam Ravnborg static struct module *find_module(char *modname) 1131da177e4SLinus Torvalds { 1141da177e4SLinus Torvalds struct module *mod; 1151da177e4SLinus Torvalds 1161da177e4SLinus Torvalds for (mod = modules; mod; mod = mod->next) 1171da177e4SLinus Torvalds if (strcmp(mod->name, modname) == 0) 1181da177e4SLinus Torvalds break; 1191da177e4SLinus Torvalds return mod; 1201da177e4SLinus Torvalds } 1211da177e4SLinus Torvalds 1225c3ead8cSSam Ravnborg static struct module *new_module(char *modname) 1231da177e4SLinus Torvalds { 1241da177e4SLinus Torvalds struct module *mod; 1251da177e4SLinus Torvalds char *p, *s; 1261da177e4SLinus Torvalds 1271da177e4SLinus Torvalds mod = NOFAIL(malloc(sizeof(*mod))); 1281da177e4SLinus Torvalds memset(mod, 0, sizeof(*mod)); 1291da177e4SLinus Torvalds p = NOFAIL(strdup(modname)); 1301da177e4SLinus Torvalds 1311da177e4SLinus Torvalds /* strip trailing .o */ 132df578e7dSSam Ravnborg s = strrchr(p, '.'); 133df578e7dSSam Ravnborg if (s != NULL) 1341da177e4SLinus Torvalds if (strcmp(s, ".o") == 0) 1351da177e4SLinus Torvalds *s = '\0'; 1361da177e4SLinus Torvalds 1371da177e4SLinus Torvalds /* add to list */ 1381da177e4SLinus Torvalds mod->name = p; 139b817f6feSSam Ravnborg mod->gpl_compatible = -1; 1401da177e4SLinus Torvalds mod->next = modules; 1411da177e4SLinus Torvalds modules = mod; 1421da177e4SLinus Torvalds 1431da177e4SLinus Torvalds return mod; 1441da177e4SLinus Torvalds } 1451da177e4SLinus Torvalds 1461da177e4SLinus Torvalds /* A hash of all exported symbols, 1471da177e4SLinus Torvalds * struct symbol is also used for lists of unresolved symbols */ 1481da177e4SLinus Torvalds 1491da177e4SLinus Torvalds #define SYMBOL_HASH_SIZE 1024 1501da177e4SLinus Torvalds 1511da177e4SLinus Torvalds struct symbol { 1521da177e4SLinus Torvalds struct symbol *next; 1531da177e4SLinus Torvalds struct module *module; 1541da177e4SLinus Torvalds unsigned int crc; 1551da177e4SLinus Torvalds int crc_valid; 1561da177e4SLinus Torvalds unsigned int weak:1; 157040fcc81SSam Ravnborg unsigned int vmlinux:1; /* 1 if symbol is defined in vmlinux */ 158040fcc81SSam Ravnborg unsigned int kernel:1; /* 1 if symbol is from kernel 159040fcc81SSam Ravnborg * (only for external modules) **/ 1608e70c458SSam Ravnborg unsigned int preloaded:1; /* 1 if symbol from Module.symvers */ 161bd5cbcedSRam Pai enum export export; /* Type of export */ 1621da177e4SLinus Torvalds char name[0]; 1631da177e4SLinus Torvalds }; 1641da177e4SLinus Torvalds 1651da177e4SLinus Torvalds static struct symbol *symbolhash[SYMBOL_HASH_SIZE]; 1661da177e4SLinus Torvalds 1671da177e4SLinus Torvalds /* This is based on the hash agorithm from gdbm, via tdb */ 1681da177e4SLinus Torvalds static inline unsigned int tdb_hash(const char *name) 1691da177e4SLinus Torvalds { 1701da177e4SLinus Torvalds unsigned value; /* Used to compute the hash value. */ 1711da177e4SLinus Torvalds unsigned i; /* Used to cycle through random values. */ 1721da177e4SLinus Torvalds 1731da177e4SLinus Torvalds /* Set the initial value from the key size. */ 1741da177e4SLinus Torvalds for (value = 0x238F13AF * strlen(name), i = 0; name[i]; i++) 1751da177e4SLinus Torvalds value = (value + (((unsigned char *)name)[i] << (i*5 % 24))); 1761da177e4SLinus Torvalds 1771da177e4SLinus Torvalds return (1103515243 * value + 12345); 1781da177e4SLinus Torvalds } 1791da177e4SLinus Torvalds 1805c3ead8cSSam Ravnborg /** 1815c3ead8cSSam Ravnborg * Allocate a new symbols for use in the hash of exported symbols or 1825c3ead8cSSam Ravnborg * the list of unresolved symbols per module 1835c3ead8cSSam Ravnborg **/ 1845c3ead8cSSam Ravnborg static struct symbol *alloc_symbol(const char *name, unsigned int weak, 1855c3ead8cSSam Ravnborg struct symbol *next) 1861da177e4SLinus Torvalds { 1871da177e4SLinus Torvalds struct symbol *s = NOFAIL(malloc(sizeof(*s) + strlen(name) + 1)); 1881da177e4SLinus Torvalds 1891da177e4SLinus Torvalds memset(s, 0, sizeof(*s)); 1901da177e4SLinus Torvalds strcpy(s->name, name); 1911da177e4SLinus Torvalds s->weak = weak; 1921da177e4SLinus Torvalds s->next = next; 1931da177e4SLinus Torvalds return s; 1941da177e4SLinus Torvalds } 1951da177e4SLinus Torvalds 1961da177e4SLinus Torvalds /* For the hash of exported symbols */ 197bd5cbcedSRam Pai static struct symbol *new_symbol(const char *name, struct module *module, 198bd5cbcedSRam Pai enum export export) 1991da177e4SLinus Torvalds { 2001da177e4SLinus Torvalds unsigned int hash; 2011da177e4SLinus Torvalds struct symbol *new; 2021da177e4SLinus Torvalds 2031da177e4SLinus Torvalds hash = tdb_hash(name) % SYMBOL_HASH_SIZE; 2041da177e4SLinus Torvalds new = symbolhash[hash] = alloc_symbol(name, 0, symbolhash[hash]); 2051da177e4SLinus Torvalds new->module = module; 206bd5cbcedSRam Pai new->export = export; 207040fcc81SSam Ravnborg return new; 2081da177e4SLinus Torvalds } 2091da177e4SLinus Torvalds 2105c3ead8cSSam Ravnborg static struct symbol *find_symbol(const char *name) 2111da177e4SLinus Torvalds { 2121da177e4SLinus Torvalds struct symbol *s; 2131da177e4SLinus Torvalds 2141da177e4SLinus Torvalds /* For our purposes, .foo matches foo. PPC64 needs this. */ 2151da177e4SLinus Torvalds if (name[0] == '.') 2161da177e4SLinus Torvalds name++; 2171da177e4SLinus Torvalds 2181da177e4SLinus Torvalds for (s = symbolhash[tdb_hash(name) % SYMBOL_HASH_SIZE]; s; s = s->next) { 2191da177e4SLinus Torvalds if (strcmp(s->name, name) == 0) 2201da177e4SLinus Torvalds return s; 2211da177e4SLinus Torvalds } 2221da177e4SLinus Torvalds return NULL; 2231da177e4SLinus Torvalds } 2241da177e4SLinus Torvalds 225bd5cbcedSRam Pai static struct { 226bd5cbcedSRam Pai const char *str; 227bd5cbcedSRam Pai enum export export; 228bd5cbcedSRam Pai } export_list[] = { 229bd5cbcedSRam Pai { .str = "EXPORT_SYMBOL", .export = export_plain }, 230c96fca21SSam Ravnborg { .str = "EXPORT_UNUSED_SYMBOL", .export = export_unused }, 231bd5cbcedSRam Pai { .str = "EXPORT_SYMBOL_GPL", .export = export_gpl }, 232c96fca21SSam Ravnborg { .str = "EXPORT_UNUSED_SYMBOL_GPL", .export = export_unused_gpl }, 233bd5cbcedSRam Pai { .str = "EXPORT_SYMBOL_GPL_FUTURE", .export = export_gpl_future }, 234bd5cbcedSRam Pai { .str = "(unknown)", .export = export_unknown }, 235bd5cbcedSRam Pai }; 236bd5cbcedSRam Pai 237bd5cbcedSRam Pai 238bd5cbcedSRam Pai static const char *export_str(enum export ex) 239bd5cbcedSRam Pai { 240bd5cbcedSRam Pai return export_list[ex].str; 241bd5cbcedSRam Pai } 242bd5cbcedSRam Pai 243bd5cbcedSRam Pai static enum export export_no(const char *s) 244bd5cbcedSRam Pai { 245bd5cbcedSRam Pai int i; 246df578e7dSSam Ravnborg 247534b89a9SSam Ravnborg if (!s) 248534b89a9SSam Ravnborg return export_unknown; 249bd5cbcedSRam Pai for (i = 0; export_list[i].export != export_unknown; i++) { 250bd5cbcedSRam Pai if (strcmp(export_list[i].str, s) == 0) 251bd5cbcedSRam Pai return export_list[i].export; 252bd5cbcedSRam Pai } 253bd5cbcedSRam Pai return export_unknown; 254bd5cbcedSRam Pai } 255bd5cbcedSRam Pai 256bd5cbcedSRam Pai static enum export export_from_sec(struct elf_info *elf, Elf_Section sec) 257bd5cbcedSRam Pai { 258bd5cbcedSRam Pai if (sec == elf->export_sec) 259bd5cbcedSRam Pai return export_plain; 260c96fca21SSam Ravnborg else if (sec == elf->export_unused_sec) 261c96fca21SSam Ravnborg return export_unused; 262bd5cbcedSRam Pai else if (sec == elf->export_gpl_sec) 263bd5cbcedSRam Pai return export_gpl; 264c96fca21SSam Ravnborg else if (sec == elf->export_unused_gpl_sec) 265c96fca21SSam Ravnborg return export_unused_gpl; 266bd5cbcedSRam Pai else if (sec == elf->export_gpl_future_sec) 267bd5cbcedSRam Pai return export_gpl_future; 268bd5cbcedSRam Pai else 269bd5cbcedSRam Pai return export_unknown; 270bd5cbcedSRam Pai } 271bd5cbcedSRam Pai 2725c3ead8cSSam Ravnborg /** 2735c3ead8cSSam Ravnborg * Add an exported symbol - it may have already been added without a 2745c3ead8cSSam Ravnborg * CRC, in this case just update the CRC 2755c3ead8cSSam Ravnborg **/ 276bd5cbcedSRam Pai static struct symbol *sym_add_exported(const char *name, struct module *mod, 277bd5cbcedSRam Pai enum export export) 2781da177e4SLinus Torvalds { 2791da177e4SLinus Torvalds struct symbol *s = find_symbol(name); 2801da177e4SLinus Torvalds 2811da177e4SLinus Torvalds if (!s) { 282bd5cbcedSRam Pai s = new_symbol(name, mod, export); 2838e70c458SSam Ravnborg } else { 2848e70c458SSam Ravnborg if (!s->preloaded) { 2857b75b13cSSam Ravnborg warn("%s: '%s' exported twice. Previous export " 2868e70c458SSam Ravnborg "was in %s%s\n", mod->name, name, 2878e70c458SSam Ravnborg s->module->name, 2888e70c458SSam Ravnborg is_vmlinux(s->module->name) ?"":".ko"); 2894b21960fSTrent Piepho } else { 2904b21960fSTrent Piepho /* In case Modules.symvers was out of date */ 2914b21960fSTrent Piepho s->module = mod; 2921da177e4SLinus Torvalds } 2931da177e4SLinus Torvalds } 2948e70c458SSam Ravnborg s->preloaded = 0; 295040fcc81SSam Ravnborg s->vmlinux = is_vmlinux(mod->name); 296040fcc81SSam Ravnborg s->kernel = 0; 297bd5cbcedSRam Pai s->export = export; 298040fcc81SSam Ravnborg return s; 2991da177e4SLinus Torvalds } 3001da177e4SLinus Torvalds 301040fcc81SSam Ravnborg static void sym_update_crc(const char *name, struct module *mod, 302bd5cbcedSRam Pai unsigned int crc, enum export export) 303040fcc81SSam Ravnborg { 304040fcc81SSam Ravnborg struct symbol *s = find_symbol(name); 305040fcc81SSam Ravnborg 306040fcc81SSam Ravnborg if (!s) 307bd5cbcedSRam Pai s = new_symbol(name, mod, export); 308040fcc81SSam Ravnborg s->crc = crc; 3091da177e4SLinus Torvalds s->crc_valid = 1; 3101da177e4SLinus Torvalds } 3111da177e4SLinus Torvalds 3125c3ead8cSSam Ravnborg void *grab_file(const char *filename, unsigned long *size) 3131da177e4SLinus Torvalds { 3141da177e4SLinus Torvalds struct stat st; 3151da177e4SLinus Torvalds void *map; 3161da177e4SLinus Torvalds int fd; 3171da177e4SLinus Torvalds 3181da177e4SLinus Torvalds fd = open(filename, O_RDONLY); 3191da177e4SLinus Torvalds if (fd < 0 || fstat(fd, &st) != 0) 3201da177e4SLinus Torvalds return NULL; 3211da177e4SLinus Torvalds 3221da177e4SLinus Torvalds *size = st.st_size; 3231da177e4SLinus Torvalds map = mmap(NULL, *size, PROT_READ|PROT_WRITE, MAP_PRIVATE, fd, 0); 3241da177e4SLinus Torvalds close(fd); 3251da177e4SLinus Torvalds 3261da177e4SLinus Torvalds if (map == MAP_FAILED) 3271da177e4SLinus Torvalds return NULL; 3281da177e4SLinus Torvalds return map; 3291da177e4SLinus Torvalds } 3301da177e4SLinus Torvalds 3315c3ead8cSSam Ravnborg /** 3325c3ead8cSSam Ravnborg * Return a copy of the next line in a mmap'ed file. 3335c3ead8cSSam Ravnborg * spaces in the beginning of the line is trimmed away. 3345c3ead8cSSam Ravnborg * Return a pointer to a static buffer. 3355c3ead8cSSam Ravnborg **/ 3365c3ead8cSSam Ravnborg char *get_next_line(unsigned long *pos, void *file, unsigned long size) 3371da177e4SLinus Torvalds { 3381da177e4SLinus Torvalds static char line[4096]; 3391da177e4SLinus Torvalds int skip = 1; 3401da177e4SLinus Torvalds size_t len = 0; 3411da177e4SLinus Torvalds signed char *p = (signed char *)file + *pos; 3421da177e4SLinus Torvalds char *s = line; 3431da177e4SLinus Torvalds 344df578e7dSSam Ravnborg for (; *pos < size ; (*pos)++) { 3451da177e4SLinus Torvalds if (skip && isspace(*p)) { 3461da177e4SLinus Torvalds p++; 3471da177e4SLinus Torvalds continue; 3481da177e4SLinus Torvalds } 3491da177e4SLinus Torvalds skip = 0; 3501da177e4SLinus Torvalds if (*p != '\n' && (*pos < size)) { 3511da177e4SLinus Torvalds len++; 3521da177e4SLinus Torvalds *s++ = *p++; 3531da177e4SLinus Torvalds if (len > 4095) 3541da177e4SLinus Torvalds break; /* Too long, stop */ 3551da177e4SLinus Torvalds } else { 3561da177e4SLinus Torvalds /* End of string */ 3571da177e4SLinus Torvalds *s = '\0'; 3581da177e4SLinus Torvalds return line; 3591da177e4SLinus Torvalds } 3601da177e4SLinus Torvalds } 3611da177e4SLinus Torvalds /* End of buffer */ 3621da177e4SLinus Torvalds return NULL; 3631da177e4SLinus Torvalds } 3641da177e4SLinus Torvalds 3655c3ead8cSSam Ravnborg void release_file(void *file, unsigned long size) 3661da177e4SLinus Torvalds { 3671da177e4SLinus Torvalds munmap(file, size); 3681da177e4SLinus Torvalds } 3691da177e4SLinus Torvalds 37085bd2fddSSam Ravnborg static int parse_elf(struct elf_info *info, const char *filename) 3711da177e4SLinus Torvalds { 3721da177e4SLinus Torvalds unsigned int i; 37385bd2fddSSam Ravnborg Elf_Ehdr *hdr; 3741da177e4SLinus Torvalds Elf_Shdr *sechdrs; 3751da177e4SLinus Torvalds Elf_Sym *sym; 3761da177e4SLinus Torvalds 3771da177e4SLinus Torvalds hdr = grab_file(filename, &info->size); 3781da177e4SLinus Torvalds if (!hdr) { 3791da177e4SLinus Torvalds perror(filename); 3806803dc0eSSam Ravnborg exit(1); 3811da177e4SLinus Torvalds } 3821da177e4SLinus Torvalds info->hdr = hdr; 38385bd2fddSSam Ravnborg if (info->size < sizeof(*hdr)) { 38485bd2fddSSam Ravnborg /* file too small, assume this is an empty .o file */ 38585bd2fddSSam Ravnborg return 0; 38685bd2fddSSam Ravnborg } 38785bd2fddSSam Ravnborg /* Is this a valid ELF file? */ 38885bd2fddSSam Ravnborg if ((hdr->e_ident[EI_MAG0] != ELFMAG0) || 38985bd2fddSSam Ravnborg (hdr->e_ident[EI_MAG1] != ELFMAG1) || 39085bd2fddSSam Ravnborg (hdr->e_ident[EI_MAG2] != ELFMAG2) || 39185bd2fddSSam Ravnborg (hdr->e_ident[EI_MAG3] != ELFMAG3)) { 39285bd2fddSSam Ravnborg /* Not an ELF file - silently ignore it */ 39385bd2fddSSam Ravnborg return 0; 39485bd2fddSSam Ravnborg } 3951da177e4SLinus Torvalds /* Fix endianness in ELF header */ 396ae4ac123SAtsushi Nemoto hdr->e_type = TO_NATIVE(hdr->e_type); 3977d875a02SAnders Kaseorg hdr->e_machine = TO_NATIVE(hdr->e_machine); 3987d875a02SAnders Kaseorg hdr->e_version = TO_NATIVE(hdr->e_version); 3997d875a02SAnders Kaseorg hdr->e_entry = TO_NATIVE(hdr->e_entry); 4007d875a02SAnders Kaseorg hdr->e_phoff = TO_NATIVE(hdr->e_phoff); 4017d875a02SAnders Kaseorg hdr->e_shoff = TO_NATIVE(hdr->e_shoff); 4027d875a02SAnders Kaseorg hdr->e_flags = TO_NATIVE(hdr->e_flags); 4037d875a02SAnders Kaseorg hdr->e_ehsize = TO_NATIVE(hdr->e_ehsize); 4047d875a02SAnders Kaseorg hdr->e_phentsize = TO_NATIVE(hdr->e_phentsize); 4057d875a02SAnders Kaseorg hdr->e_phnum = TO_NATIVE(hdr->e_phnum); 4067d875a02SAnders Kaseorg hdr->e_shentsize = TO_NATIVE(hdr->e_shentsize); 4077d875a02SAnders Kaseorg hdr->e_shnum = TO_NATIVE(hdr->e_shnum); 4087d875a02SAnders Kaseorg hdr->e_shstrndx = TO_NATIVE(hdr->e_shstrndx); 4091da177e4SLinus Torvalds sechdrs = (void *)hdr + hdr->e_shoff; 4101da177e4SLinus Torvalds info->sechdrs = sechdrs; 4111da177e4SLinus Torvalds 412a83710e5SPetr Stetiar /* Check if file offset is correct */ 413a83710e5SPetr Stetiar if (hdr->e_shoff > info->size) { 414df578e7dSSam Ravnborg fatal("section header offset=%lu in file '%s' is bigger than " 415df578e7dSSam Ravnborg "filesize=%lu\n", (unsigned long)hdr->e_shoff, 416df578e7dSSam Ravnborg filename, info->size); 417a83710e5SPetr Stetiar return 0; 418a83710e5SPetr Stetiar } 419a83710e5SPetr Stetiar 4201da177e4SLinus Torvalds /* Fix endianness in section headers */ 4211da177e4SLinus Torvalds for (i = 0; i < hdr->e_shnum; i++) { 4227d875a02SAnders Kaseorg sechdrs[i].sh_name = TO_NATIVE(sechdrs[i].sh_name); 4231da177e4SLinus Torvalds sechdrs[i].sh_type = TO_NATIVE(sechdrs[i].sh_type); 4247d875a02SAnders Kaseorg sechdrs[i].sh_flags = TO_NATIVE(sechdrs[i].sh_flags); 4257d875a02SAnders Kaseorg sechdrs[i].sh_addr = TO_NATIVE(sechdrs[i].sh_addr); 4261da177e4SLinus Torvalds sechdrs[i].sh_offset = TO_NATIVE(sechdrs[i].sh_offset); 4271da177e4SLinus Torvalds sechdrs[i].sh_size = TO_NATIVE(sechdrs[i].sh_size); 4281da177e4SLinus Torvalds sechdrs[i].sh_link = TO_NATIVE(sechdrs[i].sh_link); 429ae4ac123SAtsushi Nemoto sechdrs[i].sh_info = TO_NATIVE(sechdrs[i].sh_info); 4307d875a02SAnders Kaseorg sechdrs[i].sh_addralign = TO_NATIVE(sechdrs[i].sh_addralign); 4317d875a02SAnders Kaseorg sechdrs[i].sh_entsize = TO_NATIVE(sechdrs[i].sh_entsize); 4321da177e4SLinus Torvalds } 4331da177e4SLinus Torvalds /* Find symbol table. */ 4341da177e4SLinus Torvalds for (i = 1; i < hdr->e_shnum; i++) { 4351da177e4SLinus Torvalds const char *secstrings 4361da177e4SLinus Torvalds = (void *)hdr + sechdrs[hdr->e_shstrndx].sh_offset; 437bd5cbcedSRam Pai const char *secname; 43856fc82c5STejun Heo int nobits = sechdrs[i].sh_type == SHT_NOBITS; 4391da177e4SLinus Torvalds 44056fc82c5STejun Heo if (!nobits && sechdrs[i].sh_offset > info->size) { 441df578e7dSSam Ravnborg fatal("%s is truncated. sechdrs[i].sh_offset=%lu > " 442df578e7dSSam Ravnborg "sizeof(*hrd)=%zu\n", filename, 443df578e7dSSam Ravnborg (unsigned long)sechdrs[i].sh_offset, 444df578e7dSSam Ravnborg sizeof(*hdr)); 44585bd2fddSSam Ravnborg return 0; 44685bd2fddSSam Ravnborg } 447bd5cbcedSRam Pai secname = secstrings + sechdrs[i].sh_name; 448bd5cbcedSRam Pai if (strcmp(secname, ".modinfo") == 0) { 44956fc82c5STejun Heo if (nobits) 45056fc82c5STejun Heo fatal("%s has NOBITS .modinfo\n", filename); 4511da177e4SLinus Torvalds info->modinfo = (void *)hdr + sechdrs[i].sh_offset; 4521da177e4SLinus Torvalds info->modinfo_len = sechdrs[i].sh_size; 453bd5cbcedSRam Pai } else if (strcmp(secname, "__ksymtab") == 0) 454bd5cbcedSRam Pai info->export_sec = i; 455c96fca21SSam Ravnborg else if (strcmp(secname, "__ksymtab_unused") == 0) 456c96fca21SSam Ravnborg info->export_unused_sec = i; 457bd5cbcedSRam Pai else if (strcmp(secname, "__ksymtab_gpl") == 0) 458bd5cbcedSRam Pai info->export_gpl_sec = i; 459c96fca21SSam Ravnborg else if (strcmp(secname, "__ksymtab_unused_gpl") == 0) 460c96fca21SSam Ravnborg info->export_unused_gpl_sec = i; 461bd5cbcedSRam Pai else if (strcmp(secname, "__ksymtab_gpl_future") == 0) 462bd5cbcedSRam Pai info->export_gpl_future_sec = i; 463bd5cbcedSRam Pai 4641da177e4SLinus Torvalds if (sechdrs[i].sh_type != SHT_SYMTAB) 4651da177e4SLinus Torvalds continue; 4661da177e4SLinus Torvalds 4671da177e4SLinus Torvalds info->symtab_start = (void *)hdr + sechdrs[i].sh_offset; 4681da177e4SLinus Torvalds info->symtab_stop = (void *)hdr + sechdrs[i].sh_offset 4691da177e4SLinus Torvalds + sechdrs[i].sh_size; 4701da177e4SLinus Torvalds info->strtab = (void *)hdr + 4711da177e4SLinus Torvalds sechdrs[sechdrs[i].sh_link].sh_offset; 4721da177e4SLinus Torvalds } 473df578e7dSSam Ravnborg if (!info->symtab_start) 474cb80514dSSam Ravnborg fatal("%s has no symtab?\n", filename); 475df578e7dSSam Ravnborg 4761da177e4SLinus Torvalds /* Fix endianness in symbols */ 4771da177e4SLinus Torvalds for (sym = info->symtab_start; sym < info->symtab_stop; sym++) { 4781da177e4SLinus Torvalds sym->st_shndx = TO_NATIVE(sym->st_shndx); 4791da177e4SLinus Torvalds sym->st_name = TO_NATIVE(sym->st_name); 4801da177e4SLinus Torvalds sym->st_value = TO_NATIVE(sym->st_value); 4811da177e4SLinus Torvalds sym->st_size = TO_NATIVE(sym->st_size); 4821da177e4SLinus Torvalds } 48385bd2fddSSam Ravnborg return 1; 4841da177e4SLinus Torvalds } 4851da177e4SLinus Torvalds 4865c3ead8cSSam Ravnborg static void parse_elf_finish(struct elf_info *info) 4871da177e4SLinus Torvalds { 4881da177e4SLinus Torvalds release_file(info->hdr, info->size); 4891da177e4SLinus Torvalds } 4901da177e4SLinus Torvalds 4914d7365d6SSam Ravnborg static int ignore_undef_symbol(struct elf_info *info, const char *symname) 4924d7365d6SSam Ravnborg { 4934d7365d6SSam Ravnborg /* ignore __this_module, it will be resolved shortly */ 4944d7365d6SSam Ravnborg if (strcmp(symname, MODULE_SYMBOL_PREFIX "__this_module") == 0) 4954d7365d6SSam Ravnborg return 1; 4964d7365d6SSam Ravnborg /* ignore global offset table */ 4974d7365d6SSam Ravnborg if (strcmp(symname, "_GLOBAL_OFFSET_TABLE_") == 0) 4984d7365d6SSam Ravnborg return 1; 4994d7365d6SSam Ravnborg if (info->hdr->e_machine == EM_PPC) 5004d7365d6SSam Ravnborg /* Special register function linked on all modules during final link of .ko */ 5014d7365d6SSam Ravnborg if (strncmp(symname, "_restgpr_", sizeof("_restgpr_") - 1) == 0 || 5024d7365d6SSam Ravnborg strncmp(symname, "_savegpr_", sizeof("_savegpr_") - 1) == 0 || 5034d7365d6SSam Ravnborg strncmp(symname, "_rest32gpr_", sizeof("_rest32gpr_") - 1) == 0 || 5044d7365d6SSam Ravnborg strncmp(symname, "_save32gpr_", sizeof("_save32gpr_") - 1) == 0) 5054d7365d6SSam Ravnborg return 1; 5064d7365d6SSam Ravnborg /* Do not ignore this symbol */ 5074d7365d6SSam Ravnborg return 0; 5084d7365d6SSam Ravnborg } 5094d7365d6SSam Ravnborg 510f7b05e64SLuke Yang #define CRC_PFX MODULE_SYMBOL_PREFIX "__crc_" 511f7b05e64SLuke Yang #define KSYMTAB_PFX MODULE_SYMBOL_PREFIX "__ksymtab_" 5121da177e4SLinus Torvalds 5135c3ead8cSSam Ravnborg static void handle_modversions(struct module *mod, struct elf_info *info, 5141da177e4SLinus Torvalds Elf_Sym *sym, const char *symname) 5151da177e4SLinus Torvalds { 5161da177e4SLinus Torvalds unsigned int crc; 517bd5cbcedSRam Pai enum export export = export_from_sec(info, sym->st_shndx); 5181da177e4SLinus Torvalds 5191da177e4SLinus Torvalds switch (sym->st_shndx) { 5201da177e4SLinus Torvalds case SHN_COMMON: 521cb80514dSSam Ravnborg warn("\"%s\" [%s] is COMMON symbol\n", symname, mod->name); 5221da177e4SLinus Torvalds break; 5231da177e4SLinus Torvalds case SHN_ABS: 5241da177e4SLinus Torvalds /* CRC'd symbol */ 5258d99513cSMichal Marek if (strncmp(symname, CRC_PFX, strlen(CRC_PFX)) == 0) { 5261da177e4SLinus Torvalds crc = (unsigned int) sym->st_value; 527bd5cbcedSRam Pai sym_update_crc(symname + strlen(CRC_PFX), mod, crc, 528bd5cbcedSRam Pai export); 5291da177e4SLinus Torvalds } 5301da177e4SLinus Torvalds break; 5311da177e4SLinus Torvalds case SHN_UNDEF: 5321da177e4SLinus Torvalds /* undefined symbol */ 5331da177e4SLinus Torvalds if (ELF_ST_BIND(sym->st_info) != STB_GLOBAL && 5341da177e4SLinus Torvalds ELF_ST_BIND(sym->st_info) != STB_WEAK) 5351da177e4SLinus Torvalds break; 5364d7365d6SSam Ravnborg if (ignore_undef_symbol(info, symname)) 5371da177e4SLinus Torvalds break; 5388d529014SBen Colline /* cope with newer glibc (2.3.4 or higher) STT_ definition in elf.h */ 5398d529014SBen Colline #if defined(STT_REGISTER) || defined(STT_SPARC_REGISTER) 5408d529014SBen Colline /* add compatibility with older glibc */ 5418d529014SBen Colline #ifndef STT_SPARC_REGISTER 5428d529014SBen Colline #define STT_SPARC_REGISTER STT_REGISTER 5438d529014SBen Colline #endif 5441da177e4SLinus Torvalds if (info->hdr->e_machine == EM_SPARC || 5451da177e4SLinus Torvalds info->hdr->e_machine == EM_SPARCV9) { 5461da177e4SLinus Torvalds /* Ignore register directives. */ 5478d529014SBen Colline if (ELF_ST_TYPE(sym->st_info) == STT_SPARC_REGISTER) 5481da177e4SLinus Torvalds break; 5497caaeabbSAl Viro if (symname[0] == '.') { 5507caaeabbSAl Viro char *munged = strdup(symname); 5517caaeabbSAl Viro munged[0] = '_'; 5527caaeabbSAl Viro munged[1] = toupper(munged[1]); 5537caaeabbSAl Viro symname = munged; 5547caaeabbSAl Viro } 5551da177e4SLinus Torvalds } 5561da177e4SLinus Torvalds #endif 5571da177e4SLinus Torvalds 5581da177e4SLinus Torvalds if (memcmp(symname, MODULE_SYMBOL_PREFIX, 559df578e7dSSam Ravnborg strlen(MODULE_SYMBOL_PREFIX)) == 0) { 560df578e7dSSam Ravnborg mod->unres = 561df578e7dSSam Ravnborg alloc_symbol(symname + 5621da177e4SLinus Torvalds strlen(MODULE_SYMBOL_PREFIX), 5631da177e4SLinus Torvalds ELF_ST_BIND(sym->st_info) == STB_WEAK, 5641da177e4SLinus Torvalds mod->unres); 565df578e7dSSam Ravnborg } 5661da177e4SLinus Torvalds break; 5671da177e4SLinus Torvalds default: 5681da177e4SLinus Torvalds /* All exported symbols */ 5698d99513cSMichal Marek if (strncmp(symname, KSYMTAB_PFX, strlen(KSYMTAB_PFX)) == 0) { 570bd5cbcedSRam Pai sym_add_exported(symname + strlen(KSYMTAB_PFX), mod, 571bd5cbcedSRam Pai export); 5721da177e4SLinus Torvalds } 5731da177e4SLinus Torvalds if (strcmp(symname, MODULE_SYMBOL_PREFIX "init_module") == 0) 5741da177e4SLinus Torvalds mod->has_init = 1; 5751da177e4SLinus Torvalds if (strcmp(symname, MODULE_SYMBOL_PREFIX "cleanup_module") == 0) 5761da177e4SLinus Torvalds mod->has_cleanup = 1; 5771da177e4SLinus Torvalds break; 5781da177e4SLinus Torvalds } 5791da177e4SLinus Torvalds } 5801da177e4SLinus Torvalds 5815c3ead8cSSam Ravnborg /** 5825c3ead8cSSam Ravnborg * Parse tag=value strings from .modinfo section 5835c3ead8cSSam Ravnborg **/ 5841da177e4SLinus Torvalds static char *next_string(char *string, unsigned long *secsize) 5851da177e4SLinus Torvalds { 5861da177e4SLinus Torvalds /* Skip non-zero chars */ 5871da177e4SLinus Torvalds while (string[0]) { 5881da177e4SLinus Torvalds string++; 5891da177e4SLinus Torvalds if ((*secsize)-- <= 1) 5901da177e4SLinus Torvalds return NULL; 5911da177e4SLinus Torvalds } 5921da177e4SLinus Torvalds 5931da177e4SLinus Torvalds /* Skip any zero padding. */ 5941da177e4SLinus Torvalds while (!string[0]) { 5951da177e4SLinus Torvalds string++; 5961da177e4SLinus Torvalds if ((*secsize)-- <= 1) 5971da177e4SLinus Torvalds return NULL; 5981da177e4SLinus Torvalds } 5991da177e4SLinus Torvalds return string; 6001da177e4SLinus Torvalds } 6011da177e4SLinus Torvalds 602b817f6feSSam Ravnborg static char *get_next_modinfo(void *modinfo, unsigned long modinfo_len, 603b817f6feSSam Ravnborg const char *tag, char *info) 6041da177e4SLinus Torvalds { 6051da177e4SLinus Torvalds char *p; 6061da177e4SLinus Torvalds unsigned int taglen = strlen(tag); 6071da177e4SLinus Torvalds unsigned long size = modinfo_len; 6081da177e4SLinus Torvalds 609b817f6feSSam Ravnborg if (info) { 610b817f6feSSam Ravnborg size -= info - (char *)modinfo; 611b817f6feSSam Ravnborg modinfo = next_string(info, &size); 612b817f6feSSam Ravnborg } 613b817f6feSSam Ravnborg 6141da177e4SLinus Torvalds for (p = modinfo; p; p = next_string(p, &size)) { 6151da177e4SLinus Torvalds if (strncmp(p, tag, taglen) == 0 && p[taglen] == '=') 6161da177e4SLinus Torvalds return p + taglen + 1; 6171da177e4SLinus Torvalds } 6181da177e4SLinus Torvalds return NULL; 6191da177e4SLinus Torvalds } 6201da177e4SLinus Torvalds 621b817f6feSSam Ravnborg static char *get_modinfo(void *modinfo, unsigned long modinfo_len, 622b817f6feSSam Ravnborg const char *tag) 623b817f6feSSam Ravnborg 624b817f6feSSam Ravnborg { 625b817f6feSSam Ravnborg return get_next_modinfo(modinfo, modinfo_len, tag, NULL); 626b817f6feSSam Ravnborg } 627b817f6feSSam Ravnborg 62893684d3bSSam Ravnborg /** 6294c8fbca5SSam Ravnborg * Test if string s ends in string sub 6304c8fbca5SSam Ravnborg * return 0 if match 6314c8fbca5SSam Ravnborg **/ 6324c8fbca5SSam Ravnborg static int strrcmp(const char *s, const char *sub) 6334c8fbca5SSam Ravnborg { 6344c8fbca5SSam Ravnborg int slen, sublen; 6354c8fbca5SSam Ravnborg 6364c8fbca5SSam Ravnborg if (!s || !sub) 6374c8fbca5SSam Ravnborg return 1; 6384c8fbca5SSam Ravnborg 6394c8fbca5SSam Ravnborg slen = strlen(s); 6404c8fbca5SSam Ravnborg sublen = strlen(sub); 6414c8fbca5SSam Ravnborg 6424c8fbca5SSam Ravnborg if ((slen == 0) || (sublen == 0)) 6434c8fbca5SSam Ravnborg return 1; 6444c8fbca5SSam Ravnborg 6454c8fbca5SSam Ravnborg if (sublen > slen) 6464c8fbca5SSam Ravnborg return 1; 6474c8fbca5SSam Ravnborg 6484c8fbca5SSam Ravnborg return memcmp(s + slen - sublen, sub, sublen); 6494c8fbca5SSam Ravnborg } 6504c8fbca5SSam Ravnborg 651ff13f926SSam Ravnborg static const char *sym_name(struct elf_info *elf, Elf_Sym *sym) 652ff13f926SSam Ravnborg { 65358fb0d4fSSam Ravnborg if (sym) 654ff13f926SSam Ravnborg return elf->strtab + sym->st_name; 65558fb0d4fSSam Ravnborg else 656f666751aSSam Ravnborg return "(unknown)"; 657ff13f926SSam Ravnborg } 658ff13f926SSam Ravnborg 659ff13f926SSam Ravnborg static const char *sec_name(struct elf_info *elf, int shndx) 660ff13f926SSam Ravnborg { 661ff13f926SSam Ravnborg Elf_Shdr *sechdrs = elf->sechdrs; 662ff13f926SSam Ravnborg return (void *)elf->hdr + 663ff13f926SSam Ravnborg elf->sechdrs[elf->hdr->e_shstrndx].sh_offset + 664ff13f926SSam Ravnborg sechdrs[shndx].sh_name; 665ff13f926SSam Ravnborg } 666ff13f926SSam Ravnborg 667ff13f926SSam Ravnborg static const char *sech_name(struct elf_info *elf, Elf_Shdr *sechdr) 668ff13f926SSam Ravnborg { 669ff13f926SSam Ravnborg return (void *)elf->hdr + 670ff13f926SSam Ravnborg elf->sechdrs[elf->hdr->e_shstrndx].sh_offset + 671ff13f926SSam Ravnborg sechdr->sh_name; 672ff13f926SSam Ravnborg } 673ff13f926SSam Ravnborg 67410668220SSam Ravnborg /* if sym is empty or point to a string 67510668220SSam Ravnborg * like ".[0-9]+" then return 1. 67610668220SSam Ravnborg * This is the optional prefix added by ld to some sections 67710668220SSam Ravnborg */ 67810668220SSam Ravnborg static int number_prefix(const char *sym) 67910668220SSam Ravnborg { 68010668220SSam Ravnborg if (*sym++ == '\0') 68110668220SSam Ravnborg return 1; 68210668220SSam Ravnborg if (*sym != '.') 68310668220SSam Ravnborg return 0; 68410668220SSam Ravnborg do { 68510668220SSam Ravnborg char c = *sym++; 68610668220SSam Ravnborg if (c < '0' || c > '9') 68710668220SSam Ravnborg return 0; 68810668220SSam Ravnborg } while (*sym); 68910668220SSam Ravnborg return 1; 69010668220SSam Ravnborg } 69110668220SSam Ravnborg 69210668220SSam Ravnborg /* The pattern is an array of simple patterns. 69310668220SSam Ravnborg * "foo" will match an exact string equal to "foo" 6946c5bd235SSam Ravnborg * "*foo" will match a string that ends with "foo" 69510668220SSam Ravnborg * "foo*" will match a string that begins with "foo" 69610668220SSam Ravnborg * "foo$" will match a string equal to "foo" or "foo.1" 69710668220SSam Ravnborg * where the '1' can be any number including several digits. 69810668220SSam Ravnborg * The $ syntax is for sections where ld append a dot number 69910668220SSam Ravnborg * to make section name unique. 70010668220SSam Ravnborg */ 7015c725138STrevor Keith static int match(const char *sym, const char * const pat[]) 70210668220SSam Ravnborg { 70310668220SSam Ravnborg const char *p; 70410668220SSam Ravnborg while (*pat) { 70510668220SSam Ravnborg p = *pat++; 70610668220SSam Ravnborg const char *endp = p + strlen(p) - 1; 70710668220SSam Ravnborg 7086c5bd235SSam Ravnborg /* "*foo" */ 7096c5bd235SSam Ravnborg if (*p == '*') { 7106c5bd235SSam Ravnborg if (strrcmp(sym, p + 1) == 0) 7116c5bd235SSam Ravnborg return 1; 7126c5bd235SSam Ravnborg } 71310668220SSam Ravnborg /* "foo*" */ 7146c5bd235SSam Ravnborg else if (*endp == '*') { 71510668220SSam Ravnborg if (strncmp(sym, p, strlen(p) - 1) == 0) 71610668220SSam Ravnborg return 1; 71710668220SSam Ravnborg } 71810668220SSam Ravnborg /* "foo$" */ 71910668220SSam Ravnborg else if (*endp == '$') { 72010668220SSam Ravnborg if (strncmp(sym, p, strlen(p) - 1) == 0) { 72110668220SSam Ravnborg if (number_prefix(sym + strlen(p) - 1)) 72210668220SSam Ravnborg return 1; 72310668220SSam Ravnborg } 72410668220SSam Ravnborg } 72510668220SSam Ravnborg /* no wildcards */ 72610668220SSam Ravnborg else { 72710668220SSam Ravnborg if (strcmp(p, sym) == 0) 72810668220SSam Ravnborg return 1; 72910668220SSam Ravnborg } 73010668220SSam Ravnborg } 73110668220SSam Ravnborg /* no match */ 73210668220SSam Ravnborg return 0; 73310668220SSam Ravnborg } 73410668220SSam Ravnborg 73510668220SSam Ravnborg /* sections that we do not want to do full section mismatch check on */ 73610668220SSam Ravnborg static const char *section_white_list[] = 7374391ed6aSSam Ravnborg { 7384391ed6aSSam Ravnborg ".comment*", 7394391ed6aSSam Ravnborg ".debug*", 7404391ed6aSSam Ravnborg ".mdebug*", /* alpha, score, mips etc. */ 7414391ed6aSSam Ravnborg ".pdr", /* alpha, score, mips etc. */ 7424391ed6aSSam Ravnborg ".stab*", 7434391ed6aSSam Ravnborg ".note*", 7444391ed6aSSam Ravnborg ".got*", 7454391ed6aSSam Ravnborg ".toc*", 7464391ed6aSSam Ravnborg NULL 7474391ed6aSSam Ravnborg }; 74810668220SSam Ravnborg 749e241a630SSam Ravnborg /* 750b614a697SAnders Kaseorg * This is used to find sections missing the SHF_ALLOC flag. 751e241a630SSam Ravnborg * The cause of this is often a section specified in assembler 752b614a697SAnders Kaseorg * without "ax" / "aw". 753e241a630SSam Ravnborg */ 754b614a697SAnders Kaseorg static void check_section(const char *modname, struct elf_info *elf, 755b614a697SAnders Kaseorg Elf_Shdr *sechdr) 756e241a630SSam Ravnborg { 757b614a697SAnders Kaseorg const char *sec = sech_name(elf, sechdr); 758e241a630SSam Ravnborg 759b614a697SAnders Kaseorg if (sechdr->sh_type == SHT_PROGBITS && 760b614a697SAnders Kaseorg !(sechdr->sh_flags & SHF_ALLOC) && 761b614a697SAnders Kaseorg !match(sec, section_white_list)) { 762b614a697SAnders Kaseorg warn("%s (%s): unexpected non-allocatable section.\n" 763b614a697SAnders Kaseorg "Did you forget to use \"ax\"/\"aw\" in a .S file?\n" 764e241a630SSam Ravnborg "Note that for example <linux/init.h> contains\n" 765e241a630SSam Ravnborg "section definitions for use in .S files.\n\n", 766e241a630SSam Ravnborg modname, sec); 767e241a630SSam Ravnborg } 768e241a630SSam Ravnborg } 769e241a630SSam Ravnborg 770e241a630SSam Ravnborg 771e241a630SSam Ravnborg 772eb8f6890SSam Ravnborg #define ALL_INIT_DATA_SECTIONS \ 773fd6c3a8dSJan Beulich ".init.setup$", ".init.rodata$", \ 774fd6c3a8dSJan Beulich ".devinit.rodata$", ".cpuinit.rodata$", ".meminit.rodata$" \ 775eb8f6890SSam Ravnborg ".init.data$", ".devinit.data$", ".cpuinit.data$", ".meminit.data$" 776eb8f6890SSam Ravnborg #define ALL_EXIT_DATA_SECTIONS \ 777eb8f6890SSam Ravnborg ".exit.data$", ".devexit.data$", ".cpuexit.data$", ".memexit.data$" 77810668220SSam Ravnborg 779eb8f6890SSam Ravnborg #define ALL_INIT_TEXT_SECTIONS \ 780eb8f6890SSam Ravnborg ".init.text$", ".devinit.text$", ".cpuinit.text$", ".meminit.text$" 781eb8f6890SSam Ravnborg #define ALL_EXIT_TEXT_SECTIONS \ 782eb8f6890SSam Ravnborg ".exit.text$", ".devexit.text$", ".cpuexit.text$", ".memexit.text$" 78310668220SSam Ravnborg 7844a31a229SUwe Kleine-König #define ALL_XXXINIT_SECTIONS DEV_INIT_SECTIONS, CPU_INIT_SECTIONS, \ 7854a31a229SUwe Kleine-König MEM_INIT_SECTIONS 7864a31a229SUwe Kleine-König #define ALL_XXXEXIT_SECTIONS DEV_EXIT_SECTIONS, CPU_EXIT_SECTIONS, \ 7874a31a229SUwe Kleine-König MEM_EXIT_SECTIONS 7884a31a229SUwe Kleine-König 7894a31a229SUwe Kleine-König #define ALL_INIT_SECTIONS INIT_SECTIONS, ALL_XXXINIT_SECTIONS 7904a31a229SUwe Kleine-König #define ALL_EXIT_SECTIONS EXIT_SECTIONS, ALL_XXXEXIT_SECTIONS 79110668220SSam Ravnborg 7926c5bd235SSam Ravnborg #define DATA_SECTIONS ".data$", ".data.rel$" 79310668220SSam Ravnborg #define TEXT_SECTIONS ".text$" 79410668220SSam Ravnborg 795fd6c3a8dSJan Beulich #define INIT_SECTIONS ".init.*" 796fd6c3a8dSJan Beulich #define DEV_INIT_SECTIONS ".devinit.*" 797fd6c3a8dSJan Beulich #define CPU_INIT_SECTIONS ".cpuinit.*" 798fd6c3a8dSJan Beulich #define MEM_INIT_SECTIONS ".meminit.*" 799eb8f6890SSam Ravnborg 800fd6c3a8dSJan Beulich #define EXIT_SECTIONS ".exit.*" 801fd6c3a8dSJan Beulich #define DEV_EXIT_SECTIONS ".devexit.*" 802fd6c3a8dSJan Beulich #define CPU_EXIT_SECTIONS ".cpuexit.*" 803fd6c3a8dSJan Beulich #define MEM_EXIT_SECTIONS ".memexit.*" 804eb8f6890SSam Ravnborg 8056c5bd235SSam Ravnborg /* init data sections */ 806eb8f6890SSam Ravnborg static const char *init_data_sections[] = { ALL_INIT_DATA_SECTIONS, NULL }; 8076c5bd235SSam Ravnborg 8086c5bd235SSam Ravnborg /* all init sections */ 809eb8f6890SSam Ravnborg static const char *init_sections[] = { ALL_INIT_SECTIONS, NULL }; 8106c5bd235SSam Ravnborg 8116c5bd235SSam Ravnborg /* All init and exit sections (code + data) */ 8126c5bd235SSam Ravnborg static const char *init_exit_sections[] = 813eb8f6890SSam Ravnborg {ALL_INIT_SECTIONS, ALL_EXIT_SECTIONS, NULL }; 8146c5bd235SSam Ravnborg 8156c5bd235SSam Ravnborg /* data section */ 8166c5bd235SSam Ravnborg static const char *data_sections[] = { DATA_SECTIONS, NULL }; 8176c5bd235SSam Ravnborg 8186c5bd235SSam Ravnborg 8196c5bd235SSam Ravnborg /* symbols in .data that may refer to init/exit sections */ 8206c5bd235SSam Ravnborg static const char *symbol_white_list[] = 8216c5bd235SSam Ravnborg { 8226c5bd235SSam Ravnborg "*driver", 8236c5bd235SSam Ravnborg "*_template", /* scsi uses *_template a lot */ 8246c5bd235SSam Ravnborg "*_timer", /* arm uses ops structures named _timer a lot */ 8256c5bd235SSam Ravnborg "*_sht", /* scsi also used *_sht to some extent */ 8266c5bd235SSam Ravnborg "*_ops", 8276c5bd235SSam Ravnborg "*_probe", 8286c5bd235SSam Ravnborg "*_probe_one", 8296c5bd235SSam Ravnborg "*_console", 8306c5bd235SSam Ravnborg NULL 8316c5bd235SSam Ravnborg }; 8326c5bd235SSam Ravnborg 8336c5bd235SSam Ravnborg static const char *head_sections[] = { ".head.text*", NULL }; 8346c5bd235SSam Ravnborg static const char *linker_symbols[] = 8356c5bd235SSam Ravnborg { "__init_begin", "_sinittext", "_einittext", NULL }; 8366c5bd235SSam Ravnborg 837588ccd73SSam Ravnborg enum mismatch { 838588ccd73SSam Ravnborg NO_MISMATCH, 839*bbd3f4fbSUwe Kleine-König TEXT_TO_ANY_INIT, 840*bbd3f4fbSUwe Kleine-König DATA_TO_ANY_INIT, 841*bbd3f4fbSUwe Kleine-König TEXT_TO_ANY_EXIT, 842*bbd3f4fbSUwe Kleine-König DATA_TO_ANY_EXIT, 843*bbd3f4fbSUwe Kleine-König XXXINIT_TO_SOME_INIT, 844*bbd3f4fbSUwe Kleine-König XXXEXIT_TO_SOME_EXIT, 845*bbd3f4fbSUwe Kleine-König ANY_INIT_TO_ANY_EXIT, 846*bbd3f4fbSUwe Kleine-König ANY_EXIT_TO_ANY_INIT, 847588ccd73SSam Ravnborg EXPORT_TO_INIT_EXIT, 848588ccd73SSam Ravnborg }; 849588ccd73SSam Ravnborg 85010668220SSam Ravnborg struct sectioncheck { 85110668220SSam Ravnborg const char *fromsec[20]; 85210668220SSam Ravnborg const char *tosec[20]; 853588ccd73SSam Ravnborg enum mismatch mismatch; 85410668220SSam Ravnborg }; 85510668220SSam Ravnborg 85610668220SSam Ravnborg const struct sectioncheck sectioncheck[] = { 85710668220SSam Ravnborg /* Do not reference init/exit code/data from 85810668220SSam Ravnborg * normal code and data 85910668220SSam Ravnborg */ 86010668220SSam Ravnborg { 861588ccd73SSam Ravnborg .fromsec = { TEXT_SECTIONS, NULL }, 862588ccd73SSam Ravnborg .tosec = { ALL_INIT_SECTIONS, NULL }, 863*bbd3f4fbSUwe Kleine-König .mismatch = TEXT_TO_ANY_INIT, 864588ccd73SSam Ravnborg }, 865588ccd73SSam Ravnborg { 866588ccd73SSam Ravnborg .fromsec = { DATA_SECTIONS, NULL }, 867588ccd73SSam Ravnborg .tosec = { ALL_INIT_SECTIONS, NULL }, 868*bbd3f4fbSUwe Kleine-König .mismatch = DATA_TO_ANY_INIT, 869588ccd73SSam Ravnborg }, 870588ccd73SSam Ravnborg { 871588ccd73SSam Ravnborg .fromsec = { TEXT_SECTIONS, NULL }, 872588ccd73SSam Ravnborg .tosec = { ALL_EXIT_SECTIONS, NULL }, 873*bbd3f4fbSUwe Kleine-König .mismatch = TEXT_TO_ANY_EXIT, 874588ccd73SSam Ravnborg }, 875588ccd73SSam Ravnborg { 876588ccd73SSam Ravnborg .fromsec = { DATA_SECTIONS, NULL }, 877588ccd73SSam Ravnborg .tosec = { ALL_EXIT_SECTIONS, NULL }, 878*bbd3f4fbSUwe Kleine-König .mismatch = DATA_TO_ANY_EXIT, 879eb8f6890SSam Ravnborg }, 880eb8f6890SSam Ravnborg /* Do not reference init code/data from devinit/cpuinit/meminit code/data */ 881eb8f6890SSam Ravnborg { 8824a31a229SUwe Kleine-König .fromsec = { ALL_XXXINIT_SECTIONS, NULL }, 883588ccd73SSam Ravnborg .tosec = { INIT_SECTIONS, NULL }, 884*bbd3f4fbSUwe Kleine-König .mismatch = XXXINIT_TO_SOME_INIT, 885eb8f6890SSam Ravnborg }, 886fd6c3a8dSJan Beulich /* Do not reference cpuinit code/data from meminit code/data */ 887fd6c3a8dSJan Beulich { 888fd6c3a8dSJan Beulich .fromsec = { MEM_INIT_SECTIONS, NULL }, 889fd6c3a8dSJan Beulich .tosec = { CPU_INIT_SECTIONS, NULL }, 890*bbd3f4fbSUwe Kleine-König .mismatch = XXXINIT_TO_SOME_INIT, 891fd6c3a8dSJan Beulich }, 892fd6c3a8dSJan Beulich /* Do not reference meminit code/data from cpuinit code/data */ 893fd6c3a8dSJan Beulich { 894fd6c3a8dSJan Beulich .fromsec = { CPU_INIT_SECTIONS, NULL }, 895fd6c3a8dSJan Beulich .tosec = { MEM_INIT_SECTIONS, NULL }, 896*bbd3f4fbSUwe Kleine-König .mismatch = XXXINIT_TO_SOME_INIT, 897fd6c3a8dSJan Beulich }, 898eb8f6890SSam Ravnborg /* Do not reference exit code/data from devexit/cpuexit/memexit code/data */ 899eb8f6890SSam Ravnborg { 9004a31a229SUwe Kleine-König .fromsec = { ALL_XXXEXIT_SECTIONS, NULL }, 901588ccd73SSam Ravnborg .tosec = { EXIT_SECTIONS, NULL }, 902*bbd3f4fbSUwe Kleine-König .mismatch = XXXEXIT_TO_SOME_EXIT, 90310668220SSam Ravnborg }, 904fd6c3a8dSJan Beulich /* Do not reference cpuexit code/data from memexit code/data */ 905fd6c3a8dSJan Beulich { 906fd6c3a8dSJan Beulich .fromsec = { MEM_EXIT_SECTIONS, NULL }, 907fd6c3a8dSJan Beulich .tosec = { CPU_EXIT_SECTIONS, NULL }, 908*bbd3f4fbSUwe Kleine-König .mismatch = XXXEXIT_TO_SOME_EXIT, 909fd6c3a8dSJan Beulich }, 910fd6c3a8dSJan Beulich /* Do not reference memexit code/data from cpuexit code/data */ 911fd6c3a8dSJan Beulich { 912fd6c3a8dSJan Beulich .fromsec = { CPU_EXIT_SECTIONS, NULL }, 913fd6c3a8dSJan Beulich .tosec = { MEM_EXIT_SECTIONS, NULL }, 914*bbd3f4fbSUwe Kleine-König .mismatch = XXXEXIT_TO_SOME_EXIT, 915fd6c3a8dSJan Beulich }, 91610668220SSam Ravnborg /* Do not use exit code/data from init code */ 91710668220SSam Ravnborg { 918eb8f6890SSam Ravnborg .fromsec = { ALL_INIT_SECTIONS, NULL }, 919eb8f6890SSam Ravnborg .tosec = { ALL_EXIT_SECTIONS, NULL }, 920*bbd3f4fbSUwe Kleine-König .mismatch = ANY_INIT_TO_ANY_EXIT, 92110668220SSam Ravnborg }, 92210668220SSam Ravnborg /* Do not use init code/data from exit code */ 92310668220SSam Ravnborg { 924eb8f6890SSam Ravnborg .fromsec = { ALL_EXIT_SECTIONS, NULL }, 925588ccd73SSam Ravnborg .tosec = { ALL_INIT_SECTIONS, NULL }, 926*bbd3f4fbSUwe Kleine-König .mismatch = ANY_EXIT_TO_ANY_INIT, 92710668220SSam Ravnborg }, 92810668220SSam Ravnborg /* Do not export init/exit functions or data */ 92910668220SSam Ravnborg { 93010668220SSam Ravnborg .fromsec = { "__ksymtab*", NULL }, 931fa95eb1fSSam Ravnborg .tosec = { INIT_SECTIONS, EXIT_SECTIONS, NULL }, 932588ccd73SSam Ravnborg .mismatch = EXPORT_TO_INIT_EXIT 93310668220SSam Ravnborg } 93410668220SSam Ravnborg }; 93510668220SSam Ravnborg 93610668220SSam Ravnborg static int section_mismatch(const char *fromsec, const char *tosec) 93710668220SSam Ravnborg { 93810668220SSam Ravnborg int i; 93910668220SSam Ravnborg int elems = sizeof(sectioncheck) / sizeof(struct sectioncheck); 94010668220SSam Ravnborg const struct sectioncheck *check = §ioncheck[0]; 94110668220SSam Ravnborg 94210668220SSam Ravnborg for (i = 0; i < elems; i++) { 94310668220SSam Ravnborg if (match(fromsec, check->fromsec) && 94410668220SSam Ravnborg match(tosec, check->tosec)) 945588ccd73SSam Ravnborg return check->mismatch; 94610668220SSam Ravnborg check++; 94710668220SSam Ravnborg } 948588ccd73SSam Ravnborg return NO_MISMATCH; 94910668220SSam Ravnborg } 95010668220SSam Ravnborg 9514c8fbca5SSam Ravnborg /** 9524c8fbca5SSam Ravnborg * Whitelist to allow certain references to pass with no warning. 9530e0d314eSSam Ravnborg * 9544c8fbca5SSam Ravnborg * Pattern 1: 9554c8fbca5SSam Ravnborg * If a module parameter is declared __initdata and permissions=0 9564c8fbca5SSam Ravnborg * then this is legal despite the warning generated. 9574c8fbca5SSam Ravnborg * We cannot see value of permissions here, so just ignore 9584c8fbca5SSam Ravnborg * this pattern. 9594c8fbca5SSam Ravnborg * The pattern is identified by: 9604c8fbca5SSam Ravnborg * tosec = .init.data 9619209aed0SSam Ravnborg * fromsec = .data* 9624c8fbca5SSam Ravnborg * atsym =__param* 9634c8fbca5SSam Ravnborg * 9644c8fbca5SSam Ravnborg * Pattern 2: 96572ee59b5SRandy Dunlap * Many drivers utilise a *driver container with references to 9664c8fbca5SSam Ravnborg * add, remove, probe functions etc. 967b75dcabdSUwe Kleine-König * These functions may often be marked __devinit and we do not want to 9684c8fbca5SSam Ravnborg * warn here. 9694c8fbca5SSam Ravnborg * the pattern is identified by: 97083cda2bbSSam Ravnborg * tosec = init or exit section 97183cda2bbSSam Ravnborg * fromsec = data section 972df578e7dSSam Ravnborg * atsym = *driver, *_template, *_sht, *_ops, *_probe, 973df578e7dSSam Ravnborg * *probe_one, *_console, *_timer 974ee6a8545SVivek Goyal * 975ee6a8545SVivek Goyal * Pattern 3: 976c993971fSSam Ravnborg * Whitelist all references from .head.text to any init section 9779bf8cb9bSSam Ravnborg * 9781d8af559SSam Ravnborg * Pattern 4: 979ee6a8545SVivek Goyal * Some symbols belong to init section but still it is ok to reference 980ee6a8545SVivek Goyal * these from non-init sections as these symbols don't have any memory 981ee6a8545SVivek Goyal * allocated for them and symbol address and value are same. So even 982ee6a8545SVivek Goyal * if init section is freed, its ok to reference those symbols. 983ee6a8545SVivek Goyal * For ex. symbols marking the init section boundaries. 984ee6a8545SVivek Goyal * This pattern is identified by 985ee6a8545SVivek Goyal * refsymname = __init_begin, _sinittext, _einittext 9869bf8cb9bSSam Ravnborg * 9874c8fbca5SSam Ravnborg **/ 98858fb0d4fSSam Ravnborg static int secref_whitelist(const char *fromsec, const char *fromsym, 98958fb0d4fSSam Ravnborg const char *tosec, const char *tosym) 9904c8fbca5SSam Ravnborg { 9914c8fbca5SSam Ravnborg /* Check for pattern 1 */ 9926c5bd235SSam Ravnborg if (match(tosec, init_data_sections) && 9936c5bd235SSam Ravnborg match(fromsec, data_sections) && 99458fb0d4fSSam Ravnborg (strncmp(fromsym, "__param", strlen("__param")) == 0)) 99558fb0d4fSSam Ravnborg return 0; 9964c8fbca5SSam Ravnborg 9974c8fbca5SSam Ravnborg /* Check for pattern 2 */ 9986c5bd235SSam Ravnborg if (match(tosec, init_exit_sections) && 9996c5bd235SSam Ravnborg match(fromsec, data_sections) && 100058fb0d4fSSam Ravnborg match(fromsym, symbol_white_list)) 100158fb0d4fSSam Ravnborg return 0; 10024c8fbca5SSam Ravnborg 10039bf8cb9bSSam Ravnborg /* Check for pattern 3 */ 10046c5bd235SSam Ravnborg if (match(fromsec, head_sections) && 10056c5bd235SSam Ravnborg match(tosec, init_sections)) 100658fb0d4fSSam Ravnborg return 0; 1007f8657e1bSVivek Goyal 10081d8af559SSam Ravnborg /* Check for pattern 4 */ 100958fb0d4fSSam Ravnborg if (match(tosym, linker_symbols)) 101093659af1SSam Ravnborg return 0; 101158fb0d4fSSam Ravnborg 101258fb0d4fSSam Ravnborg return 1; 10134c8fbca5SSam Ravnborg } 10144c8fbca5SSam Ravnborg 10154c8fbca5SSam Ravnborg /** 101693684d3bSSam Ravnborg * Find symbol based on relocation record info. 101793684d3bSSam Ravnborg * In some cases the symbol supplied is a valid symbol so 101893684d3bSSam Ravnborg * return refsym. If st_name != 0 we assume this is a valid symbol. 101993684d3bSSam Ravnborg * In other cases the symbol needs to be looked up in the symbol table 102093684d3bSSam Ravnborg * based on section and address. 102193684d3bSSam Ravnborg * **/ 10229ad21c3fSSam Ravnborg static Elf_Sym *find_elf_symbol(struct elf_info *elf, Elf64_Sword addr, 102393684d3bSSam Ravnborg Elf_Sym *relsym) 102493684d3bSSam Ravnborg { 102593684d3bSSam Ravnborg Elf_Sym *sym; 10269ad21c3fSSam Ravnborg Elf_Sym *near = NULL; 10279ad21c3fSSam Ravnborg Elf64_Sword distance = 20; 10289ad21c3fSSam Ravnborg Elf64_Sword d; 102993684d3bSSam Ravnborg 103093684d3bSSam Ravnborg if (relsym->st_name != 0) 103193684d3bSSam Ravnborg return relsym; 103293684d3bSSam Ravnborg for (sym = elf->symtab_start; sym < elf->symtab_stop; sym++) { 103393684d3bSSam Ravnborg if (sym->st_shndx != relsym->st_shndx) 103493684d3bSSam Ravnborg continue; 1035ae4ac123SAtsushi Nemoto if (ELF_ST_TYPE(sym->st_info) == STT_SECTION) 1036ae4ac123SAtsushi Nemoto continue; 103793684d3bSSam Ravnborg if (sym->st_value == addr) 103893684d3bSSam Ravnborg return sym; 10399ad21c3fSSam Ravnborg /* Find a symbol nearby - addr are maybe negative */ 10409ad21c3fSSam Ravnborg d = sym->st_value - addr; 10419ad21c3fSSam Ravnborg if (d < 0) 10429ad21c3fSSam Ravnborg d = addr - sym->st_value; 10439ad21c3fSSam Ravnborg if (d < distance) { 10449ad21c3fSSam Ravnborg distance = d; 10459ad21c3fSSam Ravnborg near = sym; 104693684d3bSSam Ravnborg } 10479ad21c3fSSam Ravnborg } 10489ad21c3fSSam Ravnborg /* We need a close match */ 10499ad21c3fSSam Ravnborg if (distance < 20) 10509ad21c3fSSam Ravnborg return near; 10519ad21c3fSSam Ravnborg else 105293684d3bSSam Ravnborg return NULL; 105393684d3bSSam Ravnborg } 105493684d3bSSam Ravnborg 1055da68d61fSDavid Brownell static inline int is_arm_mapping_symbol(const char *str) 1056da68d61fSDavid Brownell { 1057da68d61fSDavid Brownell return str[0] == '$' && strchr("atd", str[1]) 1058da68d61fSDavid Brownell && (str[2] == '\0' || str[2] == '.'); 1059da68d61fSDavid Brownell } 1060da68d61fSDavid Brownell 1061da68d61fSDavid Brownell /* 1062da68d61fSDavid Brownell * If there's no name there, ignore it; likewise, ignore it if it's 1063da68d61fSDavid Brownell * one of the magic symbols emitted used by current ARM tools. 1064da68d61fSDavid Brownell * 1065da68d61fSDavid Brownell * Otherwise if find_symbols_between() returns those symbols, they'll 1066da68d61fSDavid Brownell * fail the whitelist tests and cause lots of false alarms ... fixable 1067da68d61fSDavid Brownell * only by merging __exit and __init sections into __text, bloating 1068da68d61fSDavid Brownell * the kernel (which is especially evil on embedded platforms). 1069da68d61fSDavid Brownell */ 1070da68d61fSDavid Brownell static inline int is_valid_name(struct elf_info *elf, Elf_Sym *sym) 1071da68d61fSDavid Brownell { 1072da68d61fSDavid Brownell const char *name = elf->strtab + sym->st_name; 1073da68d61fSDavid Brownell 1074da68d61fSDavid Brownell if (!name || !strlen(name)) 1075da68d61fSDavid Brownell return 0; 1076da68d61fSDavid Brownell return !is_arm_mapping_symbol(name); 1077da68d61fSDavid Brownell } 1078da68d61fSDavid Brownell 1079b39927cfSSam Ravnborg /* 108043c74d17SSam Ravnborg * Find symbols before or equal addr and after addr - in the section sec. 108143c74d17SSam Ravnborg * If we find two symbols with equal offset prefer one with a valid name. 108243c74d17SSam Ravnborg * The ELF format may have a better way to detect what type of symbol 108343c74d17SSam Ravnborg * it is, but this works for now. 1084b39927cfSSam Ravnborg **/ 1085157c23c8SSam Ravnborg static Elf_Sym *find_elf_symbol2(struct elf_info *elf, Elf_Addr addr, 1086157c23c8SSam Ravnborg const char *sec) 1087b39927cfSSam Ravnborg { 1088b39927cfSSam Ravnborg Elf_Sym *sym; 1089157c23c8SSam Ravnborg Elf_Sym *near = NULL; 1090157c23c8SSam Ravnborg Elf_Addr distance = ~0; 1091b39927cfSSam Ravnborg 1092b39927cfSSam Ravnborg for (sym = elf->symtab_start; sym < elf->symtab_stop; sym++) { 1093b39927cfSSam Ravnborg const char *symsec; 1094b39927cfSSam Ravnborg 1095b39927cfSSam Ravnborg if (sym->st_shndx >= SHN_LORESERVE) 1096b39927cfSSam Ravnborg continue; 1097ff13f926SSam Ravnborg symsec = sec_name(elf, sym->st_shndx); 1098b39927cfSSam Ravnborg if (strcmp(symsec, sec) != 0) 1099b39927cfSSam Ravnborg continue; 1100da68d61fSDavid Brownell if (!is_valid_name(elf, sym)) 1101da68d61fSDavid Brownell continue; 1102b39927cfSSam Ravnborg if (sym->st_value <= addr) { 1103157c23c8SSam Ravnborg if ((addr - sym->st_value) < distance) { 1104157c23c8SSam Ravnborg distance = addr - sym->st_value; 1105157c23c8SSam Ravnborg near = sym; 1106157c23c8SSam Ravnborg } else if ((addr - sym->st_value) == distance) { 1107157c23c8SSam Ravnborg near = sym; 110843c74d17SSam Ravnborg } 1109b39927cfSSam Ravnborg } 1110b39927cfSSam Ravnborg } 1111157c23c8SSam Ravnborg return near; 1112157c23c8SSam Ravnborg } 1113b39927cfSSam Ravnborg 111458fb0d4fSSam Ravnborg /* 1115588ccd73SSam Ravnborg * Convert a section name to the function/data attribute 1116588ccd73SSam Ravnborg * .init.text => __init 1117588ccd73SSam Ravnborg * .cpuinit.data => __cpudata 1118588ccd73SSam Ravnborg * .memexitconst => __memconst 1119588ccd73SSam Ravnborg * etc. 1120588ccd73SSam Ravnborg */ 1121588ccd73SSam Ravnborg static char *sec2annotation(const char *s) 1122588ccd73SSam Ravnborg { 1123588ccd73SSam Ravnborg if (match(s, init_exit_sections)) { 1124588ccd73SSam Ravnborg char *p = malloc(20); 1125588ccd73SSam Ravnborg char *r = p; 1126588ccd73SSam Ravnborg 1127588ccd73SSam Ravnborg *p++ = '_'; 1128588ccd73SSam Ravnborg *p++ = '_'; 1129588ccd73SSam Ravnborg if (*s == '.') 1130588ccd73SSam Ravnborg s++; 1131588ccd73SSam Ravnborg while (*s && *s != '.') 1132588ccd73SSam Ravnborg *p++ = *s++; 1133588ccd73SSam Ravnborg *p = '\0'; 1134588ccd73SSam Ravnborg if (*s == '.') 1135588ccd73SSam Ravnborg s++; 1136588ccd73SSam Ravnborg if (strstr(s, "rodata") != NULL) 1137588ccd73SSam Ravnborg strcat(p, "const "); 1138588ccd73SSam Ravnborg else if (strstr(s, "data") != NULL) 1139588ccd73SSam Ravnborg strcat(p, "data "); 1140588ccd73SSam Ravnborg else 1141588ccd73SSam Ravnborg strcat(p, " "); 1142588ccd73SSam Ravnborg return r; /* we leak her but we do not care */ 1143588ccd73SSam Ravnborg } else { 1144588ccd73SSam Ravnborg return ""; 1145588ccd73SSam Ravnborg } 1146588ccd73SSam Ravnborg } 1147588ccd73SSam Ravnborg 1148588ccd73SSam Ravnborg static int is_function(Elf_Sym *sym) 1149588ccd73SSam Ravnborg { 1150588ccd73SSam Ravnborg if (sym) 1151588ccd73SSam Ravnborg return ELF_ST_TYPE(sym->st_info) == STT_FUNC; 1152588ccd73SSam Ravnborg else 1153f666751aSSam Ravnborg return -1; 1154588ccd73SSam Ravnborg } 1155588ccd73SSam Ravnborg 1156588ccd73SSam Ravnborg /* 1157b39927cfSSam Ravnborg * Print a warning about a section mismatch. 1158b39927cfSSam Ravnborg * Try to find symbols near it so user can find it. 11594c8fbca5SSam Ravnborg * Check whitelist before warning - it may be a false positive. 116058fb0d4fSSam Ravnborg */ 1161588ccd73SSam Ravnborg static void report_sec_mismatch(const char *modname, enum mismatch mismatch, 116258fb0d4fSSam Ravnborg const char *fromsec, 116358fb0d4fSSam Ravnborg unsigned long long fromaddr, 116458fb0d4fSSam Ravnborg const char *fromsym, 1165588ccd73SSam Ravnborg int from_is_func, 1166588ccd73SSam Ravnborg const char *tosec, const char *tosym, 1167588ccd73SSam Ravnborg int to_is_func) 1168b39927cfSSam Ravnborg { 1169588ccd73SSam Ravnborg const char *from, *from_p; 1170588ccd73SSam Ravnborg const char *to, *to_p; 1171f666751aSSam Ravnborg 1172f666751aSSam Ravnborg switch (from_is_func) { 1173f666751aSSam Ravnborg case 0: from = "variable"; from_p = ""; break; 1174f666751aSSam Ravnborg case 1: from = "function"; from_p = "()"; break; 1175f666751aSSam Ravnborg default: from = "(unknown reference)"; from_p = ""; break; 1176f666751aSSam Ravnborg } 1177f666751aSSam Ravnborg switch (to_is_func) { 1178f666751aSSam Ravnborg case 0: to = "variable"; to_p = ""; break; 1179f666751aSSam Ravnborg case 1: to = "function"; to_p = "()"; break; 1180f666751aSSam Ravnborg default: to = "(unknown reference)"; to_p = ""; break; 1181f666751aSSam Ravnborg } 1182588ccd73SSam Ravnborg 1183e5f95c8bSSam Ravnborg sec_mismatch_count++; 1184e5f95c8bSSam Ravnborg if (!sec_mismatch_verbose) 1185e5f95c8bSSam Ravnborg return; 1186e5f95c8bSSam Ravnborg 11877c0ac495SGeert Uytterhoeven warn("%s(%s+0x%llx): Section mismatch in reference from the %s %s%s " 11887c0ac495SGeert Uytterhoeven "to the %s %s:%s%s\n", 11897c0ac495SGeert Uytterhoeven modname, fromsec, fromaddr, from, fromsym, from_p, to, tosec, 11907c0ac495SGeert Uytterhoeven tosym, to_p); 1191588ccd73SSam Ravnborg 1192588ccd73SSam Ravnborg switch (mismatch) { 1193*bbd3f4fbSUwe Kleine-König case TEXT_TO_ANY_INIT: 1194588ccd73SSam Ravnborg fprintf(stderr, 1195588ccd73SSam Ravnborg "The function %s%s() references\n" 1196588ccd73SSam Ravnborg "the %s %s%s%s.\n" 1197588ccd73SSam Ravnborg "This is often because %s lacks a %s\n" 1198588ccd73SSam Ravnborg "annotation or the annotation of %s is wrong.\n", 1199588ccd73SSam Ravnborg sec2annotation(fromsec), fromsym, 1200588ccd73SSam Ravnborg to, sec2annotation(tosec), tosym, to_p, 1201588ccd73SSam Ravnborg fromsym, sec2annotation(tosec), tosym); 1202588ccd73SSam Ravnborg break; 1203*bbd3f4fbSUwe Kleine-König case DATA_TO_ANY_INIT: { 1204588ccd73SSam Ravnborg const char **s = symbol_white_list; 1205588ccd73SSam Ravnborg fprintf(stderr, 1206588ccd73SSam Ravnborg "The variable %s references\n" 1207588ccd73SSam Ravnborg "the %s %s%s%s\n" 1208588ccd73SSam Ravnborg "If the reference is valid then annotate the\n" 12098b8b76c0SSam Ravnborg "variable with __init* or __refdata (see linux/init.h) " 1210588ccd73SSam Ravnborg "or name the variable:\n", 1211588ccd73SSam Ravnborg fromsym, to, sec2annotation(tosec), tosym, to_p); 1212588ccd73SSam Ravnborg while (*s) 1213588ccd73SSam Ravnborg fprintf(stderr, "%s, ", *s++); 1214588ccd73SSam Ravnborg fprintf(stderr, "\n"); 1215588ccd73SSam Ravnborg break; 121658fb0d4fSSam Ravnborg } 1217*bbd3f4fbSUwe Kleine-König case TEXT_TO_ANY_EXIT: 1218588ccd73SSam Ravnborg fprintf(stderr, 1219588ccd73SSam Ravnborg "The function %s() references a %s in an exit section.\n" 1220588ccd73SSam Ravnborg "Often the %s %s%s has valid usage outside the exit section\n" 1221588ccd73SSam Ravnborg "and the fix is to remove the %sannotation of %s.\n", 1222588ccd73SSam Ravnborg fromsym, to, to, tosym, to_p, sec2annotation(tosec), tosym); 1223588ccd73SSam Ravnborg break; 1224*bbd3f4fbSUwe Kleine-König case DATA_TO_ANY_EXIT: { 1225588ccd73SSam Ravnborg const char **s = symbol_white_list; 1226588ccd73SSam Ravnborg fprintf(stderr, 1227588ccd73SSam Ravnborg "The variable %s references\n" 1228588ccd73SSam Ravnborg "the %s %s%s%s\n" 1229588ccd73SSam Ravnborg "If the reference is valid then annotate the\n" 1230588ccd73SSam Ravnborg "variable with __exit* (see linux/init.h) or " 1231588ccd73SSam Ravnborg "name the variable:\n", 1232588ccd73SSam Ravnborg fromsym, to, sec2annotation(tosec), tosym, to_p); 1233588ccd73SSam Ravnborg while (*s) 1234588ccd73SSam Ravnborg fprintf(stderr, "%s, ", *s++); 1235588ccd73SSam Ravnborg fprintf(stderr, "\n"); 1236588ccd73SSam Ravnborg break; 1237588ccd73SSam Ravnborg } 1238*bbd3f4fbSUwe Kleine-König case XXXINIT_TO_SOME_INIT: 1239*bbd3f4fbSUwe Kleine-König case XXXEXIT_TO_SOME_EXIT: 1240588ccd73SSam Ravnborg fprintf(stderr, 1241588ccd73SSam Ravnborg "The %s %s%s%s references\n" 1242588ccd73SSam Ravnborg "a %s %s%s%s.\n" 1243588ccd73SSam Ravnborg "If %s is only used by %s then\n" 1244588ccd73SSam Ravnborg "annotate %s with a matching annotation.\n", 1245588ccd73SSam Ravnborg from, sec2annotation(fromsec), fromsym, from_p, 1246588ccd73SSam Ravnborg to, sec2annotation(tosec), tosym, to_p, 1247b1d2675aSGeert Uytterhoeven tosym, fromsym, tosym); 1248588ccd73SSam Ravnborg break; 1249*bbd3f4fbSUwe Kleine-König case ANY_INIT_TO_ANY_EXIT: 1250588ccd73SSam Ravnborg fprintf(stderr, 1251588ccd73SSam Ravnborg "The %s %s%s%s references\n" 1252588ccd73SSam Ravnborg "a %s %s%s%s.\n" 1253588ccd73SSam Ravnborg "This is often seen when error handling " 1254588ccd73SSam Ravnborg "in the init function\n" 1255588ccd73SSam Ravnborg "uses functionality in the exit path.\n" 1256588ccd73SSam Ravnborg "The fix is often to remove the %sannotation of\n" 1257588ccd73SSam Ravnborg "%s%s so it may be used outside an exit section.\n", 1258588ccd73SSam Ravnborg from, sec2annotation(fromsec), fromsym, from_p, 1259588ccd73SSam Ravnborg to, sec2annotation(tosec), tosym, to_p, 1260588ccd73SSam Ravnborg sec2annotation(tosec), tosym, to_p); 1261588ccd73SSam Ravnborg break; 1262*bbd3f4fbSUwe Kleine-König case ANY_EXIT_TO_ANY_INIT: 1263588ccd73SSam Ravnborg fprintf(stderr, 1264588ccd73SSam Ravnborg "The %s %s%s%s references\n" 1265588ccd73SSam Ravnborg "a %s %s%s%s.\n" 1266588ccd73SSam Ravnborg "This is often seen when error handling " 1267588ccd73SSam Ravnborg "in the exit function\n" 1268588ccd73SSam Ravnborg "uses functionality in the init path.\n" 1269588ccd73SSam Ravnborg "The fix is often to remove the %sannotation of\n" 1270588ccd73SSam Ravnborg "%s%s so it may be used outside an init section.\n", 1271588ccd73SSam Ravnborg from, sec2annotation(fromsec), fromsym, from_p, 1272588ccd73SSam Ravnborg to, sec2annotation(tosec), tosym, to_p, 1273588ccd73SSam Ravnborg sec2annotation(tosec), tosym, to_p); 1274588ccd73SSam Ravnborg break; 1275588ccd73SSam Ravnborg case EXPORT_TO_INIT_EXIT: 1276588ccd73SSam Ravnborg fprintf(stderr, 1277588ccd73SSam Ravnborg "The symbol %s is exported and annotated %s\n" 1278588ccd73SSam Ravnborg "Fix this by removing the %sannotation of %s " 1279588ccd73SSam Ravnborg "or drop the export.\n", 1280588ccd73SSam Ravnborg tosym, sec2annotation(tosec), sec2annotation(tosec), tosym); 1281588ccd73SSam Ravnborg case NO_MISMATCH: 1282588ccd73SSam Ravnborg /* To get warnings on missing members */ 1283588ccd73SSam Ravnborg break; 1284588ccd73SSam Ravnborg } 1285588ccd73SSam Ravnborg fprintf(stderr, "\n"); 128658fb0d4fSSam Ravnborg } 128758fb0d4fSSam Ravnborg 128858fb0d4fSSam Ravnborg static void check_section_mismatch(const char *modname, struct elf_info *elf, 128958fb0d4fSSam Ravnborg Elf_Rela *r, Elf_Sym *sym, const char *fromsec) 129058fb0d4fSSam Ravnborg { 129158fb0d4fSSam Ravnborg const char *tosec; 1292588ccd73SSam Ravnborg enum mismatch mismatch; 129358fb0d4fSSam Ravnborg 129458fb0d4fSSam Ravnborg tosec = sec_name(elf, sym->st_shndx); 1295588ccd73SSam Ravnborg mismatch = section_mismatch(fromsec, tosec); 1296588ccd73SSam Ravnborg if (mismatch != NO_MISMATCH) { 1297588ccd73SSam Ravnborg Elf_Sym *to; 1298588ccd73SSam Ravnborg Elf_Sym *from; 129958fb0d4fSSam Ravnborg const char *tosym; 1300588ccd73SSam Ravnborg const char *fromsym; 130158fb0d4fSSam Ravnborg 1302588ccd73SSam Ravnborg from = find_elf_symbol2(elf, r->r_offset, fromsec); 1303588ccd73SSam Ravnborg fromsym = sym_name(elf, from); 1304588ccd73SSam Ravnborg to = find_elf_symbol(elf, r->r_addend, sym); 1305588ccd73SSam Ravnborg tosym = sym_name(elf, to); 130658fb0d4fSSam Ravnborg 130758fb0d4fSSam Ravnborg /* check whitelist - we may ignore it */ 130858fb0d4fSSam Ravnborg if (secref_whitelist(fromsec, fromsym, tosec, tosym)) { 1309588ccd73SSam Ravnborg report_sec_mismatch(modname, mismatch, 1310588ccd73SSam Ravnborg fromsec, r->r_offset, fromsym, 1311588ccd73SSam Ravnborg is_function(from), tosec, tosym, 1312588ccd73SSam Ravnborg is_function(to)); 131358fb0d4fSSam Ravnborg } 1314b39927cfSSam Ravnborg } 1315b39927cfSSam Ravnborg } 1316b39927cfSSam Ravnborg 1317ae4ac123SAtsushi Nemoto static unsigned int *reloc_location(struct elf_info *elf, 13185b24c071SSam Ravnborg Elf_Shdr *sechdr, Elf_Rela *r) 1319ae4ac123SAtsushi Nemoto { 1320ae4ac123SAtsushi Nemoto Elf_Shdr *sechdrs = elf->sechdrs; 13215b24c071SSam Ravnborg int section = sechdr->sh_info; 1322ae4ac123SAtsushi Nemoto 1323ae4ac123SAtsushi Nemoto return (void *)elf->hdr + sechdrs[section].sh_offset + 1324ae4ac123SAtsushi Nemoto (r->r_offset - sechdrs[section].sh_addr); 1325ae4ac123SAtsushi Nemoto } 1326ae4ac123SAtsushi Nemoto 13275b24c071SSam Ravnborg static int addend_386_rel(struct elf_info *elf, Elf_Shdr *sechdr, Elf_Rela *r) 1328ae4ac123SAtsushi Nemoto { 1329ae4ac123SAtsushi Nemoto unsigned int r_typ = ELF_R_TYPE(r->r_info); 13305b24c071SSam Ravnborg unsigned int *location = reloc_location(elf, sechdr, r); 1331ae4ac123SAtsushi Nemoto 1332ae4ac123SAtsushi Nemoto switch (r_typ) { 1333ae4ac123SAtsushi Nemoto case R_386_32: 1334ae4ac123SAtsushi Nemoto r->r_addend = TO_NATIVE(*location); 1335ae4ac123SAtsushi Nemoto break; 1336ae4ac123SAtsushi Nemoto case R_386_PC32: 1337ae4ac123SAtsushi Nemoto r->r_addend = TO_NATIVE(*location) + 4; 1338ae4ac123SAtsushi Nemoto /* For CONFIG_RELOCATABLE=y */ 1339ae4ac123SAtsushi Nemoto if (elf->hdr->e_type == ET_EXEC) 1340ae4ac123SAtsushi Nemoto r->r_addend += r->r_offset; 1341ae4ac123SAtsushi Nemoto break; 1342ae4ac123SAtsushi Nemoto } 1343ae4ac123SAtsushi Nemoto return 0; 1344ae4ac123SAtsushi Nemoto } 1345ae4ac123SAtsushi Nemoto 13465b24c071SSam Ravnborg static int addend_arm_rel(struct elf_info *elf, Elf_Shdr *sechdr, Elf_Rela *r) 134756a974faSSam Ravnborg { 134856a974faSSam Ravnborg unsigned int r_typ = ELF_R_TYPE(r->r_info); 134956a974faSSam Ravnborg 135056a974faSSam Ravnborg switch (r_typ) { 135156a974faSSam Ravnborg case R_ARM_ABS32: 135256a974faSSam Ravnborg /* From ARM ABI: (S + A) | T */ 1353df578e7dSSam Ravnborg r->r_addend = (int)(long) 1354df578e7dSSam Ravnborg (elf->symtab_start + ELF_R_SYM(r->r_info)); 135556a974faSSam Ravnborg break; 135656a974faSSam Ravnborg case R_ARM_PC24: 135756a974faSSam Ravnborg /* From ARM ABI: ((S + A) | T) - P */ 1358df578e7dSSam Ravnborg r->r_addend = (int)(long)(elf->hdr + 13595b24c071SSam Ravnborg sechdr->sh_offset + 13605b24c071SSam Ravnborg (r->r_offset - sechdr->sh_addr)); 136156a974faSSam Ravnborg break; 136256a974faSSam Ravnborg default: 136356a974faSSam Ravnborg return 1; 136456a974faSSam Ravnborg } 136556a974faSSam Ravnborg return 0; 136656a974faSSam Ravnborg } 136756a974faSSam Ravnborg 13685b24c071SSam Ravnborg static int addend_mips_rel(struct elf_info *elf, Elf_Shdr *sechdr, Elf_Rela *r) 1369ae4ac123SAtsushi Nemoto { 1370ae4ac123SAtsushi Nemoto unsigned int r_typ = ELF_R_TYPE(r->r_info); 13715b24c071SSam Ravnborg unsigned int *location = reloc_location(elf, sechdr, r); 1372ae4ac123SAtsushi Nemoto unsigned int inst; 1373ae4ac123SAtsushi Nemoto 1374ae4ac123SAtsushi Nemoto if (r_typ == R_MIPS_HI16) 1375ae4ac123SAtsushi Nemoto return 1; /* skip this */ 1376ae4ac123SAtsushi Nemoto inst = TO_NATIVE(*location); 1377ae4ac123SAtsushi Nemoto switch (r_typ) { 1378ae4ac123SAtsushi Nemoto case R_MIPS_LO16: 1379ae4ac123SAtsushi Nemoto r->r_addend = inst & 0xffff; 1380ae4ac123SAtsushi Nemoto break; 1381ae4ac123SAtsushi Nemoto case R_MIPS_26: 1382ae4ac123SAtsushi Nemoto r->r_addend = (inst & 0x03ffffff) << 2; 1383ae4ac123SAtsushi Nemoto break; 1384ae4ac123SAtsushi Nemoto case R_MIPS_32: 1385ae4ac123SAtsushi Nemoto r->r_addend = inst; 1386ae4ac123SAtsushi Nemoto break; 1387ae4ac123SAtsushi Nemoto } 1388ae4ac123SAtsushi Nemoto return 0; 1389ae4ac123SAtsushi Nemoto } 1390ae4ac123SAtsushi Nemoto 13915b24c071SSam Ravnborg static void section_rela(const char *modname, struct elf_info *elf, 139210668220SSam Ravnborg Elf_Shdr *sechdr) 1393b39927cfSSam Ravnborg { 1394b39927cfSSam Ravnborg Elf_Sym *sym; 13955b24c071SSam Ravnborg Elf_Rela *rela; 13962c1a51f3SAtsushi Nemoto Elf_Rela r; 1397eae07ac6SAtsushi Nemoto unsigned int r_sym; 13985b24c071SSam Ravnborg const char *fromsec; 13995b24c071SSam Ravnborg 1400ff13f926SSam Ravnborg Elf_Rela *start = (void *)elf->hdr + sechdr->sh_offset; 14015b24c071SSam Ravnborg Elf_Rela *stop = (void *)start + sechdr->sh_size; 14025b24c071SSam Ravnborg 1403ff13f926SSam Ravnborg fromsec = sech_name(elf, sechdr); 14045b24c071SSam Ravnborg fromsec += strlen(".rela"); 14055b24c071SSam Ravnborg /* if from section (name) is know good then skip it */ 1406b614a697SAnders Kaseorg if (match(fromsec, section_white_list)) 14075b24c071SSam Ravnborg return; 1408e241a630SSam Ravnborg 1409b39927cfSSam Ravnborg for (rela = start; rela < stop; rela++) { 1410b39927cfSSam Ravnborg r.r_offset = TO_NATIVE(rela->r_offset); 1411eae07ac6SAtsushi Nemoto #if KERNEL_ELFCLASS == ELFCLASS64 1412ff13f926SSam Ravnborg if (elf->hdr->e_machine == EM_MIPS) { 1413ae4ac123SAtsushi Nemoto unsigned int r_typ; 1414eae07ac6SAtsushi Nemoto r_sym = ELF64_MIPS_R_SYM(rela->r_info); 1415eae07ac6SAtsushi Nemoto r_sym = TO_NATIVE(r_sym); 1416ae4ac123SAtsushi Nemoto r_typ = ELF64_MIPS_R_TYPE(rela->r_info); 1417ae4ac123SAtsushi Nemoto r.r_info = ELF64_R_INFO(r_sym, r_typ); 1418eae07ac6SAtsushi Nemoto } else { 1419601e7f02SLinus Torvalds r.r_info = TO_NATIVE(rela->r_info); 1420eae07ac6SAtsushi Nemoto r_sym = ELF_R_SYM(r.r_info); 1421eae07ac6SAtsushi Nemoto } 1422eae07ac6SAtsushi Nemoto #else 1423eae07ac6SAtsushi Nemoto r.r_info = TO_NATIVE(rela->r_info); 1424eae07ac6SAtsushi Nemoto r_sym = ELF_R_SYM(r.r_info); 1425eae07ac6SAtsushi Nemoto #endif 142693684d3bSSam Ravnborg r.r_addend = TO_NATIVE(rela->r_addend); 1427eae07ac6SAtsushi Nemoto sym = elf->symtab_start + r_sym; 1428b39927cfSSam Ravnborg /* Skip special sections */ 1429b39927cfSSam Ravnborg if (sym->st_shndx >= SHN_LORESERVE) 1430b39927cfSSam Ravnborg continue; 143158fb0d4fSSam Ravnborg check_section_mismatch(modname, elf, &r, sym, fromsec); 14322c1a51f3SAtsushi Nemoto } 14335b24c071SSam Ravnborg } 14345b24c071SSam Ravnborg 14355b24c071SSam Ravnborg static void section_rel(const char *modname, struct elf_info *elf, 143610668220SSam Ravnborg Elf_Shdr *sechdr) 14375b24c071SSam Ravnborg { 14385b24c071SSam Ravnborg Elf_Sym *sym; 14392c1a51f3SAtsushi Nemoto Elf_Rel *rel; 14405b24c071SSam Ravnborg Elf_Rela r; 14415b24c071SSam Ravnborg unsigned int r_sym; 14425b24c071SSam Ravnborg const char *fromsec; 14435b24c071SSam Ravnborg 1444ff13f926SSam Ravnborg Elf_Rel *start = (void *)elf->hdr + sechdr->sh_offset; 14455b24c071SSam Ravnborg Elf_Rel *stop = (void *)start + sechdr->sh_size; 14465b24c071SSam Ravnborg 1447ff13f926SSam Ravnborg fromsec = sech_name(elf, sechdr); 14485b24c071SSam Ravnborg fromsec += strlen(".rel"); 14495b24c071SSam Ravnborg /* if from section (name) is know good then skip it */ 1450b614a697SAnders Kaseorg if (match(fromsec, section_white_list)) 14515b24c071SSam Ravnborg return; 14522c1a51f3SAtsushi Nemoto 14532c1a51f3SAtsushi Nemoto for (rel = start; rel < stop; rel++) { 14542c1a51f3SAtsushi Nemoto r.r_offset = TO_NATIVE(rel->r_offset); 1455eae07ac6SAtsushi Nemoto #if KERNEL_ELFCLASS == ELFCLASS64 1456ff13f926SSam Ravnborg if (elf->hdr->e_machine == EM_MIPS) { 1457ae4ac123SAtsushi Nemoto unsigned int r_typ; 1458eae07ac6SAtsushi Nemoto r_sym = ELF64_MIPS_R_SYM(rel->r_info); 1459eae07ac6SAtsushi Nemoto r_sym = TO_NATIVE(r_sym); 1460ae4ac123SAtsushi Nemoto r_typ = ELF64_MIPS_R_TYPE(rel->r_info); 1461ae4ac123SAtsushi Nemoto r.r_info = ELF64_R_INFO(r_sym, r_typ); 1462eae07ac6SAtsushi Nemoto } else { 14632c1a51f3SAtsushi Nemoto r.r_info = TO_NATIVE(rel->r_info); 1464eae07ac6SAtsushi Nemoto r_sym = ELF_R_SYM(r.r_info); 1465eae07ac6SAtsushi Nemoto } 1466eae07ac6SAtsushi Nemoto #else 1467eae07ac6SAtsushi Nemoto r.r_info = TO_NATIVE(rel->r_info); 1468eae07ac6SAtsushi Nemoto r_sym = ELF_R_SYM(r.r_info); 1469eae07ac6SAtsushi Nemoto #endif 14702c1a51f3SAtsushi Nemoto r.r_addend = 0; 1471ff13f926SSam Ravnborg switch (elf->hdr->e_machine) { 1472ae4ac123SAtsushi Nemoto case EM_386: 14735b24c071SSam Ravnborg if (addend_386_rel(elf, sechdr, &r)) 1474ae4ac123SAtsushi Nemoto continue; 1475ae4ac123SAtsushi Nemoto break; 147656a974faSSam Ravnborg case EM_ARM: 14775b24c071SSam Ravnborg if (addend_arm_rel(elf, sechdr, &r)) 147856a974faSSam Ravnborg continue; 147956a974faSSam Ravnborg break; 1480ae4ac123SAtsushi Nemoto case EM_MIPS: 14815b24c071SSam Ravnborg if (addend_mips_rel(elf, sechdr, &r)) 1482ae4ac123SAtsushi Nemoto continue; 1483ae4ac123SAtsushi Nemoto break; 1484ae4ac123SAtsushi Nemoto } 1485eae07ac6SAtsushi Nemoto sym = elf->symtab_start + r_sym; 14862c1a51f3SAtsushi Nemoto /* Skip special sections */ 14872c1a51f3SAtsushi Nemoto if (sym->st_shndx >= SHN_LORESERVE) 14882c1a51f3SAtsushi Nemoto continue; 148958fb0d4fSSam Ravnborg check_section_mismatch(modname, elf, &r, sym, fromsec); 14902c1a51f3SAtsushi Nemoto } 1491b39927cfSSam Ravnborg } 14925b24c071SSam Ravnborg 14935b24c071SSam Ravnborg /** 14945b24c071SSam Ravnborg * A module includes a number of sections that are discarded 14955b24c071SSam Ravnborg * either when loaded or when used as built-in. 14965b24c071SSam Ravnborg * For loaded modules all functions marked __init and all data 14975b24c071SSam Ravnborg * marked __initdata will be discarded when the module has been intialized. 14985b24c071SSam Ravnborg * Likewise for modules used built-in the sections marked __exit 14995b24c071SSam Ravnborg * are discarded because __exit marked function are supposed to be called 150032be1d22SBen Dooks * only when a module is unloaded which never happens for built-in modules. 15015b24c071SSam Ravnborg * The check_sec_ref() function traverses all relocation records 15025b24c071SSam Ravnborg * to find all references to a section that reference a section that will 15035b24c071SSam Ravnborg * be discarded and warns about it. 15045b24c071SSam Ravnborg **/ 15055b24c071SSam Ravnborg static void check_sec_ref(struct module *mod, const char *modname, 150610668220SSam Ravnborg struct elf_info *elf) 15075b24c071SSam Ravnborg { 15085b24c071SSam Ravnborg int i; 15095b24c071SSam Ravnborg Elf_Shdr *sechdrs = elf->sechdrs; 15105b24c071SSam Ravnborg 15115b24c071SSam Ravnborg /* Walk through all sections */ 1512ff13f926SSam Ravnborg for (i = 0; i < elf->hdr->e_shnum; i++) { 1513b614a697SAnders Kaseorg check_section(modname, elf, &elf->sechdrs[i]); 15145b24c071SSam Ravnborg /* We want to process only relocation sections and not .init */ 15155b24c071SSam Ravnborg if (sechdrs[i].sh_type == SHT_RELA) 151610668220SSam Ravnborg section_rela(modname, elf, &elf->sechdrs[i]); 15175b24c071SSam Ravnborg else if (sechdrs[i].sh_type == SHT_REL) 151810668220SSam Ravnborg section_rel(modname, elf, &elf->sechdrs[i]); 1519b39927cfSSam Ravnborg } 1520b39927cfSSam Ravnborg } 1521b39927cfSSam Ravnborg 15225c3ead8cSSam Ravnborg static void read_symbols(char *modname) 15231da177e4SLinus Torvalds { 15241da177e4SLinus Torvalds const char *symname; 15251da177e4SLinus Torvalds char *version; 1526b817f6feSSam Ravnborg char *license; 15271da177e4SLinus Torvalds struct module *mod; 15281da177e4SLinus Torvalds struct elf_info info = { }; 15291da177e4SLinus Torvalds Elf_Sym *sym; 15301da177e4SLinus Torvalds 153185bd2fddSSam Ravnborg if (!parse_elf(&info, modname)) 153285bd2fddSSam Ravnborg return; 15331da177e4SLinus Torvalds 15341da177e4SLinus Torvalds mod = new_module(modname); 15351da177e4SLinus Torvalds 15361da177e4SLinus Torvalds /* When there's no vmlinux, don't print warnings about 15371da177e4SLinus Torvalds * unresolved symbols (since there'll be too many ;) */ 15381da177e4SLinus Torvalds if (is_vmlinux(modname)) { 15391da177e4SLinus Torvalds have_vmlinux = 1; 15401da177e4SLinus Torvalds mod->skip = 1; 15411da177e4SLinus Torvalds } 15421da177e4SLinus Torvalds 1543b817f6feSSam Ravnborg license = get_modinfo(info.modinfo, info.modinfo_len, "license"); 15442fa36568SSam Ravnborg if (info.modinfo && !license && !is_vmlinux(modname)) 15452fa36568SSam Ravnborg warn("modpost: missing MODULE_LICENSE() in %s\n" 1546eed7d279SSam Ravnborg "see include/linux/module.h for " 1547eed7d279SSam Ravnborg "more information\n", modname); 1548b817f6feSSam Ravnborg while (license) { 1549b817f6feSSam Ravnborg if (license_is_gpl_compatible(license)) 1550b817f6feSSam Ravnborg mod->gpl_compatible = 1; 1551b817f6feSSam Ravnborg else { 1552b817f6feSSam Ravnborg mod->gpl_compatible = 0; 1553b817f6feSSam Ravnborg break; 1554b817f6feSSam Ravnborg } 1555b817f6feSSam Ravnborg license = get_next_modinfo(info.modinfo, info.modinfo_len, 1556b817f6feSSam Ravnborg "license", license); 1557b817f6feSSam Ravnborg } 1558b817f6feSSam Ravnborg 15591da177e4SLinus Torvalds for (sym = info.symtab_start; sym < info.symtab_stop; sym++) { 15601da177e4SLinus Torvalds symname = info.strtab + sym->st_name; 15611da177e4SLinus Torvalds 15621da177e4SLinus Torvalds handle_modversions(mod, &info, sym, symname); 15631da177e4SLinus Torvalds handle_moddevtable(mod, &info, sym, symname); 15641da177e4SLinus Torvalds } 1565d1f25e66SSam Ravnborg if (!is_vmlinux(modname) || 156610668220SSam Ravnborg (is_vmlinux(modname) && vmlinux_section_warnings)) 156710668220SSam Ravnborg check_sec_ref(mod, modname, &info); 15681da177e4SLinus Torvalds 15691da177e4SLinus Torvalds version = get_modinfo(info.modinfo, info.modinfo_len, "version"); 15701da177e4SLinus Torvalds if (version) 15711da177e4SLinus Torvalds maybe_frob_rcs_version(modname, version, info.modinfo, 15721da177e4SLinus Torvalds version - (char *)info.hdr); 15731da177e4SLinus Torvalds if (version || (all_versions && !is_vmlinux(modname))) 15741da177e4SLinus Torvalds get_src_version(modname, mod->srcversion, 15751da177e4SLinus Torvalds sizeof(mod->srcversion)-1); 15761da177e4SLinus Torvalds 15771da177e4SLinus Torvalds parse_elf_finish(&info); 15781da177e4SLinus Torvalds 15798c8ef42aSRusty Russell /* Our trick to get versioning for module struct etc. - it's 15801da177e4SLinus Torvalds * never passed as an argument to an exported function, so 15811da177e4SLinus Torvalds * the automatic versioning doesn't pick it up, but it's really 15821da177e4SLinus Torvalds * important anyhow */ 15831da177e4SLinus Torvalds if (modversions) 15848c8ef42aSRusty Russell mod->unres = alloc_symbol("module_layout", 0, mod->unres); 15851da177e4SLinus Torvalds } 15861da177e4SLinus Torvalds 15871da177e4SLinus Torvalds #define SZ 500 15881da177e4SLinus Torvalds 15891da177e4SLinus Torvalds /* We first write the generated file into memory using the 15901da177e4SLinus Torvalds * following helper, then compare to the file on disk and 15911da177e4SLinus Torvalds * only update the later if anything changed */ 15921da177e4SLinus Torvalds 15935c3ead8cSSam Ravnborg void __attribute__((format(printf, 2, 3))) buf_printf(struct buffer *buf, 15945c3ead8cSSam Ravnborg const char *fmt, ...) 15951da177e4SLinus Torvalds { 15961da177e4SLinus Torvalds char tmp[SZ]; 15971da177e4SLinus Torvalds int len; 15981da177e4SLinus Torvalds va_list ap; 15991da177e4SLinus Torvalds 16001da177e4SLinus Torvalds va_start(ap, fmt); 16011da177e4SLinus Torvalds len = vsnprintf(tmp, SZ, fmt, ap); 16027670f023SSam Ravnborg buf_write(buf, tmp, len); 16031da177e4SLinus Torvalds va_end(ap); 16041da177e4SLinus Torvalds } 16051da177e4SLinus Torvalds 16065c3ead8cSSam Ravnborg void buf_write(struct buffer *buf, const char *s, int len) 16071da177e4SLinus Torvalds { 16081da177e4SLinus Torvalds if (buf->size - buf->pos < len) { 16097670f023SSam Ravnborg buf->size += len + SZ; 16101da177e4SLinus Torvalds buf->p = realloc(buf->p, buf->size); 16111da177e4SLinus Torvalds } 16121da177e4SLinus Torvalds strncpy(buf->p + buf->pos, s, len); 16131da177e4SLinus Torvalds buf->pos += len; 16141da177e4SLinus Torvalds } 16151da177e4SLinus Torvalds 1616c96fca21SSam Ravnborg static void check_for_gpl_usage(enum export exp, const char *m, const char *s) 1617c96fca21SSam Ravnborg { 1618c96fca21SSam Ravnborg const char *e = is_vmlinux(m) ?"":".ko"; 1619c96fca21SSam Ravnborg 1620c96fca21SSam Ravnborg switch (exp) { 1621c96fca21SSam Ravnborg case export_gpl: 1622c96fca21SSam Ravnborg fatal("modpost: GPL-incompatible module %s%s " 1623c96fca21SSam Ravnborg "uses GPL-only symbol '%s'\n", m, e, s); 1624c96fca21SSam Ravnborg break; 1625c96fca21SSam Ravnborg case export_unused_gpl: 1626c96fca21SSam Ravnborg fatal("modpost: GPL-incompatible module %s%s " 1627c96fca21SSam Ravnborg "uses GPL-only symbol marked UNUSED '%s'\n", m, e, s); 1628c96fca21SSam Ravnborg break; 1629c96fca21SSam Ravnborg case export_gpl_future: 1630c96fca21SSam Ravnborg warn("modpost: GPL-incompatible module %s%s " 1631c96fca21SSam Ravnborg "uses future GPL-only symbol '%s'\n", m, e, s); 1632c96fca21SSam Ravnborg break; 1633c96fca21SSam Ravnborg case export_plain: 1634c96fca21SSam Ravnborg case export_unused: 1635c96fca21SSam Ravnborg case export_unknown: 1636c96fca21SSam Ravnborg /* ignore */ 1637c96fca21SSam Ravnborg break; 1638c96fca21SSam Ravnborg } 1639c96fca21SSam Ravnborg } 1640c96fca21SSam Ravnborg 1641c96fca21SSam Ravnborg static void check_for_unused(enum export exp, const char *m, const char *s) 1642c96fca21SSam Ravnborg { 1643c96fca21SSam Ravnborg const char *e = is_vmlinux(m) ?"":".ko"; 1644c96fca21SSam Ravnborg 1645c96fca21SSam Ravnborg switch (exp) { 1646c96fca21SSam Ravnborg case export_unused: 1647c96fca21SSam Ravnborg case export_unused_gpl: 1648c96fca21SSam Ravnborg warn("modpost: module %s%s " 1649c96fca21SSam Ravnborg "uses symbol '%s' marked UNUSED\n", m, e, s); 1650c96fca21SSam Ravnborg break; 1651c96fca21SSam Ravnborg default: 1652c96fca21SSam Ravnborg /* ignore */ 1653c96fca21SSam Ravnborg break; 1654c96fca21SSam Ravnborg } 1655c96fca21SSam Ravnborg } 1656c96fca21SSam Ravnborg 1657c96fca21SSam Ravnborg static void check_exports(struct module *mod) 1658b817f6feSSam Ravnborg { 1659b817f6feSSam Ravnborg struct symbol *s, *exp; 1660b817f6feSSam Ravnborg 1661b817f6feSSam Ravnborg for (s = mod->unres; s; s = s->next) { 16626449bd62SAndrew Morton const char *basename; 1663b817f6feSSam Ravnborg exp = find_symbol(s->name); 1664b817f6feSSam Ravnborg if (!exp || exp->module == mod) 1665b817f6feSSam Ravnborg continue; 16666449bd62SAndrew Morton basename = strrchr(mod->name, '/'); 1667b817f6feSSam Ravnborg if (basename) 1668b817f6feSSam Ravnborg basename++; 1669c96fca21SSam Ravnborg else 1670c96fca21SSam Ravnborg basename = mod->name; 1671c96fca21SSam Ravnborg if (!mod->gpl_compatible) 1672c96fca21SSam Ravnborg check_for_gpl_usage(exp->export, basename, exp->name); 1673c96fca21SSam Ravnborg check_for_unused(exp->export, basename, exp->name); 1674b817f6feSSam Ravnborg } 1675b817f6feSSam Ravnborg } 1676b817f6feSSam Ravnborg 16775c3ead8cSSam Ravnborg /** 16785c3ead8cSSam Ravnborg * Header for the generated file 16795c3ead8cSSam Ravnborg **/ 16805c3ead8cSSam Ravnborg static void add_header(struct buffer *b, struct module *mod) 16811da177e4SLinus Torvalds { 16821da177e4SLinus Torvalds buf_printf(b, "#include <linux/module.h>\n"); 16831da177e4SLinus Torvalds buf_printf(b, "#include <linux/vermagic.h>\n"); 16841da177e4SLinus Torvalds buf_printf(b, "#include <linux/compiler.h>\n"); 16851da177e4SLinus Torvalds buf_printf(b, "\n"); 16861da177e4SLinus Torvalds buf_printf(b, "MODULE_INFO(vermagic, VERMAGIC_STRING);\n"); 16871da177e4SLinus Torvalds buf_printf(b, "\n"); 16881da177e4SLinus Torvalds buf_printf(b, "struct module __this_module\n"); 16891da177e4SLinus Torvalds buf_printf(b, "__attribute__((section(\".gnu.linkonce.this_module\"))) = {\n"); 1690f83b5e32SUstyugov Roman buf_printf(b, " .name = KBUILD_MODNAME,\n"); 16911da177e4SLinus Torvalds if (mod->has_init) 16921da177e4SLinus Torvalds buf_printf(b, " .init = init_module,\n"); 16931da177e4SLinus Torvalds if (mod->has_cleanup) 16941da177e4SLinus Torvalds buf_printf(b, "#ifdef CONFIG_MODULE_UNLOAD\n" 16951da177e4SLinus Torvalds " .exit = cleanup_module,\n" 16961da177e4SLinus Torvalds "#endif\n"); 1697e61a1c1cSRoman Zippel buf_printf(b, " .arch = MODULE_ARCH_INIT,\n"); 16981da177e4SLinus Torvalds buf_printf(b, "};\n"); 16991da177e4SLinus Torvalds } 17001da177e4SLinus Torvalds 17015c725138STrevor Keith static void add_staging_flag(struct buffer *b, const char *name) 1702a9860bf0SGreg Kroah-Hartman { 1703a9860bf0SGreg Kroah-Hartman static const char *staging_dir = "drivers/staging"; 1704a9860bf0SGreg Kroah-Hartman 1705a9860bf0SGreg Kroah-Hartman if (strncmp(staging_dir, name, strlen(staging_dir)) == 0) 1706a9860bf0SGreg Kroah-Hartman buf_printf(b, "\nMODULE_INFO(staging, \"Y\");\n"); 1707a9860bf0SGreg Kroah-Hartman } 1708a9860bf0SGreg Kroah-Hartman 17095c3ead8cSSam Ravnborg /** 17105c3ead8cSSam Ravnborg * Record CRCs for unresolved symbols 17115c3ead8cSSam Ravnborg **/ 1712c53ddacdSKirill Korotaev static int add_versions(struct buffer *b, struct module *mod) 17131da177e4SLinus Torvalds { 17141da177e4SLinus Torvalds struct symbol *s, *exp; 1715c53ddacdSKirill Korotaev int err = 0; 17161da177e4SLinus Torvalds 17171da177e4SLinus Torvalds for (s = mod->unres; s; s = s->next) { 17181da177e4SLinus Torvalds exp = find_symbol(s->name); 17191da177e4SLinus Torvalds if (!exp || exp->module == mod) { 1720c53ddacdSKirill Korotaev if (have_vmlinux && !s->weak) { 17212a116659SMatthew Wilcox if (warn_unresolved) { 1722cb80514dSSam Ravnborg warn("\"%s\" [%s.ko] undefined!\n", 1723cb80514dSSam Ravnborg s->name, mod->name); 17242a116659SMatthew Wilcox } else { 17252a116659SMatthew Wilcox merror("\"%s\" [%s.ko] undefined!\n", 17262a116659SMatthew Wilcox s->name, mod->name); 17272a116659SMatthew Wilcox err = 1; 17282a116659SMatthew Wilcox } 1729c53ddacdSKirill Korotaev } 17301da177e4SLinus Torvalds continue; 17311da177e4SLinus Torvalds } 17321da177e4SLinus Torvalds s->module = exp->module; 17331da177e4SLinus Torvalds s->crc_valid = exp->crc_valid; 17341da177e4SLinus Torvalds s->crc = exp->crc; 17351da177e4SLinus Torvalds } 17361da177e4SLinus Torvalds 17371da177e4SLinus Torvalds if (!modversions) 1738c53ddacdSKirill Korotaev return err; 17391da177e4SLinus Torvalds 17401da177e4SLinus Torvalds buf_printf(b, "\n"); 17411da177e4SLinus Torvalds buf_printf(b, "static const struct modversion_info ____versions[]\n"); 17423ff6eeccSAdrian Bunk buf_printf(b, "__used\n"); 17431da177e4SLinus Torvalds buf_printf(b, "__attribute__((section(\"__versions\"))) = {\n"); 17441da177e4SLinus Torvalds 17451da177e4SLinus Torvalds for (s = mod->unres; s; s = s->next) { 1746df578e7dSSam Ravnborg if (!s->module) 17471da177e4SLinus Torvalds continue; 17481da177e4SLinus Torvalds if (!s->crc_valid) { 1749cb80514dSSam Ravnborg warn("\"%s\" [%s.ko] has no CRC!\n", 17501da177e4SLinus Torvalds s->name, mod->name); 17511da177e4SLinus Torvalds continue; 17521da177e4SLinus Torvalds } 17531da177e4SLinus Torvalds buf_printf(b, "\t{ %#8x, \"%s\" },\n", s->crc, s->name); 17541da177e4SLinus Torvalds } 17551da177e4SLinus Torvalds 17561da177e4SLinus Torvalds buf_printf(b, "};\n"); 1757c53ddacdSKirill Korotaev 1758c53ddacdSKirill Korotaev return err; 17591da177e4SLinus Torvalds } 17601da177e4SLinus Torvalds 17615c3ead8cSSam Ravnborg static void add_depends(struct buffer *b, struct module *mod, 17625c3ead8cSSam Ravnborg struct module *modules) 17631da177e4SLinus Torvalds { 17641da177e4SLinus Torvalds struct symbol *s; 17651da177e4SLinus Torvalds struct module *m; 17661da177e4SLinus Torvalds int first = 1; 17671da177e4SLinus Torvalds 1768df578e7dSSam Ravnborg for (m = modules; m; m = m->next) 17691da177e4SLinus Torvalds m->seen = is_vmlinux(m->name); 17701da177e4SLinus Torvalds 17711da177e4SLinus Torvalds buf_printf(b, "\n"); 17721da177e4SLinus Torvalds buf_printf(b, "static const char __module_depends[]\n"); 17733ff6eeccSAdrian Bunk buf_printf(b, "__used\n"); 17741da177e4SLinus Torvalds buf_printf(b, "__attribute__((section(\".modinfo\"))) =\n"); 17751da177e4SLinus Torvalds buf_printf(b, "\"depends="); 17761da177e4SLinus Torvalds for (s = mod->unres; s; s = s->next) { 1777a61b2dfdSSam Ravnborg const char *p; 17781da177e4SLinus Torvalds if (!s->module) 17791da177e4SLinus Torvalds continue; 17801da177e4SLinus Torvalds 17811da177e4SLinus Torvalds if (s->module->seen) 17821da177e4SLinus Torvalds continue; 17831da177e4SLinus Torvalds 17841da177e4SLinus Torvalds s->module->seen = 1; 1785df578e7dSSam Ravnborg p = strrchr(s->module->name, '/'); 1786df578e7dSSam Ravnborg if (p) 1787a61b2dfdSSam Ravnborg p++; 1788a61b2dfdSSam Ravnborg else 1789a61b2dfdSSam Ravnborg p = s->module->name; 1790a61b2dfdSSam Ravnborg buf_printf(b, "%s%s", first ? "" : ",", p); 17911da177e4SLinus Torvalds first = 0; 17921da177e4SLinus Torvalds } 17931da177e4SLinus Torvalds buf_printf(b, "\";\n"); 17941da177e4SLinus Torvalds } 17951da177e4SLinus Torvalds 17965c3ead8cSSam Ravnborg static void add_srcversion(struct buffer *b, struct module *mod) 17971da177e4SLinus Torvalds { 17981da177e4SLinus Torvalds if (mod->srcversion[0]) { 17991da177e4SLinus Torvalds buf_printf(b, "\n"); 18001da177e4SLinus Torvalds buf_printf(b, "MODULE_INFO(srcversion, \"%s\");\n", 18011da177e4SLinus Torvalds mod->srcversion); 18021da177e4SLinus Torvalds } 18031da177e4SLinus Torvalds } 18041da177e4SLinus Torvalds 18055c3ead8cSSam Ravnborg static void write_if_changed(struct buffer *b, const char *fname) 18061da177e4SLinus Torvalds { 18071da177e4SLinus Torvalds char *tmp; 18081da177e4SLinus Torvalds FILE *file; 18091da177e4SLinus Torvalds struct stat st; 18101da177e4SLinus Torvalds 18111da177e4SLinus Torvalds file = fopen(fname, "r"); 18121da177e4SLinus Torvalds if (!file) 18131da177e4SLinus Torvalds goto write; 18141da177e4SLinus Torvalds 18151da177e4SLinus Torvalds if (fstat(fileno(file), &st) < 0) 18161da177e4SLinus Torvalds goto close_write; 18171da177e4SLinus Torvalds 18181da177e4SLinus Torvalds if (st.st_size != b->pos) 18191da177e4SLinus Torvalds goto close_write; 18201da177e4SLinus Torvalds 18211da177e4SLinus Torvalds tmp = NOFAIL(malloc(b->pos)); 18221da177e4SLinus Torvalds if (fread(tmp, 1, b->pos, file) != b->pos) 18231da177e4SLinus Torvalds goto free_write; 18241da177e4SLinus Torvalds 18251da177e4SLinus Torvalds if (memcmp(tmp, b->p, b->pos) != 0) 18261da177e4SLinus Torvalds goto free_write; 18271da177e4SLinus Torvalds 18281da177e4SLinus Torvalds free(tmp); 18291da177e4SLinus Torvalds fclose(file); 18301da177e4SLinus Torvalds return; 18311da177e4SLinus Torvalds 18321da177e4SLinus Torvalds free_write: 18331da177e4SLinus Torvalds free(tmp); 18341da177e4SLinus Torvalds close_write: 18351da177e4SLinus Torvalds fclose(file); 18361da177e4SLinus Torvalds write: 18371da177e4SLinus Torvalds file = fopen(fname, "w"); 18381da177e4SLinus Torvalds if (!file) { 18391da177e4SLinus Torvalds perror(fname); 18401da177e4SLinus Torvalds exit(1); 18411da177e4SLinus Torvalds } 18421da177e4SLinus Torvalds if (fwrite(b->p, 1, b->pos, file) != b->pos) { 18431da177e4SLinus Torvalds perror(fname); 18441da177e4SLinus Torvalds exit(1); 18451da177e4SLinus Torvalds } 18461da177e4SLinus Torvalds fclose(file); 18471da177e4SLinus Torvalds } 18481da177e4SLinus Torvalds 1849bd5cbcedSRam Pai /* parse Module.symvers file. line format: 1850534b89a9SSam Ravnborg * 0x12345678<tab>symbol<tab>module[[<tab>export]<tab>something] 1851bd5cbcedSRam Pai **/ 1852040fcc81SSam Ravnborg static void read_dump(const char *fname, unsigned int kernel) 18531da177e4SLinus Torvalds { 18541da177e4SLinus Torvalds unsigned long size, pos = 0; 18551da177e4SLinus Torvalds void *file = grab_file(fname, &size); 18561da177e4SLinus Torvalds char *line; 18571da177e4SLinus Torvalds 18581da177e4SLinus Torvalds if (!file) 18591da177e4SLinus Torvalds /* No symbol versions, silently ignore */ 18601da177e4SLinus Torvalds return; 18611da177e4SLinus Torvalds 18621da177e4SLinus Torvalds while ((line = get_next_line(&pos, file, size))) { 1863534b89a9SSam Ravnborg char *symname, *modname, *d, *export, *end; 18641da177e4SLinus Torvalds unsigned int crc; 18651da177e4SLinus Torvalds struct module *mod; 1866040fcc81SSam Ravnborg struct symbol *s; 18671da177e4SLinus Torvalds 18681da177e4SLinus Torvalds if (!(symname = strchr(line, '\t'))) 18691da177e4SLinus Torvalds goto fail; 18701da177e4SLinus Torvalds *symname++ = '\0'; 18711da177e4SLinus Torvalds if (!(modname = strchr(symname, '\t'))) 18721da177e4SLinus Torvalds goto fail; 18731da177e4SLinus Torvalds *modname++ = '\0'; 18749ac545b0SLaurent Riffard if ((export = strchr(modname, '\t')) != NULL) 1875bd5cbcedSRam Pai *export++ = '\0'; 1876534b89a9SSam Ravnborg if (export && ((end = strchr(export, '\t')) != NULL)) 1877534b89a9SSam Ravnborg *end = '\0'; 18781da177e4SLinus Torvalds crc = strtoul(line, &d, 16); 18791da177e4SLinus Torvalds if (*symname == '\0' || *modname == '\0' || *d != '\0') 18801da177e4SLinus Torvalds goto fail; 1881df578e7dSSam Ravnborg mod = find_module(modname); 1882df578e7dSSam Ravnborg if (!mod) { 1883df578e7dSSam Ravnborg if (is_vmlinux(modname)) 18841da177e4SLinus Torvalds have_vmlinux = 1; 18850fa3a88cSJan Beulich mod = new_module(modname); 18861da177e4SLinus Torvalds mod->skip = 1; 18871da177e4SLinus Torvalds } 1888bd5cbcedSRam Pai s = sym_add_exported(symname, mod, export_no(export)); 1889040fcc81SSam Ravnborg s->kernel = kernel; 18908e70c458SSam Ravnborg s->preloaded = 1; 1891bd5cbcedSRam Pai sym_update_crc(symname, mod, crc, export_no(export)); 18921da177e4SLinus Torvalds } 18931da177e4SLinus Torvalds return; 18941da177e4SLinus Torvalds fail: 18951da177e4SLinus Torvalds fatal("parse error in symbol dump file\n"); 18961da177e4SLinus Torvalds } 18971da177e4SLinus Torvalds 1898040fcc81SSam Ravnborg /* For normal builds always dump all symbols. 1899040fcc81SSam Ravnborg * For external modules only dump symbols 1900040fcc81SSam Ravnborg * that are not read from kernel Module.symvers. 1901040fcc81SSam Ravnborg **/ 1902040fcc81SSam Ravnborg static int dump_sym(struct symbol *sym) 1903040fcc81SSam Ravnborg { 1904040fcc81SSam Ravnborg if (!external_module) 1905040fcc81SSam Ravnborg return 1; 1906040fcc81SSam Ravnborg if (sym->vmlinux || sym->kernel) 1907040fcc81SSam Ravnborg return 0; 1908040fcc81SSam Ravnborg return 1; 1909040fcc81SSam Ravnborg } 1910040fcc81SSam Ravnborg 19115c3ead8cSSam Ravnborg static void write_dump(const char *fname) 19121da177e4SLinus Torvalds { 19131da177e4SLinus Torvalds struct buffer buf = { }; 19141da177e4SLinus Torvalds struct symbol *symbol; 19151da177e4SLinus Torvalds int n; 19161da177e4SLinus Torvalds 19171da177e4SLinus Torvalds for (n = 0; n < SYMBOL_HASH_SIZE ; n++) { 19181da177e4SLinus Torvalds symbol = symbolhash[n]; 19191da177e4SLinus Torvalds while (symbol) { 1920040fcc81SSam Ravnborg if (dump_sym(symbol)) 1921bd5cbcedSRam Pai buf_printf(&buf, "0x%08x\t%s\t%s\t%s\n", 1922040fcc81SSam Ravnborg symbol->crc, symbol->name, 1923bd5cbcedSRam Pai symbol->module->name, 1924bd5cbcedSRam Pai export_str(symbol->export)); 19251da177e4SLinus Torvalds symbol = symbol->next; 19261da177e4SLinus Torvalds } 19271da177e4SLinus Torvalds } 19281da177e4SLinus Torvalds write_if_changed(&buf, fname); 19291da177e4SLinus Torvalds } 19301da177e4SLinus Torvalds 19312d04b5aeSRichard Hacker struct ext_sym_list { 19322d04b5aeSRichard Hacker struct ext_sym_list *next; 19332d04b5aeSRichard Hacker const char *file; 19342d04b5aeSRichard Hacker }; 19352d04b5aeSRichard Hacker 19365c3ead8cSSam Ravnborg int main(int argc, char **argv) 19371da177e4SLinus Torvalds { 19381da177e4SLinus Torvalds struct module *mod; 19391da177e4SLinus Torvalds struct buffer buf = { }; 1940040fcc81SSam Ravnborg char *kernel_read = NULL, *module_read = NULL; 1941040fcc81SSam Ravnborg char *dump_write = NULL; 19421da177e4SLinus Torvalds int opt; 1943c53ddacdSKirill Korotaev int err; 19442d04b5aeSRichard Hacker struct ext_sym_list *extsym_iter; 19452d04b5aeSRichard Hacker struct ext_sym_list *extsym_start = NULL; 19461da177e4SLinus Torvalds 19472d04b5aeSRichard Hacker while ((opt = getopt(argc, argv, "i:I:e:cmsSo:awM:K:")) != -1) { 19481da177e4SLinus Torvalds switch (opt) { 19491da177e4SLinus Torvalds case 'i': 1950040fcc81SSam Ravnborg kernel_read = optarg; 1951040fcc81SSam Ravnborg break; 1952040fcc81SSam Ravnborg case 'I': 1953040fcc81SSam Ravnborg module_read = optarg; 1954040fcc81SSam Ravnborg external_module = 1; 19551da177e4SLinus Torvalds break; 19564ce6efedSSam Ravnborg case 'c': 19574ce6efedSSam Ravnborg cross_build = 1; 19584ce6efedSSam Ravnborg break; 19592d04b5aeSRichard Hacker case 'e': 19602d04b5aeSRichard Hacker external_module = 1; 19612d04b5aeSRichard Hacker extsym_iter = 19622d04b5aeSRichard Hacker NOFAIL(malloc(sizeof(*extsym_iter))); 19632d04b5aeSRichard Hacker extsym_iter->next = extsym_start; 19642d04b5aeSRichard Hacker extsym_iter->file = optarg; 19652d04b5aeSRichard Hacker extsym_start = extsym_iter; 19662d04b5aeSRichard Hacker break; 19671da177e4SLinus Torvalds case 'm': 19681da177e4SLinus Torvalds modversions = 1; 19691da177e4SLinus Torvalds break; 19701da177e4SLinus Torvalds case 'o': 19711da177e4SLinus Torvalds dump_write = optarg; 19721da177e4SLinus Torvalds break; 19731da177e4SLinus Torvalds case 'a': 19741da177e4SLinus Torvalds all_versions = 1; 19751da177e4SLinus Torvalds break; 19768d8d8289SSam Ravnborg case 's': 19778d8d8289SSam Ravnborg vmlinux_section_warnings = 0; 19788d8d8289SSam Ravnborg break; 1979588ccd73SSam Ravnborg case 'S': 1980588ccd73SSam Ravnborg sec_mismatch_verbose = 0; 1981588ccd73SSam Ravnborg break; 1982c53ddacdSKirill Korotaev case 'w': 1983c53ddacdSKirill Korotaev warn_unresolved = 1; 1984c53ddacdSKirill Korotaev break; 19851da177e4SLinus Torvalds default: 19861da177e4SLinus Torvalds exit(1); 19871da177e4SLinus Torvalds } 19881da177e4SLinus Torvalds } 19891da177e4SLinus Torvalds 1990040fcc81SSam Ravnborg if (kernel_read) 1991040fcc81SSam Ravnborg read_dump(kernel_read, 1); 1992040fcc81SSam Ravnborg if (module_read) 1993040fcc81SSam Ravnborg read_dump(module_read, 0); 19942d04b5aeSRichard Hacker while (extsym_start) { 19952d04b5aeSRichard Hacker read_dump(extsym_start->file, 0); 19962d04b5aeSRichard Hacker extsym_iter = extsym_start->next; 19972d04b5aeSRichard Hacker free(extsym_start); 19982d04b5aeSRichard Hacker extsym_start = extsym_iter; 19992d04b5aeSRichard Hacker } 20001da177e4SLinus Torvalds 2001df578e7dSSam Ravnborg while (optind < argc) 20021da177e4SLinus Torvalds read_symbols(argv[optind++]); 20031da177e4SLinus Torvalds 20041da177e4SLinus Torvalds for (mod = modules; mod; mod = mod->next) { 20051da177e4SLinus Torvalds if (mod->skip) 20061da177e4SLinus Torvalds continue; 2007c96fca21SSam Ravnborg check_exports(mod); 2008b817f6feSSam Ravnborg } 2009b817f6feSSam Ravnborg 2010c53ddacdSKirill Korotaev err = 0; 2011c53ddacdSKirill Korotaev 2012b817f6feSSam Ravnborg for (mod = modules; mod; mod = mod->next) { 2013666ab414SAndi Kleen char fname[strlen(mod->name) + 10]; 2014666ab414SAndi Kleen 2015b817f6feSSam Ravnborg if (mod->skip) 2016b817f6feSSam Ravnborg continue; 20171da177e4SLinus Torvalds 20181da177e4SLinus Torvalds buf.pos = 0; 20191da177e4SLinus Torvalds 20201da177e4SLinus Torvalds add_header(&buf, mod); 2021a9860bf0SGreg Kroah-Hartman add_staging_flag(&buf, mod->name); 2022c53ddacdSKirill Korotaev err |= add_versions(&buf, mod); 20231da177e4SLinus Torvalds add_depends(&buf, mod, modules); 20241da177e4SLinus Torvalds add_moddevtable(&buf, mod); 20251da177e4SLinus Torvalds add_srcversion(&buf, mod); 20261da177e4SLinus Torvalds 20271da177e4SLinus Torvalds sprintf(fname, "%s.mod.c", mod->name); 20281da177e4SLinus Torvalds write_if_changed(&buf, fname); 20291da177e4SLinus Torvalds } 20301da177e4SLinus Torvalds 20311da177e4SLinus Torvalds if (dump_write) 20321da177e4SLinus Torvalds write_dump(dump_write); 2033588ccd73SSam Ravnborg if (sec_mismatch_count && !sec_mismatch_verbose) 20347c0ac495SGeert Uytterhoeven warn("modpost: Found %d section mismatch(es).\n" 2035e5f95c8bSSam Ravnborg "To see full details build your kernel with:\n" 2036e5f95c8bSSam Ravnborg "'make CONFIG_DEBUG_SECTION_MISMATCH=y'\n", 2037e5f95c8bSSam Ravnborg sec_mismatch_count); 20381da177e4SLinus Torvalds 2039c53ddacdSKirill Korotaev return err; 20401da177e4SLinus Torvalds } 2041