xref: /linux-6.15/scripts/mod/modpost.c (revision bbd3f4fb)
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 = &sectioncheck[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