2018-12-28 00:32:24 -08:00
// SPDX-License-Identifier: GPL-2.0+
2005-04-16 15:20:36 -07:00
/* Kernel module help for SH.
2007-11-20 15:16:25 +09:00
SHcompact version by Kaz Kojima and Paul Mundt .
SHmedia bits :
Copyright 2004 SuperH ( UK ) Ltd
Author : Richard Curnow
Based on the sh version , and on code from the sh64 - specific parts of
modutils , originally written by Richard Curnow and Ben Gaster .
2005-04-16 15:20:36 -07:00
*/
# include <linux/moduleloader.h>
# include <linux/elf.h>
# include <linux/vmalloc.h>
2008-08-04 13:32:04 +09:00
# include <linux/bug.h>
2005-04-16 15:20:36 -07:00
# include <linux/fs.h>
# include <linux/string.h>
# include <linux/kernel.h>
2008-05-28 16:38:17 -07:00
# include <asm/unaligned.h>
2009-10-09 23:20:54 +01:00
# include <asm/dwarf.h>
2005-04-16 15:20:36 -07:00
int apply_relocate_add ( Elf32_Shdr * sechdrs ,
const char * strtab ,
unsigned int symindex ,
unsigned int relsec ,
struct module * me )
{
unsigned int i ;
Elf32_Rela * rel = ( void * ) sechdrs [ relsec ] . sh_addr ;
Elf32_Sym * sym ;
Elf32_Addr relocation ;
uint32_t * location ;
uint32_t value ;
2007-11-20 15:16:25 +09:00
pr_debug ( " Applying relocate section %u to %u \n " , relsec ,
sechdrs [ relsec ] . sh_info ) ;
2005-04-16 15:20:36 -07:00
for ( i = 0 ; i < sechdrs [ relsec ] . sh_size / sizeof ( * rel ) ; i + + ) {
/* This is where to make the change */
location = ( void * ) sechdrs [ sechdrs [ relsec ] . sh_info ] . sh_addr
+ rel [ i ] . r_offset ;
/* This is the symbol it is referring to. Note that all
undefined symbols have been resolved . */
sym = ( Elf32_Sym * ) sechdrs [ symindex ] . sh_addr
+ ELF32_R_SYM ( rel [ i ] . r_info ) ;
relocation = sym - > st_value + rel [ i ] . r_addend ;
switch ( ELF32_R_TYPE ( rel [ i ] . r_info ) ) {
2011-05-23 17:09:30 +09:00
case R_SH_NONE :
break ;
2005-04-16 15:20:36 -07:00
case R_SH_DIR32 :
2008-05-28 16:38:17 -07:00
value = get_unaligned ( location ) ;
2005-04-16 15:20:36 -07:00
value + = relocation ;
2008-05-28 16:38:17 -07:00
put_unaligned ( value , location ) ;
2005-04-16 15:20:36 -07:00
break ;
case R_SH_REL32 :
2007-11-20 15:16:25 +09:00
relocation = ( relocation - ( Elf32_Addr ) location ) ;
2008-05-28 16:38:17 -07:00
value = get_unaligned ( location ) ;
2005-04-16 15:20:36 -07:00
value + = relocation ;
2008-05-28 16:38:17 -07:00
put_unaligned ( value , location ) ;
2007-11-20 15:16:25 +09:00
break ;
case R_SH_IMM_LOW16 :
* location = ( * location & ~ 0x3fffc00 ) |
( ( relocation & 0xffff ) < < 10 ) ;
break ;
case R_SH_IMM_MEDLOW16 :
* location = ( * location & ~ 0x3fffc00 ) |
( ( ( relocation > > 16 ) & 0xffff ) < < 10 ) ;
break ;
case R_SH_IMM_LOW16_PCREL :
relocation - = ( Elf32_Addr ) location ;
* location = ( * location & ~ 0x3fffc00 ) |
( ( relocation & 0xffff ) < < 10 ) ;
break ;
case R_SH_IMM_MEDLOW16_PCREL :
relocation - = ( Elf32_Addr ) location ;
* location = ( * location & ~ 0x3fffc00 ) |
( ( ( relocation > > 16 ) & 0xffff ) < < 10 ) ;
2005-04-16 15:20:36 -07:00
break ;
default :
printk ( KERN_ERR " module %s: Unknown relocation: %u \n " ,
me - > name , ELF32_R_TYPE ( rel [ i ] . r_info ) ) ;
return - ENOEXEC ;
}
}
return 0 ;
}
int module_finalize ( const Elf_Ehdr * hdr ,
const Elf_Shdr * sechdrs ,
struct module * me )
{
2009-10-13 13:32:19 +09:00
int ret = 0 ;
2009-10-09 23:20:54 +01:00
2009-10-13 13:32:19 +09:00
ret | = module_dwarf_finalize ( hdr , sechdrs , me ) ;
2009-10-09 23:20:54 +01:00
2009-10-13 13:32:19 +09:00
return ret ;
2005-04-16 15:20:36 -07:00
}
void module_arch_cleanup ( struct module * mod )
{
2009-10-13 13:32:19 +09:00
module_dwarf_cleanup ( mod ) ;
2005-04-16 15:20:36 -07:00
}