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
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# 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
void * module_alloc ( unsigned long size )
{
if ( size = = 0 )
return NULL ;
2008-08-04 13:34:29 +09:00
return vmalloc_exec ( size ) ;
2005-04-16 15:20:36 -07:00
}
/* Free memory returned from module_alloc */
void module_free ( struct module * mod , void * module_region )
{
vfree ( module_region ) ;
}
/* We don't need anything special. */
int module_frob_arch_sections ( Elf_Ehdr * hdr ,
Elf_Shdr * sechdrs ,
char * secstrings ,
struct module * mod )
{
return 0 ;
}
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 ;
2007-11-20 15:16:25 +09:00
# ifdef CONFIG_SUPERH64
/* For text addresses, bit2 of the st_other field indicates
* whether the symbol is SHmedia ( 1 ) or SHcompact ( 0 ) . If
* SHmedia , the LSB of the symbol needs to be asserted
* for the CPU to be in SHmedia mode when it starts executing
* the branch target . */
2009-05-09 18:03:37 +09:00
relocation | = ! ! ( sym - > st_other & 4 ) ;
2007-11-20 15:16:25 +09:00
# endif
2005-04-16 15:20:36 -07:00
switch ( ELF32_R_TYPE ( rel [ i ] . r_info ) ) {
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 apply_relocate ( Elf32_Shdr * sechdrs ,
const char * strtab ,
unsigned int symindex ,
unsigned int relsec ,
struct module * me )
{
printk ( KERN_ERR " module %s: REL RELOCATION unsupported \n " ,
me - > name ) ;
return - ENOEXEC ;
}
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
}