2008-06-21 02:31:01 +10:00
/* Kernel module help for powerpc.
Copyright ( C ) 2001 , 2003 Rusty Russell IBM Corporation .
Copyright ( C ) 2008 Freescale Semiconductor , Inc .
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/elf.h>
# include <linux/moduleloader.h>
# include <linux/err.h>
# include <linux/vmalloc.h>
# include <linux/bug.h>
# include <asm/module.h>
2016-12-24 11:46:01 -08:00
# include <linux/uaccess.h>
2008-06-21 02:31:01 +10:00
# include <asm/firmware.h>
# include <linux/sort.h>
2013-10-28 09:20:51 -05:00
# include <asm/setup.h>
2008-06-21 02:31:01 +10:00
2016-09-06 15:32:40 +10:00
static LIST_HEAD ( module_bug_list ) ;
2008-06-21 02:31:01 +10:00
static const Elf_Shdr * find_section ( const Elf_Ehdr * hdr ,
const Elf_Shdr * sechdrs ,
const char * name )
{
char * secstrings ;
unsigned int i ;
secstrings = ( char * ) hdr + sechdrs [ hdr - > e_shstrndx ] . sh_offset ;
for ( i = 1 ; i < hdr - > e_shnum ; i + + )
if ( strcmp ( secstrings + sechdrs [ i ] . sh_name , name ) = = 0 )
return & sechdrs [ i ] ;
return NULL ;
}
int module_finalize ( const Elf_Ehdr * hdr ,
const Elf_Shdr * sechdrs , struct module * me )
{
const Elf_Shdr * sect ;
2016-03-03 15:26:54 +11:00
int rc ;
rc = module_finalize_ftrace ( me , sechdrs ) ;
if ( rc )
return rc ;
2008-06-21 02:31:01 +10:00
/* Apply feature fixups */
sect = find_section ( hdr , sechdrs , " __ftr_fixup " ) ;
if ( sect ! = NULL )
do_feature_fixups ( cur_cpu_spec - > cpu_features ,
( void * ) sect - > sh_addr ,
( void * ) sect - > sh_addr + sect - > sh_size ) ;
2008-12-18 19:13:32 +00:00
sect = find_section ( hdr , sechdrs , " __mmu_ftr_fixup " ) ;
if ( sect ! = NULL )
do_feature_fixups ( cur_cpu_spec - > mmu_features ,
( void * ) sect - > sh_addr ,
( void * ) sect - > sh_addr + sect - > sh_size ) ;
2008-06-21 02:31:01 +10:00
# ifdef CONFIG_PPC64
sect = find_section ( hdr , sechdrs , " __fw_ftr_fixup " ) ;
if ( sect ! = NULL )
do_feature_fixups ( powerpc_firmware_features ,
( void * ) sect - > sh_addr ,
( void * ) sect - > sh_addr + sect - > sh_size ) ;
2018-07-28 09:06:34 +10:00
# endif /* CONFIG_PPC64 */
2018-04-24 14:15:56 +10:00
2018-05-29 12:21:00 +05:30
# ifdef PPC64_ELF_ABI_v1
sect = find_section ( hdr , sechdrs , " .opd " ) ;
if ( sect ! = NULL ) {
me - > arch . start_opd = sect - > sh_addr ;
me - > arch . end_opd = sect - > sh_addr + sect - > sh_size ;
}
# endif /* PPC64_ELF_ABI_v1 */
2018-07-28 09:06:34 +10:00
# ifdef CONFIG_PPC_BARRIER_NOSPEC
2018-04-24 14:15:56 +10:00
sect = find_section ( hdr , sechdrs , " __spec_barrier_fixup " ) ;
if ( sect ! = NULL )
do_barrier_nospec_fixups_range ( barrier_nospec_enabled ,
( void * ) sect - > sh_addr ,
( void * ) sect - > sh_addr + sect - > sh_size ) ;
2018-07-28 09:06:34 +10:00
# endif /* CONFIG_PPC_BARRIER_NOSPEC */
2008-06-21 02:31:01 +10:00
2008-07-02 01:16:40 +10:00
sect = find_section ( hdr , sechdrs , " __lwsync_fixup " ) ;
if ( sect ! = NULL )
do_lwsync_fixups ( cur_cpu_spec - > cpu_features ,
( void * ) sect - > sh_addr ,
( void * ) sect - > sh_addr + sect - > sh_size ) ;
2008-06-21 02:31:01 +10:00
return 0 ;
}