xref: /linux-6.15/scripts/mod/modpost.c (revision 83cda2bb)
11da177e4SLinus Torvalds /* Postprocess module symbol versions
21da177e4SLinus Torvalds  *
31da177e4SLinus Torvalds  * Copyright 2003       Kai Germaschewski
41da177e4SLinus Torvalds  * Copyright 2002-2004  Rusty Russell, IBM Corporation
5382168f4SSam Ravnborg  * Copyright 2006       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 
141da177e4SLinus Torvalds #include <ctype.h>
151da177e4SLinus Torvalds #include "modpost.h"
16b817f6feSSam Ravnborg #include "../../include/linux/license.h"
171da177e4SLinus Torvalds 
181da177e4SLinus Torvalds /* Are we using CONFIG_MODVERSIONS? */
191da177e4SLinus Torvalds int modversions = 0;
201da177e4SLinus Torvalds /* Warn about undefined symbols? (do so if we have vmlinux) */
211da177e4SLinus Torvalds int have_vmlinux = 0;
221da177e4SLinus Torvalds /* Is CONFIG_MODULE_SRCVERSION_ALL set? */
231da177e4SLinus Torvalds static int all_versions = 0;
24040fcc81SSam Ravnborg /* If we are modposting external module set to 1 */
25040fcc81SSam Ravnborg static int external_module = 0;
268d8d8289SSam Ravnborg /* Warn about section mismatch in vmlinux if set to 1 */
278d8d8289SSam Ravnborg static int vmlinux_section_warnings = 1;
28c53ddacdSKirill Korotaev /* Only warn about unresolved symbols */
29c53ddacdSKirill Korotaev static int warn_unresolved = 0;
30bd5cbcedSRam Pai /* How a symbol is exported */
31c96fca21SSam Ravnborg enum export {
32c96fca21SSam Ravnborg 	export_plain,      export_unused,     export_gpl,
33c96fca21SSam Ravnborg 	export_unused_gpl, export_gpl_future, export_unknown
34c96fca21SSam Ravnborg };
351da177e4SLinus Torvalds 
365c3ead8cSSam Ravnborg void fatal(const char *fmt, ...)
371da177e4SLinus Torvalds {
381da177e4SLinus Torvalds 	va_list arglist;
391da177e4SLinus Torvalds 
401da177e4SLinus Torvalds 	fprintf(stderr, "FATAL: ");
411da177e4SLinus Torvalds 
421da177e4SLinus Torvalds 	va_start(arglist, fmt);
431da177e4SLinus Torvalds 	vfprintf(stderr, fmt, arglist);
441da177e4SLinus Torvalds 	va_end(arglist);
451da177e4SLinus Torvalds 
461da177e4SLinus Torvalds 	exit(1);
471da177e4SLinus Torvalds }
481da177e4SLinus Torvalds 
495c3ead8cSSam Ravnborg void warn(const char *fmt, ...)
501da177e4SLinus Torvalds {
511da177e4SLinus Torvalds 	va_list arglist;
521da177e4SLinus Torvalds 
531da177e4SLinus Torvalds 	fprintf(stderr, "WARNING: ");
541da177e4SLinus Torvalds 
551da177e4SLinus Torvalds 	va_start(arglist, fmt);
561da177e4SLinus Torvalds 	vfprintf(stderr, fmt, arglist);
571da177e4SLinus Torvalds 	va_end(arglist);
581da177e4SLinus Torvalds }
591da177e4SLinus Torvalds 
602a116659SMatthew Wilcox void merror(const char *fmt, ...)
612a116659SMatthew Wilcox {
622a116659SMatthew Wilcox 	va_list arglist;
632a116659SMatthew Wilcox 
642a116659SMatthew Wilcox 	fprintf(stderr, "ERROR: ");
652a116659SMatthew Wilcox 
662a116659SMatthew Wilcox 	va_start(arglist, fmt);
672a116659SMatthew Wilcox 	vfprintf(stderr, fmt, arglist);
682a116659SMatthew Wilcox 	va_end(arglist);
692a116659SMatthew Wilcox }
702a116659SMatthew Wilcox 
71040fcc81SSam Ravnborg static int is_vmlinux(const char *modname)
72040fcc81SSam Ravnborg {
73040fcc81SSam Ravnborg 	const char *myname;
74040fcc81SSam Ravnborg 
75040fcc81SSam Ravnborg 	if ((myname = strrchr(modname, '/')))
76040fcc81SSam Ravnborg 		myname++;
77040fcc81SSam Ravnborg 	else
78040fcc81SSam Ravnborg 		myname = modname;
79040fcc81SSam Ravnborg 
80741f98feSSam Ravnborg 	return (strcmp(myname, "vmlinux") == 0) ||
81741f98feSSam Ravnborg 	       (strcmp(myname, "vmlinux.o") == 0);
82040fcc81SSam Ravnborg }
83040fcc81SSam Ravnborg 
841da177e4SLinus Torvalds void *do_nofail(void *ptr, const char *expr)
851da177e4SLinus Torvalds {
861da177e4SLinus Torvalds 	if (!ptr) {
871da177e4SLinus Torvalds 		fatal("modpost: Memory allocation failure: %s.\n", expr);
881da177e4SLinus Torvalds 	}
891da177e4SLinus Torvalds 	return ptr;
901da177e4SLinus Torvalds }
911da177e4SLinus Torvalds 
921da177e4SLinus Torvalds /* A list of all modules we processed */
931da177e4SLinus Torvalds 
941da177e4SLinus Torvalds static struct module *modules;
951da177e4SLinus Torvalds 
965c3ead8cSSam Ravnborg static struct module *find_module(char *modname)
971da177e4SLinus Torvalds {
981da177e4SLinus Torvalds 	struct module *mod;
991da177e4SLinus Torvalds 
1001da177e4SLinus Torvalds 	for (mod = modules; mod; mod = mod->next)
1011da177e4SLinus Torvalds 		if (strcmp(mod->name, modname) == 0)
1021da177e4SLinus Torvalds 			break;
1031da177e4SLinus Torvalds 	return mod;
1041da177e4SLinus Torvalds }
1051da177e4SLinus Torvalds 
1065c3ead8cSSam Ravnborg static struct module *new_module(char *modname)
1071da177e4SLinus Torvalds {
1081da177e4SLinus Torvalds 	struct module *mod;
1091da177e4SLinus Torvalds 	char *p, *s;
1101da177e4SLinus Torvalds 
1111da177e4SLinus Torvalds 	mod = NOFAIL(malloc(sizeof(*mod)));
1121da177e4SLinus Torvalds 	memset(mod, 0, sizeof(*mod));
1131da177e4SLinus Torvalds 	p = NOFAIL(strdup(modname));
1141da177e4SLinus Torvalds 
1151da177e4SLinus Torvalds 	/* strip trailing .o */
1161da177e4SLinus Torvalds 	if ((s = strrchr(p, '.')) != NULL)
1171da177e4SLinus Torvalds 		if (strcmp(s, ".o") == 0)
1181da177e4SLinus Torvalds 			*s = '\0';
1191da177e4SLinus Torvalds 
1201da177e4SLinus Torvalds 	/* add to list */
1211da177e4SLinus Torvalds 	mod->name = p;
122b817f6feSSam Ravnborg 	mod->gpl_compatible = -1;
1231da177e4SLinus Torvalds 	mod->next = modules;
1241da177e4SLinus Torvalds 	modules = mod;
1251da177e4SLinus Torvalds 
1261da177e4SLinus Torvalds 	return mod;
1271da177e4SLinus Torvalds }
1281da177e4SLinus Torvalds 
1291da177e4SLinus Torvalds /* A hash of all exported symbols,
1301da177e4SLinus Torvalds  * struct symbol is also used for lists of unresolved symbols */
1311da177e4SLinus Torvalds 
1321da177e4SLinus Torvalds #define SYMBOL_HASH_SIZE 1024
1331da177e4SLinus Torvalds 
1341da177e4SLinus Torvalds struct symbol {
1351da177e4SLinus Torvalds 	struct symbol *next;
1361da177e4SLinus Torvalds 	struct module *module;
1371da177e4SLinus Torvalds 	unsigned int crc;
1381da177e4SLinus Torvalds 	int crc_valid;
1391da177e4SLinus Torvalds 	unsigned int weak:1;
140040fcc81SSam Ravnborg 	unsigned int vmlinux:1;    /* 1 if symbol is defined in vmlinux */
141040fcc81SSam Ravnborg 	unsigned int kernel:1;     /* 1 if symbol is from kernel
142040fcc81SSam Ravnborg 				    *  (only for external modules) **/
1438e70c458SSam Ravnborg 	unsigned int preloaded:1;  /* 1 if symbol from Module.symvers */
144bd5cbcedSRam Pai 	enum export  export;       /* Type of export */
1451da177e4SLinus Torvalds 	char name[0];
1461da177e4SLinus Torvalds };
1471da177e4SLinus Torvalds 
1481da177e4SLinus Torvalds static struct symbol *symbolhash[SYMBOL_HASH_SIZE];
1491da177e4SLinus Torvalds 
1501da177e4SLinus Torvalds /* This is based on the hash agorithm from gdbm, via tdb */
1511da177e4SLinus Torvalds static inline unsigned int tdb_hash(const char *name)
1521da177e4SLinus Torvalds {
1531da177e4SLinus Torvalds 	unsigned value;	/* Used to compute the hash value.  */
1541da177e4SLinus Torvalds 	unsigned   i;	/* Used to cycle through random values. */
1551da177e4SLinus Torvalds 
1561da177e4SLinus Torvalds 	/* Set the initial value from the key size. */
1571da177e4SLinus Torvalds 	for (value = 0x238F13AF * strlen(name), i=0; name[i]; i++)
1581da177e4SLinus Torvalds 		value = (value + (((unsigned char *)name)[i] << (i*5 % 24)));
1591da177e4SLinus Torvalds 
1601da177e4SLinus Torvalds 	return (1103515243 * value + 12345);
1611da177e4SLinus Torvalds }
1621da177e4SLinus Torvalds 
1635c3ead8cSSam Ravnborg /**
1645c3ead8cSSam Ravnborg  * Allocate a new symbols for use in the hash of exported symbols or
1655c3ead8cSSam Ravnborg  * the list of unresolved symbols per module
1665c3ead8cSSam Ravnborg  **/
1675c3ead8cSSam Ravnborg static struct symbol *alloc_symbol(const char *name, unsigned int weak,
1685c3ead8cSSam Ravnborg 				   struct symbol *next)
1691da177e4SLinus Torvalds {
1701da177e4SLinus Torvalds 	struct symbol *s = NOFAIL(malloc(sizeof(*s) + strlen(name) + 1));
1711da177e4SLinus Torvalds 
1721da177e4SLinus Torvalds 	memset(s, 0, sizeof(*s));
1731da177e4SLinus Torvalds 	strcpy(s->name, name);
1741da177e4SLinus Torvalds 	s->weak = weak;
1751da177e4SLinus Torvalds 	s->next = next;
1761da177e4SLinus Torvalds 	return s;
1771da177e4SLinus Torvalds }
1781da177e4SLinus Torvalds 
1791da177e4SLinus Torvalds /* For the hash of exported symbols */
180bd5cbcedSRam Pai static struct symbol *new_symbol(const char *name, struct module *module,
181bd5cbcedSRam Pai 				 enum export export)
1821da177e4SLinus Torvalds {
1831da177e4SLinus Torvalds 	unsigned int hash;
1841da177e4SLinus Torvalds 	struct symbol *new;
1851da177e4SLinus Torvalds 
1861da177e4SLinus Torvalds 	hash = tdb_hash(name) % SYMBOL_HASH_SIZE;
1871da177e4SLinus Torvalds 	new = symbolhash[hash] = alloc_symbol(name, 0, symbolhash[hash]);
1881da177e4SLinus Torvalds 	new->module = module;
189bd5cbcedSRam Pai 	new->export = export;
190040fcc81SSam Ravnborg 	return new;
1911da177e4SLinus Torvalds }
1921da177e4SLinus Torvalds 
1935c3ead8cSSam Ravnborg static struct symbol *find_symbol(const char *name)
1941da177e4SLinus Torvalds {
1951da177e4SLinus Torvalds 	struct symbol *s;
1961da177e4SLinus Torvalds 
1971da177e4SLinus Torvalds 	/* For our purposes, .foo matches foo.  PPC64 needs this. */
1981da177e4SLinus Torvalds 	if (name[0] == '.')
1991da177e4SLinus Torvalds 		name++;
2001da177e4SLinus Torvalds 
2011da177e4SLinus Torvalds 	for (s = symbolhash[tdb_hash(name) % SYMBOL_HASH_SIZE]; s; s=s->next) {
2021da177e4SLinus Torvalds 		if (strcmp(s->name, name) == 0)
2031da177e4SLinus Torvalds 			return s;
2041da177e4SLinus Torvalds 	}
2051da177e4SLinus Torvalds 	return NULL;
2061da177e4SLinus Torvalds }
2071da177e4SLinus Torvalds 
208bd5cbcedSRam Pai static struct {
209bd5cbcedSRam Pai 	const char *str;
210bd5cbcedSRam Pai 	enum export export;
211bd5cbcedSRam Pai } export_list[] = {
212bd5cbcedSRam Pai 	{ .str = "EXPORT_SYMBOL",            .export = export_plain },
213c96fca21SSam Ravnborg 	{ .str = "EXPORT_UNUSED_SYMBOL",     .export = export_unused },
214bd5cbcedSRam Pai 	{ .str = "EXPORT_SYMBOL_GPL",        .export = export_gpl },
215c96fca21SSam Ravnborg 	{ .str = "EXPORT_UNUSED_SYMBOL_GPL", .export = export_unused_gpl },
216bd5cbcedSRam Pai 	{ .str = "EXPORT_SYMBOL_GPL_FUTURE", .export = export_gpl_future },
217bd5cbcedSRam Pai 	{ .str = "(unknown)",                .export = export_unknown },
218bd5cbcedSRam Pai };
219bd5cbcedSRam Pai 
220bd5cbcedSRam Pai 
221bd5cbcedSRam Pai static const char *export_str(enum export ex)
222bd5cbcedSRam Pai {
223bd5cbcedSRam Pai 	return export_list[ex].str;
224bd5cbcedSRam Pai }
225bd5cbcedSRam Pai 
226bd5cbcedSRam Pai static enum export export_no(const char * s)
227bd5cbcedSRam Pai {
228bd5cbcedSRam Pai 	int i;
229534b89a9SSam Ravnborg 	if (!s)
230534b89a9SSam Ravnborg 		return export_unknown;
231bd5cbcedSRam Pai 	for (i = 0; export_list[i].export != export_unknown; i++) {
232bd5cbcedSRam Pai 		if (strcmp(export_list[i].str, s) == 0)
233bd5cbcedSRam Pai 			return export_list[i].export;
234bd5cbcedSRam Pai 	}
235bd5cbcedSRam Pai 	return export_unknown;
236bd5cbcedSRam Pai }
237bd5cbcedSRam Pai 
238bd5cbcedSRam Pai static enum export export_from_sec(struct elf_info *elf, Elf_Section sec)
239bd5cbcedSRam Pai {
240bd5cbcedSRam Pai 	if (sec == elf->export_sec)
241bd5cbcedSRam Pai 		return export_plain;
242c96fca21SSam Ravnborg 	else if (sec == elf->export_unused_sec)
243c96fca21SSam Ravnborg 		return export_unused;
244bd5cbcedSRam Pai 	else if (sec == elf->export_gpl_sec)
245bd5cbcedSRam Pai 		return export_gpl;
246c96fca21SSam Ravnborg 	else if (sec == elf->export_unused_gpl_sec)
247c96fca21SSam Ravnborg 		return export_unused_gpl;
248bd5cbcedSRam Pai 	else if (sec == elf->export_gpl_future_sec)
249bd5cbcedSRam Pai 		return export_gpl_future;
250bd5cbcedSRam Pai 	else
251bd5cbcedSRam Pai 		return export_unknown;
252bd5cbcedSRam Pai }
253bd5cbcedSRam Pai 
2545c3ead8cSSam Ravnborg /**
2555c3ead8cSSam Ravnborg  * Add an exported symbol - it may have already been added without a
2565c3ead8cSSam Ravnborg  * CRC, in this case just update the CRC
2575c3ead8cSSam Ravnborg  **/
258bd5cbcedSRam Pai static struct symbol *sym_add_exported(const char *name, struct module *mod,
259bd5cbcedSRam Pai 				       enum export export)
2601da177e4SLinus Torvalds {
2611da177e4SLinus Torvalds 	struct symbol *s = find_symbol(name);
2621da177e4SLinus Torvalds 
2631da177e4SLinus Torvalds 	if (!s) {
264bd5cbcedSRam Pai 		s = new_symbol(name, mod, export);
2658e70c458SSam Ravnborg 	} else {
2668e70c458SSam Ravnborg 		if (!s->preloaded) {
2677b75b13cSSam Ravnborg 			warn("%s: '%s' exported twice. Previous export "
2688e70c458SSam Ravnborg 			     "was in %s%s\n", mod->name, name,
2698e70c458SSam Ravnborg 			     s->module->name,
2708e70c458SSam Ravnborg 			     is_vmlinux(s->module->name) ?"":".ko");
2711da177e4SLinus Torvalds 		}
2721da177e4SLinus Torvalds 	}
2738e70c458SSam Ravnborg 	s->preloaded = 0;
274040fcc81SSam Ravnborg 	s->vmlinux   = is_vmlinux(mod->name);
275040fcc81SSam Ravnborg 	s->kernel    = 0;
276bd5cbcedSRam Pai 	s->export    = export;
277040fcc81SSam Ravnborg 	return s;
2781da177e4SLinus Torvalds }
2791da177e4SLinus Torvalds 
280040fcc81SSam Ravnborg static void sym_update_crc(const char *name, struct module *mod,
281bd5cbcedSRam Pai 			   unsigned int crc, enum export export)
282040fcc81SSam Ravnborg {
283040fcc81SSam Ravnborg 	struct symbol *s = find_symbol(name);
284040fcc81SSam Ravnborg 
285040fcc81SSam Ravnborg 	if (!s)
286bd5cbcedSRam Pai 		s = new_symbol(name, mod, export);
287040fcc81SSam Ravnborg 	s->crc = crc;
2881da177e4SLinus Torvalds 	s->crc_valid = 1;
2891da177e4SLinus Torvalds }
2901da177e4SLinus Torvalds 
2915c3ead8cSSam Ravnborg void *grab_file(const char *filename, unsigned long *size)
2921da177e4SLinus Torvalds {
2931da177e4SLinus Torvalds 	struct stat st;
2941da177e4SLinus Torvalds 	void *map;
2951da177e4SLinus Torvalds 	int fd;
2961da177e4SLinus Torvalds 
2971da177e4SLinus Torvalds 	fd = open(filename, O_RDONLY);
2981da177e4SLinus Torvalds 	if (fd < 0 || fstat(fd, &st) != 0)
2991da177e4SLinus Torvalds 		return NULL;
3001da177e4SLinus Torvalds 
3011da177e4SLinus Torvalds 	*size = st.st_size;
3021da177e4SLinus Torvalds 	map = mmap(NULL, *size, PROT_READ|PROT_WRITE, MAP_PRIVATE, fd, 0);
3031da177e4SLinus Torvalds 	close(fd);
3041da177e4SLinus Torvalds 
3051da177e4SLinus Torvalds 	if (map == MAP_FAILED)
3061da177e4SLinus Torvalds 		return NULL;
3071da177e4SLinus Torvalds 	return map;
3081da177e4SLinus Torvalds }
3091da177e4SLinus Torvalds 
3105c3ead8cSSam Ravnborg /**
3115c3ead8cSSam Ravnborg   * Return a copy of the next line in a mmap'ed file.
3125c3ead8cSSam Ravnborg   * spaces in the beginning of the line is trimmed away.
3135c3ead8cSSam Ravnborg   * Return a pointer to a static buffer.
3145c3ead8cSSam Ravnborg   **/
3155c3ead8cSSam Ravnborg char* get_next_line(unsigned long *pos, void *file, unsigned long size)
3161da177e4SLinus Torvalds {
3171da177e4SLinus Torvalds 	static char line[4096];
3181da177e4SLinus Torvalds 	int skip = 1;
3191da177e4SLinus Torvalds 	size_t len = 0;
3201da177e4SLinus Torvalds 	signed char *p = (signed char *)file + *pos;
3211da177e4SLinus Torvalds 	char *s = line;
3221da177e4SLinus Torvalds 
3231da177e4SLinus Torvalds 	for (; *pos < size ; (*pos)++)
3241da177e4SLinus Torvalds 	{
3251da177e4SLinus Torvalds 		if (skip && isspace(*p)) {
3261da177e4SLinus Torvalds 			p++;
3271da177e4SLinus Torvalds 			continue;
3281da177e4SLinus Torvalds 		}
3291da177e4SLinus Torvalds 		skip = 0;
3301da177e4SLinus Torvalds 		if (*p != '\n' && (*pos < size)) {
3311da177e4SLinus Torvalds 			len++;
3321da177e4SLinus Torvalds 			*s++ = *p++;
3331da177e4SLinus Torvalds 			if (len > 4095)
3341da177e4SLinus Torvalds 				break; /* Too long, stop */
3351da177e4SLinus Torvalds 		} else {
3361da177e4SLinus Torvalds 			/* End of string */
3371da177e4SLinus Torvalds 			*s = '\0';
3381da177e4SLinus Torvalds 			return line;
3391da177e4SLinus Torvalds 		}
3401da177e4SLinus Torvalds 	}
3411da177e4SLinus Torvalds 	/* End of buffer */
3421da177e4SLinus Torvalds 	return NULL;
3431da177e4SLinus Torvalds }
3441da177e4SLinus Torvalds 
3455c3ead8cSSam Ravnborg void release_file(void *file, unsigned long size)
3461da177e4SLinus Torvalds {
3471da177e4SLinus Torvalds 	munmap(file, size);
3481da177e4SLinus Torvalds }
3491da177e4SLinus Torvalds 
35085bd2fddSSam Ravnborg static int parse_elf(struct elf_info *info, const char *filename)
3511da177e4SLinus Torvalds {
3521da177e4SLinus Torvalds 	unsigned int i;
35385bd2fddSSam Ravnborg 	Elf_Ehdr *hdr;
3541da177e4SLinus Torvalds 	Elf_Shdr *sechdrs;
3551da177e4SLinus Torvalds 	Elf_Sym  *sym;
3561da177e4SLinus Torvalds 
3571da177e4SLinus Torvalds 	hdr = grab_file(filename, &info->size);
3581da177e4SLinus Torvalds 	if (!hdr) {
3591da177e4SLinus Torvalds 		perror(filename);
3606803dc0eSSam Ravnborg 		exit(1);
3611da177e4SLinus Torvalds 	}
3621da177e4SLinus Torvalds 	info->hdr = hdr;
36385bd2fddSSam Ravnborg 	if (info->size < sizeof(*hdr)) {
36485bd2fddSSam Ravnborg 		/* file too small, assume this is an empty .o file */
36585bd2fddSSam Ravnborg 		return 0;
36685bd2fddSSam Ravnborg 	}
36785bd2fddSSam Ravnborg 	/* Is this a valid ELF file? */
36885bd2fddSSam Ravnborg 	if ((hdr->e_ident[EI_MAG0] != ELFMAG0) ||
36985bd2fddSSam Ravnborg 	    (hdr->e_ident[EI_MAG1] != ELFMAG1) ||
37085bd2fddSSam Ravnborg 	    (hdr->e_ident[EI_MAG2] != ELFMAG2) ||
37185bd2fddSSam Ravnborg 	    (hdr->e_ident[EI_MAG3] != ELFMAG3)) {
37285bd2fddSSam Ravnborg 		/* Not an ELF file - silently ignore it */
37385bd2fddSSam Ravnborg 		return 0;
37485bd2fddSSam Ravnborg 	}
3751da177e4SLinus Torvalds 	/* Fix endianness in ELF header */
3761da177e4SLinus Torvalds 	hdr->e_shoff    = TO_NATIVE(hdr->e_shoff);
3771da177e4SLinus Torvalds 	hdr->e_shstrndx = TO_NATIVE(hdr->e_shstrndx);
3781da177e4SLinus Torvalds 	hdr->e_shnum    = TO_NATIVE(hdr->e_shnum);
3791da177e4SLinus Torvalds 	hdr->e_machine  = TO_NATIVE(hdr->e_machine);
380ae4ac123SAtsushi Nemoto 	hdr->e_type     = TO_NATIVE(hdr->e_type);
3811da177e4SLinus Torvalds 	sechdrs = (void *)hdr + hdr->e_shoff;
3821da177e4SLinus Torvalds 	info->sechdrs = sechdrs;
3831da177e4SLinus Torvalds 
3841da177e4SLinus Torvalds 	/* Fix endianness in section headers */
3851da177e4SLinus Torvalds 	for (i = 0; i < hdr->e_shnum; i++) {
3861da177e4SLinus Torvalds 		sechdrs[i].sh_type   = TO_NATIVE(sechdrs[i].sh_type);
3871da177e4SLinus Torvalds 		sechdrs[i].sh_offset = TO_NATIVE(sechdrs[i].sh_offset);
3881da177e4SLinus Torvalds 		sechdrs[i].sh_size   = TO_NATIVE(sechdrs[i].sh_size);
3891da177e4SLinus Torvalds 		sechdrs[i].sh_link   = TO_NATIVE(sechdrs[i].sh_link);
3901da177e4SLinus Torvalds 		sechdrs[i].sh_name   = TO_NATIVE(sechdrs[i].sh_name);
391ae4ac123SAtsushi Nemoto 		sechdrs[i].sh_info   = TO_NATIVE(sechdrs[i].sh_info);
392ae4ac123SAtsushi Nemoto 		sechdrs[i].sh_addr   = TO_NATIVE(sechdrs[i].sh_addr);
3931da177e4SLinus Torvalds 	}
3941da177e4SLinus Torvalds 	/* Find symbol table. */
3951da177e4SLinus Torvalds 	for (i = 1; i < hdr->e_shnum; i++) {
3961da177e4SLinus Torvalds 		const char *secstrings
3971da177e4SLinus Torvalds 			= (void *)hdr + sechdrs[hdr->e_shstrndx].sh_offset;
398bd5cbcedSRam Pai 		const char *secname;
3991da177e4SLinus Torvalds 
40085bd2fddSSam Ravnborg 		if (sechdrs[i].sh_offset > info->size) {
40185bd2fddSSam Ravnborg 			fatal("%s is truncated. sechdrs[i].sh_offset=%u > sizeof(*hrd)=%ul\n", filename, (unsigned int)sechdrs[i].sh_offset, sizeof(*hdr));
40285bd2fddSSam Ravnborg 			return 0;
40385bd2fddSSam Ravnborg 		}
404bd5cbcedSRam Pai 		secname = secstrings + sechdrs[i].sh_name;
405bd5cbcedSRam Pai 		if (strcmp(secname, ".modinfo") == 0) {
4061da177e4SLinus Torvalds 			info->modinfo = (void *)hdr + sechdrs[i].sh_offset;
4071da177e4SLinus Torvalds 			info->modinfo_len = sechdrs[i].sh_size;
408bd5cbcedSRam Pai 		} else if (strcmp(secname, "__ksymtab") == 0)
409bd5cbcedSRam Pai 			info->export_sec = i;
410c96fca21SSam Ravnborg 		else if (strcmp(secname, "__ksymtab_unused") == 0)
411c96fca21SSam Ravnborg 			info->export_unused_sec = i;
412bd5cbcedSRam Pai 		else if (strcmp(secname, "__ksymtab_gpl") == 0)
413bd5cbcedSRam Pai 			info->export_gpl_sec = i;
414c96fca21SSam Ravnborg 		else if (strcmp(secname, "__ksymtab_unused_gpl") == 0)
415c96fca21SSam Ravnborg 			info->export_unused_gpl_sec = i;
416bd5cbcedSRam Pai 		else if (strcmp(secname, "__ksymtab_gpl_future") == 0)
417bd5cbcedSRam Pai 			info->export_gpl_future_sec = i;
418bd5cbcedSRam Pai 
4191da177e4SLinus Torvalds 		if (sechdrs[i].sh_type != SHT_SYMTAB)
4201da177e4SLinus Torvalds 			continue;
4211da177e4SLinus Torvalds 
4221da177e4SLinus Torvalds 		info->symtab_start = (void *)hdr + sechdrs[i].sh_offset;
4231da177e4SLinus Torvalds 		info->symtab_stop  = (void *)hdr + sechdrs[i].sh_offset
4241da177e4SLinus Torvalds 			                         + sechdrs[i].sh_size;
4251da177e4SLinus Torvalds 		info->strtab       = (void *)hdr +
4261da177e4SLinus Torvalds 			             sechdrs[sechdrs[i].sh_link].sh_offset;
4271da177e4SLinus Torvalds 	}
4281da177e4SLinus Torvalds 	if (!info->symtab_start) {
429cb80514dSSam Ravnborg 		fatal("%s has no symtab?\n", filename);
4301da177e4SLinus Torvalds 	}
4311da177e4SLinus Torvalds 	/* Fix endianness in symbols */
4321da177e4SLinus Torvalds 	for (sym = info->symtab_start; sym < info->symtab_stop; sym++) {
4331da177e4SLinus Torvalds 		sym->st_shndx = TO_NATIVE(sym->st_shndx);
4341da177e4SLinus Torvalds 		sym->st_name  = TO_NATIVE(sym->st_name);
4351da177e4SLinus Torvalds 		sym->st_value = TO_NATIVE(sym->st_value);
4361da177e4SLinus Torvalds 		sym->st_size  = TO_NATIVE(sym->st_size);
4371da177e4SLinus Torvalds 	}
43885bd2fddSSam Ravnborg 	return 1;
4391da177e4SLinus Torvalds }
4401da177e4SLinus Torvalds 
4415c3ead8cSSam Ravnborg static void parse_elf_finish(struct elf_info *info)
4421da177e4SLinus Torvalds {
4431da177e4SLinus Torvalds 	release_file(info->hdr, info->size);
4441da177e4SLinus Torvalds }
4451da177e4SLinus Torvalds 
446f7b05e64SLuke Yang #define CRC_PFX     MODULE_SYMBOL_PREFIX "__crc_"
447f7b05e64SLuke Yang #define KSYMTAB_PFX MODULE_SYMBOL_PREFIX "__ksymtab_"
4481da177e4SLinus Torvalds 
4495c3ead8cSSam Ravnborg static void handle_modversions(struct module *mod, struct elf_info *info,
4501da177e4SLinus Torvalds 			       Elf_Sym *sym, const char *symname)
4511da177e4SLinus Torvalds {
4521da177e4SLinus Torvalds 	unsigned int crc;
453bd5cbcedSRam Pai 	enum export export = export_from_sec(info, sym->st_shndx);
4541da177e4SLinus Torvalds 
4551da177e4SLinus Torvalds 	switch (sym->st_shndx) {
4561da177e4SLinus Torvalds 	case SHN_COMMON:
457cb80514dSSam Ravnborg 		warn("\"%s\" [%s] is COMMON symbol\n", symname, mod->name);
4581da177e4SLinus Torvalds 		break;
4591da177e4SLinus Torvalds 	case SHN_ABS:
4601da177e4SLinus Torvalds 		/* CRC'd symbol */
4611da177e4SLinus Torvalds 		if (memcmp(symname, CRC_PFX, strlen(CRC_PFX)) == 0) {
4621da177e4SLinus Torvalds 			crc = (unsigned int) sym->st_value;
463bd5cbcedSRam Pai 			sym_update_crc(symname + strlen(CRC_PFX), mod, crc,
464bd5cbcedSRam Pai 					export);
4651da177e4SLinus Torvalds 		}
4661da177e4SLinus Torvalds 		break;
4671da177e4SLinus Torvalds 	case SHN_UNDEF:
4681da177e4SLinus Torvalds 		/* undefined symbol */
4691da177e4SLinus Torvalds 		if (ELF_ST_BIND(sym->st_info) != STB_GLOBAL &&
4701da177e4SLinus Torvalds 		    ELF_ST_BIND(sym->st_info) != STB_WEAK)
4711da177e4SLinus Torvalds 			break;
4721da177e4SLinus Torvalds 		/* ignore global offset table */
4731da177e4SLinus Torvalds 		if (strcmp(symname, "_GLOBAL_OFFSET_TABLE_") == 0)
4741da177e4SLinus Torvalds 			break;
4751da177e4SLinus Torvalds 		/* ignore __this_module, it will be resolved shortly */
4761da177e4SLinus Torvalds 		if (strcmp(symname, MODULE_SYMBOL_PREFIX "__this_module") == 0)
4771da177e4SLinus Torvalds 			break;
4788d529014SBen Colline /* cope with newer glibc (2.3.4 or higher) STT_ definition in elf.h */
4798d529014SBen Colline #if defined(STT_REGISTER) || defined(STT_SPARC_REGISTER)
4808d529014SBen Colline /* add compatibility with older glibc */
4818d529014SBen Colline #ifndef STT_SPARC_REGISTER
4828d529014SBen Colline #define STT_SPARC_REGISTER STT_REGISTER
4838d529014SBen Colline #endif
4841da177e4SLinus Torvalds 		if (info->hdr->e_machine == EM_SPARC ||
4851da177e4SLinus Torvalds 		    info->hdr->e_machine == EM_SPARCV9) {
4861da177e4SLinus Torvalds 			/* Ignore register directives. */
4878d529014SBen Colline 			if (ELF_ST_TYPE(sym->st_info) == STT_SPARC_REGISTER)
4881da177e4SLinus Torvalds 				break;
4897caaeabbSAl Viro 			if (symname[0] == '.') {
4907caaeabbSAl Viro 				char *munged = strdup(symname);
4917caaeabbSAl Viro 				munged[0] = '_';
4927caaeabbSAl Viro 				munged[1] = toupper(munged[1]);
4937caaeabbSAl Viro 				symname = munged;
4947caaeabbSAl Viro 			}
4951da177e4SLinus Torvalds 		}
4961da177e4SLinus Torvalds #endif
4971da177e4SLinus Torvalds 
4981da177e4SLinus Torvalds 		if (memcmp(symname, MODULE_SYMBOL_PREFIX,
4991da177e4SLinus Torvalds 			   strlen(MODULE_SYMBOL_PREFIX)) == 0)
5001da177e4SLinus Torvalds 			mod->unres = alloc_symbol(symname +
5011da177e4SLinus Torvalds 						  strlen(MODULE_SYMBOL_PREFIX),
5021da177e4SLinus Torvalds 						  ELF_ST_BIND(sym->st_info) == STB_WEAK,
5031da177e4SLinus Torvalds 						  mod->unres);
5041da177e4SLinus Torvalds 		break;
5051da177e4SLinus Torvalds 	default:
5061da177e4SLinus Torvalds 		/* All exported symbols */
5071da177e4SLinus Torvalds 		if (memcmp(symname, KSYMTAB_PFX, strlen(KSYMTAB_PFX)) == 0) {
508bd5cbcedSRam Pai 			sym_add_exported(symname + strlen(KSYMTAB_PFX), mod,
509bd5cbcedSRam Pai 					export);
5101da177e4SLinus Torvalds 		}
5111da177e4SLinus Torvalds 		if (strcmp(symname, MODULE_SYMBOL_PREFIX "init_module") == 0)
5121da177e4SLinus Torvalds 			mod->has_init = 1;
5131da177e4SLinus Torvalds 		if (strcmp(symname, MODULE_SYMBOL_PREFIX "cleanup_module") == 0)
5141da177e4SLinus Torvalds 			mod->has_cleanup = 1;
5151da177e4SLinus Torvalds 		break;
5161da177e4SLinus Torvalds 	}
5171da177e4SLinus Torvalds }
5181da177e4SLinus Torvalds 
5195c3ead8cSSam Ravnborg /**
5205c3ead8cSSam Ravnborg  * Parse tag=value strings from .modinfo section
5215c3ead8cSSam Ravnborg  **/
5221da177e4SLinus Torvalds static char *next_string(char *string, unsigned long *secsize)
5231da177e4SLinus Torvalds {
5241da177e4SLinus Torvalds 	/* Skip non-zero chars */
5251da177e4SLinus Torvalds 	while (string[0]) {
5261da177e4SLinus Torvalds 		string++;
5271da177e4SLinus Torvalds 		if ((*secsize)-- <= 1)
5281da177e4SLinus Torvalds 			return NULL;
5291da177e4SLinus Torvalds 	}
5301da177e4SLinus Torvalds 
5311da177e4SLinus Torvalds 	/* Skip any zero padding. */
5321da177e4SLinus Torvalds 	while (!string[0]) {
5331da177e4SLinus Torvalds 		string++;
5341da177e4SLinus Torvalds 		if ((*secsize)-- <= 1)
5351da177e4SLinus Torvalds 			return NULL;
5361da177e4SLinus Torvalds 	}
5371da177e4SLinus Torvalds 	return string;
5381da177e4SLinus Torvalds }
5391da177e4SLinus Torvalds 
540b817f6feSSam Ravnborg static char *get_next_modinfo(void *modinfo, unsigned long modinfo_len,
541b817f6feSSam Ravnborg 			      const char *tag, char *info)
5421da177e4SLinus Torvalds {
5431da177e4SLinus Torvalds 	char *p;
5441da177e4SLinus Torvalds 	unsigned int taglen = strlen(tag);
5451da177e4SLinus Torvalds 	unsigned long size = modinfo_len;
5461da177e4SLinus Torvalds 
547b817f6feSSam Ravnborg 	if (info) {
548b817f6feSSam Ravnborg 		size -= info - (char *)modinfo;
549b817f6feSSam Ravnborg 		modinfo = next_string(info, &size);
550b817f6feSSam Ravnborg 	}
551b817f6feSSam Ravnborg 
5521da177e4SLinus Torvalds 	for (p = modinfo; p; p = next_string(p, &size)) {
5531da177e4SLinus Torvalds 		if (strncmp(p, tag, taglen) == 0 && p[taglen] == '=')
5541da177e4SLinus Torvalds 			return p + taglen + 1;
5551da177e4SLinus Torvalds 	}
5561da177e4SLinus Torvalds 	return NULL;
5571da177e4SLinus Torvalds }
5581da177e4SLinus Torvalds 
559b817f6feSSam Ravnborg static char *get_modinfo(void *modinfo, unsigned long modinfo_len,
560b817f6feSSam Ravnborg 			 const char *tag)
561b817f6feSSam Ravnborg 
562b817f6feSSam Ravnborg {
563b817f6feSSam Ravnborg 	return get_next_modinfo(modinfo, modinfo_len, tag, NULL);
564b817f6feSSam Ravnborg }
565b817f6feSSam Ravnborg 
56693684d3bSSam Ravnborg /**
5674c8fbca5SSam Ravnborg  * Test if string s ends in string sub
5684c8fbca5SSam Ravnborg  * return 0 if match
5694c8fbca5SSam Ravnborg  **/
5704c8fbca5SSam Ravnborg static int strrcmp(const char *s, const char *sub)
5714c8fbca5SSam Ravnborg {
5724c8fbca5SSam Ravnborg         int slen, sublen;
5734c8fbca5SSam Ravnborg 
5744c8fbca5SSam Ravnborg 	if (!s || !sub)
5754c8fbca5SSam Ravnborg 		return 1;
5764c8fbca5SSam Ravnborg 
5774c8fbca5SSam Ravnborg 	slen = strlen(s);
5784c8fbca5SSam Ravnborg         sublen = strlen(sub);
5794c8fbca5SSam Ravnborg 
5804c8fbca5SSam Ravnborg 	if ((slen == 0) || (sublen == 0))
5814c8fbca5SSam Ravnborg 		return 1;
5824c8fbca5SSam Ravnborg 
5834c8fbca5SSam Ravnborg         if (sublen > slen)
5844c8fbca5SSam Ravnborg                 return 1;
5854c8fbca5SSam Ravnborg 
5864c8fbca5SSam Ravnborg         return memcmp(s + slen - sublen, sub, sublen);
5874c8fbca5SSam Ravnborg }
5884c8fbca5SSam Ravnborg 
5892f5ee619SSam Ravnborg /*
5902f5ee619SSam Ravnborg  * Functions used only during module init is marked __init and is stored in
5912f5ee619SSam Ravnborg  * a .init.text section. Likewise data is marked __initdata and stored in
5922f5ee619SSam Ravnborg  * a .init.data section.
5932f5ee619SSam Ravnborg  * If this section is one of these sections return 1
5942f5ee619SSam Ravnborg  * See include/linux/init.h for the details
5952f5ee619SSam Ravnborg  */
5962f5ee619SSam Ravnborg static int init_section(const char *name)
5972f5ee619SSam Ravnborg {
5982f5ee619SSam Ravnborg 	if (strcmp(name, ".init") == 0)
5992f5ee619SSam Ravnborg 		return 1;
6002f5ee619SSam Ravnborg 	if (strncmp(name, ".init.", strlen(".init.")) == 0)
6012f5ee619SSam Ravnborg 		return 1;
6022f5ee619SSam Ravnborg 	return 0;
6032f5ee619SSam Ravnborg }
6042f5ee619SSam Ravnborg 
6052f5ee619SSam Ravnborg /*
6062f5ee619SSam Ravnborg  * Functions used only during module exit is marked __exit and is stored in
6072f5ee619SSam Ravnborg  * a .exit.text section. Likewise data is marked __exitdata and stored in
6082f5ee619SSam Ravnborg  * a .exit.data section.
6092f5ee619SSam Ravnborg  * If this section is one of these sections return 1
6102f5ee619SSam Ravnborg  * See include/linux/init.h for the details
6112f5ee619SSam Ravnborg  **/
6122f5ee619SSam Ravnborg static int exit_section(const char *name)
6132f5ee619SSam Ravnborg {
6142f5ee619SSam Ravnborg 	if (strcmp(name, ".exit.text") == 0)
6152f5ee619SSam Ravnborg 		return 1;
6162f5ee619SSam Ravnborg 	if (strcmp(name, ".exit.data") == 0)
6172f5ee619SSam Ravnborg 		return 1;
6182f5ee619SSam Ravnborg 	return 0;
6192f5ee619SSam Ravnborg 
6202f5ee619SSam Ravnborg }
6212f5ee619SSam Ravnborg 
6222f5ee619SSam Ravnborg /*
6232f5ee619SSam Ravnborg  * Data sections are named like this:
6242f5ee619SSam Ravnborg  * .data | .data.rel | .data.rel.*
6252f5ee619SSam Ravnborg  * Return 1 if the specified section is a data section
6262f5ee619SSam Ravnborg  */
6272f5ee619SSam Ravnborg static int data_section(const char *name)
6282f5ee619SSam Ravnborg {
6292f5ee619SSam Ravnborg 	if ((strcmp(name, ".data") == 0) ||
6302f5ee619SSam Ravnborg 	    (strcmp(name, ".data.rel") == 0) ||
6312f5ee619SSam Ravnborg 	    (strncmp(name, ".data.rel.", strlen(".data.rel.")) == 0))
6322f5ee619SSam Ravnborg 		return 1;
6332f5ee619SSam Ravnborg 	else
6342f5ee619SSam Ravnborg 		return 0;
6352f5ee619SSam Ravnborg }
6362f5ee619SSam Ravnborg 
6374c8fbca5SSam Ravnborg /**
6384c8fbca5SSam Ravnborg  * Whitelist to allow certain references to pass with no warning.
6390e0d314eSSam Ravnborg  *
6400e0d314eSSam Ravnborg  * Pattern 0:
6410e0d314eSSam Ravnborg  *   Do not warn if funtion/data are marked with __init_refok/__initdata_refok.
6420e0d314eSSam Ravnborg  *   The pattern is identified by:
6430e0d314eSSam Ravnborg  *   fromsec = .text.init.refok | .data.init.refok
6440e0d314eSSam Ravnborg  *
6454c8fbca5SSam Ravnborg  * Pattern 1:
6464c8fbca5SSam Ravnborg  *   If a module parameter is declared __initdata and permissions=0
6474c8fbca5SSam Ravnborg  *   then this is legal despite the warning generated.
6484c8fbca5SSam Ravnborg  *   We cannot see value of permissions here, so just ignore
6494c8fbca5SSam Ravnborg  *   this pattern.
6504c8fbca5SSam Ravnborg  *   The pattern is identified by:
6514c8fbca5SSam Ravnborg  *   tosec   = .init.data
6529209aed0SSam Ravnborg  *   fromsec = .data*
6534c8fbca5SSam Ravnborg  *   atsym   =__param*
6544c8fbca5SSam Ravnborg  *
6554c8fbca5SSam Ravnborg  * Pattern 2:
65672ee59b5SRandy Dunlap  *   Many drivers utilise a *driver container with references to
6574c8fbca5SSam Ravnborg  *   add, remove, probe functions etc.
6584c8fbca5SSam Ravnborg  *   These functions may often be marked __init and we do not want to
6594c8fbca5SSam Ravnborg  *   warn here.
6604c8fbca5SSam Ravnborg  *   the pattern is identified by:
661*83cda2bbSSam Ravnborg  *   tosec   = init or exit section
662*83cda2bbSSam Ravnborg  *   fromsec = data section
6631e29a706SSam Ravnborg  *   atsym = *driver, *_template, *_sht, *_ops, *_probe, *probe_one, *_console, *_timer
664ee6a8545SVivek Goyal  *
665ee6a8545SVivek Goyal  * Pattern 3:
6669bf8cb9bSSam Ravnborg  *   Whitelist all refereces from .text.head to .init.data
6679bf8cb9bSSam Ravnborg  *   Whitelist all refereces from .text.head to .init.text
6689bf8cb9bSSam Ravnborg  *
6691d8af559SSam Ravnborg  * Pattern 4:
670ee6a8545SVivek Goyal  *   Some symbols belong to init section but still it is ok to reference
671ee6a8545SVivek Goyal  *   these from non-init sections as these symbols don't have any memory
672ee6a8545SVivek Goyal  *   allocated for them and symbol address and value are same. So even
673ee6a8545SVivek Goyal  *   if init section is freed, its ok to reference those symbols.
674ee6a8545SVivek Goyal  *   For ex. symbols marking the init section boundaries.
675ee6a8545SVivek Goyal  *   This pattern is identified by
676ee6a8545SVivek Goyal  *   refsymname = __init_begin, _sinittext, _einittext
6779bf8cb9bSSam Ravnborg  *
6784c8fbca5SSam Ravnborg  **/
6799e157a5aSMagnus Damm static int secref_whitelist(const char *modname, const char *tosec,
680ee6a8545SVivek Goyal 			    const char *fromsec, const char *atsym,
681ee6a8545SVivek Goyal 			    const char *refsymname)
6824c8fbca5SSam Ravnborg {
6834c8fbca5SSam Ravnborg 	const char **s;
6844c8fbca5SSam Ravnborg 	const char *pat2sym[] = {
68572ee59b5SRandy Dunlap 		"driver",
6865ecdd0f6SSam Ravnborg 		"_template", /* scsi uses *_template a lot */
6871e29a706SSam Ravnborg 		"_timer",    /* arm uses ops structures named _timer a lot */
6885ecdd0f6SSam Ravnborg 		"_sht",      /* scsi also used *_sht to some extent */
6894c8fbca5SSam Ravnborg 		"_ops",
6904c8fbca5SSam Ravnborg 		"_probe",
6914c8fbca5SSam Ravnborg 		"_probe_one",
692118c0aceSVivek Goyal 		"_console",
6934c8fbca5SSam Ravnborg 		NULL
6944c8fbca5SSam Ravnborg 	};
6954c8fbca5SSam Ravnborg 
696ee6a8545SVivek Goyal 	const char *pat3refsym[] = {
697ee6a8545SVivek Goyal 		"__init_begin",
698ee6a8545SVivek Goyal 		"_sinittext",
699ee6a8545SVivek Goyal 		"_einittext",
700ee6a8545SVivek Goyal 		NULL
701ee6a8545SVivek Goyal 	};
702ee6a8545SVivek Goyal 
7030e0d314eSSam Ravnborg 	/* Check for pattern 0 */
7040e0d314eSSam Ravnborg 	if ((strcmp(fromsec, ".text.init.refok") == 0) ||
7050e0d314eSSam Ravnborg 	    (strcmp(fromsec, ".data.init.refok") == 0))
7060e0d314eSSam Ravnborg 		return 1;
7070e0d314eSSam Ravnborg 
7084c8fbca5SSam Ravnborg 	/* Check for pattern 1 */
709*83cda2bbSSam Ravnborg 	if ((strcmp(tosec, ".init.data") == 0) &&
710*83cda2bbSSam Ravnborg 	    (strncmp(fromsec, ".data", strlen(".data")) == 0) &&
711*83cda2bbSSam Ravnborg 	    (strncmp(atsym, "__param", strlen("__param")) == 0))
712*83cda2bbSSam Ravnborg 		return 1;
7134c8fbca5SSam Ravnborg 
7144c8fbca5SSam Ravnborg 	/* Check for pattern 2 */
715*83cda2bbSSam Ravnborg 	if ((init_section(tosec) || exit_section(tosec)) && data_section(fromsec))
7164c8fbca5SSam Ravnborg 		for (s = pat2sym; *s; s++)
7174c8fbca5SSam Ravnborg 			if (strrcmp(atsym, *s) == 0)
7189e157a5aSMagnus Damm 				return 1;
7194c8fbca5SSam Ravnborg 
7209bf8cb9bSSam Ravnborg 	/* Check for pattern 3 */
721f8657e1bSVivek Goyal 	if ((strcmp(fromsec, ".text.head") == 0) &&
722f8657e1bSVivek Goyal 		((strcmp(tosec, ".init.data") == 0) ||
723f8657e1bSVivek Goyal 		(strcmp(tosec, ".init.text") == 0)))
724f8657e1bSVivek Goyal 	return 1;
725f8657e1bSVivek Goyal 
7261d8af559SSam Ravnborg 	/* Check for pattern 4 */
727ee6a8545SVivek Goyal 	for (s = pat3refsym; *s; s++)
728ee6a8545SVivek Goyal 		if (strcmp(refsymname, *s) == 0)
729ee6a8545SVivek Goyal 			return 1;
7309bf8cb9bSSam Ravnborg 
73193659af1SSam Ravnborg 	return 0;
7324c8fbca5SSam Ravnborg }
7334c8fbca5SSam Ravnborg 
7344c8fbca5SSam Ravnborg /**
73593684d3bSSam Ravnborg  * Find symbol based on relocation record info.
73693684d3bSSam Ravnborg  * In some cases the symbol supplied is a valid symbol so
73793684d3bSSam Ravnborg  * return refsym. If st_name != 0 we assume this is a valid symbol.
73893684d3bSSam Ravnborg  * In other cases the symbol needs to be looked up in the symbol table
73993684d3bSSam Ravnborg  * based on section and address.
74093684d3bSSam Ravnborg  *  **/
74193684d3bSSam Ravnborg static Elf_Sym *find_elf_symbol(struct elf_info *elf, Elf_Addr addr,
74293684d3bSSam Ravnborg 				Elf_Sym *relsym)
74393684d3bSSam Ravnborg {
74493684d3bSSam Ravnborg 	Elf_Sym *sym;
74593684d3bSSam Ravnborg 
74693684d3bSSam Ravnborg 	if (relsym->st_name != 0)
74793684d3bSSam Ravnborg 		return relsym;
74893684d3bSSam Ravnborg 	for (sym = elf->symtab_start; sym < elf->symtab_stop; sym++) {
74993684d3bSSam Ravnborg 		if (sym->st_shndx != relsym->st_shndx)
75093684d3bSSam Ravnborg 			continue;
751ae4ac123SAtsushi Nemoto 		if (ELF_ST_TYPE(sym->st_info) == STT_SECTION)
752ae4ac123SAtsushi Nemoto 			continue;
75393684d3bSSam Ravnborg 		if (sym->st_value == addr)
75493684d3bSSam Ravnborg 			return sym;
75593684d3bSSam Ravnborg 	}
75693684d3bSSam Ravnborg 	return NULL;
75793684d3bSSam Ravnborg }
75893684d3bSSam Ravnborg 
759da68d61fSDavid Brownell static inline int is_arm_mapping_symbol(const char *str)
760da68d61fSDavid Brownell {
761da68d61fSDavid Brownell 	return str[0] == '$' && strchr("atd", str[1])
762da68d61fSDavid Brownell 	       && (str[2] == '\0' || str[2] == '.');
763da68d61fSDavid Brownell }
764da68d61fSDavid Brownell 
765da68d61fSDavid Brownell /*
766da68d61fSDavid Brownell  * If there's no name there, ignore it; likewise, ignore it if it's
767da68d61fSDavid Brownell  * one of the magic symbols emitted used by current ARM tools.
768da68d61fSDavid Brownell  *
769da68d61fSDavid Brownell  * Otherwise if find_symbols_between() returns those symbols, they'll
770da68d61fSDavid Brownell  * fail the whitelist tests and cause lots of false alarms ... fixable
771da68d61fSDavid Brownell  * only by merging __exit and __init sections into __text, bloating
772da68d61fSDavid Brownell  * the kernel (which is especially evil on embedded platforms).
773da68d61fSDavid Brownell  */
774da68d61fSDavid Brownell static inline int is_valid_name(struct elf_info *elf, Elf_Sym *sym)
775da68d61fSDavid Brownell {
776da68d61fSDavid Brownell 	const char *name = elf->strtab + sym->st_name;
777da68d61fSDavid Brownell 
778da68d61fSDavid Brownell 	if (!name || !strlen(name))
779da68d61fSDavid Brownell 		return 0;
780da68d61fSDavid Brownell 	return !is_arm_mapping_symbol(name);
781da68d61fSDavid Brownell }
782da68d61fSDavid Brownell 
783b39927cfSSam Ravnborg /*
78443c74d17SSam Ravnborg  * Find symbols before or equal addr and after addr - in the section sec.
78543c74d17SSam Ravnborg  * If we find two symbols with equal offset prefer one with a valid name.
78643c74d17SSam Ravnborg  * The ELF format may have a better way to detect what type of symbol
78743c74d17SSam Ravnborg  * it is, but this works for now.
788b39927cfSSam Ravnborg  **/
789b39927cfSSam Ravnborg static void find_symbols_between(struct elf_info *elf, Elf_Addr addr,
790b39927cfSSam Ravnborg 				 const char *sec,
791b39927cfSSam Ravnborg 			         Elf_Sym **before, Elf_Sym **after)
792b39927cfSSam Ravnborg {
793b39927cfSSam Ravnborg 	Elf_Sym *sym;
794b39927cfSSam Ravnborg 	Elf_Ehdr *hdr = elf->hdr;
795b39927cfSSam Ravnborg 	Elf_Addr beforediff = ~0;
796b39927cfSSam Ravnborg 	Elf_Addr afterdiff = ~0;
797b39927cfSSam Ravnborg 	const char *secstrings = (void *)hdr +
798b39927cfSSam Ravnborg 				 elf->sechdrs[hdr->e_shstrndx].sh_offset;
799b39927cfSSam Ravnborg 
800b39927cfSSam Ravnborg 	*before = NULL;
801b39927cfSSam Ravnborg 	*after = NULL;
802b39927cfSSam Ravnborg 
803b39927cfSSam Ravnborg 	for (sym = elf->symtab_start; sym < elf->symtab_stop; sym++) {
804b39927cfSSam Ravnborg 		const char *symsec;
805b39927cfSSam Ravnborg 
806b39927cfSSam Ravnborg 		if (sym->st_shndx >= SHN_LORESERVE)
807b39927cfSSam Ravnborg 			continue;
808b39927cfSSam Ravnborg 		symsec = secstrings + elf->sechdrs[sym->st_shndx].sh_name;
809b39927cfSSam Ravnborg 		if (strcmp(symsec, sec) != 0)
810b39927cfSSam Ravnborg 			continue;
811da68d61fSDavid Brownell 		if (!is_valid_name(elf, sym))
812da68d61fSDavid Brownell 			continue;
813b39927cfSSam Ravnborg 		if (sym->st_value <= addr) {
814b39927cfSSam Ravnborg 			if ((addr - sym->st_value) < beforediff) {
815b39927cfSSam Ravnborg 				beforediff = addr - sym->st_value;
816b39927cfSSam Ravnborg 				*before = sym;
817b39927cfSSam Ravnborg 			}
81843c74d17SSam Ravnborg 			else if ((addr - sym->st_value) == beforediff) {
81943c74d17SSam Ravnborg 				*before = sym;
82043c74d17SSam Ravnborg 			}
821b39927cfSSam Ravnborg 		}
822b39927cfSSam Ravnborg 		else
823b39927cfSSam Ravnborg 		{
824b39927cfSSam Ravnborg 			if ((sym->st_value - addr) < afterdiff) {
825b39927cfSSam Ravnborg 				afterdiff = sym->st_value - addr;
826b39927cfSSam Ravnborg 				*after = sym;
827b39927cfSSam Ravnborg 			}
82843c74d17SSam Ravnborg 			else if ((sym->st_value - addr) == afterdiff) {
82943c74d17SSam Ravnborg 				*after = sym;
83043c74d17SSam Ravnborg 			}
831b39927cfSSam Ravnborg 		}
832b39927cfSSam Ravnborg 	}
833b39927cfSSam Ravnborg }
834b39927cfSSam Ravnborg 
835b39927cfSSam Ravnborg /**
836b39927cfSSam Ravnborg  * Print a warning about a section mismatch.
837b39927cfSSam Ravnborg  * Try to find symbols near it so user can find it.
8384c8fbca5SSam Ravnborg  * Check whitelist before warning - it may be a false positive.
839b39927cfSSam Ravnborg  **/
840b39927cfSSam Ravnborg static void warn_sec_mismatch(const char *modname, const char *fromsec,
841b39927cfSSam Ravnborg 			      struct elf_info *elf, Elf_Sym *sym, Elf_Rela r)
842b39927cfSSam Ravnborg {
84393684d3bSSam Ravnborg 	const char *refsymname = "";
84493684d3bSSam Ravnborg 	Elf_Sym *before, *after;
84593684d3bSSam Ravnborg 	Elf_Sym *refsym;
846b39927cfSSam Ravnborg 	Elf_Ehdr *hdr = elf->hdr;
847b39927cfSSam Ravnborg 	Elf_Shdr *sechdrs = elf->sechdrs;
848b39927cfSSam Ravnborg 	const char *secstrings = (void *)hdr +
849b39927cfSSam Ravnborg 				 sechdrs[hdr->e_shstrndx].sh_offset;
850b39927cfSSam Ravnborg 	const char *secname = secstrings + sechdrs[sym->st_shndx].sh_name;
851b39927cfSSam Ravnborg 
852b39927cfSSam Ravnborg 	find_symbols_between(elf, r.r_offset, fromsec, &before, &after);
853b39927cfSSam Ravnborg 
85493684d3bSSam Ravnborg 	refsym = find_elf_symbol(elf, r.r_addend, sym);
85593684d3bSSam Ravnborg 	if (refsym && strlen(elf->strtab + refsym->st_name))
85693684d3bSSam Ravnborg 		refsymname = elf->strtab + refsym->st_name;
85793684d3bSSam Ravnborg 
8584c8fbca5SSam Ravnborg 	/* check whitelist - we may ignore it */
8594c8fbca5SSam Ravnborg 	if (before &&
8609e157a5aSMagnus Damm 	    secref_whitelist(modname, secname, fromsec,
861ee6a8545SVivek Goyal 			     elf->strtab + before->st_name, refsymname))
8624c8fbca5SSam Ravnborg 		return;
8634c8fbca5SSam Ravnborg 
864b39927cfSSam Ravnborg 	if (before && after) {
86525601209SRussell King 		warn("%s(%s+0x%llx): Section mismatch: reference to %s:%s "
86625601209SRussell King 		     "(between '%s' and '%s')\n",
86725601209SRussell King 		     modname, fromsec, (unsigned long long)r.r_offset,
86825601209SRussell King 		     secname, refsymname,
869b39927cfSSam Ravnborg 		     elf->strtab + before->st_name,
870b39927cfSSam Ravnborg 		     elf->strtab + after->st_name);
871b39927cfSSam Ravnborg 	} else if (before) {
87225601209SRussell King 		warn("%s(%s+0x%llx): Section mismatch: reference to %s:%s "
87325601209SRussell King 		     "(after '%s')\n",
87425601209SRussell King 		     modname, fromsec, (unsigned long long)r.r_offset,
87525601209SRussell King 		     secname, refsymname,
87625601209SRussell King 		     elf->strtab + before->st_name);
877b39927cfSSam Ravnborg 	} else if (after) {
87825601209SRussell King 		warn("%s(%s+0x%llx): Section mismatch: reference to %s:%s "
87993684d3bSSam Ravnborg 		     "before '%s' (at offset -0x%llx)\n",
88025601209SRussell King 		     modname, fromsec, (unsigned long long)r.r_offset,
88125601209SRussell King 		     secname, refsymname,
88225601209SRussell King 		     elf->strtab + after->st_name);
883b39927cfSSam Ravnborg 	} else {
88425601209SRussell King 		warn("%s(%s+0x%llx): Section mismatch: reference to %s:%s\n",
88525601209SRussell King 		     modname, fromsec, (unsigned long long)r.r_offset,
88625601209SRussell King 		     secname, refsymname);
887b39927cfSSam Ravnborg 	}
888b39927cfSSam Ravnborg }
889b39927cfSSam Ravnborg 
890ae4ac123SAtsushi Nemoto static unsigned int *reloc_location(struct elf_info *elf,
891ae4ac123SAtsushi Nemoto 					   int rsection, Elf_Rela *r)
892ae4ac123SAtsushi Nemoto {
893ae4ac123SAtsushi Nemoto 	Elf_Shdr *sechdrs = elf->sechdrs;
894ae4ac123SAtsushi Nemoto 	int section = sechdrs[rsection].sh_info;
895ae4ac123SAtsushi Nemoto 
896ae4ac123SAtsushi Nemoto 	return (void *)elf->hdr + sechdrs[section].sh_offset +
897ae4ac123SAtsushi Nemoto 		(r->r_offset - sechdrs[section].sh_addr);
898ae4ac123SAtsushi Nemoto }
899ae4ac123SAtsushi Nemoto 
900ae4ac123SAtsushi Nemoto static int addend_386_rel(struct elf_info *elf, int rsection, Elf_Rela *r)
901ae4ac123SAtsushi Nemoto {
902ae4ac123SAtsushi Nemoto 	unsigned int r_typ = ELF_R_TYPE(r->r_info);
903ae4ac123SAtsushi Nemoto 	unsigned int *location = reloc_location(elf, rsection, r);
904ae4ac123SAtsushi Nemoto 
905ae4ac123SAtsushi Nemoto 	switch (r_typ) {
906ae4ac123SAtsushi Nemoto 	case R_386_32:
907ae4ac123SAtsushi Nemoto 		r->r_addend = TO_NATIVE(*location);
908ae4ac123SAtsushi Nemoto 		break;
909ae4ac123SAtsushi Nemoto 	case R_386_PC32:
910ae4ac123SAtsushi Nemoto 		r->r_addend = TO_NATIVE(*location) + 4;
911ae4ac123SAtsushi Nemoto 		/* For CONFIG_RELOCATABLE=y */
912ae4ac123SAtsushi Nemoto 		if (elf->hdr->e_type == ET_EXEC)
913ae4ac123SAtsushi Nemoto 			r->r_addend += r->r_offset;
914ae4ac123SAtsushi Nemoto 		break;
915ae4ac123SAtsushi Nemoto 	}
916ae4ac123SAtsushi Nemoto 	return 0;
917ae4ac123SAtsushi Nemoto }
918ae4ac123SAtsushi Nemoto 
91956a974faSSam Ravnborg static int addend_arm_rel(struct elf_info *elf, int rsection, Elf_Rela *r)
92056a974faSSam Ravnborg {
92156a974faSSam Ravnborg 	unsigned int r_typ = ELF_R_TYPE(r->r_info);
92256a974faSSam Ravnborg 
92356a974faSSam Ravnborg 	switch (r_typ) {
92456a974faSSam Ravnborg 	case R_ARM_ABS32:
92556a974faSSam Ravnborg 		/* From ARM ABI: (S + A) | T */
92656a974faSSam Ravnborg 		r->r_addend = (int)(long)(elf->symtab_start + ELF_R_SYM(r->r_info));
92756a974faSSam Ravnborg 		break;
92856a974faSSam Ravnborg 	case R_ARM_PC24:
92956a974faSSam Ravnborg 		/* From ARM ABI: ((S + A) | T) - P */
93056a974faSSam Ravnborg 		r->r_addend = (int)(long)(elf->hdr + elf->sechdrs[rsection].sh_offset +
93156a974faSSam Ravnborg 		                          (r->r_offset - elf->sechdrs[rsection].sh_addr));
93256a974faSSam Ravnborg 		break;
93356a974faSSam Ravnborg 	default:
93456a974faSSam Ravnborg 		return 1;
93556a974faSSam Ravnborg 	}
93656a974faSSam Ravnborg 	return 0;
93756a974faSSam Ravnborg }
93856a974faSSam Ravnborg 
939ae4ac123SAtsushi Nemoto static int addend_mips_rel(struct elf_info *elf, int rsection, Elf_Rela *r)
940ae4ac123SAtsushi Nemoto {
941ae4ac123SAtsushi Nemoto 	unsigned int r_typ = ELF_R_TYPE(r->r_info);
942ae4ac123SAtsushi Nemoto 	unsigned int *location = reloc_location(elf, rsection, r);
943ae4ac123SAtsushi Nemoto 	unsigned int inst;
944ae4ac123SAtsushi Nemoto 
945ae4ac123SAtsushi Nemoto 	if (r_typ == R_MIPS_HI16)
946ae4ac123SAtsushi Nemoto 		return 1;	/* skip this */
947ae4ac123SAtsushi Nemoto 	inst = TO_NATIVE(*location);
948ae4ac123SAtsushi Nemoto 	switch (r_typ) {
949ae4ac123SAtsushi Nemoto 	case R_MIPS_LO16:
950ae4ac123SAtsushi Nemoto 		r->r_addend = inst & 0xffff;
951ae4ac123SAtsushi Nemoto 		break;
952ae4ac123SAtsushi Nemoto 	case R_MIPS_26:
953ae4ac123SAtsushi Nemoto 		r->r_addend = (inst & 0x03ffffff) << 2;
954ae4ac123SAtsushi Nemoto 		break;
955ae4ac123SAtsushi Nemoto 	case R_MIPS_32:
956ae4ac123SAtsushi Nemoto 		r->r_addend = inst;
957ae4ac123SAtsushi Nemoto 		break;
958ae4ac123SAtsushi Nemoto 	}
959ae4ac123SAtsushi Nemoto 	return 0;
960ae4ac123SAtsushi Nemoto }
961ae4ac123SAtsushi Nemoto 
962b39927cfSSam Ravnborg /**
963b39927cfSSam Ravnborg  * A module includes a number of sections that are discarded
964b39927cfSSam Ravnborg  * either when loaded or when used as built-in.
965b39927cfSSam Ravnborg  * For loaded modules all functions marked __init and all data
966b39927cfSSam Ravnborg  * marked __initdata will be discarded when the module has been intialized.
967b39927cfSSam Ravnborg  * Likewise for modules used built-in the sections marked __exit
968b39927cfSSam Ravnborg  * are discarded because __exit marked function are supposed to be called
969b39927cfSSam Ravnborg  * only when a moduel is unloaded which never happes for built-in modules.
970b39927cfSSam Ravnborg  * The check_sec_ref() function traverses all relocation records
971b39927cfSSam Ravnborg  * to find all references to a section that reference a section that will
972b39927cfSSam Ravnborg  * be discarded and warns about it.
973b39927cfSSam Ravnborg  **/
974b39927cfSSam Ravnborg static void check_sec_ref(struct module *mod, const char *modname,
975b39927cfSSam Ravnborg 			  struct elf_info *elf,
976b39927cfSSam Ravnborg 			  int section(const char*),
977b39927cfSSam Ravnborg 			  int section_ref_ok(const char *))
978b39927cfSSam Ravnborg {
979b39927cfSSam Ravnborg 	int i;
980b39927cfSSam Ravnborg 	Elf_Sym  *sym;
981b39927cfSSam Ravnborg 	Elf_Ehdr *hdr = elf->hdr;
982b39927cfSSam Ravnborg 	Elf_Shdr *sechdrs = elf->sechdrs;
983b39927cfSSam Ravnborg 	const char *secstrings = (void *)hdr +
984b39927cfSSam Ravnborg 				 sechdrs[hdr->e_shstrndx].sh_offset;
985b39927cfSSam Ravnborg 
986b39927cfSSam Ravnborg 	/* Walk through all sections */
987b39927cfSSam Ravnborg 	for (i = 0; i < hdr->e_shnum; i++) {
9882c1a51f3SAtsushi Nemoto 		const char *name = secstrings + sechdrs[i].sh_name;
9892c1a51f3SAtsushi Nemoto 		const char *secname;
9902c1a51f3SAtsushi Nemoto 		Elf_Rela r;
991eae07ac6SAtsushi Nemoto 		unsigned int r_sym;
9922c1a51f3SAtsushi Nemoto 		/* We want to process only relocation sections and not .init */
9932c1a51f3SAtsushi Nemoto 		if (sechdrs[i].sh_type == SHT_RELA) {
994fededcd2S[email protected] 			Elf_Rela *rela;
995fededcd2S[email protected] 			Elf_Rela *start = (void *)hdr + sechdrs[i].sh_offset;
996fededcd2S[email protected] 			Elf_Rela *stop  = (void*)start + sechdrs[i].sh_size;
9972c1a51f3SAtsushi Nemoto 			name += strlen(".rela");
9982c1a51f3SAtsushi Nemoto 			if (section_ref_ok(name))
999b39927cfSSam Ravnborg 				continue;
1000b39927cfSSam Ravnborg 
1001b39927cfSSam Ravnborg 			for (rela = start; rela < stop; rela++) {
1002b39927cfSSam Ravnborg 				r.r_offset = TO_NATIVE(rela->r_offset);
1003eae07ac6SAtsushi Nemoto #if KERNEL_ELFCLASS == ELFCLASS64
1004eae07ac6SAtsushi Nemoto 				if (hdr->e_machine == EM_MIPS) {
1005ae4ac123SAtsushi Nemoto 					unsigned int r_typ;
1006eae07ac6SAtsushi Nemoto 					r_sym = ELF64_MIPS_R_SYM(rela->r_info);
1007eae07ac6SAtsushi Nemoto 					r_sym = TO_NATIVE(r_sym);
1008ae4ac123SAtsushi Nemoto 					r_typ = ELF64_MIPS_R_TYPE(rela->r_info);
1009ae4ac123SAtsushi Nemoto 					r.r_info = ELF64_R_INFO(r_sym, r_typ);
1010eae07ac6SAtsushi Nemoto 				} else {
1011601e7f02SLinus Torvalds 					r.r_info = TO_NATIVE(rela->r_info);
1012eae07ac6SAtsushi Nemoto 					r_sym = ELF_R_SYM(r.r_info);
1013eae07ac6SAtsushi Nemoto 				}
1014eae07ac6SAtsushi Nemoto #else
1015eae07ac6SAtsushi Nemoto 				r.r_info = TO_NATIVE(rela->r_info);
1016eae07ac6SAtsushi Nemoto 				r_sym = ELF_R_SYM(r.r_info);
1017eae07ac6SAtsushi Nemoto #endif
101893684d3bSSam Ravnborg 				r.r_addend = TO_NATIVE(rela->r_addend);
1019eae07ac6SAtsushi Nemoto 				sym = elf->symtab_start + r_sym;
1020b39927cfSSam Ravnborg 				/* Skip special sections */
1021b39927cfSSam Ravnborg 				if (sym->st_shndx >= SHN_LORESERVE)
1022b39927cfSSam Ravnborg 					continue;
1023b39927cfSSam Ravnborg 
10242c1a51f3SAtsushi Nemoto 				secname = secstrings +
10252c1a51f3SAtsushi Nemoto 					sechdrs[sym->st_shndx].sh_name;
1026b39927cfSSam Ravnborg 				if (section(secname))
10272c1a51f3SAtsushi Nemoto 					warn_sec_mismatch(modname, name,
10282c1a51f3SAtsushi Nemoto 							  elf, sym, r);
10292c1a51f3SAtsushi Nemoto 			}
10302c1a51f3SAtsushi Nemoto 		} else if (sechdrs[i].sh_type == SHT_REL) {
10312c1a51f3SAtsushi Nemoto 			Elf_Rel *rel;
10322c1a51f3SAtsushi Nemoto 			Elf_Rel *start = (void *)hdr + sechdrs[i].sh_offset;
10332c1a51f3SAtsushi Nemoto 			Elf_Rel *stop  = (void*)start + sechdrs[i].sh_size;
10342c1a51f3SAtsushi Nemoto 			name += strlen(".rel");
10352c1a51f3SAtsushi Nemoto 			if (section_ref_ok(name))
10362c1a51f3SAtsushi Nemoto 				continue;
10372c1a51f3SAtsushi Nemoto 
10382c1a51f3SAtsushi Nemoto 			for (rel = start; rel < stop; rel++) {
10392c1a51f3SAtsushi Nemoto 				r.r_offset = TO_NATIVE(rel->r_offset);
1040eae07ac6SAtsushi Nemoto #if KERNEL_ELFCLASS == ELFCLASS64
1041eae07ac6SAtsushi Nemoto 				if (hdr->e_machine == EM_MIPS) {
1042ae4ac123SAtsushi Nemoto 					unsigned int r_typ;
1043eae07ac6SAtsushi Nemoto 					r_sym = ELF64_MIPS_R_SYM(rel->r_info);
1044eae07ac6SAtsushi Nemoto 					r_sym = TO_NATIVE(r_sym);
1045ae4ac123SAtsushi Nemoto 					r_typ = ELF64_MIPS_R_TYPE(rel->r_info);
1046ae4ac123SAtsushi Nemoto 					r.r_info = ELF64_R_INFO(r_sym, r_typ);
1047eae07ac6SAtsushi Nemoto 				} else {
10482c1a51f3SAtsushi Nemoto 					r.r_info = TO_NATIVE(rel->r_info);
1049eae07ac6SAtsushi Nemoto 					r_sym = ELF_R_SYM(r.r_info);
1050eae07ac6SAtsushi Nemoto 				}
1051eae07ac6SAtsushi Nemoto #else
1052eae07ac6SAtsushi Nemoto 				r.r_info = TO_NATIVE(rel->r_info);
1053eae07ac6SAtsushi Nemoto 				r_sym = ELF_R_SYM(r.r_info);
1054eae07ac6SAtsushi Nemoto #endif
10552c1a51f3SAtsushi Nemoto 				r.r_addend = 0;
1056ae4ac123SAtsushi Nemoto 				switch (hdr->e_machine) {
1057ae4ac123SAtsushi Nemoto 				case EM_386:
1058ae4ac123SAtsushi Nemoto 					if (addend_386_rel(elf, i, &r))
1059ae4ac123SAtsushi Nemoto 						continue;
1060ae4ac123SAtsushi Nemoto 					break;
106156a974faSSam Ravnborg 				case EM_ARM:
106256a974faSSam Ravnborg 					if(addend_arm_rel(elf, i, &r))
106356a974faSSam Ravnborg 						continue;
106456a974faSSam Ravnborg 					break;
1065ae4ac123SAtsushi Nemoto 				case EM_MIPS:
1066ae4ac123SAtsushi Nemoto 					if (addend_mips_rel(elf, i, &r))
1067ae4ac123SAtsushi Nemoto 						continue;
1068ae4ac123SAtsushi Nemoto 					break;
1069ae4ac123SAtsushi Nemoto 				}
1070eae07ac6SAtsushi Nemoto 				sym = elf->symtab_start + r_sym;
10712c1a51f3SAtsushi Nemoto 				/* Skip special sections */
10722c1a51f3SAtsushi Nemoto 				if (sym->st_shndx >= SHN_LORESERVE)
10732c1a51f3SAtsushi Nemoto 					continue;
10742c1a51f3SAtsushi Nemoto 
10752c1a51f3SAtsushi Nemoto 				secname = secstrings +
10762c1a51f3SAtsushi Nemoto 					sechdrs[sym->st_shndx].sh_name;
10772c1a51f3SAtsushi Nemoto 				if (section(secname))
10782c1a51f3SAtsushi Nemoto 					warn_sec_mismatch(modname, name,
10792c1a51f3SAtsushi Nemoto 							  elf, sym, r);
10802c1a51f3SAtsushi Nemoto 			}
1081b39927cfSSam Ravnborg 		}
1082b39927cfSSam Ravnborg 	}
1083b39927cfSSam Ravnborg }
1084b39927cfSSam Ravnborg 
10851087247bSSam Ravnborg /*
10861087247bSSam Ravnborg  * Identify sections from which references to either a
10871087247bSSam Ravnborg  * .init or a .exit section is OK.
10881087247bSSam Ravnborg  *
10891087247bSSam Ravnborg  * [OPD] Keith Ownes <[email protected]> commented:
10901087247bSSam Ravnborg  * For our future {in}sanity, add a comment that this is the ppc .opd
10911087247bSSam Ravnborg  * section, not the ia64 .opd section.
10921087247bSSam Ravnborg  * ia64 .opd should not point to discarded sections.
10931087247bSSam Ravnborg  * [.rodata] like for .init.text we ignore .rodata references -same reason
10941d8af559SSam Ravnborg  */
10951087247bSSam Ravnborg static int initexit_section_ref_ok(const char *name)
10961087247bSSam Ravnborg {
10971087247bSSam Ravnborg 	const char **s;
10981087247bSSam Ravnborg 	/* Absolute section names */
10991087247bSSam Ravnborg 	const char *namelist1[] = {
11001087247bSSam Ravnborg 		"__bug_table",		/* used by powerpc for BUG() */
11011087247bSSam Ravnborg 		"__ex_table",
11021087247bSSam Ravnborg 		".altinstructions",
11031087247bSSam Ravnborg 		".cranges",		/* used by sh64 */
11041087247bSSam Ravnborg 		".fixup",
11051d8af559SSam Ravnborg 		".machvec",		/* ia64 + powerpc uses these */
11061d8af559SSam Ravnborg 		".machine.desc",
11071087247bSSam Ravnborg 		".opd",			/* See comment [OPD] */
11081087247bSSam Ravnborg 		".parainstructions",
11091087247bSSam Ravnborg 		".pdr",
11101087247bSSam Ravnborg 		".plt",			/* seen on ARCH=um build on x86_64. Harmless */
11111087247bSSam Ravnborg 		".smp_locks",
11121087247bSSam Ravnborg 		".stab",
11133a5df1d4SAl Viro 		".m68k_fixup",
11141087247bSSam Ravnborg 		NULL
11151087247bSSam Ravnborg 	};
11161087247bSSam Ravnborg 	/* Start of section names */
11171087247bSSam Ravnborg 	const char *namelist2[] = {
11181087247bSSam Ravnborg 		".debug",
11191087247bSSam Ravnborg 		".eh_frame",
11201087247bSSam Ravnborg 		".note",		/* ignore ELF notes - may contain anything */
11211087247bSSam Ravnborg 		".got",			/* powerpc - global offset table */
11221087247bSSam Ravnborg 		".toc",			/* powerpc - table of contents */
11231087247bSSam Ravnborg 		NULL
11241087247bSSam Ravnborg 	};
11251087247bSSam Ravnborg 	/* part of section name */
11261087247bSSam Ravnborg 	const char *namelist3 [] = {
11271087247bSSam Ravnborg 		".unwind",  /* Sample: IA_64.unwind.exit.text */
11281087247bSSam Ravnborg 		NULL
11291087247bSSam Ravnborg 	};
11301087247bSSam Ravnborg 
11311087247bSSam Ravnborg 	for (s = namelist1; *s; s++)
11321087247bSSam Ravnborg 		if (strcmp(*s, name) == 0)
11331087247bSSam Ravnborg 			return 1;
11341087247bSSam Ravnborg 	for (s = namelist2; *s; s++)
11351087247bSSam Ravnborg 		if (strncmp(*s, name, strlen(*s)) == 0)
11361087247bSSam Ravnborg 			return 1;
11371087247bSSam Ravnborg 	for (s = namelist3; *s; s++)
11381087247bSSam Ravnborg 		if (strstr(name, *s) != NULL)
11391087247bSSam Ravnborg 			return 1;
11401087247bSSam Ravnborg 	return 0;
11411087247bSSam Ravnborg }
11421087247bSSam Ravnborg 
1143b39927cfSSam Ravnborg 
11441087247bSSam Ravnborg /*
1145b39927cfSSam Ravnborg  * Identify sections from which references to a .init section is OK.
1146b39927cfSSam Ravnborg  *
1147b39927cfSSam Ravnborg  * Unfortunately references to read only data that referenced .init
1148b39927cfSSam Ravnborg  * sections had to be excluded. Almost all of these are false
1149b39927cfSSam Ravnborg  * positives, they are created by gcc. The downside of excluding rodata
1150b39927cfSSam Ravnborg  * is that there really are some user references from rodata to
1151b39927cfSSam Ravnborg  * init code, e.g. drivers/video/vgacon.c:
1152b39927cfSSam Ravnborg  *
1153b39927cfSSam Ravnborg  * const struct consw vga_con = {
1154b39927cfSSam Ravnborg  *        con_startup:            vgacon_startup,
1155b39927cfSSam Ravnborg  *
1156b39927cfSSam Ravnborg  * where vgacon_startup is __init.  If you want to wade through the false
1157b39927cfSSam Ravnborg  * positives, take out the check for rodata.
11581087247bSSam Ravnborg  */
1159b39927cfSSam Ravnborg static int init_section_ref_ok(const char *name)
1160b39927cfSSam Ravnborg {
1161b39927cfSSam Ravnborg 	const char **s;
1162b39927cfSSam Ravnborg 	/* Absolute section names */
1163b39927cfSSam Ravnborg 	const char *namelist1[] = {
1164eec73e88SRalf Baechle 		"__dbe_table",		/* MIPS generate these */
116521c4ff80SBenjamin Herrenschmidt 		"__ftr_fixup",		/* powerpc cpu feature fixup */
116621c4ff80SBenjamin Herrenschmidt 		"__fw_ftr_fixup",	/* powerpc firmware feature fixup */
11671087247bSSam Ravnborg 		"__param",
11681087247bSSam Ravnborg 		".data.rel.ro",		/* used by parisc64 */
11691087247bSSam Ravnborg 		".init",
11701087247bSSam Ravnborg 		".text.lock",
1171b39927cfSSam Ravnborg 		NULL
1172b39927cfSSam Ravnborg 	};
1173b39927cfSSam Ravnborg 	/* Start of section names */
1174b39927cfSSam Ravnborg 	const char *namelist2[] = {
1175b39927cfSSam Ravnborg 		".init.",
11761087247bSSam Ravnborg 		".pci_fixup",
1177742433b0SMatthew Wilcox 		".rodata",
1178b39927cfSSam Ravnborg 		NULL
1179b39927cfSSam Ravnborg 	};
11801087247bSSam Ravnborg 
11811087247bSSam Ravnborg 	if (initexit_section_ref_ok(name))
11821087247bSSam Ravnborg 		return 1;
1183b39927cfSSam Ravnborg 
1184b39927cfSSam Ravnborg 	for (s = namelist1; *s; s++)
1185b39927cfSSam Ravnborg 		if (strcmp(*s, name) == 0)
1186b39927cfSSam Ravnborg 			return 1;
1187b39927cfSSam Ravnborg 	for (s = namelist2; *s; s++)
1188b39927cfSSam Ravnborg 		if (strncmp(*s, name, strlen(*s)) == 0)
1189b39927cfSSam Ravnborg 			return 1;
11901087247bSSam Ravnborg 
11911087247bSSam Ravnborg 	/* If section name ends with ".init" we allow references
11921087247bSSam Ravnborg 	 * as is the case with .initcallN.init, .early_param.init, .taglist.init etc
11931087247bSSam Ravnborg 	 */
1194468d9494SAl Viro 	if (strrcmp(name, ".init") == 0)
1195468d9494SAl Viro 		return 1;
1196b39927cfSSam Ravnborg 	return 0;
1197b39927cfSSam Ravnborg }
1198b39927cfSSam Ravnborg 
1199b39927cfSSam Ravnborg /*
1200b39927cfSSam Ravnborg  * Identify sections from which references to a .exit section is OK.
12011087247bSSam Ravnborg  */
1202b39927cfSSam Ravnborg static int exit_section_ref_ok(const char *name)
1203b39927cfSSam Ravnborg {
1204b39927cfSSam Ravnborg 	const char **s;
1205b39927cfSSam Ravnborg 	/* Absolute section names */
1206b39927cfSSam Ravnborg 	const char *namelist1[] = {
1207b39927cfSSam Ravnborg 		".exit.data",
12081087247bSSam Ravnborg 		".exit.text",
12091087247bSSam Ravnborg 		".exitcall.exit",
12105ecdd0f6SSam Ravnborg 		".rodata",
12116e10133fSSam Ravnborg 		NULL
12126e10133fSSam Ravnborg 	};
1213b39927cfSSam Ravnborg 
12141087247bSSam Ravnborg 	if (initexit_section_ref_ok(name))
12151087247bSSam Ravnborg 		return 1;
12161087247bSSam Ravnborg 
1217b39927cfSSam Ravnborg 	for (s = namelist1; *s; s++)
1218b39927cfSSam Ravnborg 		if (strcmp(*s, name) == 0)
1219b39927cfSSam Ravnborg 			return 1;
1220b39927cfSSam Ravnborg 	return 0;
1221b39927cfSSam Ravnborg }
1222b39927cfSSam Ravnborg 
12235c3ead8cSSam Ravnborg static void read_symbols(char *modname)
12241da177e4SLinus Torvalds {
12251da177e4SLinus Torvalds 	const char *symname;
12261da177e4SLinus Torvalds 	char *version;
1227b817f6feSSam Ravnborg 	char *license;
12281da177e4SLinus Torvalds 	struct module *mod;
12291da177e4SLinus Torvalds 	struct elf_info info = { };
12301da177e4SLinus Torvalds 	Elf_Sym *sym;
12311da177e4SLinus Torvalds 
123285bd2fddSSam Ravnborg 	if (!parse_elf(&info, modname))
123385bd2fddSSam Ravnborg 		return;
12341da177e4SLinus Torvalds 
12351da177e4SLinus Torvalds 	mod = new_module(modname);
12361da177e4SLinus Torvalds 
12371da177e4SLinus Torvalds 	/* When there's no vmlinux, don't print warnings about
12381da177e4SLinus Torvalds 	 * unresolved symbols (since there'll be too many ;) */
12391da177e4SLinus Torvalds 	if (is_vmlinux(modname)) {
12401da177e4SLinus Torvalds 		have_vmlinux = 1;
12411da177e4SLinus Torvalds 		mod->skip = 1;
12421da177e4SLinus Torvalds 	}
12431da177e4SLinus Torvalds 
1244b817f6feSSam Ravnborg 	license = get_modinfo(info.modinfo, info.modinfo_len, "license");
1245b817f6feSSam Ravnborg 	while (license) {
1246b817f6feSSam Ravnborg 		if (license_is_gpl_compatible(license))
1247b817f6feSSam Ravnborg 			mod->gpl_compatible = 1;
1248b817f6feSSam Ravnborg 		else {
1249b817f6feSSam Ravnborg 			mod->gpl_compatible = 0;
1250b817f6feSSam Ravnborg 			break;
1251b817f6feSSam Ravnborg 		}
1252b817f6feSSam Ravnborg 		license = get_next_modinfo(info.modinfo, info.modinfo_len,
1253b817f6feSSam Ravnborg 					   "license", license);
1254b817f6feSSam Ravnborg 	}
1255b817f6feSSam Ravnborg 
12561da177e4SLinus Torvalds 	for (sym = info.symtab_start; sym < info.symtab_stop; sym++) {
12571da177e4SLinus Torvalds 		symname = info.strtab + sym->st_name;
12581da177e4SLinus Torvalds 
12591da177e4SLinus Torvalds 		handle_modversions(mod, &info, sym, symname);
12601da177e4SLinus Torvalds 		handle_moddevtable(mod, &info, sym, symname);
12611da177e4SLinus Torvalds 	}
12628d8d8289SSam Ravnborg 	if (is_vmlinux(modname) && vmlinux_section_warnings) {
1263b39927cfSSam Ravnborg 		check_sec_ref(mod, modname, &info, init_section, init_section_ref_ok);
1264b39927cfSSam Ravnborg 		check_sec_ref(mod, modname, &info, exit_section, exit_section_ref_ok);
12658d8d8289SSam Ravnborg 	}
12661da177e4SLinus Torvalds 
12671da177e4SLinus Torvalds 	version = get_modinfo(info.modinfo, info.modinfo_len, "version");
12681da177e4SLinus Torvalds 	if (version)
12691da177e4SLinus Torvalds 		maybe_frob_rcs_version(modname, version, info.modinfo,
12701da177e4SLinus Torvalds 				       version - (char *)info.hdr);
12711da177e4SLinus Torvalds 	if (version || (all_versions && !is_vmlinux(modname)))
12721da177e4SLinus Torvalds 		get_src_version(modname, mod->srcversion,
12731da177e4SLinus Torvalds 				sizeof(mod->srcversion)-1);
12741da177e4SLinus Torvalds 
12751da177e4SLinus Torvalds 	parse_elf_finish(&info);
12761da177e4SLinus Torvalds 
12771da177e4SLinus Torvalds 	/* Our trick to get versioning for struct_module - it's
12781da177e4SLinus Torvalds 	 * never passed as an argument to an exported function, so
12791da177e4SLinus Torvalds 	 * the automatic versioning doesn't pick it up, but it's really
12801da177e4SLinus Torvalds 	 * important anyhow */
12811da177e4SLinus Torvalds 	if (modversions)
12821da177e4SLinus Torvalds 		mod->unres = alloc_symbol("struct_module", 0, mod->unres);
12831da177e4SLinus Torvalds }
12841da177e4SLinus Torvalds 
12851da177e4SLinus Torvalds #define SZ 500
12861da177e4SLinus Torvalds 
12871da177e4SLinus Torvalds /* We first write the generated file into memory using the
12881da177e4SLinus Torvalds  * following helper, then compare to the file on disk and
12891da177e4SLinus Torvalds  * only update the later if anything changed */
12901da177e4SLinus Torvalds 
12915c3ead8cSSam Ravnborg void __attribute__((format(printf, 2, 3))) buf_printf(struct buffer *buf,
12925c3ead8cSSam Ravnborg 						      const char *fmt, ...)
12931da177e4SLinus Torvalds {
12941da177e4SLinus Torvalds 	char tmp[SZ];
12951da177e4SLinus Torvalds 	int len;
12961da177e4SLinus Torvalds 	va_list ap;
12971da177e4SLinus Torvalds 
12981da177e4SLinus Torvalds 	va_start(ap, fmt);
12991da177e4SLinus Torvalds 	len = vsnprintf(tmp, SZ, fmt, ap);
13007670f023SSam Ravnborg 	buf_write(buf, tmp, len);
13011da177e4SLinus Torvalds 	va_end(ap);
13021da177e4SLinus Torvalds }
13031da177e4SLinus Torvalds 
13045c3ead8cSSam Ravnborg void buf_write(struct buffer *buf, const char *s, int len)
13051da177e4SLinus Torvalds {
13061da177e4SLinus Torvalds 	if (buf->size - buf->pos < len) {
13077670f023SSam Ravnborg 		buf->size += len + SZ;
13081da177e4SLinus Torvalds 		buf->p = realloc(buf->p, buf->size);
13091da177e4SLinus Torvalds 	}
13101da177e4SLinus Torvalds 	strncpy(buf->p + buf->pos, s, len);
13111da177e4SLinus Torvalds 	buf->pos += len;
13121da177e4SLinus Torvalds }
13131da177e4SLinus Torvalds 
1314c96fca21SSam Ravnborg static void check_for_gpl_usage(enum export exp, const char *m, const char *s)
1315c96fca21SSam Ravnborg {
1316c96fca21SSam Ravnborg 	const char *e = is_vmlinux(m) ?"":".ko";
1317c96fca21SSam Ravnborg 
1318c96fca21SSam Ravnborg 	switch (exp) {
1319c96fca21SSam Ravnborg 	case export_gpl:
1320c96fca21SSam Ravnborg 		fatal("modpost: GPL-incompatible module %s%s "
1321c96fca21SSam Ravnborg 		      "uses GPL-only symbol '%s'\n", m, e, s);
1322c96fca21SSam Ravnborg 		break;
1323c96fca21SSam Ravnborg 	case export_unused_gpl:
1324c96fca21SSam Ravnborg 		fatal("modpost: GPL-incompatible module %s%s "
1325c96fca21SSam Ravnborg 		      "uses GPL-only symbol marked UNUSED '%s'\n", m, e, s);
1326c96fca21SSam Ravnborg 		break;
1327c96fca21SSam Ravnborg 	case export_gpl_future:
1328c96fca21SSam Ravnborg 		warn("modpost: GPL-incompatible module %s%s "
1329c96fca21SSam Ravnborg 		      "uses future GPL-only symbol '%s'\n", m, e, s);
1330c96fca21SSam Ravnborg 		break;
1331c96fca21SSam Ravnborg 	case export_plain:
1332c96fca21SSam Ravnborg 	case export_unused:
1333c96fca21SSam Ravnborg 	case export_unknown:
1334c96fca21SSam Ravnborg 		/* ignore */
1335c96fca21SSam Ravnborg 		break;
1336c96fca21SSam Ravnborg 	}
1337c96fca21SSam Ravnborg }
1338c96fca21SSam Ravnborg 
1339c96fca21SSam Ravnborg static void check_for_unused(enum export exp, const char* m, const char* s)
1340c96fca21SSam Ravnborg {
1341c96fca21SSam Ravnborg 	const char *e = is_vmlinux(m) ?"":".ko";
1342c96fca21SSam Ravnborg 
1343c96fca21SSam Ravnborg 	switch (exp) {
1344c96fca21SSam Ravnborg 	case export_unused:
1345c96fca21SSam Ravnborg 	case export_unused_gpl:
1346c96fca21SSam Ravnborg 		warn("modpost: module %s%s "
1347c96fca21SSam Ravnborg 		      "uses symbol '%s' marked UNUSED\n", m, e, s);
1348c96fca21SSam Ravnborg 		break;
1349c96fca21SSam Ravnborg 	default:
1350c96fca21SSam Ravnborg 		/* ignore */
1351c96fca21SSam Ravnborg 		break;
1352c96fca21SSam Ravnborg 	}
1353c96fca21SSam Ravnborg }
1354c96fca21SSam Ravnborg 
1355c96fca21SSam Ravnborg static void check_exports(struct module *mod)
1356b817f6feSSam Ravnborg {
1357b817f6feSSam Ravnborg 	struct symbol *s, *exp;
1358b817f6feSSam Ravnborg 
1359b817f6feSSam Ravnborg 	for (s = mod->unres; s; s = s->next) {
13606449bd62SAndrew Morton 		const char *basename;
1361b817f6feSSam Ravnborg 		exp = find_symbol(s->name);
1362b817f6feSSam Ravnborg 		if (!exp || exp->module == mod)
1363b817f6feSSam Ravnborg 			continue;
13646449bd62SAndrew Morton 		basename = strrchr(mod->name, '/');
1365b817f6feSSam Ravnborg 		if (basename)
1366b817f6feSSam Ravnborg 			basename++;
1367c96fca21SSam Ravnborg 		else
1368c96fca21SSam Ravnborg 			basename = mod->name;
1369c96fca21SSam Ravnborg 		if (!mod->gpl_compatible)
1370c96fca21SSam Ravnborg 			check_for_gpl_usage(exp->export, basename, exp->name);
1371c96fca21SSam Ravnborg 		check_for_unused(exp->export, basename, exp->name);
1372b817f6feSSam Ravnborg         }
1373b817f6feSSam Ravnborg }
1374b817f6feSSam Ravnborg 
13755c3ead8cSSam Ravnborg /**
13765c3ead8cSSam Ravnborg  * Header for the generated file
13775c3ead8cSSam Ravnborg  **/
13785c3ead8cSSam Ravnborg static void add_header(struct buffer *b, struct module *mod)
13791da177e4SLinus Torvalds {
13801da177e4SLinus Torvalds 	buf_printf(b, "#include <linux/module.h>\n");
13811da177e4SLinus Torvalds 	buf_printf(b, "#include <linux/vermagic.h>\n");
13821da177e4SLinus Torvalds 	buf_printf(b, "#include <linux/compiler.h>\n");
13831da177e4SLinus Torvalds 	buf_printf(b, "\n");
13841da177e4SLinus Torvalds 	buf_printf(b, "MODULE_INFO(vermagic, VERMAGIC_STRING);\n");
13851da177e4SLinus Torvalds 	buf_printf(b, "\n");
13861da177e4SLinus Torvalds 	buf_printf(b, "struct module __this_module\n");
13871da177e4SLinus Torvalds 	buf_printf(b, "__attribute__((section(\".gnu.linkonce.this_module\"))) = {\n");
1388f83b5e32SUstyugov Roman 	buf_printf(b, " .name = KBUILD_MODNAME,\n");
13891da177e4SLinus Torvalds 	if (mod->has_init)
13901da177e4SLinus Torvalds 		buf_printf(b, " .init = init_module,\n");
13911da177e4SLinus Torvalds 	if (mod->has_cleanup)
13921da177e4SLinus Torvalds 		buf_printf(b, "#ifdef CONFIG_MODULE_UNLOAD\n"
13931da177e4SLinus Torvalds 			      " .exit = cleanup_module,\n"
13941da177e4SLinus Torvalds 			      "#endif\n");
1395e61a1c1cSRoman Zippel 	buf_printf(b, " .arch = MODULE_ARCH_INIT,\n");
13961da177e4SLinus Torvalds 	buf_printf(b, "};\n");
13971da177e4SLinus Torvalds }
13981da177e4SLinus Torvalds 
13995c3ead8cSSam Ravnborg /**
14005c3ead8cSSam Ravnborg  * Record CRCs for unresolved symbols
14015c3ead8cSSam Ravnborg  **/
1402c53ddacdSKirill Korotaev static int add_versions(struct buffer *b, struct module *mod)
14031da177e4SLinus Torvalds {
14041da177e4SLinus Torvalds 	struct symbol *s, *exp;
1405c53ddacdSKirill Korotaev 	int err = 0;
14061da177e4SLinus Torvalds 
14071da177e4SLinus Torvalds 	for (s = mod->unres; s; s = s->next) {
14081da177e4SLinus Torvalds 		exp = find_symbol(s->name);
14091da177e4SLinus Torvalds 		if (!exp || exp->module == mod) {
1410c53ddacdSKirill Korotaev 			if (have_vmlinux && !s->weak) {
14112a116659SMatthew Wilcox 				if (warn_unresolved) {
1412cb80514dSSam Ravnborg 					warn("\"%s\" [%s.ko] undefined!\n",
1413cb80514dSSam Ravnborg 					     s->name, mod->name);
14142a116659SMatthew Wilcox 				} else {
14152a116659SMatthew Wilcox 					merror("\"%s\" [%s.ko] undefined!\n",
14162a116659SMatthew Wilcox 					          s->name, mod->name);
14172a116659SMatthew Wilcox 					err = 1;
14182a116659SMatthew Wilcox 				}
1419c53ddacdSKirill Korotaev 			}
14201da177e4SLinus Torvalds 			continue;
14211da177e4SLinus Torvalds 		}
14221da177e4SLinus Torvalds 		s->module = exp->module;
14231da177e4SLinus Torvalds 		s->crc_valid = exp->crc_valid;
14241da177e4SLinus Torvalds 		s->crc = exp->crc;
14251da177e4SLinus Torvalds 	}
14261da177e4SLinus Torvalds 
14271da177e4SLinus Torvalds 	if (!modversions)
1428c53ddacdSKirill Korotaev 		return err;
14291da177e4SLinus Torvalds 
14301da177e4SLinus Torvalds 	buf_printf(b, "\n");
14311da177e4SLinus Torvalds 	buf_printf(b, "static const struct modversion_info ____versions[]\n");
14321da177e4SLinus Torvalds 	buf_printf(b, "__attribute_used__\n");
14331da177e4SLinus Torvalds 	buf_printf(b, "__attribute__((section(\"__versions\"))) = {\n");
14341da177e4SLinus Torvalds 
14351da177e4SLinus Torvalds 	for (s = mod->unres; s; s = s->next) {
14361da177e4SLinus Torvalds 		if (!s->module) {
14371da177e4SLinus Torvalds 			continue;
14381da177e4SLinus Torvalds 		}
14391da177e4SLinus Torvalds 		if (!s->crc_valid) {
1440cb80514dSSam Ravnborg 			warn("\"%s\" [%s.ko] has no CRC!\n",
14411da177e4SLinus Torvalds 				s->name, mod->name);
14421da177e4SLinus Torvalds 			continue;
14431da177e4SLinus Torvalds 		}
14441da177e4SLinus Torvalds 		buf_printf(b, "\t{ %#8x, \"%s\" },\n", s->crc, s->name);
14451da177e4SLinus Torvalds 	}
14461da177e4SLinus Torvalds 
14471da177e4SLinus Torvalds 	buf_printf(b, "};\n");
1448c53ddacdSKirill Korotaev 
1449c53ddacdSKirill Korotaev 	return err;
14501da177e4SLinus Torvalds }
14511da177e4SLinus Torvalds 
14525c3ead8cSSam Ravnborg static void add_depends(struct buffer *b, struct module *mod,
14535c3ead8cSSam Ravnborg 			struct module *modules)
14541da177e4SLinus Torvalds {
14551da177e4SLinus Torvalds 	struct symbol *s;
14561da177e4SLinus Torvalds 	struct module *m;
14571da177e4SLinus Torvalds 	int first = 1;
14581da177e4SLinus Torvalds 
14591da177e4SLinus Torvalds 	for (m = modules; m; m = m->next) {
14601da177e4SLinus Torvalds 		m->seen = is_vmlinux(m->name);
14611da177e4SLinus Torvalds 	}
14621da177e4SLinus Torvalds 
14631da177e4SLinus Torvalds 	buf_printf(b, "\n");
14641da177e4SLinus Torvalds 	buf_printf(b, "static const char __module_depends[]\n");
14651da177e4SLinus Torvalds 	buf_printf(b, "__attribute_used__\n");
14661da177e4SLinus Torvalds 	buf_printf(b, "__attribute__((section(\".modinfo\"))) =\n");
14671da177e4SLinus Torvalds 	buf_printf(b, "\"depends=");
14681da177e4SLinus Torvalds 	for (s = mod->unres; s; s = s->next) {
1469a61b2dfdSSam Ravnborg 		const char *p;
14701da177e4SLinus Torvalds 		if (!s->module)
14711da177e4SLinus Torvalds 			continue;
14721da177e4SLinus Torvalds 
14731da177e4SLinus Torvalds 		if (s->module->seen)
14741da177e4SLinus Torvalds 			continue;
14751da177e4SLinus Torvalds 
14761da177e4SLinus Torvalds 		s->module->seen = 1;
1477a61b2dfdSSam Ravnborg 		if ((p = strrchr(s->module->name, '/')) != NULL)
1478a61b2dfdSSam Ravnborg 			p++;
1479a61b2dfdSSam Ravnborg 		else
1480a61b2dfdSSam Ravnborg 			p = s->module->name;
1481a61b2dfdSSam Ravnborg 		buf_printf(b, "%s%s", first ? "" : ",", p);
14821da177e4SLinus Torvalds 		first = 0;
14831da177e4SLinus Torvalds 	}
14841da177e4SLinus Torvalds 	buf_printf(b, "\";\n");
14851da177e4SLinus Torvalds }
14861da177e4SLinus Torvalds 
14875c3ead8cSSam Ravnborg static void add_srcversion(struct buffer *b, struct module *mod)
14881da177e4SLinus Torvalds {
14891da177e4SLinus Torvalds 	if (mod->srcversion[0]) {
14901da177e4SLinus Torvalds 		buf_printf(b, "\n");
14911da177e4SLinus Torvalds 		buf_printf(b, "MODULE_INFO(srcversion, \"%s\");\n",
14921da177e4SLinus Torvalds 			   mod->srcversion);
14931da177e4SLinus Torvalds 	}
14941da177e4SLinus Torvalds }
14951da177e4SLinus Torvalds 
14965c3ead8cSSam Ravnborg static void write_if_changed(struct buffer *b, const char *fname)
14971da177e4SLinus Torvalds {
14981da177e4SLinus Torvalds 	char *tmp;
14991da177e4SLinus Torvalds 	FILE *file;
15001da177e4SLinus Torvalds 	struct stat st;
15011da177e4SLinus Torvalds 
15021da177e4SLinus Torvalds 	file = fopen(fname, "r");
15031da177e4SLinus Torvalds 	if (!file)
15041da177e4SLinus Torvalds 		goto write;
15051da177e4SLinus Torvalds 
15061da177e4SLinus Torvalds 	if (fstat(fileno(file), &st) < 0)
15071da177e4SLinus Torvalds 		goto close_write;
15081da177e4SLinus Torvalds 
15091da177e4SLinus Torvalds 	if (st.st_size != b->pos)
15101da177e4SLinus Torvalds 		goto close_write;
15111da177e4SLinus Torvalds 
15121da177e4SLinus Torvalds 	tmp = NOFAIL(malloc(b->pos));
15131da177e4SLinus Torvalds 	if (fread(tmp, 1, b->pos, file) != b->pos)
15141da177e4SLinus Torvalds 		goto free_write;
15151da177e4SLinus Torvalds 
15161da177e4SLinus Torvalds 	if (memcmp(tmp, b->p, b->pos) != 0)
15171da177e4SLinus Torvalds 		goto free_write;
15181da177e4SLinus Torvalds 
15191da177e4SLinus Torvalds 	free(tmp);
15201da177e4SLinus Torvalds 	fclose(file);
15211da177e4SLinus Torvalds 	return;
15221da177e4SLinus Torvalds 
15231da177e4SLinus Torvalds  free_write:
15241da177e4SLinus Torvalds 	free(tmp);
15251da177e4SLinus Torvalds  close_write:
15261da177e4SLinus Torvalds 	fclose(file);
15271da177e4SLinus Torvalds  write:
15281da177e4SLinus Torvalds 	file = fopen(fname, "w");
15291da177e4SLinus Torvalds 	if (!file) {
15301da177e4SLinus Torvalds 		perror(fname);
15311da177e4SLinus Torvalds 		exit(1);
15321da177e4SLinus Torvalds 	}
15331da177e4SLinus Torvalds 	if (fwrite(b->p, 1, b->pos, file) != b->pos) {
15341da177e4SLinus Torvalds 		perror(fname);
15351da177e4SLinus Torvalds 		exit(1);
15361da177e4SLinus Torvalds 	}
15371da177e4SLinus Torvalds 	fclose(file);
15381da177e4SLinus Torvalds }
15391da177e4SLinus Torvalds 
1540bd5cbcedSRam Pai /* parse Module.symvers file. line format:
1541534b89a9SSam Ravnborg  * 0x12345678<tab>symbol<tab>module[[<tab>export]<tab>something]
1542bd5cbcedSRam Pai  **/
1543040fcc81SSam Ravnborg static void read_dump(const char *fname, unsigned int kernel)
15441da177e4SLinus Torvalds {
15451da177e4SLinus Torvalds 	unsigned long size, pos = 0;
15461da177e4SLinus Torvalds 	void *file = grab_file(fname, &size);
15471da177e4SLinus Torvalds 	char *line;
15481da177e4SLinus Torvalds 
15491da177e4SLinus Torvalds         if (!file)
15501da177e4SLinus Torvalds 		/* No symbol versions, silently ignore */
15511da177e4SLinus Torvalds 		return;
15521da177e4SLinus Torvalds 
15531da177e4SLinus Torvalds 	while ((line = get_next_line(&pos, file, size))) {
1554534b89a9SSam Ravnborg 		char *symname, *modname, *d, *export, *end;
15551da177e4SLinus Torvalds 		unsigned int crc;
15561da177e4SLinus Torvalds 		struct module *mod;
1557040fcc81SSam Ravnborg 		struct symbol *s;
15581da177e4SLinus Torvalds 
15591da177e4SLinus Torvalds 		if (!(symname = strchr(line, '\t')))
15601da177e4SLinus Torvalds 			goto fail;
15611da177e4SLinus Torvalds 		*symname++ = '\0';
15621da177e4SLinus Torvalds 		if (!(modname = strchr(symname, '\t')))
15631da177e4SLinus Torvalds 			goto fail;
15641da177e4SLinus Torvalds 		*modname++ = '\0';
15659ac545b0SLaurent Riffard 		if ((export = strchr(modname, '\t')) != NULL)
1566bd5cbcedSRam Pai 			*export++ = '\0';
1567534b89a9SSam Ravnborg 		if (export && ((end = strchr(export, '\t')) != NULL))
1568534b89a9SSam Ravnborg 			*end = '\0';
15691da177e4SLinus Torvalds 		crc = strtoul(line, &d, 16);
15701da177e4SLinus Torvalds 		if (*symname == '\0' || *modname == '\0' || *d != '\0')
15711da177e4SLinus Torvalds 			goto fail;
15721da177e4SLinus Torvalds 
15731da177e4SLinus Torvalds 		if (!(mod = find_module(modname))) {
15741da177e4SLinus Torvalds 			if (is_vmlinux(modname)) {
15751da177e4SLinus Torvalds 				have_vmlinux = 1;
15761da177e4SLinus Torvalds 			}
15771da177e4SLinus Torvalds 			mod = new_module(NOFAIL(strdup(modname)));
15781da177e4SLinus Torvalds 			mod->skip = 1;
15791da177e4SLinus Torvalds 		}
1580bd5cbcedSRam Pai 		s = sym_add_exported(symname, mod, export_no(export));
1581040fcc81SSam Ravnborg 		s->kernel    = kernel;
15828e70c458SSam Ravnborg 		s->preloaded = 1;
1583bd5cbcedSRam Pai 		sym_update_crc(symname, mod, crc, export_no(export));
15841da177e4SLinus Torvalds 	}
15851da177e4SLinus Torvalds 	return;
15861da177e4SLinus Torvalds fail:
15871da177e4SLinus Torvalds 	fatal("parse error in symbol dump file\n");
15881da177e4SLinus Torvalds }
15891da177e4SLinus Torvalds 
1590040fcc81SSam Ravnborg /* For normal builds always dump all symbols.
1591040fcc81SSam Ravnborg  * For external modules only dump symbols
1592040fcc81SSam Ravnborg  * that are not read from kernel Module.symvers.
1593040fcc81SSam Ravnborg  **/
1594040fcc81SSam Ravnborg static int dump_sym(struct symbol *sym)
1595040fcc81SSam Ravnborg {
1596040fcc81SSam Ravnborg 	if (!external_module)
1597040fcc81SSam Ravnborg 		return 1;
1598040fcc81SSam Ravnborg 	if (sym->vmlinux || sym->kernel)
1599040fcc81SSam Ravnborg 		return 0;
1600040fcc81SSam Ravnborg 	return 1;
1601040fcc81SSam Ravnborg }
1602040fcc81SSam Ravnborg 
16035c3ead8cSSam Ravnborg static void write_dump(const char *fname)
16041da177e4SLinus Torvalds {
16051da177e4SLinus Torvalds 	struct buffer buf = { };
16061da177e4SLinus Torvalds 	struct symbol *symbol;
16071da177e4SLinus Torvalds 	int n;
16081da177e4SLinus Torvalds 
16091da177e4SLinus Torvalds 	for (n = 0; n < SYMBOL_HASH_SIZE ; n++) {
16101da177e4SLinus Torvalds 		symbol = symbolhash[n];
16111da177e4SLinus Torvalds 		while (symbol) {
1612040fcc81SSam Ravnborg 			if (dump_sym(symbol))
1613bd5cbcedSRam Pai 				buf_printf(&buf, "0x%08x\t%s\t%s\t%s\n",
1614040fcc81SSam Ravnborg 					symbol->crc, symbol->name,
1615bd5cbcedSRam Pai 					symbol->module->name,
1616bd5cbcedSRam Pai 					export_str(symbol->export));
16171da177e4SLinus Torvalds 			symbol = symbol->next;
16181da177e4SLinus Torvalds 		}
16191da177e4SLinus Torvalds 	}
16201da177e4SLinus Torvalds 	write_if_changed(&buf, fname);
16211da177e4SLinus Torvalds }
16221da177e4SLinus Torvalds 
16235c3ead8cSSam Ravnborg int main(int argc, char **argv)
16241da177e4SLinus Torvalds {
16251da177e4SLinus Torvalds 	struct module *mod;
16261da177e4SLinus Torvalds 	struct buffer buf = { };
16271da177e4SLinus Torvalds 	char fname[SZ];
1628040fcc81SSam Ravnborg 	char *kernel_read = NULL, *module_read = NULL;
1629040fcc81SSam Ravnborg 	char *dump_write = NULL;
16301da177e4SLinus Torvalds 	int opt;
1631c53ddacdSKirill Korotaev 	int err;
16321da177e4SLinus Torvalds 
16338d8d8289SSam Ravnborg 	while ((opt = getopt(argc, argv, "i:I:mso:aw")) != -1) {
16341da177e4SLinus Torvalds 		switch(opt) {
16351da177e4SLinus Torvalds 			case 'i':
1636040fcc81SSam Ravnborg 				kernel_read = optarg;
1637040fcc81SSam Ravnborg 				break;
1638040fcc81SSam Ravnborg 			case 'I':
1639040fcc81SSam Ravnborg 				module_read = optarg;
1640040fcc81SSam Ravnborg 				external_module = 1;
16411da177e4SLinus Torvalds 				break;
16421da177e4SLinus Torvalds 			case 'm':
16431da177e4SLinus Torvalds 				modversions = 1;
16441da177e4SLinus Torvalds 				break;
16451da177e4SLinus Torvalds 			case 'o':
16461da177e4SLinus Torvalds 				dump_write = optarg;
16471da177e4SLinus Torvalds 				break;
16481da177e4SLinus Torvalds 			case 'a':
16491da177e4SLinus Torvalds 				all_versions = 1;
16501da177e4SLinus Torvalds 				break;
16518d8d8289SSam Ravnborg 			case 's':
16528d8d8289SSam Ravnborg 				vmlinux_section_warnings = 0;
16538d8d8289SSam Ravnborg 				break;
1654c53ddacdSKirill Korotaev 			case 'w':
1655c53ddacdSKirill Korotaev 				warn_unresolved = 1;
1656c53ddacdSKirill Korotaev 				break;
16571da177e4SLinus Torvalds 			default:
16581da177e4SLinus Torvalds 				exit(1);
16591da177e4SLinus Torvalds 		}
16601da177e4SLinus Torvalds 	}
16611da177e4SLinus Torvalds 
1662040fcc81SSam Ravnborg 	if (kernel_read)
1663040fcc81SSam Ravnborg 		read_dump(kernel_read, 1);
1664040fcc81SSam Ravnborg 	if (module_read)
1665040fcc81SSam Ravnborg 		read_dump(module_read, 0);
16661da177e4SLinus Torvalds 
16671da177e4SLinus Torvalds 	while (optind < argc) {
16681da177e4SLinus Torvalds 		read_symbols(argv[optind++]);
16691da177e4SLinus Torvalds 	}
16701da177e4SLinus Torvalds 
16711da177e4SLinus Torvalds 	for (mod = modules; mod; mod = mod->next) {
16721da177e4SLinus Torvalds 		if (mod->skip)
16731da177e4SLinus Torvalds 			continue;
1674c96fca21SSam Ravnborg 		check_exports(mod);
1675b817f6feSSam Ravnborg 	}
1676b817f6feSSam Ravnborg 
1677c53ddacdSKirill Korotaev 	err = 0;
1678c53ddacdSKirill Korotaev 
1679b817f6feSSam Ravnborg 	for (mod = modules; mod; mod = mod->next) {
1680b817f6feSSam Ravnborg 		if (mod->skip)
1681b817f6feSSam Ravnborg 			continue;
16821da177e4SLinus Torvalds 
16831da177e4SLinus Torvalds 		buf.pos = 0;
16841da177e4SLinus Torvalds 
16851da177e4SLinus Torvalds 		add_header(&buf, mod);
1686c53ddacdSKirill Korotaev 		err |= add_versions(&buf, mod);
16871da177e4SLinus Torvalds 		add_depends(&buf, mod, modules);
16881da177e4SLinus Torvalds 		add_moddevtable(&buf, mod);
16891da177e4SLinus Torvalds 		add_srcversion(&buf, mod);
16901da177e4SLinus Torvalds 
16911da177e4SLinus Torvalds 		sprintf(fname, "%s.mod.c", mod->name);
16921da177e4SLinus Torvalds 		write_if_changed(&buf, fname);
16931da177e4SLinus Torvalds 	}
16941da177e4SLinus Torvalds 
16951da177e4SLinus Torvalds 	if (dump_write)
16961da177e4SLinus Torvalds 		write_dump(dump_write);
16971da177e4SLinus Torvalds 
1698c53ddacdSKirill Korotaev 	return err;
16991da177e4SLinus Torvalds }
1700