2005-04-16 15:20:36 -07:00
/* Postprocess module symbol versions
*
* Copyright 2003 Kai Germaschewski
* Copyright 2002 - 2004 Rusty Russell , IBM Corporation
2006-02-26 20:11:17 +01:00
* Copyright 2006 Sam Ravnborg
2005-04-16 15:20:36 -07:00
* Based in part on module - init - tools / depmod . c , file2alias
*
* This software may be used and distributed according to the terms
* of the GNU General Public License , incorporated herein by reference .
*
* Usage : modpost vmlinux module1 . o module2 . o . . .
*/
# include <ctype.h>
# include "modpost.h"
/* Are we using CONFIG_MODVERSIONS? */
int modversions = 0 ;
/* Warn about undefined symbols? (do so if we have vmlinux) */
int have_vmlinux = 0 ;
/* Is CONFIG_MODULE_SRCVERSION_ALL set? */
static int all_versions = 0 ;
2006-01-28 22:15:55 +01:00
/* If we are modposting external module set to 1 */
static int external_module = 0 ;
2005-04-16 15:20:36 -07:00
2006-01-28 17:19:35 +01:00
void fatal ( const char * fmt , . . . )
2005-04-16 15:20:36 -07:00
{
va_list arglist ;
fprintf ( stderr , " FATAL: " ) ;
va_start ( arglist , fmt ) ;
vfprintf ( stderr , fmt , arglist ) ;
va_end ( arglist ) ;
exit ( 1 ) ;
}
2006-01-28 17:19:35 +01:00
void warn ( const char * fmt , . . . )
2005-04-16 15:20:36 -07:00
{
va_list arglist ;
fprintf ( stderr , " WARNING: " ) ;
va_start ( arglist , fmt ) ;
vfprintf ( stderr , fmt , arglist ) ;
va_end ( arglist ) ;
}
2006-01-28 22:15:55 +01:00
static int is_vmlinux ( const char * modname )
{
const char * myname ;
if ( ( myname = strrchr ( modname , ' / ' ) ) )
myname + + ;
else
myname = modname ;
return strcmp ( myname , " vmlinux " ) = = 0 ;
}
2005-04-16 15:20:36 -07:00
void * do_nofail ( void * ptr , const char * expr )
{
if ( ! ptr ) {
fatal ( " modpost: Memory allocation failure: %s. \n " , expr ) ;
}
return ptr ;
}
/* A list of all modules we processed */
static struct module * modules ;
2006-01-28 17:19:35 +01:00
static struct module * find_module ( char * modname )
2005-04-16 15:20:36 -07:00
{
struct module * mod ;
for ( mod = modules ; mod ; mod = mod - > next )
if ( strcmp ( mod - > name , modname ) = = 0 )
break ;
return mod ;
}
2006-01-28 17:19:35 +01:00
static struct module * new_module ( char * modname )
2005-04-16 15:20:36 -07:00
{
struct module * mod ;
char * p , * s ;
2006-03-03 16:46:04 +01:00
2005-04-16 15:20:36 -07:00
mod = NOFAIL ( malloc ( sizeof ( * mod ) ) ) ;
memset ( mod , 0 , sizeof ( * mod ) ) ;
p = NOFAIL ( strdup ( modname ) ) ;
/* strip trailing .o */
if ( ( s = strrchr ( p , ' . ' ) ) ! = NULL )
if ( strcmp ( s , " .o " ) = = 0 )
* s = ' \0 ' ;
/* add to list */
mod - > name = p ;
mod - > next = modules ;
modules = mod ;
return mod ;
}
/* A hash of all exported symbols,
* struct symbol is also used for lists of unresolved symbols */
# define SYMBOL_HASH_SIZE 1024
struct symbol {
struct symbol * next ;
struct module * module ;
unsigned int crc ;
int crc_valid ;
unsigned int weak : 1 ;
2006-01-28 22:15:55 +01:00
unsigned int vmlinux : 1 ; /* 1 if symbol is defined in vmlinux */
unsigned int kernel : 1 ; /* 1 if symbol is from kernel
* ( only for external modules ) * */
2006-01-28 22:22:33 +01:00
unsigned int preloaded : 1 ; /* 1 if symbol from Module.symvers */
2005-04-16 15:20:36 -07:00
char name [ 0 ] ;
} ;
static struct symbol * symbolhash [ SYMBOL_HASH_SIZE ] ;
/* This is based on the hash agorithm from gdbm, via tdb */
static inline unsigned int tdb_hash ( const char * name )
{
unsigned value ; /* Used to compute the hash value. */
unsigned i ; /* Used to cycle through random values. */
/* Set the initial value from the key size. */
for ( value = 0x238F13AF * strlen ( name ) , i = 0 ; name [ i ] ; i + + )
value = ( value + ( ( ( unsigned char * ) name ) [ i ] < < ( i * 5 % 24 ) ) ) ;
return ( 1103515243 * value + 12345 ) ;
}
2006-01-28 17:19:35 +01:00
/**
* Allocate a new symbols for use in the hash of exported symbols or
* the list of unresolved symbols per module
* */
static struct symbol * alloc_symbol ( const char * name , unsigned int weak ,
struct symbol * next )
2005-04-16 15:20:36 -07:00
{
struct symbol * s = NOFAIL ( malloc ( sizeof ( * s ) + strlen ( name ) + 1 ) ) ;
memset ( s , 0 , sizeof ( * s ) ) ;
strcpy ( s - > name , name ) ;
s - > weak = weak ;
s - > next = next ;
return s ;
}
/* For the hash of exported symbols */
2006-01-28 22:15:55 +01:00
static struct symbol * new_symbol ( const char * name , struct module * module )
2005-04-16 15:20:36 -07:00
{
unsigned int hash ;
struct symbol * new ;
hash = tdb_hash ( name ) % SYMBOL_HASH_SIZE ;
new = symbolhash [ hash ] = alloc_symbol ( name , 0 , symbolhash [ hash ] ) ;
new - > module = module ;
2006-01-28 22:15:55 +01:00
return new ;
2005-04-16 15:20:36 -07:00
}
2006-01-28 17:19:35 +01:00
static struct symbol * find_symbol ( const char * name )
2005-04-16 15:20:36 -07:00
{
struct symbol * s ;
/* For our purposes, .foo matches foo. PPC64 needs this. */
if ( name [ 0 ] = = ' . ' )
name + + ;
for ( s = symbolhash [ tdb_hash ( name ) % SYMBOL_HASH_SIZE ] ; s ; s = s - > next ) {
if ( strcmp ( s - > name , name ) = = 0 )
return s ;
}
return NULL ;
}
2006-01-28 17:19:35 +01:00
/**
* Add an exported symbol - it may have already been added without a
* CRC , in this case just update the CRC
* */
2006-01-28 22:15:55 +01:00
static struct symbol * sym_add_exported ( const char * name , struct module * mod )
2005-04-16 15:20:36 -07:00
{
struct symbol * s = find_symbol ( name ) ;
if ( ! s ) {
2006-01-28 22:15:55 +01:00
s = new_symbol ( name , mod ) ;
2006-01-28 22:22:33 +01:00
} else {
if ( ! s - > preloaded ) {
2006-03-05 13:48:58 +01:00
warn ( " %s: '%s' exported twice. Previous export "
2006-01-28 22:22:33 +01:00
" was in %s%s \n " , mod - > name , name ,
s - > module - > name ,
is_vmlinux ( s - > module - > name ) ? " " : " .ko " ) ;
}
2005-04-16 15:20:36 -07:00
}
2006-01-28 22:22:33 +01:00
s - > preloaded = 0 ;
2006-01-28 22:15:55 +01:00
s - > vmlinux = is_vmlinux ( mod - > name ) ;
s - > kernel = 0 ;
return s ;
}
static void sym_update_crc ( const char * name , struct module * mod ,
unsigned int crc )
{
struct symbol * s = find_symbol ( name ) ;
if ( ! s )
s = new_symbol ( name , mod ) ;
s - > crc = crc ;
s - > crc_valid = 1 ;
2005-04-16 15:20:36 -07:00
}
2006-01-28 17:19:35 +01:00
void * grab_file ( const char * filename , unsigned long * size )
2005-04-16 15:20:36 -07:00
{
struct stat st ;
void * map ;
int fd ;
fd = open ( filename , O_RDONLY ) ;
if ( fd < 0 | | fstat ( fd , & st ) ! = 0 )
return NULL ;
* size = st . st_size ;
map = mmap ( NULL , * size , PROT_READ | PROT_WRITE , MAP_PRIVATE , fd , 0 ) ;
close ( fd ) ;
if ( map = = MAP_FAILED )
return NULL ;
return map ;
}
2006-01-28 17:19:35 +01:00
/**
* Return a copy of the next line in a mmap ' ed file .
* spaces in the beginning of the line is trimmed away .
* Return a pointer to a static buffer .
* */
char * get_next_line ( unsigned long * pos , void * file , unsigned long size )
2005-04-16 15:20:36 -07:00
{
static char line [ 4096 ] ;
int skip = 1 ;
size_t len = 0 ;
signed char * p = ( signed char * ) file + * pos ;
char * s = line ;
for ( ; * pos < size ; ( * pos ) + + )
{
if ( skip & & isspace ( * p ) ) {
p + + ;
continue ;
}
skip = 0 ;
if ( * p ! = ' \n ' & & ( * pos < size ) ) {
len + + ;
* s + + = * p + + ;
if ( len > 4095 )
break ; /* Too long, stop */
} else {
/* End of string */
* s = ' \0 ' ;
return line ;
}
}
/* End of buffer */
return NULL ;
}
2006-01-28 17:19:35 +01:00
void release_file ( void * file , unsigned long size )
2005-04-16 15:20:36 -07:00
{
munmap ( file , size ) ;
}
2006-01-28 17:19:35 +01:00
static void parse_elf ( struct elf_info * info , const char * filename )
2005-04-16 15:20:36 -07:00
{
unsigned int i ;
Elf_Ehdr * hdr = info - > hdr ;
Elf_Shdr * sechdrs ;
Elf_Sym * sym ;
hdr = grab_file ( filename , & info - > size ) ;
if ( ! hdr ) {
perror ( filename ) ;
abort ( ) ;
}
info - > hdr = hdr ;
if ( info - > size < sizeof ( * hdr ) )
goto truncated ;
/* Fix endianness in ELF header */
hdr - > e_shoff = TO_NATIVE ( hdr - > e_shoff ) ;
hdr - > e_shstrndx = TO_NATIVE ( hdr - > e_shstrndx ) ;
hdr - > e_shnum = TO_NATIVE ( hdr - > e_shnum ) ;
hdr - > e_machine = TO_NATIVE ( hdr - > e_machine ) ;
sechdrs = ( void * ) hdr + hdr - > e_shoff ;
info - > sechdrs = sechdrs ;
/* Fix endianness in section headers */
for ( i = 0 ; i < hdr - > e_shnum ; i + + ) {
sechdrs [ i ] . sh_type = TO_NATIVE ( sechdrs [ i ] . sh_type ) ;
sechdrs [ i ] . sh_offset = TO_NATIVE ( sechdrs [ i ] . sh_offset ) ;
sechdrs [ i ] . sh_size = TO_NATIVE ( sechdrs [ i ] . sh_size ) ;
sechdrs [ i ] . sh_link = TO_NATIVE ( sechdrs [ i ] . sh_link ) ;
sechdrs [ i ] . sh_name = TO_NATIVE ( sechdrs [ i ] . sh_name ) ;
}
/* Find symbol table. */
for ( i = 1 ; i < hdr - > e_shnum ; i + + ) {
const char * secstrings
= ( void * ) hdr + sechdrs [ hdr - > e_shstrndx ] . sh_offset ;
if ( sechdrs [ i ] . sh_offset > info - > size )
goto truncated ;
if ( strcmp ( secstrings + sechdrs [ i ] . sh_name , " .modinfo " ) = = 0 ) {
info - > modinfo = ( void * ) hdr + sechdrs [ i ] . sh_offset ;
info - > modinfo_len = sechdrs [ i ] . sh_size ;
}
if ( sechdrs [ i ] . sh_type ! = SHT_SYMTAB )
continue ;
info - > symtab_start = ( void * ) hdr + sechdrs [ i ] . sh_offset ;
2006-03-03 16:46:04 +01:00
info - > symtab_stop = ( void * ) hdr + sechdrs [ i ] . sh_offset
2005-04-16 15:20:36 -07:00
+ sechdrs [ i ] . sh_size ;
2006-03-03 16:46:04 +01:00
info - > strtab = ( void * ) hdr +
2005-04-16 15:20:36 -07:00
sechdrs [ sechdrs [ i ] . sh_link ] . sh_offset ;
}
if ( ! info - > symtab_start ) {
2006-01-28 16:57:26 +01:00
fatal ( " %s has no symtab? \n " , filename ) ;
2005-04-16 15:20:36 -07:00
}
/* Fix endianness in symbols */
for ( sym = info - > symtab_start ; sym < info - > symtab_stop ; sym + + ) {
sym - > st_shndx = TO_NATIVE ( sym - > st_shndx ) ;
sym - > st_name = TO_NATIVE ( sym - > st_name ) ;
sym - > st_value = TO_NATIVE ( sym - > st_value ) ;
sym - > st_size = TO_NATIVE ( sym - > st_size ) ;
}
return ;
truncated :
2006-01-28 16:57:26 +01:00
fatal ( " %s is truncated. \n " , filename ) ;
2005-04-16 15:20:36 -07:00
}
2006-01-28 17:19:35 +01:00
static void parse_elf_finish ( struct elf_info * info )
2005-04-16 15:20:36 -07:00
{
release_file ( info - > hdr , info - > size ) ;
}
2006-03-02 18:35:31 +08:00
# define CRC_PFX MODULE_SYMBOL_PREFIX "__crc_"
# define KSYMTAB_PFX MODULE_SYMBOL_PREFIX "__ksymtab_"
2005-04-16 15:20:36 -07:00
2006-01-28 17:19:35 +01:00
static void handle_modversions ( struct module * mod , struct elf_info * info ,
Elf_Sym * sym , const char * symname )
2005-04-16 15:20:36 -07:00
{
unsigned int crc ;
switch ( sym - > st_shndx ) {
case SHN_COMMON :
2006-01-28 16:57:26 +01:00
warn ( " \" %s \" [%s] is COMMON symbol \n " , symname , mod - > name ) ;
2005-04-16 15:20:36 -07:00
break ;
case SHN_ABS :
/* CRC'd symbol */
if ( memcmp ( symname , CRC_PFX , strlen ( CRC_PFX ) ) = = 0 ) {
crc = ( unsigned int ) sym - > st_value ;
2006-01-28 22:15:55 +01:00
sym_update_crc ( symname + strlen ( CRC_PFX ) , mod , crc ) ;
2005-04-16 15:20:36 -07:00
}
break ;
case SHN_UNDEF :
/* undefined symbol */
if ( ELF_ST_BIND ( sym - > st_info ) ! = STB_GLOBAL & &
ELF_ST_BIND ( sym - > st_info ) ! = STB_WEAK )
break ;
/* ignore global offset table */
if ( strcmp ( symname , " _GLOBAL_OFFSET_TABLE_ " ) = = 0 )
break ;
/* ignore __this_module, it will be resolved shortly */
if ( strcmp ( symname , MODULE_SYMBOL_PREFIX " __this_module " ) = = 0 )
break ;
2005-08-19 13:44:57 -07:00
/* cope with newer glibc (2.3.4 or higher) STT_ definition in elf.h */
# if defined(STT_REGISTER) || defined(STT_SPARC_REGISTER)
/* add compatibility with older glibc */
# ifndef STT_SPARC_REGISTER
# define STT_SPARC_REGISTER STT_REGISTER
# endif
2005-04-16 15:20:36 -07:00
if ( info - > hdr - > e_machine = = EM_SPARC | |
info - > hdr - > e_machine = = EM_SPARCV9 ) {
/* Ignore register directives. */
2005-08-19 13:44:57 -07:00
if ( ELF_ST_TYPE ( sym - > st_info ) = = STT_SPARC_REGISTER )
2005-04-16 15:20:36 -07:00
break ;
2006-03-03 16:46:04 +01:00
if ( symname [ 0 ] = = ' . ' ) {
char * munged = strdup ( symname ) ;
munged [ 0 ] = ' _ ' ;
munged [ 1 ] = toupper ( munged [ 1 ] ) ;
symname = munged ;
}
2005-04-16 15:20:36 -07:00
}
# endif
2006-03-03 16:46:04 +01:00
2005-04-16 15:20:36 -07:00
if ( memcmp ( symname , MODULE_SYMBOL_PREFIX ,
strlen ( MODULE_SYMBOL_PREFIX ) ) = = 0 )
mod - > unres = alloc_symbol ( symname +
strlen ( MODULE_SYMBOL_PREFIX ) ,
ELF_ST_BIND ( sym - > st_info ) = = STB_WEAK ,
mod - > unres ) ;
break ;
default :
/* All exported symbols */
if ( memcmp ( symname , KSYMTAB_PFX , strlen ( KSYMTAB_PFX ) ) = = 0 ) {
2006-01-28 22:15:55 +01:00
sym_add_exported ( symname + strlen ( KSYMTAB_PFX ) , mod ) ;
2005-04-16 15:20:36 -07:00
}
if ( strcmp ( symname , MODULE_SYMBOL_PREFIX " init_module " ) = = 0 )
mod - > has_init = 1 ;
if ( strcmp ( symname , MODULE_SYMBOL_PREFIX " cleanup_module " ) = = 0 )
mod - > has_cleanup = 1 ;
break ;
}
}
2006-01-28 17:19:35 +01:00
/**
* Parse tag = value strings from . modinfo section
* */
2005-04-16 15:20:36 -07:00
static char * next_string ( char * string , unsigned long * secsize )
{
/* Skip non-zero chars */
while ( string [ 0 ] ) {
string + + ;
if ( ( * secsize ) - - < = 1 )
return NULL ;
}
/* Skip any zero padding. */
while ( ! string [ 0 ] ) {
string + + ;
if ( ( * secsize ) - - < = 1 )
return NULL ;
}
return string ;
}
static char * get_modinfo ( void * modinfo , unsigned long modinfo_len ,
const char * tag )
{
char * p ;
unsigned int taglen = strlen ( tag ) ;
unsigned long size = modinfo_len ;
for ( p = modinfo ; p ; p = next_string ( p , & size ) ) {
if ( strncmp ( p , tag , taglen ) = = 0 & & p [ taglen ] = = ' = ' )
return p + taglen + 1 ;
}
return NULL ;
}
2006-02-26 22:18:11 +01:00
/**
* Test if string s ends in string sub
* return 0 if match
* */
static int strrcmp ( const char * s , const char * sub )
{
int slen , sublen ;
2006-03-03 16:46:04 +01:00
2006-02-26 22:18:11 +01:00
if ( ! s | | ! sub )
return 1 ;
2006-03-03 16:46:04 +01:00
2006-02-26 22:18:11 +01:00
slen = strlen ( s ) ;
sublen = strlen ( sub ) ;
2006-03-03 16:46:04 +01:00
2006-02-26 22:18:11 +01:00
if ( ( slen = = 0 ) | | ( sublen = = 0 ) )
return 1 ;
if ( sublen > slen )
return 1 ;
return memcmp ( s + slen - sublen , sub , sublen ) ;
}
/**
* Whitelist to allow certain references to pass with no warning .
* Pattern 1 :
* If a module parameter is declared __initdata and permissions = 0
* then this is legal despite the warning generated .
* We cannot see value of permissions here , so just ignore
* this pattern .
* The pattern is identified by :
* tosec = . init . data
2006-03-05 00:16:26 +01:00
* fromsec = . data *
2006-02-26 22:18:11 +01:00
* atsym = __param *
2006-03-03 16:46:04 +01:00
*
2006-02-26 22:18:11 +01:00
* Pattern 2 :
* Many drivers utilise a * _driver container with references to
* add , remove , probe functions etc .
* These functions may often be marked __init and we do not want to
* warn here .
* the pattern is identified by :
* tosec = . init . text | . exit . text
* fromsec = . data
* atsym = * _driver , * _ops , * _probe , * probe_one
* */
static int secref_whitelist ( const char * tosec , const char * fromsec ,
const char * atsym )
{
int f1 = 1 , f2 = 1 ;
const char * * s ;
const char * pat2sym [ ] = {
" _driver " ,
" _ops " ,
" _probe " ,
" _probe_one " ,
NULL
} ;
2006-03-03 16:46:04 +01:00
2006-02-26 22:18:11 +01:00
/* Check for pattern 1 */
if ( strcmp ( tosec , " .init.data " ) ! = 0 )
f1 = 0 ;
2006-03-05 00:16:26 +01:00
if ( strncmp ( fromsec , " .data " , strlen ( " .data " ) ) ! = 0 )
2006-02-26 22:18:11 +01:00
f1 = 0 ;
if ( strncmp ( atsym , " __param " , strlen ( " __param " ) ) ! = 0 )
f1 = 0 ;
if ( f1 )
return f1 ;
/* Check for pattern 2 */
2006-03-03 16:46:04 +01:00
if ( ( strcmp ( tosec , " .init.text " ) ! = 0 ) & &
2006-02-26 22:18:11 +01:00
( strcmp ( tosec , " .exit.text " ) ! = 0 ) )
f2 = 0 ;
if ( strcmp ( fromsec , " .data " ) ! = 0 )
f2 = 0 ;
for ( s = pat2sym ; * s ; s + + )
if ( strrcmp ( atsym , * s ) = = 0 )
f1 = 1 ;
return f1 & & f2 ;
}
2006-02-19 11:53:35 +01:00
/**
* Find symbol based on relocation record info .
* In some cases the symbol supplied is a valid symbol so
* return refsym . If st_name ! = 0 we assume this is a valid symbol .
* In other cases the symbol needs to be looked up in the symbol table
* based on section and address .
* * */
static Elf_Sym * find_elf_symbol ( struct elf_info * elf , Elf_Addr addr ,
Elf_Sym * relsym )
{
Elf_Sym * sym ;
if ( relsym - > st_name ! = 0 )
return relsym ;
for ( sym = elf - > symtab_start ; sym < elf - > symtab_stop ; sym + + ) {
if ( sym - > st_shndx ! = relsym - > st_shndx )
continue ;
if ( sym - > st_value = = addr )
return sym ;
}
return NULL ;
}
2006-02-17 22:42:02 +01:00
/*
2006-03-05 12:02:46 +01:00
* Find symbols before or equal addr and after addr - in the section sec .
* If we find two symbols with equal offset prefer one with a valid name .
* The ELF format may have a better way to detect what type of symbol
* it is , but this works for now .
2006-02-17 22:42:02 +01:00
* */
static void find_symbols_between ( struct elf_info * elf , Elf_Addr addr ,
const char * sec ,
Elf_Sym * * before , Elf_Sym * * after )
{
Elf_Sym * sym ;
Elf_Ehdr * hdr = elf - > hdr ;
Elf_Addr beforediff = ~ 0 ;
Elf_Addr afterdiff = ~ 0 ;
const char * secstrings = ( void * ) hdr +
elf - > sechdrs [ hdr - > e_shstrndx ] . sh_offset ;
2006-03-03 16:46:04 +01:00
2006-02-17 22:42:02 +01:00
* before = NULL ;
* after = NULL ;
for ( sym = elf - > symtab_start ; sym < elf - > symtab_stop ; sym + + ) {
const char * symsec ;
if ( sym - > st_shndx > = SHN_LORESERVE )
continue ;
symsec = secstrings + elf - > sechdrs [ sym - > st_shndx ] . sh_name ;
if ( strcmp ( symsec , sec ) ! = 0 )
continue ;
if ( sym - > st_value < = addr ) {
if ( ( addr - sym - > st_value ) < beforediff ) {
beforediff = addr - sym - > st_value ;
* before = sym ;
}
2006-03-05 12:02:46 +01:00
else if ( ( addr - sym - > st_value ) = = beforediff ) {
/* equal offset, valid name? */
const char * name = elf - > strtab + sym - > st_name ;
if ( name & & strlen ( name ) )
* before = sym ;
}
2006-02-17 22:42:02 +01:00
}
else
{
if ( ( sym - > st_value - addr ) < afterdiff ) {
afterdiff = sym - > st_value - addr ;
* after = sym ;
}
2006-03-05 12:02:46 +01:00
else if ( ( sym - > st_value - addr ) = = afterdiff ) {
/* equal offset, valid name? */
const char * name = elf - > strtab + sym - > st_name ;
if ( name & & strlen ( name ) )
* after = sym ;
}
2006-02-17 22:42:02 +01:00
}
}
}
/**
* Print a warning about a section mismatch .
* Try to find symbols near it so user can find it .
2006-02-26 22:18:11 +01:00
* Check whitelist before warning - it may be a false positive .
2006-02-17 22:42:02 +01:00
* */
static void warn_sec_mismatch ( const char * modname , const char * fromsec ,
struct elf_info * elf , Elf_Sym * sym , Elf_Rela r )
{
2006-02-19 11:53:35 +01:00
const char * refsymname = " " ;
Elf_Sym * before , * after ;
Elf_Sym * refsym ;
2006-02-17 22:42:02 +01:00
Elf_Ehdr * hdr = elf - > hdr ;
Elf_Shdr * sechdrs = elf - > sechdrs ;
const char * secstrings = ( void * ) hdr +
sechdrs [ hdr - > e_shstrndx ] . sh_offset ;
const char * secname = secstrings + sechdrs [ sym - > st_shndx ] . sh_name ;
2006-03-03 16:46:04 +01:00
2006-02-17 22:42:02 +01:00
find_symbols_between ( elf , r . r_offset , fromsec , & before , & after ) ;
2006-02-19 11:53:35 +01:00
refsym = find_elf_symbol ( elf , r . r_addend , sym ) ;
if ( refsym & & strlen ( elf - > strtab + refsym - > st_name ) )
refsymname = elf - > strtab + refsym - > st_name ;
2006-02-26 22:18:11 +01:00
/* check whitelist - we may ignore it */
2006-03-03 16:46:04 +01:00
if ( before & &
2006-02-26 22:18:11 +01:00
secref_whitelist ( secname , fromsec , elf - > strtab + before - > st_name ) )
return ;
2006-03-03 16:46:04 +01:00
2006-02-17 22:42:02 +01:00
if ( before & & after ) {
2006-02-19 11:53:35 +01:00
warn ( " %s - Section mismatch: reference to %s:%s from %s "
" between '%s' (at offset 0x%llx) and '%s' \n " ,
modname , secname , refsymname , fromsec ,
2006-02-17 22:42:02 +01:00
elf - > strtab + before - > st_name ,
2006-02-19 11:53:35 +01:00
( long long ) r . r_offset ,
2006-02-17 22:42:02 +01:00
elf - > strtab + after - > st_name ) ;
} else if ( before ) {
2006-02-19 11:53:35 +01:00
warn ( " %s - Section mismatch: reference to %s:%s from %s "
" after '%s' (at offset 0x%llx) \n " ,
2006-03-03 16:46:04 +01:00
modname , secname , refsymname , fromsec ,
2006-02-17 22:42:02 +01:00
elf - > strtab + before - > st_name ,
2006-02-19 11:53:35 +01:00
( long long ) r . r_offset ) ;
2006-02-17 22:42:02 +01:00
} else if ( after ) {
2006-02-19 11:53:35 +01:00
warn ( " %s - Section mismatch: reference to %s:%s from %s "
" before '%s' (at offset -0x%llx) \n " ,
2006-03-03 16:46:04 +01:00
modname , secname , refsymname , fromsec ,
2006-02-17 22:42:02 +01:00
elf - > strtab + before - > st_name ,
2006-02-19 11:53:35 +01:00
( long long ) r . r_offset ) ;
2006-02-17 22:42:02 +01:00
} else {
2006-02-19 11:53:35 +01:00
warn ( " %s - Section mismatch: reference to %s:%s from %s "
" (offset 0x%llx) \n " ,
modname , secname , fromsec , refsymname ,
( long long ) r . r_offset ) ;
2006-02-17 22:42:02 +01:00
}
}
/**
* A module includes a number of sections that are discarded
* either when loaded or when used as built - in .
* For loaded modules all functions marked __init and all data
* marked __initdata will be discarded when the module has been intialized .
* Likewise for modules used built - in the sections marked __exit
* are discarded because __exit marked function are supposed to be called
* only when a moduel is unloaded which never happes for built - in modules .
* The check_sec_ref ( ) function traverses all relocation records
* to find all references to a section that reference a section that will
* be discarded and warns about it .
* */
static void check_sec_ref ( struct module * mod , const char * modname ,
struct elf_info * elf ,
int section ( const char * ) ,
int section_ref_ok ( const char * ) )
{
int i ;
Elf_Sym * sym ;
Elf_Ehdr * hdr = elf - > hdr ;
Elf_Shdr * sechdrs = elf - > sechdrs ;
const char * secstrings = ( void * ) hdr +
sechdrs [ hdr - > e_shstrndx ] . sh_offset ;
2006-03-03 16:46:04 +01:00
2006-02-17 22:42:02 +01:00
/* Walk through all sections */
for ( i = 0 ; i < hdr - > e_shnum ; i + + ) {
2006-02-22 03:19:54 -08:00
Elf_Rela * rela ;
Elf_Rela * start = ( void * ) hdr + sechdrs [ i ] . sh_offset ;
Elf_Rela * stop = ( void * ) start + sechdrs [ i ] . sh_size ;
2006-02-17 22:42:02 +01:00
const char * name = secstrings + sechdrs [ i ] . sh_name +
strlen ( " .rela " ) ;
/* We want to process only relocation sections and not .init */
if ( section_ref_ok ( name ) | | ( sechdrs [ i ] . sh_type ! = SHT_RELA ) )
continue ;
for ( rela = start ; rela < stop ; rela + + ) {
Elf_Rela r ;
const char * secname ;
r . r_offset = TO_NATIVE ( rela - > r_offset ) ;
r . r_info = TO_NATIVE ( rela - > r_info ) ;
2006-02-19 11:53:35 +01:00
r . r_addend = TO_NATIVE ( rela - > r_addend ) ;
2006-02-17 22:42:02 +01:00
sym = elf - > symtab_start + ELF_R_SYM ( r . r_info ) ;
/* Skip special sections */
if ( sym - > st_shndx > = SHN_LORESERVE )
continue ;
2006-02-19 09:56:18 +01:00
secname = secstrings + sechdrs [ sym - > st_shndx ] . sh_name ;
2006-02-17 22:42:02 +01:00
if ( section ( secname ) )
warn_sec_mismatch ( modname , name , elf , sym , r ) ;
}
}
}
/**
* Functions used only during module init is marked __init and is stored in
* a . init . text section . Likewise data is marked __initdata and stored in
* a . init . data section .
* If this section is one of these sections return 1
* See include / linux / init . h for the details
* */
static int init_section ( const char * name )
{
if ( strcmp ( name , " .init " ) = = 0 )
return 1 ;
if ( strncmp ( name , " .init. " , strlen ( " .init. " ) ) = = 0 )
return 1 ;
return 0 ;
}
/**
* Identify sections from which references to a . init section is OK .
2006-03-03 16:46:04 +01:00
*
2006-02-17 22:42:02 +01:00
* Unfortunately references to read only data that referenced . init
* sections had to be excluded . Almost all of these are false
* positives , they are created by gcc . The downside of excluding rodata
* is that there really are some user references from rodata to
* init code , e . g . drivers / video / vgacon . c :
2006-03-03 16:46:04 +01:00
*
2006-02-17 22:42:02 +01:00
* const struct consw vga_con = {
* con_startup : vgacon_startup ,
*
* where vgacon_startup is __init . If you want to wade through the false
* positives , take out the check for rodata .
* */
static int init_section_ref_ok ( const char * name )
{
const char * * s ;
/* Absolute section names */
const char * namelist1 [ ] = {
" .init " ,
2006-03-05 00:16:26 +01:00
" .opd " , /* see comment [OPD] at exit_section_ref_ok() */
" .toc1 " , /* used by ppc64 */
2006-02-17 22:42:02 +01:00
" .stab " ,
" .rodata " ,
" .text.lock " ,
2006-03-05 00:16:26 +01:00
" __bug_table " , /* used by powerpc for BUG() */
2006-02-17 22:42:02 +01:00
" .pci_fixup_header " ,
" .pci_fixup_final " ,
" .pdr " ,
" __param " ,
NULL
} ;
/* Start of section names */
const char * namelist2 [ ] = {
" .init. " ,
" .altinstructions " ,
" .eh_frame " ,
" .debug " ,
NULL
} ;
2006-02-22 21:24:50 +01:00
/* part of section name */
const char * namelist3 [ ] = {
" .unwind " , /* sample: IA_64.unwind.init.text */
NULL
} ;
2006-02-17 22:42:02 +01:00
for ( s = namelist1 ; * s ; s + + )
if ( strcmp ( * s , name ) = = 0 )
return 1 ;
2006-03-03 16:46:04 +01:00
for ( s = namelist2 ; * s ; s + + )
2006-02-17 22:42:02 +01:00
if ( strncmp ( * s , name , strlen ( * s ) ) = = 0 )
return 1 ;
2006-03-03 16:46:04 +01:00
for ( s = namelist3 ; * s ; s + + )
2006-03-05 11:34:15 +01:00
if ( strstr ( name , * s ) ! = NULL )
2006-02-22 21:24:50 +01:00
return 1 ;
2006-02-17 22:42:02 +01:00
return 0 ;
}
/*
* Functions used only during module exit is marked __exit and is stored in
* a . exit . text section . Likewise data is marked __exitdata and stored in
* a . exit . data section .
* If this section is one of these sections return 1
* See include / linux / init . h for the details
* */
static int exit_section ( const char * name )
{
if ( strcmp ( name , " .exit.text " ) = = 0 )
return 1 ;
if ( strcmp ( name , " .exit.data " ) = = 0 )
return 1 ;
return 0 ;
2006-03-03 16:46:04 +01:00
2006-02-17 22:42:02 +01:00
}
/*
* Identify sections from which references to a . exit section is OK .
2006-03-03 16:46:04 +01:00
*
2006-02-17 22:42:02 +01:00
* [ OPD ] Keith Ownes < kaos @ sgi . com > commented :
* For our future { in } sanity , add a comment that this is the ppc . opd
* section , not the ia64 . opd section .
* ia64 . opd should not point to discarded sections .
* */
static int exit_section_ref_ok ( const char * name )
{
const char * * s ;
/* Absolute section names */
const char * namelist1 [ ] = {
" .exit.text " ,
" .exit.data " ,
" .init.text " ,
" .opd " , /* See comment [OPD] */
2006-03-05 00:16:26 +01:00
" .toc1 " , /* used by ppc64 */
2006-02-17 22:42:02 +01:00
" .altinstructions " ,
" .pdr " ,
2006-03-05 00:16:26 +01:00
" __bug_table " , /* used by powerpc for BUG() */
2006-02-17 22:42:02 +01:00
" .exitcall.exit " ,
" .eh_frame " ,
" .stab " ,
NULL
} ;
/* Start of section names */
const char * namelist2 [ ] = {
" .debug " ,
NULL
} ;
2006-02-22 21:24:50 +01:00
/* part of section name */
const char * namelist3 [ ] = {
" .unwind " , /* Sample: IA_64.unwind.exit.text */
NULL
} ;
2006-03-03 16:46:04 +01:00
2006-02-17 22:42:02 +01:00
for ( s = namelist1 ; * s ; s + + )
if ( strcmp ( * s , name ) = = 0 )
return 1 ;
2006-03-03 16:46:04 +01:00
for ( s = namelist2 ; * s ; s + + )
2006-02-17 22:42:02 +01:00
if ( strncmp ( * s , name , strlen ( * s ) ) = = 0 )
return 1 ;
2006-03-03 16:46:04 +01:00
for ( s = namelist3 ; * s ; s + + )
2006-03-05 11:34:15 +01:00
if ( strstr ( name , * s ) ! = NULL )
2006-02-22 21:24:50 +01:00
return 1 ;
2006-02-17 22:42:02 +01:00
return 0 ;
}
2006-01-28 17:19:35 +01:00
static void read_symbols ( char * modname )
2005-04-16 15:20:36 -07:00
{
const char * symname ;
char * version ;
struct module * mod ;
struct elf_info info = { } ;
Elf_Sym * sym ;
parse_elf ( & info , modname ) ;
mod = new_module ( modname ) ;
/* When there's no vmlinux, don't print warnings about
* unresolved symbols ( since there ' ll be too many ; ) */
if ( is_vmlinux ( modname ) ) {
have_vmlinux = 1 ;
mod - > skip = 1 ;
}
for ( sym = info . symtab_start ; sym < info . symtab_stop ; sym + + ) {
symname = info . strtab + sym - > st_name ;
handle_modversions ( mod , & info , sym , symname ) ;
handle_moddevtable ( mod , & info , sym , symname ) ;
}
2006-02-17 22:42:02 +01:00
check_sec_ref ( mod , modname , & info , init_section , init_section_ref_ok ) ;
check_sec_ref ( mod , modname , & info , exit_section , exit_section_ref_ok ) ;
2005-04-16 15:20:36 -07:00
version = get_modinfo ( info . modinfo , info . modinfo_len , " version " ) ;
if ( version )
maybe_frob_rcs_version ( modname , version , info . modinfo ,
version - ( char * ) info . hdr ) ;
if ( version | | ( all_versions & & ! is_vmlinux ( modname ) ) )
get_src_version ( modname , mod - > srcversion ,
sizeof ( mod - > srcversion ) - 1 ) ;
parse_elf_finish ( & info ) ;
/* Our trick to get versioning for struct_module - it's
* never passed as an argument to an exported function , so
* the automatic versioning doesn ' t pick it up , but it ' s really
* important anyhow */
if ( modversions )
mod - > unres = alloc_symbol ( " struct_module " , 0 , mod - > unres ) ;
}
# define SZ 500
/* We first write the generated file into memory using the
* following helper , then compare to the file on disk and
* only update the later if anything changed */
2006-01-28 17:19:35 +01:00
void __attribute__ ( ( format ( printf , 2 , 3 ) ) ) buf_printf ( struct buffer * buf ,
const char * fmt , . . . )
2005-04-16 15:20:36 -07:00
{
char tmp [ SZ ] ;
int len ;
va_list ap ;
2006-03-03 16:46:04 +01:00
2005-04-16 15:20:36 -07:00
va_start ( ap , fmt ) ;
len = vsnprintf ( tmp , SZ , fmt , ap ) ;
2006-03-16 23:04:08 -08:00
buf_write ( buf , tmp , len ) ;
2005-04-16 15:20:36 -07:00
va_end ( ap ) ;
}
2006-01-28 17:19:35 +01:00
void buf_write ( struct buffer * buf , const char * s , int len )
2005-04-16 15:20:36 -07:00
{
if ( buf - > size - buf - > pos < len ) {
2006-03-16 23:04:08 -08:00
buf - > size + = len + SZ ;
2005-04-16 15:20:36 -07:00
buf - > p = realloc ( buf - > p , buf - > size ) ;
}
strncpy ( buf - > p + buf - > pos , s , len ) ;
buf - > pos + = len ;
}
2006-01-28 17:19:35 +01:00
/**
* Header for the generated file
* */
static void add_header ( struct buffer * b , struct module * mod )
2005-04-16 15:20:36 -07:00
{
buf_printf ( b , " #include <linux/module.h> \n " ) ;
buf_printf ( b , " #include <linux/vermagic.h> \n " ) ;
buf_printf ( b , " #include <linux/compiler.h> \n " ) ;
buf_printf ( b , " \n " ) ;
buf_printf ( b , " MODULE_INFO(vermagic, VERMAGIC_STRING); \n " ) ;
buf_printf ( b , " \n " ) ;
buf_printf ( b , " struct module __this_module \n " ) ;
buf_printf ( b , " __attribute__((section( \" .gnu.linkonce.this_module \" ))) = { \n " ) ;
2005-09-23 08:42:11 +04:00
buf_printf ( b , " .name = KBUILD_MODNAME, \n " ) ;
2005-04-16 15:20:36 -07:00
if ( mod - > has_init )
buf_printf ( b , " .init = init_module, \n " ) ;
if ( mod - > has_cleanup )
buf_printf ( b , " #ifdef CONFIG_MODULE_UNLOAD \n "
" .exit = cleanup_module, \n "
" #endif \n " ) ;
buf_printf ( b , " }; \n " ) ;
}
2006-01-28 17:19:35 +01:00
/**
* Record CRCs for unresolved symbols
* */
static void add_versions ( struct buffer * b , struct module * mod )
2005-04-16 15:20:36 -07:00
{
struct symbol * s , * exp ;
for ( s = mod - > unres ; s ; s = s - > next ) {
exp = find_symbol ( s - > name ) ;
if ( ! exp | | exp - > module = = mod ) {
if ( have_vmlinux & & ! s - > weak )
2006-01-28 16:57:26 +01:00
warn ( " \" %s \" [%s.ko] undefined! \n " ,
s - > name , mod - > name ) ;
2005-04-16 15:20:36 -07:00
continue ;
}
s - > module = exp - > module ;
s - > crc_valid = exp - > crc_valid ;
s - > crc = exp - > crc ;
}
if ( ! modversions )
return ;
buf_printf ( b , " \n " ) ;
buf_printf ( b , " static const struct modversion_info ____versions[] \n " ) ;
buf_printf ( b , " __attribute_used__ \n " ) ;
buf_printf ( b , " __attribute__((section( \" __versions \" ))) = { \n " ) ;
for ( s = mod - > unres ; s ; s = s - > next ) {
if ( ! s - > module ) {
continue ;
}
if ( ! s - > crc_valid ) {
2006-01-28 16:57:26 +01:00
warn ( " \" %s \" [%s.ko] has no CRC! \n " ,
2005-04-16 15:20:36 -07:00
s - > name , mod - > name ) ;
continue ;
}
buf_printf ( b , " \t { %#8x, \" %s \" }, \n " , s - > crc , s - > name ) ;
}
buf_printf ( b , " }; \n " ) ;
}
2006-01-28 17:19:35 +01:00
static void add_depends ( struct buffer * b , struct module * mod ,
struct module * modules )
2005-04-16 15:20:36 -07:00
{
struct symbol * s ;
struct module * m ;
int first = 1 ;
for ( m = modules ; m ; m = m - > next ) {
m - > seen = is_vmlinux ( m - > name ) ;
}
buf_printf ( b , " \n " ) ;
buf_printf ( b , " static const char __module_depends[] \n " ) ;
buf_printf ( b , " __attribute_used__ \n " ) ;
buf_printf ( b , " __attribute__((section( \" .modinfo \" ))) = \n " ) ;
buf_printf ( b , " \" depends= " ) ;
for ( s = mod - > unres ; s ; s = s - > next ) {
if ( ! s - > module )
continue ;
if ( s - > module - > seen )
continue ;
s - > module - > seen = 1 ;
buf_printf ( b , " %s%s " , first ? " " : " , " ,
strrchr ( s - > module - > name , ' / ' ) + 1 ) ;
first = 0 ;
}
buf_printf ( b , " \" ; \n " ) ;
}
2006-01-28 17:19:35 +01:00
static void add_srcversion ( struct buffer * b , struct module * mod )
2005-04-16 15:20:36 -07:00
{
if ( mod - > srcversion [ 0 ] ) {
buf_printf ( b , " \n " ) ;
buf_printf ( b , " MODULE_INFO(srcversion, \" %s \" ); \n " ,
mod - > srcversion ) ;
}
}
2006-01-28 17:19:35 +01:00
static void write_if_changed ( struct buffer * b , const char * fname )
2005-04-16 15:20:36 -07:00
{
char * tmp ;
FILE * file ;
struct stat st ;
file = fopen ( fname , " r " ) ;
if ( ! file )
goto write ;
if ( fstat ( fileno ( file ) , & st ) < 0 )
goto close_write ;
if ( st . st_size ! = b - > pos )
goto close_write ;
tmp = NOFAIL ( malloc ( b - > pos ) ) ;
if ( fread ( tmp , 1 , b - > pos , file ) ! = b - > pos )
goto free_write ;
if ( memcmp ( tmp , b - > p , b - > pos ) ! = 0 )
goto free_write ;
free ( tmp ) ;
fclose ( file ) ;
return ;
free_write :
free ( tmp ) ;
close_write :
fclose ( file ) ;
write :
file = fopen ( fname , " w " ) ;
if ( ! file ) {
perror ( fname ) ;
exit ( 1 ) ;
}
if ( fwrite ( b - > p , 1 , b - > pos , file ) ! = b - > pos ) {
perror ( fname ) ;
exit ( 1 ) ;
}
fclose ( file ) ;
}
2006-01-28 22:15:55 +01:00
static void read_dump ( const char * fname , unsigned int kernel )
2005-04-16 15:20:36 -07:00
{
unsigned long size , pos = 0 ;
void * file = grab_file ( fname , & size ) ;
char * line ;
if ( ! file )
/* No symbol versions, silently ignore */
return ;
while ( ( line = get_next_line ( & pos , file , size ) ) ) {
char * symname , * modname , * d ;
unsigned int crc ;
struct module * mod ;
2006-01-28 22:15:55 +01:00
struct symbol * s ;
2005-04-16 15:20:36 -07:00
if ( ! ( symname = strchr ( line , ' \t ' ) ) )
goto fail ;
* symname + + = ' \0 ' ;
if ( ! ( modname = strchr ( symname , ' \t ' ) ) )
goto fail ;
* modname + + = ' \0 ' ;
if ( strchr ( modname , ' \t ' ) )
goto fail ;
crc = strtoul ( line , & d , 16 ) ;
if ( * symname = = ' \0 ' | | * modname = = ' \0 ' | | * d ! = ' \0 ' )
goto fail ;
if ( ! ( mod = find_module ( modname ) ) ) {
if ( is_vmlinux ( modname ) ) {
have_vmlinux = 1 ;
}
mod = new_module ( NOFAIL ( strdup ( modname ) ) ) ;
mod - > skip = 1 ;
}
2006-01-28 22:15:55 +01:00
s = sym_add_exported ( symname , mod ) ;
2006-01-28 22:22:33 +01:00
s - > kernel = kernel ;
s - > preloaded = 1 ;
2006-01-28 22:15:55 +01:00
sym_update_crc ( symname , mod , crc ) ;
2005-04-16 15:20:36 -07:00
}
return ;
fail :
fatal ( " parse error in symbol dump file \n " ) ;
}
2006-01-28 22:15:55 +01:00
/* For normal builds always dump all symbols.
* For external modules only dump symbols
* that are not read from kernel Module . symvers .
* */
static int dump_sym ( struct symbol * sym )
{
if ( ! external_module )
return 1 ;
if ( sym - > vmlinux | | sym - > kernel )
return 0 ;
return 1 ;
}
2006-03-03 16:46:04 +01:00
2006-01-28 17:19:35 +01:00
static void write_dump ( const char * fname )
2005-04-16 15:20:36 -07:00
{
struct buffer buf = { } ;
struct symbol * symbol ;
int n ;
for ( n = 0 ; n < SYMBOL_HASH_SIZE ; n + + ) {
symbol = symbolhash [ n ] ;
while ( symbol ) {
2006-01-28 22:15:55 +01:00
if ( dump_sym ( symbol ) )
buf_printf ( & buf , " 0x%08x \t %s \t %s \n " ,
2006-03-03 16:46:04 +01:00
symbol - > crc , symbol - > name ,
2006-01-28 22:15:55 +01:00
symbol - > module - > name ) ;
2005-04-16 15:20:36 -07:00
symbol = symbol - > next ;
}
}
write_if_changed ( & buf , fname ) ;
}
2006-01-28 17:19:35 +01:00
int main ( int argc , char * * argv )
2005-04-16 15:20:36 -07:00
{
struct module * mod ;
struct buffer buf = { } ;
char fname [ SZ ] ;
2006-01-28 22:15:55 +01:00
char * kernel_read = NULL , * module_read = NULL ;
char * dump_write = NULL ;
2005-04-16 15:20:36 -07:00
int opt ;
2006-01-28 22:15:55 +01:00
while ( ( opt = getopt ( argc , argv , " i:I:mo:a " ) ) ! = - 1 ) {
2005-04-16 15:20:36 -07:00
switch ( opt ) {
case ' i ' :
2006-01-28 22:15:55 +01:00
kernel_read = optarg ;
break ;
case ' I ' :
module_read = optarg ;
external_module = 1 ;
2005-04-16 15:20:36 -07:00
break ;
case ' m ' :
modversions = 1 ;
break ;
case ' o ' :
dump_write = optarg ;
break ;
case ' a ' :
all_versions = 1 ;
break ;
default :
exit ( 1 ) ;
}
}
2006-01-28 22:15:55 +01:00
if ( kernel_read )
read_dump ( kernel_read , 1 ) ;
if ( module_read )
read_dump ( module_read , 0 ) ;
2005-04-16 15:20:36 -07:00
while ( optind < argc ) {
read_symbols ( argv [ optind + + ] ) ;
}
for ( mod = modules ; mod ; mod = mod - > next ) {
if ( mod - > skip )
continue ;
buf . pos = 0 ;
add_header ( & buf , mod ) ;
add_versions ( & buf , mod ) ;
add_depends ( & buf , mod , modules ) ;
add_moddevtable ( & buf , mod ) ;
add_srcversion ( & buf , mod ) ;
sprintf ( fname , " %s.mod.c " , mod - > name ) ;
write_if_changed ( & buf , fname ) ;
}
if ( dump_write )
write_dump ( dump_write ) ;
return 0 ;
}