xref: /linux-6.15/scripts/mod/modpost.c (revision a61b2dfd)
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;
26c53ddacdSKirill Korotaev /* Only warn about unresolved symbols */
27c53ddacdSKirill Korotaev static int warn_unresolved = 0;
28bd5cbcedSRam Pai /* How a symbol is exported */
29c96fca21SSam Ravnborg enum export {
30c96fca21SSam Ravnborg 	export_plain,      export_unused,     export_gpl,
31c96fca21SSam Ravnborg 	export_unused_gpl, export_gpl_future, export_unknown
32c96fca21SSam Ravnborg };
331da177e4SLinus Torvalds 
345c3ead8cSSam Ravnborg void fatal(const char *fmt, ...)
351da177e4SLinus Torvalds {
361da177e4SLinus Torvalds 	va_list arglist;
371da177e4SLinus Torvalds 
381da177e4SLinus Torvalds 	fprintf(stderr, "FATAL: ");
391da177e4SLinus Torvalds 
401da177e4SLinus Torvalds 	va_start(arglist, fmt);
411da177e4SLinus Torvalds 	vfprintf(stderr, fmt, arglist);
421da177e4SLinus Torvalds 	va_end(arglist);
431da177e4SLinus Torvalds 
441da177e4SLinus Torvalds 	exit(1);
451da177e4SLinus Torvalds }
461da177e4SLinus Torvalds 
475c3ead8cSSam Ravnborg void warn(const char *fmt, ...)
481da177e4SLinus Torvalds {
491da177e4SLinus Torvalds 	va_list arglist;
501da177e4SLinus Torvalds 
511da177e4SLinus Torvalds 	fprintf(stderr, "WARNING: ");
521da177e4SLinus Torvalds 
531da177e4SLinus Torvalds 	va_start(arglist, fmt);
541da177e4SLinus Torvalds 	vfprintf(stderr, fmt, arglist);
551da177e4SLinus Torvalds 	va_end(arglist);
561da177e4SLinus Torvalds }
571da177e4SLinus Torvalds 
58040fcc81SSam Ravnborg static int is_vmlinux(const char *modname)
59040fcc81SSam Ravnborg {
60040fcc81SSam Ravnborg 	const char *myname;
61040fcc81SSam Ravnborg 
62040fcc81SSam Ravnborg 	if ((myname = strrchr(modname, '/')))
63040fcc81SSam Ravnborg 		myname++;
64040fcc81SSam Ravnborg 	else
65040fcc81SSam Ravnborg 		myname = modname;
66040fcc81SSam Ravnborg 
67040fcc81SSam Ravnborg 	return strcmp(myname, "vmlinux") == 0;
68040fcc81SSam Ravnborg }
69040fcc81SSam Ravnborg 
701da177e4SLinus Torvalds void *do_nofail(void *ptr, const char *expr)
711da177e4SLinus Torvalds {
721da177e4SLinus Torvalds 	if (!ptr) {
731da177e4SLinus Torvalds 		fatal("modpost: Memory allocation failure: %s.\n", expr);
741da177e4SLinus Torvalds 	}
751da177e4SLinus Torvalds 	return ptr;
761da177e4SLinus Torvalds }
771da177e4SLinus Torvalds 
781da177e4SLinus Torvalds /* A list of all modules we processed */
791da177e4SLinus Torvalds 
801da177e4SLinus Torvalds static struct module *modules;
811da177e4SLinus Torvalds 
825c3ead8cSSam Ravnborg static struct module *find_module(char *modname)
831da177e4SLinus Torvalds {
841da177e4SLinus Torvalds 	struct module *mod;
851da177e4SLinus Torvalds 
861da177e4SLinus Torvalds 	for (mod = modules; mod; mod = mod->next)
871da177e4SLinus Torvalds 		if (strcmp(mod->name, modname) == 0)
881da177e4SLinus Torvalds 			break;
891da177e4SLinus Torvalds 	return mod;
901da177e4SLinus Torvalds }
911da177e4SLinus Torvalds 
925c3ead8cSSam Ravnborg static struct module *new_module(char *modname)
931da177e4SLinus Torvalds {
941da177e4SLinus Torvalds 	struct module *mod;
951da177e4SLinus Torvalds 	char *p, *s;
961da177e4SLinus Torvalds 
971da177e4SLinus Torvalds 	mod = NOFAIL(malloc(sizeof(*mod)));
981da177e4SLinus Torvalds 	memset(mod, 0, sizeof(*mod));
991da177e4SLinus Torvalds 	p = NOFAIL(strdup(modname));
1001da177e4SLinus Torvalds 
1011da177e4SLinus Torvalds 	/* strip trailing .o */
1021da177e4SLinus Torvalds 	if ((s = strrchr(p, '.')) != NULL)
1031da177e4SLinus Torvalds 		if (strcmp(s, ".o") == 0)
1041da177e4SLinus Torvalds 			*s = '\0';
1051da177e4SLinus Torvalds 
1061da177e4SLinus Torvalds 	/* add to list */
1071da177e4SLinus Torvalds 	mod->name = p;
108b817f6feSSam Ravnborg 	mod->gpl_compatible = -1;
1091da177e4SLinus Torvalds 	mod->next = modules;
1101da177e4SLinus Torvalds 	modules = mod;
1111da177e4SLinus Torvalds 
1121da177e4SLinus Torvalds 	return mod;
1131da177e4SLinus Torvalds }
1141da177e4SLinus Torvalds 
1151da177e4SLinus Torvalds /* A hash of all exported symbols,
1161da177e4SLinus Torvalds  * struct symbol is also used for lists of unresolved symbols */
1171da177e4SLinus Torvalds 
1181da177e4SLinus Torvalds #define SYMBOL_HASH_SIZE 1024
1191da177e4SLinus Torvalds 
1201da177e4SLinus Torvalds struct symbol {
1211da177e4SLinus Torvalds 	struct symbol *next;
1221da177e4SLinus Torvalds 	struct module *module;
1231da177e4SLinus Torvalds 	unsigned int crc;
1241da177e4SLinus Torvalds 	int crc_valid;
1251da177e4SLinus Torvalds 	unsigned int weak:1;
126040fcc81SSam Ravnborg 	unsigned int vmlinux:1;    /* 1 if symbol is defined in vmlinux */
127040fcc81SSam Ravnborg 	unsigned int kernel:1;     /* 1 if symbol is from kernel
128040fcc81SSam Ravnborg 				    *  (only for external modules) **/
1298e70c458SSam Ravnborg 	unsigned int preloaded:1;  /* 1 if symbol from Module.symvers */
130bd5cbcedSRam Pai 	enum export  export;       /* Type of export */
1311da177e4SLinus Torvalds 	char name[0];
1321da177e4SLinus Torvalds };
1331da177e4SLinus Torvalds 
1341da177e4SLinus Torvalds static struct symbol *symbolhash[SYMBOL_HASH_SIZE];
1351da177e4SLinus Torvalds 
1361da177e4SLinus Torvalds /* This is based on the hash agorithm from gdbm, via tdb */
1371da177e4SLinus Torvalds static inline unsigned int tdb_hash(const char *name)
1381da177e4SLinus Torvalds {
1391da177e4SLinus Torvalds 	unsigned value;	/* Used to compute the hash value.  */
1401da177e4SLinus Torvalds 	unsigned   i;	/* Used to cycle through random values. */
1411da177e4SLinus Torvalds 
1421da177e4SLinus Torvalds 	/* Set the initial value from the key size. */
1431da177e4SLinus Torvalds 	for (value = 0x238F13AF * strlen(name), i=0; name[i]; i++)
1441da177e4SLinus Torvalds 		value = (value + (((unsigned char *)name)[i] << (i*5 % 24)));
1451da177e4SLinus Torvalds 
1461da177e4SLinus Torvalds 	return (1103515243 * value + 12345);
1471da177e4SLinus Torvalds }
1481da177e4SLinus Torvalds 
1495c3ead8cSSam Ravnborg /**
1505c3ead8cSSam Ravnborg  * Allocate a new symbols for use in the hash of exported symbols or
1515c3ead8cSSam Ravnborg  * the list of unresolved symbols per module
1525c3ead8cSSam Ravnborg  **/
1535c3ead8cSSam Ravnborg static struct symbol *alloc_symbol(const char *name, unsigned int weak,
1545c3ead8cSSam Ravnborg 				   struct symbol *next)
1551da177e4SLinus Torvalds {
1561da177e4SLinus Torvalds 	struct symbol *s = NOFAIL(malloc(sizeof(*s) + strlen(name) + 1));
1571da177e4SLinus Torvalds 
1581da177e4SLinus Torvalds 	memset(s, 0, sizeof(*s));
1591da177e4SLinus Torvalds 	strcpy(s->name, name);
1601da177e4SLinus Torvalds 	s->weak = weak;
1611da177e4SLinus Torvalds 	s->next = next;
1621da177e4SLinus Torvalds 	return s;
1631da177e4SLinus Torvalds }
1641da177e4SLinus Torvalds 
1651da177e4SLinus Torvalds /* For the hash of exported symbols */
166bd5cbcedSRam Pai static struct symbol *new_symbol(const char *name, struct module *module,
167bd5cbcedSRam Pai 				 enum export export)
1681da177e4SLinus Torvalds {
1691da177e4SLinus Torvalds 	unsigned int hash;
1701da177e4SLinus Torvalds 	struct symbol *new;
1711da177e4SLinus Torvalds 
1721da177e4SLinus Torvalds 	hash = tdb_hash(name) % SYMBOL_HASH_SIZE;
1731da177e4SLinus Torvalds 	new = symbolhash[hash] = alloc_symbol(name, 0, symbolhash[hash]);
1741da177e4SLinus Torvalds 	new->module = module;
175bd5cbcedSRam Pai 	new->export = export;
176040fcc81SSam Ravnborg 	return new;
1771da177e4SLinus Torvalds }
1781da177e4SLinus Torvalds 
1795c3ead8cSSam Ravnborg static struct symbol *find_symbol(const char *name)
1801da177e4SLinus Torvalds {
1811da177e4SLinus Torvalds 	struct symbol *s;
1821da177e4SLinus Torvalds 
1831da177e4SLinus Torvalds 	/* For our purposes, .foo matches foo.  PPC64 needs this. */
1841da177e4SLinus Torvalds 	if (name[0] == '.')
1851da177e4SLinus Torvalds 		name++;
1861da177e4SLinus Torvalds 
1871da177e4SLinus Torvalds 	for (s = symbolhash[tdb_hash(name) % SYMBOL_HASH_SIZE]; s; s=s->next) {
1881da177e4SLinus Torvalds 		if (strcmp(s->name, name) == 0)
1891da177e4SLinus Torvalds 			return s;
1901da177e4SLinus Torvalds 	}
1911da177e4SLinus Torvalds 	return NULL;
1921da177e4SLinus Torvalds }
1931da177e4SLinus Torvalds 
194bd5cbcedSRam Pai static struct {
195bd5cbcedSRam Pai 	const char *str;
196bd5cbcedSRam Pai 	enum export export;
197bd5cbcedSRam Pai } export_list[] = {
198bd5cbcedSRam Pai 	{ .str = "EXPORT_SYMBOL",            .export = export_plain },
199c96fca21SSam Ravnborg 	{ .str = "EXPORT_UNUSED_SYMBOL",     .export = export_unused },
200bd5cbcedSRam Pai 	{ .str = "EXPORT_SYMBOL_GPL",        .export = export_gpl },
201c96fca21SSam Ravnborg 	{ .str = "EXPORT_UNUSED_SYMBOL_GPL", .export = export_unused_gpl },
202bd5cbcedSRam Pai 	{ .str = "EXPORT_SYMBOL_GPL_FUTURE", .export = export_gpl_future },
203bd5cbcedSRam Pai 	{ .str = "(unknown)",                .export = export_unknown },
204bd5cbcedSRam Pai };
205bd5cbcedSRam Pai 
206bd5cbcedSRam Pai 
207bd5cbcedSRam Pai static const char *export_str(enum export ex)
208bd5cbcedSRam Pai {
209bd5cbcedSRam Pai 	return export_list[ex].str;
210bd5cbcedSRam Pai }
211bd5cbcedSRam Pai 
212bd5cbcedSRam Pai static enum export export_no(const char * s)
213bd5cbcedSRam Pai {
214bd5cbcedSRam Pai 	int i;
215534b89a9SSam Ravnborg 	if (!s)
216534b89a9SSam Ravnborg 		return export_unknown;
217bd5cbcedSRam Pai 	for (i = 0; export_list[i].export != export_unknown; i++) {
218bd5cbcedSRam Pai 		if (strcmp(export_list[i].str, s) == 0)
219bd5cbcedSRam Pai 			return export_list[i].export;
220bd5cbcedSRam Pai 	}
221bd5cbcedSRam Pai 	return export_unknown;
222bd5cbcedSRam Pai }
223bd5cbcedSRam Pai 
224bd5cbcedSRam Pai static enum export export_from_sec(struct elf_info *elf, Elf_Section sec)
225bd5cbcedSRam Pai {
226bd5cbcedSRam Pai 	if (sec == elf->export_sec)
227bd5cbcedSRam Pai 		return export_plain;
228c96fca21SSam Ravnborg 	else if (sec == elf->export_unused_sec)
229c96fca21SSam Ravnborg 		return export_unused;
230bd5cbcedSRam Pai 	else if (sec == elf->export_gpl_sec)
231bd5cbcedSRam Pai 		return export_gpl;
232c96fca21SSam Ravnborg 	else if (sec == elf->export_unused_gpl_sec)
233c96fca21SSam Ravnborg 		return export_unused_gpl;
234bd5cbcedSRam Pai 	else if (sec == elf->export_gpl_future_sec)
235bd5cbcedSRam Pai 		return export_gpl_future;
236bd5cbcedSRam Pai 	else
237bd5cbcedSRam Pai 		return export_unknown;
238bd5cbcedSRam Pai }
239bd5cbcedSRam Pai 
2405c3ead8cSSam Ravnborg /**
2415c3ead8cSSam Ravnborg  * Add an exported symbol - it may have already been added without a
2425c3ead8cSSam Ravnborg  * CRC, in this case just update the CRC
2435c3ead8cSSam Ravnborg  **/
244bd5cbcedSRam Pai static struct symbol *sym_add_exported(const char *name, struct module *mod,
245bd5cbcedSRam Pai 				       enum export export)
2461da177e4SLinus Torvalds {
2471da177e4SLinus Torvalds 	struct symbol *s = find_symbol(name);
2481da177e4SLinus Torvalds 
2491da177e4SLinus Torvalds 	if (!s) {
250bd5cbcedSRam Pai 		s = new_symbol(name, mod, export);
2518e70c458SSam Ravnborg 	} else {
2528e70c458SSam Ravnborg 		if (!s->preloaded) {
2537b75b13cSSam Ravnborg 			warn("%s: '%s' exported twice. Previous export "
2548e70c458SSam Ravnborg 			     "was in %s%s\n", mod->name, name,
2558e70c458SSam Ravnborg 			     s->module->name,
2568e70c458SSam Ravnborg 			     is_vmlinux(s->module->name) ?"":".ko");
2571da177e4SLinus Torvalds 		}
2581da177e4SLinus Torvalds 	}
2598e70c458SSam Ravnborg 	s->preloaded = 0;
260040fcc81SSam Ravnborg 	s->vmlinux   = is_vmlinux(mod->name);
261040fcc81SSam Ravnborg 	s->kernel    = 0;
262bd5cbcedSRam Pai 	s->export    = export;
263040fcc81SSam Ravnborg 	return s;
2641da177e4SLinus Torvalds }
2651da177e4SLinus Torvalds 
266040fcc81SSam Ravnborg static void sym_update_crc(const char *name, struct module *mod,
267bd5cbcedSRam Pai 			   unsigned int crc, enum export export)
268040fcc81SSam Ravnborg {
269040fcc81SSam Ravnborg 	struct symbol *s = find_symbol(name);
270040fcc81SSam Ravnborg 
271040fcc81SSam Ravnborg 	if (!s)
272bd5cbcedSRam Pai 		s = new_symbol(name, mod, export);
273040fcc81SSam Ravnborg 	s->crc = crc;
2741da177e4SLinus Torvalds 	s->crc_valid = 1;
2751da177e4SLinus Torvalds }
2761da177e4SLinus Torvalds 
2775c3ead8cSSam Ravnborg void *grab_file(const char *filename, unsigned long *size)
2781da177e4SLinus Torvalds {
2791da177e4SLinus Torvalds 	struct stat st;
2801da177e4SLinus Torvalds 	void *map;
2811da177e4SLinus Torvalds 	int fd;
2821da177e4SLinus Torvalds 
2831da177e4SLinus Torvalds 	fd = open(filename, O_RDONLY);
2841da177e4SLinus Torvalds 	if (fd < 0 || fstat(fd, &st) != 0)
2851da177e4SLinus Torvalds 		return NULL;
2861da177e4SLinus Torvalds 
2871da177e4SLinus Torvalds 	*size = st.st_size;
2881da177e4SLinus Torvalds 	map = mmap(NULL, *size, PROT_READ|PROT_WRITE, MAP_PRIVATE, fd, 0);
2891da177e4SLinus Torvalds 	close(fd);
2901da177e4SLinus Torvalds 
2911da177e4SLinus Torvalds 	if (map == MAP_FAILED)
2921da177e4SLinus Torvalds 		return NULL;
2931da177e4SLinus Torvalds 	return map;
2941da177e4SLinus Torvalds }
2951da177e4SLinus Torvalds 
2965c3ead8cSSam Ravnborg /**
2975c3ead8cSSam Ravnborg   * Return a copy of the next line in a mmap'ed file.
2985c3ead8cSSam Ravnborg   * spaces in the beginning of the line is trimmed away.
2995c3ead8cSSam Ravnborg   * Return a pointer to a static buffer.
3005c3ead8cSSam Ravnborg   **/
3015c3ead8cSSam Ravnborg char* get_next_line(unsigned long *pos, void *file, unsigned long size)
3021da177e4SLinus Torvalds {
3031da177e4SLinus Torvalds 	static char line[4096];
3041da177e4SLinus Torvalds 	int skip = 1;
3051da177e4SLinus Torvalds 	size_t len = 0;
3061da177e4SLinus Torvalds 	signed char *p = (signed char *)file + *pos;
3071da177e4SLinus Torvalds 	char *s = line;
3081da177e4SLinus Torvalds 
3091da177e4SLinus Torvalds 	for (; *pos < size ; (*pos)++)
3101da177e4SLinus Torvalds 	{
3111da177e4SLinus Torvalds 		if (skip && isspace(*p)) {
3121da177e4SLinus Torvalds 			p++;
3131da177e4SLinus Torvalds 			continue;
3141da177e4SLinus Torvalds 		}
3151da177e4SLinus Torvalds 		skip = 0;
3161da177e4SLinus Torvalds 		if (*p != '\n' && (*pos < size)) {
3171da177e4SLinus Torvalds 			len++;
3181da177e4SLinus Torvalds 			*s++ = *p++;
3191da177e4SLinus Torvalds 			if (len > 4095)
3201da177e4SLinus Torvalds 				break; /* Too long, stop */
3211da177e4SLinus Torvalds 		} else {
3221da177e4SLinus Torvalds 			/* End of string */
3231da177e4SLinus Torvalds 			*s = '\0';
3241da177e4SLinus Torvalds 			return line;
3251da177e4SLinus Torvalds 		}
3261da177e4SLinus Torvalds 	}
3271da177e4SLinus Torvalds 	/* End of buffer */
3281da177e4SLinus Torvalds 	return NULL;
3291da177e4SLinus Torvalds }
3301da177e4SLinus Torvalds 
3315c3ead8cSSam Ravnborg void release_file(void *file, unsigned long size)
3321da177e4SLinus Torvalds {
3331da177e4SLinus Torvalds 	munmap(file, size);
3341da177e4SLinus Torvalds }
3351da177e4SLinus Torvalds 
33685bd2fddSSam Ravnborg static int parse_elf(struct elf_info *info, const char *filename)
3371da177e4SLinus Torvalds {
3381da177e4SLinus Torvalds 	unsigned int i;
33985bd2fddSSam Ravnborg 	Elf_Ehdr *hdr;
3401da177e4SLinus Torvalds 	Elf_Shdr *sechdrs;
3411da177e4SLinus Torvalds 	Elf_Sym  *sym;
3421da177e4SLinus Torvalds 
3431da177e4SLinus Torvalds 	hdr = grab_file(filename, &info->size);
3441da177e4SLinus Torvalds 	if (!hdr) {
3451da177e4SLinus Torvalds 		perror(filename);
3466803dc0eSSam Ravnborg 		exit(1);
3471da177e4SLinus Torvalds 	}
3481da177e4SLinus Torvalds 	info->hdr = hdr;
34985bd2fddSSam Ravnborg 	if (info->size < sizeof(*hdr)) {
35085bd2fddSSam Ravnborg 		/* file too small, assume this is an empty .o file */
35185bd2fddSSam Ravnborg 		return 0;
35285bd2fddSSam Ravnborg 	}
35385bd2fddSSam Ravnborg 	/* Is this a valid ELF file? */
35485bd2fddSSam Ravnborg 	if ((hdr->e_ident[EI_MAG0] != ELFMAG0) ||
35585bd2fddSSam Ravnborg 	    (hdr->e_ident[EI_MAG1] != ELFMAG1) ||
35685bd2fddSSam Ravnborg 	    (hdr->e_ident[EI_MAG2] != ELFMAG2) ||
35785bd2fddSSam Ravnborg 	    (hdr->e_ident[EI_MAG3] != ELFMAG3)) {
35885bd2fddSSam Ravnborg 		/* Not an ELF file - silently ignore it */
35985bd2fddSSam Ravnborg 		return 0;
36085bd2fddSSam Ravnborg 	}
3611da177e4SLinus Torvalds 	/* Fix endianness in ELF header */
3621da177e4SLinus Torvalds 	hdr->e_shoff    = TO_NATIVE(hdr->e_shoff);
3631da177e4SLinus Torvalds 	hdr->e_shstrndx = TO_NATIVE(hdr->e_shstrndx);
3641da177e4SLinus Torvalds 	hdr->e_shnum    = TO_NATIVE(hdr->e_shnum);
3651da177e4SLinus Torvalds 	hdr->e_machine  = TO_NATIVE(hdr->e_machine);
3661da177e4SLinus Torvalds 	sechdrs = (void *)hdr + hdr->e_shoff;
3671da177e4SLinus Torvalds 	info->sechdrs = sechdrs;
3681da177e4SLinus Torvalds 
3691da177e4SLinus Torvalds 	/* Fix endianness in section headers */
3701da177e4SLinus Torvalds 	for (i = 0; i < hdr->e_shnum; i++) {
3711da177e4SLinus Torvalds 		sechdrs[i].sh_type   = TO_NATIVE(sechdrs[i].sh_type);
3721da177e4SLinus Torvalds 		sechdrs[i].sh_offset = TO_NATIVE(sechdrs[i].sh_offset);
3731da177e4SLinus Torvalds 		sechdrs[i].sh_size   = TO_NATIVE(sechdrs[i].sh_size);
3741da177e4SLinus Torvalds 		sechdrs[i].sh_link   = TO_NATIVE(sechdrs[i].sh_link);
3751da177e4SLinus Torvalds 		sechdrs[i].sh_name   = TO_NATIVE(sechdrs[i].sh_name);
3761da177e4SLinus Torvalds 	}
3771da177e4SLinus Torvalds 	/* Find symbol table. */
3781da177e4SLinus Torvalds 	for (i = 1; i < hdr->e_shnum; i++) {
3791da177e4SLinus Torvalds 		const char *secstrings
3801da177e4SLinus Torvalds 			= (void *)hdr + sechdrs[hdr->e_shstrndx].sh_offset;
381bd5cbcedSRam Pai 		const char *secname;
3821da177e4SLinus Torvalds 
38385bd2fddSSam Ravnborg 		if (sechdrs[i].sh_offset > info->size) {
38485bd2fddSSam Ravnborg 			fatal("%s is truncated. sechdrs[i].sh_offset=%u > sizeof(*hrd)=%ul\n", filename, (unsigned int)sechdrs[i].sh_offset, sizeof(*hdr));
38585bd2fddSSam Ravnborg 			return 0;
38685bd2fddSSam Ravnborg 		}
387bd5cbcedSRam Pai 		secname = secstrings + sechdrs[i].sh_name;
388bd5cbcedSRam Pai 		if (strcmp(secname, ".modinfo") == 0) {
3891da177e4SLinus Torvalds 			info->modinfo = (void *)hdr + sechdrs[i].sh_offset;
3901da177e4SLinus Torvalds 			info->modinfo_len = sechdrs[i].sh_size;
391bd5cbcedSRam Pai 		} else if (strcmp(secname, "__ksymtab") == 0)
392bd5cbcedSRam Pai 			info->export_sec = i;
393c96fca21SSam Ravnborg 		else if (strcmp(secname, "__ksymtab_unused") == 0)
394c96fca21SSam Ravnborg 			info->export_unused_sec = i;
395bd5cbcedSRam Pai 		else if (strcmp(secname, "__ksymtab_gpl") == 0)
396bd5cbcedSRam Pai 			info->export_gpl_sec = i;
397c96fca21SSam Ravnborg 		else if (strcmp(secname, "__ksymtab_unused_gpl") == 0)
398c96fca21SSam Ravnborg 			info->export_unused_gpl_sec = i;
399bd5cbcedSRam Pai 		else if (strcmp(secname, "__ksymtab_gpl_future") == 0)
400bd5cbcedSRam Pai 			info->export_gpl_future_sec = i;
401bd5cbcedSRam Pai 
4021da177e4SLinus Torvalds 		if (sechdrs[i].sh_type != SHT_SYMTAB)
4031da177e4SLinus Torvalds 			continue;
4041da177e4SLinus Torvalds 
4051da177e4SLinus Torvalds 		info->symtab_start = (void *)hdr + sechdrs[i].sh_offset;
4061da177e4SLinus Torvalds 		info->symtab_stop  = (void *)hdr + sechdrs[i].sh_offset
4071da177e4SLinus Torvalds 			                         + sechdrs[i].sh_size;
4081da177e4SLinus Torvalds 		info->strtab       = (void *)hdr +
4091da177e4SLinus Torvalds 			             sechdrs[sechdrs[i].sh_link].sh_offset;
4101da177e4SLinus Torvalds 	}
4111da177e4SLinus Torvalds 	if (!info->symtab_start) {
412cb80514dSSam Ravnborg 		fatal("%s has no symtab?\n", filename);
4131da177e4SLinus Torvalds 	}
4141da177e4SLinus Torvalds 	/* Fix endianness in symbols */
4151da177e4SLinus Torvalds 	for (sym = info->symtab_start; sym < info->symtab_stop; sym++) {
4161da177e4SLinus Torvalds 		sym->st_shndx = TO_NATIVE(sym->st_shndx);
4171da177e4SLinus Torvalds 		sym->st_name  = TO_NATIVE(sym->st_name);
4181da177e4SLinus Torvalds 		sym->st_value = TO_NATIVE(sym->st_value);
4191da177e4SLinus Torvalds 		sym->st_size  = TO_NATIVE(sym->st_size);
4201da177e4SLinus Torvalds 	}
42185bd2fddSSam Ravnborg 	return 1;
4221da177e4SLinus Torvalds }
4231da177e4SLinus Torvalds 
4245c3ead8cSSam Ravnborg static void parse_elf_finish(struct elf_info *info)
4251da177e4SLinus Torvalds {
4261da177e4SLinus Torvalds 	release_file(info->hdr, info->size);
4271da177e4SLinus Torvalds }
4281da177e4SLinus Torvalds 
429f7b05e64SLuke Yang #define CRC_PFX     MODULE_SYMBOL_PREFIX "__crc_"
430f7b05e64SLuke Yang #define KSYMTAB_PFX MODULE_SYMBOL_PREFIX "__ksymtab_"
4311da177e4SLinus Torvalds 
4325c3ead8cSSam Ravnborg static void handle_modversions(struct module *mod, struct elf_info *info,
4331da177e4SLinus Torvalds 			       Elf_Sym *sym, const char *symname)
4341da177e4SLinus Torvalds {
4351da177e4SLinus Torvalds 	unsigned int crc;
436bd5cbcedSRam Pai 	enum export export = export_from_sec(info, sym->st_shndx);
4371da177e4SLinus Torvalds 
4381da177e4SLinus Torvalds 	switch (sym->st_shndx) {
4391da177e4SLinus Torvalds 	case SHN_COMMON:
440cb80514dSSam Ravnborg 		warn("\"%s\" [%s] is COMMON symbol\n", symname, mod->name);
4411da177e4SLinus Torvalds 		break;
4421da177e4SLinus Torvalds 	case SHN_ABS:
4431da177e4SLinus Torvalds 		/* CRC'd symbol */
4441da177e4SLinus Torvalds 		if (memcmp(symname, CRC_PFX, strlen(CRC_PFX)) == 0) {
4451da177e4SLinus Torvalds 			crc = (unsigned int) sym->st_value;
446bd5cbcedSRam Pai 			sym_update_crc(symname + strlen(CRC_PFX), mod, crc,
447bd5cbcedSRam Pai 					export);
4481da177e4SLinus Torvalds 		}
4491da177e4SLinus Torvalds 		break;
4501da177e4SLinus Torvalds 	case SHN_UNDEF:
4511da177e4SLinus Torvalds 		/* undefined symbol */
4521da177e4SLinus Torvalds 		if (ELF_ST_BIND(sym->st_info) != STB_GLOBAL &&
4531da177e4SLinus Torvalds 		    ELF_ST_BIND(sym->st_info) != STB_WEAK)
4541da177e4SLinus Torvalds 			break;
4551da177e4SLinus Torvalds 		/* ignore global offset table */
4561da177e4SLinus Torvalds 		if (strcmp(symname, "_GLOBAL_OFFSET_TABLE_") == 0)
4571da177e4SLinus Torvalds 			break;
4581da177e4SLinus Torvalds 		/* ignore __this_module, it will be resolved shortly */
4591da177e4SLinus Torvalds 		if (strcmp(symname, MODULE_SYMBOL_PREFIX "__this_module") == 0)
4601da177e4SLinus Torvalds 			break;
4618d529014SBen Colline /* cope with newer glibc (2.3.4 or higher) STT_ definition in elf.h */
4628d529014SBen Colline #if defined(STT_REGISTER) || defined(STT_SPARC_REGISTER)
4638d529014SBen Colline /* add compatibility with older glibc */
4648d529014SBen Colline #ifndef STT_SPARC_REGISTER
4658d529014SBen Colline #define STT_SPARC_REGISTER STT_REGISTER
4668d529014SBen Colline #endif
4671da177e4SLinus Torvalds 		if (info->hdr->e_machine == EM_SPARC ||
4681da177e4SLinus Torvalds 		    info->hdr->e_machine == EM_SPARCV9) {
4691da177e4SLinus Torvalds 			/* Ignore register directives. */
4708d529014SBen Colline 			if (ELF_ST_TYPE(sym->st_info) == STT_SPARC_REGISTER)
4711da177e4SLinus Torvalds 				break;
4727caaeabbSAl Viro 			if (symname[0] == '.') {
4737caaeabbSAl Viro 				char *munged = strdup(symname);
4747caaeabbSAl Viro 				munged[0] = '_';
4757caaeabbSAl Viro 				munged[1] = toupper(munged[1]);
4767caaeabbSAl Viro 				symname = munged;
4777caaeabbSAl Viro 			}
4781da177e4SLinus Torvalds 		}
4791da177e4SLinus Torvalds #endif
4801da177e4SLinus Torvalds 
4811da177e4SLinus Torvalds 		if (memcmp(symname, MODULE_SYMBOL_PREFIX,
4821da177e4SLinus Torvalds 			   strlen(MODULE_SYMBOL_PREFIX)) == 0)
4831da177e4SLinus Torvalds 			mod->unres = alloc_symbol(symname +
4841da177e4SLinus Torvalds 						  strlen(MODULE_SYMBOL_PREFIX),
4851da177e4SLinus Torvalds 						  ELF_ST_BIND(sym->st_info) == STB_WEAK,
4861da177e4SLinus Torvalds 						  mod->unres);
4871da177e4SLinus Torvalds 		break;
4881da177e4SLinus Torvalds 	default:
4891da177e4SLinus Torvalds 		/* All exported symbols */
4901da177e4SLinus Torvalds 		if (memcmp(symname, KSYMTAB_PFX, strlen(KSYMTAB_PFX)) == 0) {
491bd5cbcedSRam Pai 			sym_add_exported(symname + strlen(KSYMTAB_PFX), mod,
492bd5cbcedSRam Pai 					export);
4931da177e4SLinus Torvalds 		}
4941da177e4SLinus Torvalds 		if (strcmp(symname, MODULE_SYMBOL_PREFIX "init_module") == 0)
4951da177e4SLinus Torvalds 			mod->has_init = 1;
4961da177e4SLinus Torvalds 		if (strcmp(symname, MODULE_SYMBOL_PREFIX "cleanup_module") == 0)
4971da177e4SLinus Torvalds 			mod->has_cleanup = 1;
4981da177e4SLinus Torvalds 		break;
4991da177e4SLinus Torvalds 	}
5001da177e4SLinus Torvalds }
5011da177e4SLinus Torvalds 
5025c3ead8cSSam Ravnborg /**
5035c3ead8cSSam Ravnborg  * Parse tag=value strings from .modinfo section
5045c3ead8cSSam Ravnborg  **/
5051da177e4SLinus Torvalds static char *next_string(char *string, unsigned long *secsize)
5061da177e4SLinus Torvalds {
5071da177e4SLinus Torvalds 	/* Skip non-zero chars */
5081da177e4SLinus Torvalds 	while (string[0]) {
5091da177e4SLinus Torvalds 		string++;
5101da177e4SLinus Torvalds 		if ((*secsize)-- <= 1)
5111da177e4SLinus Torvalds 			return NULL;
5121da177e4SLinus Torvalds 	}
5131da177e4SLinus Torvalds 
5141da177e4SLinus Torvalds 	/* Skip any zero padding. */
5151da177e4SLinus Torvalds 	while (!string[0]) {
5161da177e4SLinus Torvalds 		string++;
5171da177e4SLinus Torvalds 		if ((*secsize)-- <= 1)
5181da177e4SLinus Torvalds 			return NULL;
5191da177e4SLinus Torvalds 	}
5201da177e4SLinus Torvalds 	return string;
5211da177e4SLinus Torvalds }
5221da177e4SLinus Torvalds 
523b817f6feSSam Ravnborg static char *get_next_modinfo(void *modinfo, unsigned long modinfo_len,
524b817f6feSSam Ravnborg 			      const char *tag, char *info)
5251da177e4SLinus Torvalds {
5261da177e4SLinus Torvalds 	char *p;
5271da177e4SLinus Torvalds 	unsigned int taglen = strlen(tag);
5281da177e4SLinus Torvalds 	unsigned long size = modinfo_len;
5291da177e4SLinus Torvalds 
530b817f6feSSam Ravnborg 	if (info) {
531b817f6feSSam Ravnborg 		size -= info - (char *)modinfo;
532b817f6feSSam Ravnborg 		modinfo = next_string(info, &size);
533b817f6feSSam Ravnborg 	}
534b817f6feSSam Ravnborg 
5351da177e4SLinus Torvalds 	for (p = modinfo; p; p = next_string(p, &size)) {
5361da177e4SLinus Torvalds 		if (strncmp(p, tag, taglen) == 0 && p[taglen] == '=')
5371da177e4SLinus Torvalds 			return p + taglen + 1;
5381da177e4SLinus Torvalds 	}
5391da177e4SLinus Torvalds 	return NULL;
5401da177e4SLinus Torvalds }
5411da177e4SLinus Torvalds 
542b817f6feSSam Ravnborg static char *get_modinfo(void *modinfo, unsigned long modinfo_len,
543b817f6feSSam Ravnborg 			 const char *tag)
544b817f6feSSam Ravnborg 
545b817f6feSSam Ravnborg {
546b817f6feSSam Ravnborg 	return get_next_modinfo(modinfo, modinfo_len, tag, NULL);
547b817f6feSSam Ravnborg }
548b817f6feSSam Ravnborg 
54993684d3bSSam Ravnborg /**
5504c8fbca5SSam Ravnborg  * Test if string s ends in string sub
5514c8fbca5SSam Ravnborg  * return 0 if match
5524c8fbca5SSam Ravnborg  **/
5534c8fbca5SSam Ravnborg static int strrcmp(const char *s, const char *sub)
5544c8fbca5SSam Ravnborg {
5554c8fbca5SSam Ravnborg         int slen, sublen;
5564c8fbca5SSam Ravnborg 
5574c8fbca5SSam Ravnborg 	if (!s || !sub)
5584c8fbca5SSam Ravnborg 		return 1;
5594c8fbca5SSam Ravnborg 
5604c8fbca5SSam Ravnborg 	slen = strlen(s);
5614c8fbca5SSam Ravnborg         sublen = strlen(sub);
5624c8fbca5SSam Ravnborg 
5634c8fbca5SSam Ravnborg 	if ((slen == 0) || (sublen == 0))
5644c8fbca5SSam Ravnborg 		return 1;
5654c8fbca5SSam Ravnborg 
5664c8fbca5SSam Ravnborg         if (sublen > slen)
5674c8fbca5SSam Ravnborg                 return 1;
5684c8fbca5SSam Ravnborg 
5694c8fbca5SSam Ravnborg         return memcmp(s + slen - sublen, sub, sublen);
5704c8fbca5SSam Ravnborg }
5714c8fbca5SSam Ravnborg 
5724c8fbca5SSam Ravnborg /**
5734c8fbca5SSam Ravnborg  * Whitelist to allow certain references to pass with no warning.
5744c8fbca5SSam Ravnborg  * Pattern 1:
5754c8fbca5SSam Ravnborg  *   If a module parameter is declared __initdata and permissions=0
5764c8fbca5SSam Ravnborg  *   then this is legal despite the warning generated.
5774c8fbca5SSam Ravnborg  *   We cannot see value of permissions here, so just ignore
5784c8fbca5SSam Ravnborg  *   this pattern.
5794c8fbca5SSam Ravnborg  *   The pattern is identified by:
5804c8fbca5SSam Ravnborg  *   tosec   = .init.data
5819209aed0SSam Ravnborg  *   fromsec = .data*
5824c8fbca5SSam Ravnborg  *   atsym   =__param*
5834c8fbca5SSam Ravnborg  *
5844c8fbca5SSam Ravnborg  * Pattern 2:
58572ee59b5SRandy Dunlap  *   Many drivers utilise a *driver container with references to
5864c8fbca5SSam Ravnborg  *   add, remove, probe functions etc.
5874c8fbca5SSam Ravnborg  *   These functions may often be marked __init and we do not want to
5884c8fbca5SSam Ravnborg  *   warn here.
5894c8fbca5SSam Ravnborg  *   the pattern is identified by:
5905ecdd0f6SSam Ravnborg  *   tosec   = .init.text | .exit.text | .init.data
5914c8fbca5SSam Ravnborg  *   fromsec = .data
592aae5f662SSam Ravnborg  *   atsym = *driver, *_template, *_sht, *_ops, *_probe, *probe_one, *_console
593ee6a8545SVivek Goyal  *
594ee6a8545SVivek Goyal  * Pattern 3:
5959bf8cb9bSSam Ravnborg  *   Whitelist all references from .pci_fixup* section to .init.text
5969bf8cb9bSSam Ravnborg  *   This is part of the PCI init when built-in
5979bf8cb9bSSam Ravnborg  *
5989bf8cb9bSSam Ravnborg  * Pattern 4:
5999bf8cb9bSSam Ravnborg  *   Whitelist all refereces from .text.head to .init.data
6009bf8cb9bSSam Ravnborg  *   Whitelist all refereces from .text.head to .init.text
6019bf8cb9bSSam Ravnborg  *
6029bf8cb9bSSam Ravnborg  * Pattern 5:
603ee6a8545SVivek Goyal  *   Some symbols belong to init section but still it is ok to reference
604ee6a8545SVivek Goyal  *   these from non-init sections as these symbols don't have any memory
605ee6a8545SVivek Goyal  *   allocated for them and symbol address and value are same. So even
606ee6a8545SVivek Goyal  *   if init section is freed, its ok to reference those symbols.
607ee6a8545SVivek Goyal  *   For ex. symbols marking the init section boundaries.
608ee6a8545SVivek Goyal  *   This pattern is identified by
609ee6a8545SVivek Goyal  *   refsymname = __init_begin, _sinittext, _einittext
6109bf8cb9bSSam Ravnborg  *
6119bf8cb9bSSam Ravnborg  * Pattern 6:
612aae5f662SSam Ravnborg  *   During the early init phase we have references from .init.text to
613aae5f662SSam Ravnborg  *   .text we have an intended section mismatch - do not warn about it.
614aae5f662SSam Ravnborg  *   See kernel_init() in init/main.c
615aae5f662SSam Ravnborg  *   tosec   = .init.text
616aae5f662SSam Ravnborg  *   fromsec = .text
617aae5f662SSam Ravnborg  *   atsym = kernel_init
618aae5f662SSam Ravnborg  *   Some symbols belong to init section but still it is ok to reference
6194c8fbca5SSam Ravnborg  **/
6209e157a5aSMagnus Damm static int secref_whitelist(const char *modname, const char *tosec,
621ee6a8545SVivek Goyal 			    const char *fromsec, const char *atsym,
622ee6a8545SVivek Goyal 			    const char *refsymname)
6234c8fbca5SSam Ravnborg {
6244c8fbca5SSam Ravnborg 	int f1 = 1, f2 = 1;
6254c8fbca5SSam Ravnborg 	const char **s;
6264c8fbca5SSam Ravnborg 	const char *pat2sym[] = {
62772ee59b5SRandy Dunlap 		"driver",
6285ecdd0f6SSam Ravnborg 		"_template", /* scsi uses *_template a lot */
6295ecdd0f6SSam Ravnborg 		"_sht",      /* scsi also used *_sht to some extent */
6304c8fbca5SSam Ravnborg 		"_ops",
6314c8fbca5SSam Ravnborg 		"_probe",
6324c8fbca5SSam Ravnborg 		"_probe_one",
633118c0aceSVivek Goyal 		"_console",
6344c8fbca5SSam Ravnborg 		NULL
6354c8fbca5SSam Ravnborg 	};
6364c8fbca5SSam Ravnborg 
637ee6a8545SVivek Goyal 	const char *pat3refsym[] = {
638ee6a8545SVivek Goyal 		"__init_begin",
639ee6a8545SVivek Goyal 		"_sinittext",
640ee6a8545SVivek Goyal 		"_einittext",
641ee6a8545SVivek Goyal 		NULL
642ee6a8545SVivek Goyal 	};
643ee6a8545SVivek Goyal 
6444c8fbca5SSam Ravnborg 	/* Check for pattern 1 */
6454c8fbca5SSam Ravnborg 	if (strcmp(tosec, ".init.data") != 0)
6464c8fbca5SSam Ravnborg 		f1 = 0;
6479209aed0SSam Ravnborg 	if (strncmp(fromsec, ".data", strlen(".data")) != 0)
6484c8fbca5SSam Ravnborg 		f1 = 0;
6494c8fbca5SSam Ravnborg 	if (strncmp(atsym, "__param", strlen("__param")) != 0)
6504c8fbca5SSam Ravnborg 		f1 = 0;
6514c8fbca5SSam Ravnborg 
6524c8fbca5SSam Ravnborg 	if (f1)
6534c8fbca5SSam Ravnborg 		return f1;
6544c8fbca5SSam Ravnborg 
6554c8fbca5SSam Ravnborg 	/* Check for pattern 2 */
6564c8fbca5SSam Ravnborg 	if ((strcmp(tosec, ".init.text") != 0) &&
6575ecdd0f6SSam Ravnborg 	    (strcmp(tosec, ".exit.text") != 0) &&
6585ecdd0f6SSam Ravnborg 	    (strcmp(tosec, ".init.data") != 0))
6594c8fbca5SSam Ravnborg 		f2 = 0;
6604c8fbca5SSam Ravnborg 	if (strcmp(fromsec, ".data") != 0)
6614c8fbca5SSam Ravnborg 		f2 = 0;
6624c8fbca5SSam Ravnborg 
6634c8fbca5SSam Ravnborg 	for (s = pat2sym; *s; s++)
6644c8fbca5SSam Ravnborg 		if (strrcmp(atsym, *s) == 0)
6654c8fbca5SSam Ravnborg 			f1 = 1;
6669e157a5aSMagnus Damm 	if (f1 && f2)
6679e157a5aSMagnus Damm 		return 1;
6684c8fbca5SSam Ravnborg 
6699bf8cb9bSSam Ravnborg 	/* Check for pattern 3 */
6709bf8cb9bSSam Ravnborg 	if ((strncmp(fromsec, ".pci_fixup", strlen(".pci_fixup")) == 0) &&
6719e157a5aSMagnus Damm 	    (strcmp(tosec, ".init.text") == 0))
6729e157a5aSMagnus Damm 	return 1;
673ee6a8545SVivek Goyal 
6749bf8cb9bSSam Ravnborg 	/* Check for pattern 4 */
675f8657e1bSVivek Goyal 	if ((strcmp(fromsec, ".text.head") == 0) &&
676f8657e1bSVivek Goyal 		((strcmp(tosec, ".init.data") == 0) ||
677f8657e1bSVivek Goyal 		(strcmp(tosec, ".init.text") == 0)))
678f8657e1bSVivek Goyal 	return 1;
679f8657e1bSVivek Goyal 
6809bf8cb9bSSam Ravnborg 	/* Check for pattern 5 */
681ee6a8545SVivek Goyal 	for (s = pat3refsym; *s; s++)
682ee6a8545SVivek Goyal 		if (strcmp(refsymname, *s) == 0)
683ee6a8545SVivek Goyal 			return 1;
6849bf8cb9bSSam Ravnborg 
6859bf8cb9bSSam Ravnborg 	/* Check for pattern 6 */
686aae5f662SSam Ravnborg 	if ((strcmp(tosec, ".init.text") == 0) &&
687aae5f662SSam Ravnborg 	    (strcmp(fromsec, ".text") == 0) &&
688aae5f662SSam Ravnborg 	    (strcmp(refsymname, "kernel_init") == 0))
689aae5f662SSam Ravnborg 		return 1;
69093659af1SSam Ravnborg 	return 0;
6914c8fbca5SSam Ravnborg }
6924c8fbca5SSam Ravnborg 
6934c8fbca5SSam Ravnborg /**
69493684d3bSSam Ravnborg  * Find symbol based on relocation record info.
69593684d3bSSam Ravnborg  * In some cases the symbol supplied is a valid symbol so
69693684d3bSSam Ravnborg  * return refsym. If st_name != 0 we assume this is a valid symbol.
69793684d3bSSam Ravnborg  * In other cases the symbol needs to be looked up in the symbol table
69893684d3bSSam Ravnborg  * based on section and address.
69993684d3bSSam Ravnborg  *  **/
70093684d3bSSam Ravnborg static Elf_Sym *find_elf_symbol(struct elf_info *elf, Elf_Addr addr,
70193684d3bSSam Ravnborg 				Elf_Sym *relsym)
70293684d3bSSam Ravnborg {
70393684d3bSSam Ravnborg 	Elf_Sym *sym;
70493684d3bSSam Ravnborg 
70593684d3bSSam Ravnborg 	if (relsym->st_name != 0)
70693684d3bSSam Ravnborg 		return relsym;
70793684d3bSSam Ravnborg 	for (sym = elf->symtab_start; sym < elf->symtab_stop; sym++) {
70893684d3bSSam Ravnborg 		if (sym->st_shndx != relsym->st_shndx)
70993684d3bSSam Ravnborg 			continue;
71093684d3bSSam Ravnborg 		if (sym->st_value == addr)
71193684d3bSSam Ravnborg 			return sym;
71293684d3bSSam Ravnborg 	}
71393684d3bSSam Ravnborg 	return NULL;
71493684d3bSSam Ravnborg }
71593684d3bSSam Ravnborg 
716da68d61fSDavid Brownell static inline int is_arm_mapping_symbol(const char *str)
717da68d61fSDavid Brownell {
718da68d61fSDavid Brownell 	return str[0] == '$' && strchr("atd", str[1])
719da68d61fSDavid Brownell 	       && (str[2] == '\0' || str[2] == '.');
720da68d61fSDavid Brownell }
721da68d61fSDavid Brownell 
722da68d61fSDavid Brownell /*
723da68d61fSDavid Brownell  * If there's no name there, ignore it; likewise, ignore it if it's
724da68d61fSDavid Brownell  * one of the magic symbols emitted used by current ARM tools.
725da68d61fSDavid Brownell  *
726da68d61fSDavid Brownell  * Otherwise if find_symbols_between() returns those symbols, they'll
727da68d61fSDavid Brownell  * fail the whitelist tests and cause lots of false alarms ... fixable
728da68d61fSDavid Brownell  * only by merging __exit and __init sections into __text, bloating
729da68d61fSDavid Brownell  * the kernel (which is especially evil on embedded platforms).
730da68d61fSDavid Brownell  */
731da68d61fSDavid Brownell static inline int is_valid_name(struct elf_info *elf, Elf_Sym *sym)
732da68d61fSDavid Brownell {
733da68d61fSDavid Brownell 	const char *name = elf->strtab + sym->st_name;
734da68d61fSDavid Brownell 
735da68d61fSDavid Brownell 	if (!name || !strlen(name))
736da68d61fSDavid Brownell 		return 0;
737da68d61fSDavid Brownell 	return !is_arm_mapping_symbol(name);
738da68d61fSDavid Brownell }
739da68d61fSDavid Brownell 
740b39927cfSSam Ravnborg /*
74143c74d17SSam Ravnborg  * Find symbols before or equal addr and after addr - in the section sec.
74243c74d17SSam Ravnborg  * If we find two symbols with equal offset prefer one with a valid name.
74343c74d17SSam Ravnborg  * The ELF format may have a better way to detect what type of symbol
74443c74d17SSam Ravnborg  * it is, but this works for now.
745b39927cfSSam Ravnborg  **/
746b39927cfSSam Ravnborg static void find_symbols_between(struct elf_info *elf, Elf_Addr addr,
747b39927cfSSam Ravnborg 				 const char *sec,
748b39927cfSSam Ravnborg 			         Elf_Sym **before, Elf_Sym **after)
749b39927cfSSam Ravnborg {
750b39927cfSSam Ravnborg 	Elf_Sym *sym;
751b39927cfSSam Ravnborg 	Elf_Ehdr *hdr = elf->hdr;
752b39927cfSSam Ravnborg 	Elf_Addr beforediff = ~0;
753b39927cfSSam Ravnborg 	Elf_Addr afterdiff = ~0;
754b39927cfSSam Ravnborg 	const char *secstrings = (void *)hdr +
755b39927cfSSam Ravnborg 				 elf->sechdrs[hdr->e_shstrndx].sh_offset;
756b39927cfSSam Ravnborg 
757b39927cfSSam Ravnborg 	*before = NULL;
758b39927cfSSam Ravnborg 	*after = NULL;
759b39927cfSSam Ravnborg 
760b39927cfSSam Ravnborg 	for (sym = elf->symtab_start; sym < elf->symtab_stop; sym++) {
761b39927cfSSam Ravnborg 		const char *symsec;
762b39927cfSSam Ravnborg 
763b39927cfSSam Ravnborg 		if (sym->st_shndx >= SHN_LORESERVE)
764b39927cfSSam Ravnborg 			continue;
765b39927cfSSam Ravnborg 		symsec = secstrings + elf->sechdrs[sym->st_shndx].sh_name;
766b39927cfSSam Ravnborg 		if (strcmp(symsec, sec) != 0)
767b39927cfSSam Ravnborg 			continue;
768da68d61fSDavid Brownell 		if (!is_valid_name(elf, sym))
769da68d61fSDavid Brownell 			continue;
770b39927cfSSam Ravnborg 		if (sym->st_value <= addr) {
771b39927cfSSam Ravnborg 			if ((addr - sym->st_value) < beforediff) {
772b39927cfSSam Ravnborg 				beforediff = addr - sym->st_value;
773b39927cfSSam Ravnborg 				*before = sym;
774b39927cfSSam Ravnborg 			}
77543c74d17SSam Ravnborg 			else if ((addr - sym->st_value) == beforediff) {
77643c74d17SSam Ravnborg 				*before = sym;
77743c74d17SSam Ravnborg 			}
778b39927cfSSam Ravnborg 		}
779b39927cfSSam Ravnborg 		else
780b39927cfSSam Ravnborg 		{
781b39927cfSSam Ravnborg 			if ((sym->st_value - addr) < afterdiff) {
782b39927cfSSam Ravnborg 				afterdiff = sym->st_value - addr;
783b39927cfSSam Ravnborg 				*after = sym;
784b39927cfSSam Ravnborg 			}
78543c74d17SSam Ravnborg 			else if ((sym->st_value - addr) == afterdiff) {
78643c74d17SSam Ravnborg 				*after = sym;
78743c74d17SSam Ravnborg 			}
788b39927cfSSam Ravnborg 		}
789b39927cfSSam Ravnborg 	}
790b39927cfSSam Ravnborg }
791b39927cfSSam Ravnborg 
792b39927cfSSam Ravnborg /**
793b39927cfSSam Ravnborg  * Print a warning about a section mismatch.
794b39927cfSSam Ravnborg  * Try to find symbols near it so user can find it.
7954c8fbca5SSam Ravnborg  * Check whitelist before warning - it may be a false positive.
796b39927cfSSam Ravnborg  **/
797b39927cfSSam Ravnborg static void warn_sec_mismatch(const char *modname, const char *fromsec,
798b39927cfSSam Ravnborg 			      struct elf_info *elf, Elf_Sym *sym, Elf_Rela r)
799b39927cfSSam Ravnborg {
80093684d3bSSam Ravnborg 	const char *refsymname = "";
80193684d3bSSam Ravnborg 	Elf_Sym *before, *after;
80293684d3bSSam Ravnborg 	Elf_Sym *refsym;
803b39927cfSSam Ravnborg 	Elf_Ehdr *hdr = elf->hdr;
804b39927cfSSam Ravnborg 	Elf_Shdr *sechdrs = elf->sechdrs;
805b39927cfSSam Ravnborg 	const char *secstrings = (void *)hdr +
806b39927cfSSam Ravnborg 				 sechdrs[hdr->e_shstrndx].sh_offset;
807b39927cfSSam Ravnborg 	const char *secname = secstrings + sechdrs[sym->st_shndx].sh_name;
808b39927cfSSam Ravnborg 
809b39927cfSSam Ravnborg 	find_symbols_between(elf, r.r_offset, fromsec, &before, &after);
810b39927cfSSam Ravnborg 
81193684d3bSSam Ravnborg 	refsym = find_elf_symbol(elf, r.r_addend, sym);
81293684d3bSSam Ravnborg 	if (refsym && strlen(elf->strtab + refsym->st_name))
81393684d3bSSam Ravnborg 		refsymname = elf->strtab + refsym->st_name;
81493684d3bSSam Ravnborg 
8154c8fbca5SSam Ravnborg 	/* check whitelist - we may ignore it */
8164c8fbca5SSam Ravnborg 	if (before &&
8179e157a5aSMagnus Damm 	    secref_whitelist(modname, secname, fromsec,
818ee6a8545SVivek Goyal 			     elf->strtab + before->st_name, refsymname))
8194c8fbca5SSam Ravnborg 		return;
8204c8fbca5SSam Ravnborg 
821b39927cfSSam Ravnborg 	if (before && after) {
82293684d3bSSam Ravnborg 		warn("%s - Section mismatch: reference to %s:%s from %s "
82393684d3bSSam Ravnborg 		     "between '%s' (at offset 0x%llx) and '%s'\n",
82493684d3bSSam Ravnborg 		     modname, secname, refsymname, fromsec,
825b39927cfSSam Ravnborg 		     elf->strtab + before->st_name,
82693684d3bSSam Ravnborg 		     (long long)r.r_offset,
827b39927cfSSam Ravnborg 		     elf->strtab + after->st_name);
828b39927cfSSam Ravnborg 	} else if (before) {
82993684d3bSSam Ravnborg 		warn("%s - Section mismatch: reference to %s:%s from %s "
83093684d3bSSam Ravnborg 		     "after '%s' (at offset 0x%llx)\n",
83193684d3bSSam Ravnborg 		     modname, secname, refsymname, fromsec,
832b39927cfSSam Ravnborg 		     elf->strtab + before->st_name,
83393684d3bSSam Ravnborg 		     (long long)r.r_offset);
834b39927cfSSam Ravnborg 	} else if (after) {
83593684d3bSSam Ravnborg 		warn("%s - Section mismatch: reference to %s:%s from %s "
83693684d3bSSam Ravnborg 		     "before '%s' (at offset -0x%llx)\n",
83793684d3bSSam Ravnborg 		     modname, secname, refsymname, fromsec,
838eaaae38cSEric Sesterhenn 		     elf->strtab + after->st_name,
83993684d3bSSam Ravnborg 		     (long long)r.r_offset);
840b39927cfSSam Ravnborg 	} else {
84193684d3bSSam Ravnborg 		warn("%s - Section mismatch: reference to %s:%s from %s "
84293684d3bSSam Ravnborg 		     "(offset 0x%llx)\n",
84393684d3bSSam Ravnborg 		     modname, secname, fromsec, refsymname,
84493684d3bSSam Ravnborg 		     (long long)r.r_offset);
845b39927cfSSam Ravnborg 	}
846b39927cfSSam Ravnborg }
847b39927cfSSam Ravnborg 
848b39927cfSSam Ravnborg /**
849b39927cfSSam Ravnborg  * A module includes a number of sections that are discarded
850b39927cfSSam Ravnborg  * either when loaded or when used as built-in.
851b39927cfSSam Ravnborg  * For loaded modules all functions marked __init and all data
852b39927cfSSam Ravnborg  * marked __initdata will be discarded when the module has been intialized.
853b39927cfSSam Ravnborg  * Likewise for modules used built-in the sections marked __exit
854b39927cfSSam Ravnborg  * are discarded because __exit marked function are supposed to be called
855b39927cfSSam Ravnborg  * only when a moduel is unloaded which never happes for built-in modules.
856b39927cfSSam Ravnborg  * The check_sec_ref() function traverses all relocation records
857b39927cfSSam Ravnborg  * to find all references to a section that reference a section that will
858b39927cfSSam Ravnborg  * be discarded and warns about it.
859b39927cfSSam Ravnborg  **/
860b39927cfSSam Ravnborg static void check_sec_ref(struct module *mod, const char *modname,
861b39927cfSSam Ravnborg 			  struct elf_info *elf,
862b39927cfSSam Ravnborg 			  int section(const char*),
863b39927cfSSam Ravnborg 			  int section_ref_ok(const char *))
864b39927cfSSam Ravnborg {
865b39927cfSSam Ravnborg 	int i;
866b39927cfSSam Ravnborg 	Elf_Sym  *sym;
867b39927cfSSam Ravnborg 	Elf_Ehdr *hdr = elf->hdr;
868b39927cfSSam Ravnborg 	Elf_Shdr *sechdrs = elf->sechdrs;
869b39927cfSSam Ravnborg 	const char *secstrings = (void *)hdr +
870b39927cfSSam Ravnborg 				 sechdrs[hdr->e_shstrndx].sh_offset;
871b39927cfSSam Ravnborg 
872b39927cfSSam Ravnborg 	/* Walk through all sections */
873b39927cfSSam Ravnborg 	for (i = 0; i < hdr->e_shnum; i++) {
8742c1a51f3SAtsushi Nemoto 		const char *name = secstrings + sechdrs[i].sh_name;
8752c1a51f3SAtsushi Nemoto 		const char *secname;
8762c1a51f3SAtsushi Nemoto 		Elf_Rela r;
877eae07ac6SAtsushi Nemoto 		unsigned int r_sym;
8782c1a51f3SAtsushi Nemoto 		/* We want to process only relocation sections and not .init */
8792c1a51f3SAtsushi Nemoto 		if (sechdrs[i].sh_type == SHT_RELA) {
880fededcd2S[email protected] 			Elf_Rela *rela;
881fededcd2S[email protected] 			Elf_Rela *start = (void *)hdr + sechdrs[i].sh_offset;
882fededcd2S[email protected] 			Elf_Rela *stop  = (void*)start + sechdrs[i].sh_size;
8832c1a51f3SAtsushi Nemoto 			name += strlen(".rela");
8842c1a51f3SAtsushi Nemoto 			if (section_ref_ok(name))
885b39927cfSSam Ravnborg 				continue;
886b39927cfSSam Ravnborg 
887b39927cfSSam Ravnborg 			for (rela = start; rela < stop; rela++) {
888b39927cfSSam Ravnborg 				r.r_offset = TO_NATIVE(rela->r_offset);
889eae07ac6SAtsushi Nemoto #if KERNEL_ELFCLASS == ELFCLASS64
890eae07ac6SAtsushi Nemoto 				if (hdr->e_machine == EM_MIPS) {
891eae07ac6SAtsushi Nemoto 					r_sym = ELF64_MIPS_R_SYM(rela->r_info);
892eae07ac6SAtsushi Nemoto 					r_sym = TO_NATIVE(r_sym);
893eae07ac6SAtsushi Nemoto 				} else {
894601e7f02SLinus Torvalds 					r.r_info = TO_NATIVE(rela->r_info);
895eae07ac6SAtsushi Nemoto 					r_sym = ELF_R_SYM(r.r_info);
896eae07ac6SAtsushi Nemoto 				}
897eae07ac6SAtsushi Nemoto #else
898eae07ac6SAtsushi Nemoto 				r.r_info = TO_NATIVE(rela->r_info);
899eae07ac6SAtsushi Nemoto 				r_sym = ELF_R_SYM(r.r_info);
900eae07ac6SAtsushi Nemoto #endif
90193684d3bSSam Ravnborg 				r.r_addend = TO_NATIVE(rela->r_addend);
902eae07ac6SAtsushi Nemoto 				sym = elf->symtab_start + r_sym;
903b39927cfSSam Ravnborg 				/* Skip special sections */
904b39927cfSSam Ravnborg 				if (sym->st_shndx >= SHN_LORESERVE)
905b39927cfSSam Ravnborg 					continue;
906b39927cfSSam Ravnborg 
9072c1a51f3SAtsushi Nemoto 				secname = secstrings +
9082c1a51f3SAtsushi Nemoto 					sechdrs[sym->st_shndx].sh_name;
909b39927cfSSam Ravnborg 				if (section(secname))
9102c1a51f3SAtsushi Nemoto 					warn_sec_mismatch(modname, name,
9112c1a51f3SAtsushi Nemoto 							  elf, sym, r);
9122c1a51f3SAtsushi Nemoto 			}
9132c1a51f3SAtsushi Nemoto 		} else if (sechdrs[i].sh_type == SHT_REL) {
9142c1a51f3SAtsushi Nemoto 			Elf_Rel *rel;
9152c1a51f3SAtsushi Nemoto 			Elf_Rel *start = (void *)hdr + sechdrs[i].sh_offset;
9162c1a51f3SAtsushi Nemoto 			Elf_Rel *stop  = (void*)start + sechdrs[i].sh_size;
9172c1a51f3SAtsushi Nemoto 			name += strlen(".rel");
9182c1a51f3SAtsushi Nemoto 			if (section_ref_ok(name))
9192c1a51f3SAtsushi Nemoto 				continue;
9202c1a51f3SAtsushi Nemoto 
9212c1a51f3SAtsushi Nemoto 			for (rel = start; rel < stop; rel++) {
9222c1a51f3SAtsushi Nemoto 				r.r_offset = TO_NATIVE(rel->r_offset);
923eae07ac6SAtsushi Nemoto #if KERNEL_ELFCLASS == ELFCLASS64
924eae07ac6SAtsushi Nemoto 				if (hdr->e_machine == EM_MIPS) {
925eae07ac6SAtsushi Nemoto 					r_sym = ELF64_MIPS_R_SYM(rel->r_info);
926eae07ac6SAtsushi Nemoto 					r_sym = TO_NATIVE(r_sym);
927eae07ac6SAtsushi Nemoto 				} else {
9282c1a51f3SAtsushi Nemoto 					r.r_info = TO_NATIVE(rel->r_info);
929eae07ac6SAtsushi Nemoto 					r_sym = ELF_R_SYM(r.r_info);
930eae07ac6SAtsushi Nemoto 				}
931eae07ac6SAtsushi Nemoto #else
932eae07ac6SAtsushi Nemoto 				r.r_info = TO_NATIVE(rel->r_info);
933eae07ac6SAtsushi Nemoto 				r_sym = ELF_R_SYM(r.r_info);
934eae07ac6SAtsushi Nemoto #endif
9352c1a51f3SAtsushi Nemoto 				r.r_addend = 0;
936eae07ac6SAtsushi Nemoto 				sym = elf->symtab_start + r_sym;
9372c1a51f3SAtsushi Nemoto 				/* Skip special sections */
9382c1a51f3SAtsushi Nemoto 				if (sym->st_shndx >= SHN_LORESERVE)
9392c1a51f3SAtsushi Nemoto 					continue;
9402c1a51f3SAtsushi Nemoto 
9412c1a51f3SAtsushi Nemoto 				secname = secstrings +
9422c1a51f3SAtsushi Nemoto 					sechdrs[sym->st_shndx].sh_name;
9432c1a51f3SAtsushi Nemoto 				if (section(secname))
9442c1a51f3SAtsushi Nemoto 					warn_sec_mismatch(modname, name,
9452c1a51f3SAtsushi Nemoto 							  elf, sym, r);
9462c1a51f3SAtsushi Nemoto 			}
947b39927cfSSam Ravnborg 		}
948b39927cfSSam Ravnborg 	}
949b39927cfSSam Ravnborg }
950b39927cfSSam Ravnborg 
951b39927cfSSam Ravnborg /**
952b39927cfSSam Ravnborg  * Functions used only during module init is marked __init and is stored in
953b39927cfSSam Ravnborg  * a .init.text section. Likewise data is marked __initdata and stored in
954b39927cfSSam Ravnborg  * a .init.data section.
955b39927cfSSam Ravnborg  * If this section is one of these sections return 1
956b39927cfSSam Ravnborg  * See include/linux/init.h for the details
957b39927cfSSam Ravnborg  **/
958b39927cfSSam Ravnborg static int init_section(const char *name)
959b39927cfSSam Ravnborg {
960b39927cfSSam Ravnborg 	if (strcmp(name, ".init") == 0)
961b39927cfSSam Ravnborg 		return 1;
962b39927cfSSam Ravnborg 	if (strncmp(name, ".init.", strlen(".init.")) == 0)
963b39927cfSSam Ravnborg 		return 1;
964b39927cfSSam Ravnborg 	return 0;
965b39927cfSSam Ravnborg }
966b39927cfSSam Ravnborg 
967b39927cfSSam Ravnborg /**
968b39927cfSSam Ravnborg  * Identify sections from which references to a .init section is OK.
969b39927cfSSam Ravnborg  *
970b39927cfSSam Ravnborg  * Unfortunately references to read only data that referenced .init
971b39927cfSSam Ravnborg  * sections had to be excluded. Almost all of these are false
972b39927cfSSam Ravnborg  * positives, they are created by gcc. The downside of excluding rodata
973b39927cfSSam Ravnborg  * is that there really are some user references from rodata to
974b39927cfSSam Ravnborg  * init code, e.g. drivers/video/vgacon.c:
975b39927cfSSam Ravnborg  *
976b39927cfSSam Ravnborg  * const struct consw vga_con = {
977b39927cfSSam Ravnborg  *        con_startup:            vgacon_startup,
978b39927cfSSam Ravnborg  *
979b39927cfSSam Ravnborg  * where vgacon_startup is __init.  If you want to wade through the false
980b39927cfSSam Ravnborg  * positives, take out the check for rodata.
981b39927cfSSam Ravnborg  **/
982b39927cfSSam Ravnborg static int init_section_ref_ok(const char *name)
983b39927cfSSam Ravnborg {
984b39927cfSSam Ravnborg 	const char **s;
985b39927cfSSam Ravnborg 	/* Absolute section names */
986b39927cfSSam Ravnborg 	const char *namelist1[] = {
987b39927cfSSam Ravnborg 		".init",
9889209aed0SSam Ravnborg 		".opd",   /* see comment [OPD] at exit_section_ref_ok() */
9899209aed0SSam Ravnborg 		".toc1",  /* used by ppc64 */
990b39927cfSSam Ravnborg 		".stab",
991742433b0SMatthew Wilcox 		".data.rel.ro", /* used by parisc64 */
992139ec7c4SRusty Russell 		".parainstructions",
993b39927cfSSam Ravnborg 		".text.lock",
9949209aed0SSam Ravnborg 		"__bug_table", /* used by powerpc for BUG() */
995b39927cfSSam Ravnborg 		".pci_fixup_header",
996b39927cfSSam Ravnborg 		".pci_fixup_final",
997b39927cfSSam Ravnborg 		".pdr",
998b39927cfSSam Ravnborg 		"__param",
999468d9494SAl Viro 		"__ex_table",
1000468d9494SAl Viro 		".fixup",
100135899c57SRandy Dunlap 		".smp_locks",
1002909252d2SSam Ravnborg 		".plt",  /* seen on ARCH=um build on x86_64. Harmless */
100321c4ff80SBenjamin Herrenschmidt 		"__ftr_fixup",		/* powerpc cpu feature fixup */
100421c4ff80SBenjamin Herrenschmidt 		"__fw_ftr_fixup",	/* powerpc firmware feature fixup */
1005b39927cfSSam Ravnborg 		NULL
1006b39927cfSSam Ravnborg 	};
1007b39927cfSSam Ravnborg 	/* Start of section names */
1008b39927cfSSam Ravnborg 	const char *namelist2[] = {
1009b39927cfSSam Ravnborg 		".init.",
1010b39927cfSSam Ravnborg 		".altinstructions",
1011b39927cfSSam Ravnborg 		".eh_frame",
1012b39927cfSSam Ravnborg 		".debug",
1013139ec7c4SRusty Russell 		".parainstructions",
1014742433b0SMatthew Wilcox 		".rodata",
1015b39927cfSSam Ravnborg 		NULL
1016b39927cfSSam Ravnborg 	};
10176e10133fSSam Ravnborg 	/* part of section name */
10186e10133fSSam Ravnborg 	const char *namelist3 [] = {
10196e10133fSSam Ravnborg 		".unwind",  /* sample: IA_64.unwind.init.text */
10206e10133fSSam Ravnborg 		NULL
10216e10133fSSam Ravnborg 	};
1022b39927cfSSam Ravnborg 
1023b39927cfSSam Ravnborg 	for (s = namelist1; *s; s++)
1024b39927cfSSam Ravnborg 		if (strcmp(*s, name) == 0)
1025b39927cfSSam Ravnborg 			return 1;
1026b39927cfSSam Ravnborg 	for (s = namelist2; *s; s++)
1027b39927cfSSam Ravnborg 		if (strncmp(*s, name, strlen(*s)) == 0)
1028b39927cfSSam Ravnborg 			return 1;
10296e10133fSSam Ravnborg 	for (s = namelist3; *s; s++)
1030e835a39cSSam Ravnborg 		if (strstr(name, *s) != NULL)
10316e10133fSSam Ravnborg 			return 1;
1032468d9494SAl Viro 	if (strrcmp(name, ".init") == 0)
1033468d9494SAl Viro 		return 1;
1034b39927cfSSam Ravnborg 	return 0;
1035b39927cfSSam Ravnborg }
1036b39927cfSSam Ravnborg 
1037b39927cfSSam Ravnborg /*
1038b39927cfSSam Ravnborg  * Functions used only during module exit is marked __exit and is stored in
1039b39927cfSSam Ravnborg  * a .exit.text section. Likewise data is marked __exitdata and stored in
1040b39927cfSSam Ravnborg  * a .exit.data section.
1041b39927cfSSam Ravnborg  * If this section is one of these sections return 1
1042b39927cfSSam Ravnborg  * See include/linux/init.h for the details
1043b39927cfSSam Ravnborg  **/
1044b39927cfSSam Ravnborg static int exit_section(const char *name)
1045b39927cfSSam Ravnborg {
1046b39927cfSSam Ravnborg 	if (strcmp(name, ".exit.text") == 0)
1047b39927cfSSam Ravnborg 		return 1;
1048b39927cfSSam Ravnborg 	if (strcmp(name, ".exit.data") == 0)
1049b39927cfSSam Ravnborg 		return 1;
1050b39927cfSSam Ravnborg 	return 0;
1051b39927cfSSam Ravnborg 
1052b39927cfSSam Ravnborg }
1053b39927cfSSam Ravnborg 
1054b39927cfSSam Ravnborg /*
1055b39927cfSSam Ravnborg  * Identify sections from which references to a .exit section is OK.
1056b39927cfSSam Ravnborg  *
1057b39927cfSSam Ravnborg  * [OPD] Keith Ownes <[email protected]> commented:
1058b39927cfSSam Ravnborg  * For our future {in}sanity, add a comment that this is the ppc .opd
1059b39927cfSSam Ravnborg  * section, not the ia64 .opd section.
1060b39927cfSSam Ravnborg  * ia64 .opd should not point to discarded sections.
10615ecdd0f6SSam Ravnborg  * [.rodata] like for .init.text we ignore .rodata references -same reason
1062b39927cfSSam Ravnborg  **/
1063b39927cfSSam Ravnborg static int exit_section_ref_ok(const char *name)
1064b39927cfSSam Ravnborg {
1065b39927cfSSam Ravnborg 	const char **s;
1066b39927cfSSam Ravnborg 	/* Absolute section names */
1067b39927cfSSam Ravnborg 	const char *namelist1[] = {
1068b39927cfSSam Ravnborg 		".exit.text",
1069b39927cfSSam Ravnborg 		".exit.data",
1070b39927cfSSam Ravnborg 		".init.text",
10715ecdd0f6SSam Ravnborg 		".rodata",
1072b39927cfSSam Ravnborg 		".opd", /* See comment [OPD] */
10739209aed0SSam Ravnborg 		".toc1",  /* used by ppc64 */
1074b39927cfSSam Ravnborg 		".altinstructions",
1075b39927cfSSam Ravnborg 		".pdr",
10769209aed0SSam Ravnborg 		"__bug_table", /* used by powerpc for BUG() */
1077b39927cfSSam Ravnborg 		".exitcall.exit",
1078b39927cfSSam Ravnborg 		".eh_frame",
1079acd19499SRandy Dunlap 		".parainstructions",
1080b39927cfSSam Ravnborg 		".stab",
1081468d9494SAl Viro 		"__ex_table",
1082468d9494SAl Viro 		".fixup",
108335899c57SRandy Dunlap 		".smp_locks",
1084909252d2SSam Ravnborg 		".plt",  /* seen on ARCH=um build on x86_64. Harmless */
1085b39927cfSSam Ravnborg 		NULL
1086b39927cfSSam Ravnborg 	};
1087b39927cfSSam Ravnborg 	/* Start of section names */
1088b39927cfSSam Ravnborg 	const char *namelist2[] = {
1089b39927cfSSam Ravnborg 		".debug",
1090b39927cfSSam Ravnborg 		NULL
1091b39927cfSSam Ravnborg 	};
10926e10133fSSam Ravnborg 	/* part of section name */
10936e10133fSSam Ravnborg 	const char *namelist3 [] = {
10946e10133fSSam Ravnborg 		".unwind",  /* Sample: IA_64.unwind.exit.text */
10956e10133fSSam Ravnborg 		NULL
10966e10133fSSam Ravnborg 	};
1097b39927cfSSam Ravnborg 
1098b39927cfSSam Ravnborg 	for (s = namelist1; *s; s++)
1099b39927cfSSam Ravnborg 		if (strcmp(*s, name) == 0)
1100b39927cfSSam Ravnborg 			return 1;
1101b39927cfSSam Ravnborg 	for (s = namelist2; *s; s++)
1102b39927cfSSam Ravnborg 		if (strncmp(*s, name, strlen(*s)) == 0)
1103b39927cfSSam Ravnborg 			return 1;
11046e10133fSSam Ravnborg 	for (s = namelist3; *s; s++)
1105e835a39cSSam Ravnborg 		if (strstr(name, *s) != NULL)
11066e10133fSSam Ravnborg 			return 1;
1107b39927cfSSam Ravnborg 	return 0;
1108b39927cfSSam Ravnborg }
1109b39927cfSSam Ravnborg 
11105c3ead8cSSam Ravnborg static void read_symbols(char *modname)
11111da177e4SLinus Torvalds {
11121da177e4SLinus Torvalds 	const char *symname;
11131da177e4SLinus Torvalds 	char *version;
1114b817f6feSSam Ravnborg 	char *license;
11151da177e4SLinus Torvalds 	struct module *mod;
11161da177e4SLinus Torvalds 	struct elf_info info = { };
11171da177e4SLinus Torvalds 	Elf_Sym *sym;
11181da177e4SLinus Torvalds 
111985bd2fddSSam Ravnborg 	if (!parse_elf(&info, modname))
112085bd2fddSSam Ravnborg 		return;
11211da177e4SLinus Torvalds 
11221da177e4SLinus Torvalds 	mod = new_module(modname);
11231da177e4SLinus Torvalds 
11241da177e4SLinus Torvalds 	/* When there's no vmlinux, don't print warnings about
11251da177e4SLinus Torvalds 	 * unresolved symbols (since there'll be too many ;) */
11261da177e4SLinus Torvalds 	if (is_vmlinux(modname)) {
11271da177e4SLinus Torvalds 		have_vmlinux = 1;
11281da177e4SLinus Torvalds 		mod->skip = 1;
11291da177e4SLinus Torvalds 	}
11301da177e4SLinus Torvalds 
1131b817f6feSSam Ravnborg 	license = get_modinfo(info.modinfo, info.modinfo_len, "license");
1132b817f6feSSam Ravnborg 	while (license) {
1133b817f6feSSam Ravnborg 		if (license_is_gpl_compatible(license))
1134b817f6feSSam Ravnborg 			mod->gpl_compatible = 1;
1135b817f6feSSam Ravnborg 		else {
1136b817f6feSSam Ravnborg 			mod->gpl_compatible = 0;
1137b817f6feSSam Ravnborg 			break;
1138b817f6feSSam Ravnborg 		}
1139b817f6feSSam Ravnborg 		license = get_next_modinfo(info.modinfo, info.modinfo_len,
1140b817f6feSSam Ravnborg 					   "license", license);
1141b817f6feSSam Ravnborg 	}
1142b817f6feSSam Ravnborg 
11431da177e4SLinus Torvalds 	for (sym = info.symtab_start; sym < info.symtab_stop; sym++) {
11441da177e4SLinus Torvalds 		symname = info.strtab + sym->st_name;
11451da177e4SLinus Torvalds 
11461da177e4SLinus Torvalds 		handle_modversions(mod, &info, sym, symname);
11471da177e4SLinus Torvalds 		handle_moddevtable(mod, &info, sym, symname);
11481da177e4SLinus Torvalds 	}
1149b39927cfSSam Ravnborg 	check_sec_ref(mod, modname, &info, init_section, init_section_ref_ok);
1150b39927cfSSam Ravnborg 	check_sec_ref(mod, modname, &info, exit_section, exit_section_ref_ok);
11511da177e4SLinus Torvalds 
11521da177e4SLinus Torvalds 	version = get_modinfo(info.modinfo, info.modinfo_len, "version");
11531da177e4SLinus Torvalds 	if (version)
11541da177e4SLinus Torvalds 		maybe_frob_rcs_version(modname, version, info.modinfo,
11551da177e4SLinus Torvalds 				       version - (char *)info.hdr);
11561da177e4SLinus Torvalds 	if (version || (all_versions && !is_vmlinux(modname)))
11571da177e4SLinus Torvalds 		get_src_version(modname, mod->srcversion,
11581da177e4SLinus Torvalds 				sizeof(mod->srcversion)-1);
11591da177e4SLinus Torvalds 
11601da177e4SLinus Torvalds 	parse_elf_finish(&info);
11611da177e4SLinus Torvalds 
11621da177e4SLinus Torvalds 	/* Our trick to get versioning for struct_module - it's
11631da177e4SLinus Torvalds 	 * never passed as an argument to an exported function, so
11641da177e4SLinus Torvalds 	 * the automatic versioning doesn't pick it up, but it's really
11651da177e4SLinus Torvalds 	 * important anyhow */
11661da177e4SLinus Torvalds 	if (modversions)
11671da177e4SLinus Torvalds 		mod->unres = alloc_symbol("struct_module", 0, mod->unres);
11681da177e4SLinus Torvalds }
11691da177e4SLinus Torvalds 
11701da177e4SLinus Torvalds #define SZ 500
11711da177e4SLinus Torvalds 
11721da177e4SLinus Torvalds /* We first write the generated file into memory using the
11731da177e4SLinus Torvalds  * following helper, then compare to the file on disk and
11741da177e4SLinus Torvalds  * only update the later if anything changed */
11751da177e4SLinus Torvalds 
11765c3ead8cSSam Ravnborg void __attribute__((format(printf, 2, 3))) buf_printf(struct buffer *buf,
11775c3ead8cSSam Ravnborg 						      const char *fmt, ...)
11781da177e4SLinus Torvalds {
11791da177e4SLinus Torvalds 	char tmp[SZ];
11801da177e4SLinus Torvalds 	int len;
11811da177e4SLinus Torvalds 	va_list ap;
11821da177e4SLinus Torvalds 
11831da177e4SLinus Torvalds 	va_start(ap, fmt);
11841da177e4SLinus Torvalds 	len = vsnprintf(tmp, SZ, fmt, ap);
11857670f023SSam Ravnborg 	buf_write(buf, tmp, len);
11861da177e4SLinus Torvalds 	va_end(ap);
11871da177e4SLinus Torvalds }
11881da177e4SLinus Torvalds 
11895c3ead8cSSam Ravnborg void buf_write(struct buffer *buf, const char *s, int len)
11901da177e4SLinus Torvalds {
11911da177e4SLinus Torvalds 	if (buf->size - buf->pos < len) {
11927670f023SSam Ravnborg 		buf->size += len + SZ;
11931da177e4SLinus Torvalds 		buf->p = realloc(buf->p, buf->size);
11941da177e4SLinus Torvalds 	}
11951da177e4SLinus Torvalds 	strncpy(buf->p + buf->pos, s, len);
11961da177e4SLinus Torvalds 	buf->pos += len;
11971da177e4SLinus Torvalds }
11981da177e4SLinus Torvalds 
1199c96fca21SSam Ravnborg static void check_for_gpl_usage(enum export exp, const char *m, const char *s)
1200c96fca21SSam Ravnborg {
1201c96fca21SSam Ravnborg 	const char *e = is_vmlinux(m) ?"":".ko";
1202c96fca21SSam Ravnborg 
1203c96fca21SSam Ravnborg 	switch (exp) {
1204c96fca21SSam Ravnborg 	case export_gpl:
1205c96fca21SSam Ravnborg 		fatal("modpost: GPL-incompatible module %s%s "
1206c96fca21SSam Ravnborg 		      "uses GPL-only symbol '%s'\n", m, e, s);
1207c96fca21SSam Ravnborg 		break;
1208c96fca21SSam Ravnborg 	case export_unused_gpl:
1209c96fca21SSam Ravnborg 		fatal("modpost: GPL-incompatible module %s%s "
1210c96fca21SSam Ravnborg 		      "uses GPL-only symbol marked UNUSED '%s'\n", m, e, s);
1211c96fca21SSam Ravnborg 		break;
1212c96fca21SSam Ravnborg 	case export_gpl_future:
1213c96fca21SSam Ravnborg 		warn("modpost: GPL-incompatible module %s%s "
1214c96fca21SSam Ravnborg 		      "uses future GPL-only symbol '%s'\n", m, e, s);
1215c96fca21SSam Ravnborg 		break;
1216c96fca21SSam Ravnborg 	case export_plain:
1217c96fca21SSam Ravnborg 	case export_unused:
1218c96fca21SSam Ravnborg 	case export_unknown:
1219c96fca21SSam Ravnborg 		/* ignore */
1220c96fca21SSam Ravnborg 		break;
1221c96fca21SSam Ravnborg 	}
1222c96fca21SSam Ravnborg }
1223c96fca21SSam Ravnborg 
1224c96fca21SSam Ravnborg static void check_for_unused(enum export exp, const char* m, const char* s)
1225c96fca21SSam Ravnborg {
1226c96fca21SSam Ravnborg 	const char *e = is_vmlinux(m) ?"":".ko";
1227c96fca21SSam Ravnborg 
1228c96fca21SSam Ravnborg 	switch (exp) {
1229c96fca21SSam Ravnborg 	case export_unused:
1230c96fca21SSam Ravnborg 	case export_unused_gpl:
1231c96fca21SSam Ravnborg 		warn("modpost: module %s%s "
1232c96fca21SSam Ravnborg 		      "uses symbol '%s' marked UNUSED\n", m, e, s);
1233c96fca21SSam Ravnborg 		break;
1234c96fca21SSam Ravnborg 	default:
1235c96fca21SSam Ravnborg 		/* ignore */
1236c96fca21SSam Ravnborg 		break;
1237c96fca21SSam Ravnborg 	}
1238c96fca21SSam Ravnborg }
1239c96fca21SSam Ravnborg 
1240c96fca21SSam Ravnborg static void check_exports(struct module *mod)
1241b817f6feSSam Ravnborg {
1242b817f6feSSam Ravnborg 	struct symbol *s, *exp;
1243b817f6feSSam Ravnborg 
1244b817f6feSSam Ravnborg 	for (s = mod->unres; s; s = s->next) {
12456449bd62SAndrew Morton 		const char *basename;
1246b817f6feSSam Ravnborg 		exp = find_symbol(s->name);
1247b817f6feSSam Ravnborg 		if (!exp || exp->module == mod)
1248b817f6feSSam Ravnborg 			continue;
12496449bd62SAndrew Morton 		basename = strrchr(mod->name, '/');
1250b817f6feSSam Ravnborg 		if (basename)
1251b817f6feSSam Ravnborg 			basename++;
1252c96fca21SSam Ravnborg 		else
1253c96fca21SSam Ravnborg 			basename = mod->name;
1254c96fca21SSam Ravnborg 		if (!mod->gpl_compatible)
1255c96fca21SSam Ravnborg 			check_for_gpl_usage(exp->export, basename, exp->name);
1256c96fca21SSam Ravnborg 		check_for_unused(exp->export, basename, exp->name);
1257b817f6feSSam Ravnborg         }
1258b817f6feSSam Ravnborg }
1259b817f6feSSam Ravnborg 
12605c3ead8cSSam Ravnborg /**
12615c3ead8cSSam Ravnborg  * Header for the generated file
12625c3ead8cSSam Ravnborg  **/
12635c3ead8cSSam Ravnborg static void add_header(struct buffer *b, struct module *mod)
12641da177e4SLinus Torvalds {
12651da177e4SLinus Torvalds 	buf_printf(b, "#include <linux/module.h>\n");
12661da177e4SLinus Torvalds 	buf_printf(b, "#include <linux/vermagic.h>\n");
12671da177e4SLinus Torvalds 	buf_printf(b, "#include <linux/compiler.h>\n");
12681da177e4SLinus Torvalds 	buf_printf(b, "\n");
12691da177e4SLinus Torvalds 	buf_printf(b, "MODULE_INFO(vermagic, VERMAGIC_STRING);\n");
12701da177e4SLinus Torvalds 	buf_printf(b, "\n");
12711da177e4SLinus Torvalds 	buf_printf(b, "struct module __this_module\n");
12721da177e4SLinus Torvalds 	buf_printf(b, "__attribute__((section(\".gnu.linkonce.this_module\"))) = {\n");
1273f83b5e32SUstyugov Roman 	buf_printf(b, " .name = KBUILD_MODNAME,\n");
12741da177e4SLinus Torvalds 	if (mod->has_init)
12751da177e4SLinus Torvalds 		buf_printf(b, " .init = init_module,\n");
12761da177e4SLinus Torvalds 	if (mod->has_cleanup)
12771da177e4SLinus Torvalds 		buf_printf(b, "#ifdef CONFIG_MODULE_UNLOAD\n"
12781da177e4SLinus Torvalds 			      " .exit = cleanup_module,\n"
12791da177e4SLinus Torvalds 			      "#endif\n");
12801da177e4SLinus Torvalds 	buf_printf(b, "};\n");
12811da177e4SLinus Torvalds }
12821da177e4SLinus Torvalds 
12835c3ead8cSSam Ravnborg /**
12845c3ead8cSSam Ravnborg  * Record CRCs for unresolved symbols
12855c3ead8cSSam Ravnborg  **/
1286c53ddacdSKirill Korotaev static int add_versions(struct buffer *b, struct module *mod)
12871da177e4SLinus Torvalds {
12881da177e4SLinus Torvalds 	struct symbol *s, *exp;
1289c53ddacdSKirill Korotaev 	int err = 0;
12901da177e4SLinus Torvalds 
12911da177e4SLinus Torvalds 	for (s = mod->unres; s; s = s->next) {
12921da177e4SLinus Torvalds 		exp = find_symbol(s->name);
12931da177e4SLinus Torvalds 		if (!exp || exp->module == mod) {
1294c53ddacdSKirill Korotaev 			if (have_vmlinux && !s->weak) {
1295cb80514dSSam Ravnborg 				warn("\"%s\" [%s.ko] undefined!\n",
1296cb80514dSSam Ravnborg 				     s->name, mod->name);
1297c53ddacdSKirill Korotaev 				err = warn_unresolved ? 0 : 1;
1298c53ddacdSKirill Korotaev 			}
12991da177e4SLinus Torvalds 			continue;
13001da177e4SLinus Torvalds 		}
13011da177e4SLinus Torvalds 		s->module = exp->module;
13021da177e4SLinus Torvalds 		s->crc_valid = exp->crc_valid;
13031da177e4SLinus Torvalds 		s->crc = exp->crc;
13041da177e4SLinus Torvalds 	}
13051da177e4SLinus Torvalds 
13061da177e4SLinus Torvalds 	if (!modversions)
1307c53ddacdSKirill Korotaev 		return err;
13081da177e4SLinus Torvalds 
13091da177e4SLinus Torvalds 	buf_printf(b, "\n");
13101da177e4SLinus Torvalds 	buf_printf(b, "static const struct modversion_info ____versions[]\n");
13111da177e4SLinus Torvalds 	buf_printf(b, "__attribute_used__\n");
13121da177e4SLinus Torvalds 	buf_printf(b, "__attribute__((section(\"__versions\"))) = {\n");
13131da177e4SLinus Torvalds 
13141da177e4SLinus Torvalds 	for (s = mod->unres; s; s = s->next) {
13151da177e4SLinus Torvalds 		if (!s->module) {
13161da177e4SLinus Torvalds 			continue;
13171da177e4SLinus Torvalds 		}
13181da177e4SLinus Torvalds 		if (!s->crc_valid) {
1319cb80514dSSam Ravnborg 			warn("\"%s\" [%s.ko] has no CRC!\n",
13201da177e4SLinus Torvalds 				s->name, mod->name);
13211da177e4SLinus Torvalds 			continue;
13221da177e4SLinus Torvalds 		}
13231da177e4SLinus Torvalds 		buf_printf(b, "\t{ %#8x, \"%s\" },\n", s->crc, s->name);
13241da177e4SLinus Torvalds 	}
13251da177e4SLinus Torvalds 
13261da177e4SLinus Torvalds 	buf_printf(b, "};\n");
1327c53ddacdSKirill Korotaev 
1328c53ddacdSKirill Korotaev 	return err;
13291da177e4SLinus Torvalds }
13301da177e4SLinus Torvalds 
13315c3ead8cSSam Ravnborg static void add_depends(struct buffer *b, struct module *mod,
13325c3ead8cSSam Ravnborg 			struct module *modules)
13331da177e4SLinus Torvalds {
13341da177e4SLinus Torvalds 	struct symbol *s;
13351da177e4SLinus Torvalds 	struct module *m;
13361da177e4SLinus Torvalds 	int first = 1;
13371da177e4SLinus Torvalds 
13381da177e4SLinus Torvalds 	for (m = modules; m; m = m->next) {
13391da177e4SLinus Torvalds 		m->seen = is_vmlinux(m->name);
13401da177e4SLinus Torvalds 	}
13411da177e4SLinus Torvalds 
13421da177e4SLinus Torvalds 	buf_printf(b, "\n");
13431da177e4SLinus Torvalds 	buf_printf(b, "static const char __module_depends[]\n");
13441da177e4SLinus Torvalds 	buf_printf(b, "__attribute_used__\n");
13451da177e4SLinus Torvalds 	buf_printf(b, "__attribute__((section(\".modinfo\"))) =\n");
13461da177e4SLinus Torvalds 	buf_printf(b, "\"depends=");
13471da177e4SLinus Torvalds 	for (s = mod->unres; s; s = s->next) {
1348*a61b2dfdSSam Ravnborg 		const char *p;
13491da177e4SLinus Torvalds 		if (!s->module)
13501da177e4SLinus Torvalds 			continue;
13511da177e4SLinus Torvalds 
13521da177e4SLinus Torvalds 		if (s->module->seen)
13531da177e4SLinus Torvalds 			continue;
13541da177e4SLinus Torvalds 
13551da177e4SLinus Torvalds 		s->module->seen = 1;
1356*a61b2dfdSSam Ravnborg 		if ((p = strrchr(s->module->name, '/')) != NULL)
1357*a61b2dfdSSam Ravnborg 			p++;
1358*a61b2dfdSSam Ravnborg 		else
1359*a61b2dfdSSam Ravnborg 			p = s->module->name;
1360*a61b2dfdSSam Ravnborg 		buf_printf(b, "%s%s", first ? "" : ",", p);
13611da177e4SLinus Torvalds 		first = 0;
13621da177e4SLinus Torvalds 	}
13631da177e4SLinus Torvalds 	buf_printf(b, "\";\n");
13641da177e4SLinus Torvalds }
13651da177e4SLinus Torvalds 
13665c3ead8cSSam Ravnborg static void add_srcversion(struct buffer *b, struct module *mod)
13671da177e4SLinus Torvalds {
13681da177e4SLinus Torvalds 	if (mod->srcversion[0]) {
13691da177e4SLinus Torvalds 		buf_printf(b, "\n");
13701da177e4SLinus Torvalds 		buf_printf(b, "MODULE_INFO(srcversion, \"%s\");\n",
13711da177e4SLinus Torvalds 			   mod->srcversion);
13721da177e4SLinus Torvalds 	}
13731da177e4SLinus Torvalds }
13741da177e4SLinus Torvalds 
13755c3ead8cSSam Ravnborg static void write_if_changed(struct buffer *b, const char *fname)
13761da177e4SLinus Torvalds {
13771da177e4SLinus Torvalds 	char *tmp;
13781da177e4SLinus Torvalds 	FILE *file;
13791da177e4SLinus Torvalds 	struct stat st;
13801da177e4SLinus Torvalds 
13811da177e4SLinus Torvalds 	file = fopen(fname, "r");
13821da177e4SLinus Torvalds 	if (!file)
13831da177e4SLinus Torvalds 		goto write;
13841da177e4SLinus Torvalds 
13851da177e4SLinus Torvalds 	if (fstat(fileno(file), &st) < 0)
13861da177e4SLinus Torvalds 		goto close_write;
13871da177e4SLinus Torvalds 
13881da177e4SLinus Torvalds 	if (st.st_size != b->pos)
13891da177e4SLinus Torvalds 		goto close_write;
13901da177e4SLinus Torvalds 
13911da177e4SLinus Torvalds 	tmp = NOFAIL(malloc(b->pos));
13921da177e4SLinus Torvalds 	if (fread(tmp, 1, b->pos, file) != b->pos)
13931da177e4SLinus Torvalds 		goto free_write;
13941da177e4SLinus Torvalds 
13951da177e4SLinus Torvalds 	if (memcmp(tmp, b->p, b->pos) != 0)
13961da177e4SLinus Torvalds 		goto free_write;
13971da177e4SLinus Torvalds 
13981da177e4SLinus Torvalds 	free(tmp);
13991da177e4SLinus Torvalds 	fclose(file);
14001da177e4SLinus Torvalds 	return;
14011da177e4SLinus Torvalds 
14021da177e4SLinus Torvalds  free_write:
14031da177e4SLinus Torvalds 	free(tmp);
14041da177e4SLinus Torvalds  close_write:
14051da177e4SLinus Torvalds 	fclose(file);
14061da177e4SLinus Torvalds  write:
14071da177e4SLinus Torvalds 	file = fopen(fname, "w");
14081da177e4SLinus Torvalds 	if (!file) {
14091da177e4SLinus Torvalds 		perror(fname);
14101da177e4SLinus Torvalds 		exit(1);
14111da177e4SLinus Torvalds 	}
14121da177e4SLinus Torvalds 	if (fwrite(b->p, 1, b->pos, file) != b->pos) {
14131da177e4SLinus Torvalds 		perror(fname);
14141da177e4SLinus Torvalds 		exit(1);
14151da177e4SLinus Torvalds 	}
14161da177e4SLinus Torvalds 	fclose(file);
14171da177e4SLinus Torvalds }
14181da177e4SLinus Torvalds 
1419bd5cbcedSRam Pai /* parse Module.symvers file. line format:
1420534b89a9SSam Ravnborg  * 0x12345678<tab>symbol<tab>module[[<tab>export]<tab>something]
1421bd5cbcedSRam Pai  **/
1422040fcc81SSam Ravnborg static void read_dump(const char *fname, unsigned int kernel)
14231da177e4SLinus Torvalds {
14241da177e4SLinus Torvalds 	unsigned long size, pos = 0;
14251da177e4SLinus Torvalds 	void *file = grab_file(fname, &size);
14261da177e4SLinus Torvalds 	char *line;
14271da177e4SLinus Torvalds 
14281da177e4SLinus Torvalds         if (!file)
14291da177e4SLinus Torvalds 		/* No symbol versions, silently ignore */
14301da177e4SLinus Torvalds 		return;
14311da177e4SLinus Torvalds 
14321da177e4SLinus Torvalds 	while ((line = get_next_line(&pos, file, size))) {
1433534b89a9SSam Ravnborg 		char *symname, *modname, *d, *export, *end;
14341da177e4SLinus Torvalds 		unsigned int crc;
14351da177e4SLinus Torvalds 		struct module *mod;
1436040fcc81SSam Ravnborg 		struct symbol *s;
14371da177e4SLinus Torvalds 
14381da177e4SLinus Torvalds 		if (!(symname = strchr(line, '\t')))
14391da177e4SLinus Torvalds 			goto fail;
14401da177e4SLinus Torvalds 		*symname++ = '\0';
14411da177e4SLinus Torvalds 		if (!(modname = strchr(symname, '\t')))
14421da177e4SLinus Torvalds 			goto fail;
14431da177e4SLinus Torvalds 		*modname++ = '\0';
14449ac545b0SLaurent Riffard 		if ((export = strchr(modname, '\t')) != NULL)
1445bd5cbcedSRam Pai 			*export++ = '\0';
1446534b89a9SSam Ravnborg 		if (export && ((end = strchr(export, '\t')) != NULL))
1447534b89a9SSam Ravnborg 			*end = '\0';
14481da177e4SLinus Torvalds 		crc = strtoul(line, &d, 16);
14491da177e4SLinus Torvalds 		if (*symname == '\0' || *modname == '\0' || *d != '\0')
14501da177e4SLinus Torvalds 			goto fail;
14511da177e4SLinus Torvalds 
14521da177e4SLinus Torvalds 		if (!(mod = find_module(modname))) {
14531da177e4SLinus Torvalds 			if (is_vmlinux(modname)) {
14541da177e4SLinus Torvalds 				have_vmlinux = 1;
14551da177e4SLinus Torvalds 			}
14561da177e4SLinus Torvalds 			mod = new_module(NOFAIL(strdup(modname)));
14571da177e4SLinus Torvalds 			mod->skip = 1;
14581da177e4SLinus Torvalds 		}
1459bd5cbcedSRam Pai 		s = sym_add_exported(symname, mod, export_no(export));
1460040fcc81SSam Ravnborg 		s->kernel    = kernel;
14618e70c458SSam Ravnborg 		s->preloaded = 1;
1462bd5cbcedSRam Pai 		sym_update_crc(symname, mod, crc, export_no(export));
14631da177e4SLinus Torvalds 	}
14641da177e4SLinus Torvalds 	return;
14651da177e4SLinus Torvalds fail:
14661da177e4SLinus Torvalds 	fatal("parse error in symbol dump file\n");
14671da177e4SLinus Torvalds }
14681da177e4SLinus Torvalds 
1469040fcc81SSam Ravnborg /* For normal builds always dump all symbols.
1470040fcc81SSam Ravnborg  * For external modules only dump symbols
1471040fcc81SSam Ravnborg  * that are not read from kernel Module.symvers.
1472040fcc81SSam Ravnborg  **/
1473040fcc81SSam Ravnborg static int dump_sym(struct symbol *sym)
1474040fcc81SSam Ravnborg {
1475040fcc81SSam Ravnborg 	if (!external_module)
1476040fcc81SSam Ravnborg 		return 1;
1477040fcc81SSam Ravnborg 	if (sym->vmlinux || sym->kernel)
1478040fcc81SSam Ravnborg 		return 0;
1479040fcc81SSam Ravnborg 	return 1;
1480040fcc81SSam Ravnborg }
1481040fcc81SSam Ravnborg 
14825c3ead8cSSam Ravnborg static void write_dump(const char *fname)
14831da177e4SLinus Torvalds {
14841da177e4SLinus Torvalds 	struct buffer buf = { };
14851da177e4SLinus Torvalds 	struct symbol *symbol;
14861da177e4SLinus Torvalds 	int n;
14871da177e4SLinus Torvalds 
14881da177e4SLinus Torvalds 	for (n = 0; n < SYMBOL_HASH_SIZE ; n++) {
14891da177e4SLinus Torvalds 		symbol = symbolhash[n];
14901da177e4SLinus Torvalds 		while (symbol) {
1491040fcc81SSam Ravnborg 			if (dump_sym(symbol))
1492bd5cbcedSRam Pai 				buf_printf(&buf, "0x%08x\t%s\t%s\t%s\n",
1493040fcc81SSam Ravnborg 					symbol->crc, symbol->name,
1494bd5cbcedSRam Pai 					symbol->module->name,
1495bd5cbcedSRam Pai 					export_str(symbol->export));
14961da177e4SLinus Torvalds 			symbol = symbol->next;
14971da177e4SLinus Torvalds 		}
14981da177e4SLinus Torvalds 	}
14991da177e4SLinus Torvalds 	write_if_changed(&buf, fname);
15001da177e4SLinus Torvalds }
15011da177e4SLinus Torvalds 
15025c3ead8cSSam Ravnborg int main(int argc, char **argv)
15031da177e4SLinus Torvalds {
15041da177e4SLinus Torvalds 	struct module *mod;
15051da177e4SLinus Torvalds 	struct buffer buf = { };
15061da177e4SLinus Torvalds 	char fname[SZ];
1507040fcc81SSam Ravnborg 	char *kernel_read = NULL, *module_read = NULL;
1508040fcc81SSam Ravnborg 	char *dump_write = NULL;
15091da177e4SLinus Torvalds 	int opt;
1510c53ddacdSKirill Korotaev 	int err;
15111da177e4SLinus Torvalds 
1512c53ddacdSKirill Korotaev 	while ((opt = getopt(argc, argv, "i:I:mo:aw")) != -1) {
15131da177e4SLinus Torvalds 		switch(opt) {
15141da177e4SLinus Torvalds 			case 'i':
1515040fcc81SSam Ravnborg 				kernel_read = optarg;
1516040fcc81SSam Ravnborg 				break;
1517040fcc81SSam Ravnborg 			case 'I':
1518040fcc81SSam Ravnborg 				module_read = optarg;
1519040fcc81SSam Ravnborg 				external_module = 1;
15201da177e4SLinus Torvalds 				break;
15211da177e4SLinus Torvalds 			case 'm':
15221da177e4SLinus Torvalds 				modversions = 1;
15231da177e4SLinus Torvalds 				break;
15241da177e4SLinus Torvalds 			case 'o':
15251da177e4SLinus Torvalds 				dump_write = optarg;
15261da177e4SLinus Torvalds 				break;
15271da177e4SLinus Torvalds 			case 'a':
15281da177e4SLinus Torvalds 				all_versions = 1;
15291da177e4SLinus Torvalds 				break;
1530c53ddacdSKirill Korotaev 			case 'w':
1531c53ddacdSKirill Korotaev 				warn_unresolved = 1;
1532c53ddacdSKirill Korotaev 				break;
15331da177e4SLinus Torvalds 			default:
15341da177e4SLinus Torvalds 				exit(1);
15351da177e4SLinus Torvalds 		}
15361da177e4SLinus Torvalds 	}
15371da177e4SLinus Torvalds 
1538040fcc81SSam Ravnborg 	if (kernel_read)
1539040fcc81SSam Ravnborg 		read_dump(kernel_read, 1);
1540040fcc81SSam Ravnborg 	if (module_read)
1541040fcc81SSam Ravnborg 		read_dump(module_read, 0);
15421da177e4SLinus Torvalds 
15431da177e4SLinus Torvalds 	while (optind < argc) {
15441da177e4SLinus Torvalds 		read_symbols(argv[optind++]);
15451da177e4SLinus Torvalds 	}
15461da177e4SLinus Torvalds 
15471da177e4SLinus Torvalds 	for (mod = modules; mod; mod = mod->next) {
15481da177e4SLinus Torvalds 		if (mod->skip)
15491da177e4SLinus Torvalds 			continue;
1550c96fca21SSam Ravnborg 		check_exports(mod);
1551b817f6feSSam Ravnborg 	}
1552b817f6feSSam Ravnborg 
1553c53ddacdSKirill Korotaev 	err = 0;
1554c53ddacdSKirill Korotaev 
1555b817f6feSSam Ravnborg 	for (mod = modules; mod; mod = mod->next) {
1556b817f6feSSam Ravnborg 		if (mod->skip)
1557b817f6feSSam Ravnborg 			continue;
15581da177e4SLinus Torvalds 
15591da177e4SLinus Torvalds 		buf.pos = 0;
15601da177e4SLinus Torvalds 
15611da177e4SLinus Torvalds 		add_header(&buf, mod);
1562c53ddacdSKirill Korotaev 		err |= add_versions(&buf, mod);
15631da177e4SLinus Torvalds 		add_depends(&buf, mod, modules);
15641da177e4SLinus Torvalds 		add_moddevtable(&buf, mod);
15651da177e4SLinus Torvalds 		add_srcversion(&buf, mod);
15661da177e4SLinus Torvalds 
15671da177e4SLinus Torvalds 		sprintf(fname, "%s.mod.c", mod->name);
15681da177e4SLinus Torvalds 		write_if_changed(&buf, fname);
15691da177e4SLinus Torvalds 	}
15701da177e4SLinus Torvalds 
15711da177e4SLinus Torvalds 	if (dump_write)
15721da177e4SLinus Torvalds 		write_dump(dump_write);
15731da177e4SLinus Torvalds 
1574c53ddacdSKirill Korotaev 	return err;
15751da177e4SLinus Torvalds }
1576