2019-05-31 11:09:38 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2010-10-13 23:12:54 +04:00
/*
* recordmcount . c : construct a table of the locations of calls to ' mcount '
* so that ftrace can find them quickly .
* Copyright 2009 John F . Reiser < jreiser @ BitWagon . com > . All rights reserved .
*
* Restructured to fit Linux format , as well as other updates :
* Copyright 2010 Steven Rostedt < srostedt @ redhat . com > , Red Hat Inc .
*/
/*
* Strategy : alter the . o file in - place .
*
* Append a new STRTAB that has the new section names , followed by a new array
* ElfXX_Shdr [ ] that has the new section headers , followed by the section
* contents for __mcount_loc and its relocations . The old shstrtab strings ,
* and the old ElfXX_Shdr [ ] array , remain as " garbage " ( commonly , a couple
* kilobytes . ) Subsequent processing by / bin / ld ( or the kernel module loader )
* will ignore the garbage regions , because they are not designated by the
* new . e_shoff nor the new ElfXX_Shdr [ ] . [ In order to remove the garbage ,
* then use " ld -r " to create a new file that omits the garbage . ]
*/
# include <sys/types.h>
# include <sys/mman.h>
# include <sys/stat.h>
2011-04-13 02:53:25 +04:00
# include <getopt.h>
2010-10-13 23:12:54 +04:00
# include <elf.h>
# include <fcntl.h>
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
# include <unistd.h>
2014-04-30 13:54:32 +04:00
# ifndef EM_AARCH64
# define EM_AARCH64 183
2015-10-30 11:31:04 +03:00
# define R_AARCH64_NONE 0
2014-04-30 13:54:32 +04:00
# define R_AARCH64_ABS64 257
# endif
2010-10-13 23:12:54 +04:00
static int fd_map ; /* File descriptor for file being modified. */
static int mmap_failed ; /* Boolean flag. */
static char gpfx ; /* prefix for global symbol name (sometimes '_') */
static struct stat sb ; /* Remember .st_size, etc. */
2010-11-30 19:36:48 +03:00
static const char * altmcount ; /* alternate mcount symbol name */
2011-04-13 02:53:25 +04:00
static int warn_on_notrace_sect ; /* warn when section has mcount not being recorded */
2015-12-16 00:06:10 +03:00
static void * file_map ; /* pointer of the mapped file */
static void * file_end ; /* pointer to the end of the mapped file */
static int file_updated ; /* flag to state file was changed */
static void * file_ptr ; /* current file pointer location */
2019-07-31 21:24:16 +03:00
2015-12-16 00:06:10 +03:00
static void * file_append ; /* added to the end of the file */
static size_t file_append_size ; /* how much is added to end of file */
2010-10-13 23:12:54 +04:00
/* Per-file resource cleanup when multiple files. */
2019-07-31 21:24:16 +03:00
static void file_append_cleanup ( void )
{
free ( file_append ) ;
file_append = NULL ;
file_append_size = 0 ;
file_updated = 0 ;
}
static void mmap_cleanup ( void )
2010-10-13 23:12:54 +04:00
{
if ( ! mmap_failed )
2015-12-16 00:06:10 +03:00
munmap ( file_map , sb . st_size ) ;
2010-10-13 23:12:54 +04:00
else
2015-12-16 00:06:10 +03:00
free ( file_map ) ;
file_map = NULL ;
2010-10-13 23:12:54 +04:00
}
2019-07-25 00:04:56 +03:00
/* ulseek, uwrite, ...: Check return value for errors. */
2010-10-13 23:12:54 +04:00
2019-07-31 21:24:13 +03:00
static off_t ulseek ( off_t const offset , int const whence )
2010-10-13 23:12:54 +04:00
{
2015-12-16 00:06:10 +03:00
switch ( whence ) {
case SEEK_SET :
file_ptr = file_map + offset ;
break ;
case SEEK_CUR :
file_ptr + = offset ;
break ;
case SEEK_END :
file_ptr = file_map + ( sb . st_size - offset ) ;
break ;
}
if ( file_ptr < file_map ) {
fprintf ( stderr , " lseek: seek before file \n " ) ;
2019-07-31 21:24:12 +03:00
return - 1 ;
2010-10-13 23:12:54 +04:00
}
2015-12-16 00:06:10 +03:00
return file_ptr - file_map ;
2010-10-13 23:12:54 +04:00
}
2019-07-31 21:24:13 +03:00
static ssize_t uwrite ( void const * const buf , size_t const count )
2010-10-13 23:12:54 +04:00
{
2015-12-16 00:06:10 +03:00
size_t cnt = count ;
off_t idx = 0 ;
file_updated = 1 ;
if ( file_ptr + count > = file_end ) {
off_t aoffset = ( file_ptr + count ) - file_end ;
if ( aoffset > file_append_size ) {
file_append = realloc ( file_append , aoffset ) ;
file_append_size = aoffset ;
}
if ( ! file_append ) {
perror ( " write " ) ;
2019-07-31 21:24:16 +03:00
file_append_cleanup ( ) ;
mmap_cleanup ( ) ;
2019-07-31 21:24:12 +03:00
return - 1 ;
2015-12-16 00:06:10 +03:00
}
if ( file_ptr < file_end ) {
cnt = file_end - file_ptr ;
} else {
cnt = 0 ;
idx = aoffset - count ;
}
2010-10-13 23:12:54 +04:00
}
2015-12-16 00:06:10 +03:00
if ( cnt )
memcpy ( file_ptr , buf , cnt ) ;
if ( cnt < count )
memcpy ( file_append + idx , buf + cnt , count - cnt ) ;
file_ptr + = count ;
return count ;
2010-10-13 23:12:54 +04:00
}
2019-07-31 21:24:13 +03:00
static void * umalloc ( size_t size )
2010-10-13 23:12:54 +04:00
{
void * const addr = malloc ( size ) ;
2011-04-06 21:21:17 +04:00
if ( addr = = 0 ) {
2010-10-13 23:12:54 +04:00
fprintf ( stderr , " malloc failed: %zu bytes \n " , size ) ;
2019-07-31 21:24:16 +03:00
file_append_cleanup ( ) ;
mmap_cleanup ( ) ;
2019-07-31 21:24:12 +03:00
return NULL ;
2010-10-13 23:12:54 +04:00
}
return addr ;
}
2019-07-31 21:24:16 +03:00
/*
* Get the whole file as a programming convenience in order to avoid
* malloc + lseek + read + free of many pieces . If successful , then mmap
* avoids copying unused pieces ; else just read the whole file .
* Open for both read and write ; new info will be appended to the file .
* Use MAP_PRIVATE so that a few changes to the in - memory ElfXX_Ehdr
* do not propagate to the file until an explicit overwrite at the last .
* This preserves most aspects of consistency ( all except . st_size )
* for simultaneous readers of the file while we are appending to it .
* However , multiple writers still are bad . We choose not to use
* locking because it is expensive and the use case of kernel build
* makes multiple writers unlikely .
*/
static void * mmap_file ( char const * fname )
{
/* Avoid problems if early cleanup() */
fd_map = - 1 ;
mmap_failed = 1 ;
file_map = NULL ;
file_ptr = NULL ;
file_updated = 0 ;
sb . st_size = 0 ;
fd_map = open ( fname , O_RDONLY ) ;
if ( fd_map < 0 ) {
perror ( fname ) ;
return NULL ;
}
if ( fstat ( fd_map , & sb ) < 0 ) {
perror ( fname ) ;
goto out ;
}
if ( ! S_ISREG ( sb . st_mode ) ) {
fprintf ( stderr , " not a regular file: %s \n " , fname ) ;
goto out ;
}
file_map = mmap ( 0 , sb . st_size , PROT_READ | PROT_WRITE , MAP_PRIVATE ,
fd_map , 0 ) ;
if ( file_map = = MAP_FAILED ) {
mmap_failed = 1 ;
file_map = umalloc ( sb . st_size ) ;
if ( ! file_map ) {
perror ( fname ) ;
goto out ;
}
if ( read ( fd_map , file_map , sb . st_size ) ! = sb . st_size ) {
perror ( fname ) ;
free ( file_map ) ;
file_map = NULL ;
goto out ;
}
} else
mmap_failed = 0 ;
out :
close ( fd_map ) ;
fd_map = - 1 ;
file_end = file_map + sb . st_size ;
return file_map ;
}
2011-04-08 11:58:48 +04:00
static unsigned char ideal_nop5_x86_64 [ 5 ] = { 0x0f , 0x1f , 0x44 , 0x00 , 0x00 } ;
static unsigned char ideal_nop5_x86_32 [ 5 ] = { 0x3e , 0x8d , 0x74 , 0x26 , 0x00 } ;
static unsigned char * ideal_nop ;
static char rel_type_nop ;
static int ( * make_nop ) ( void * map , size_t const offset ) ;
static int make_nop_x86 ( void * map , size_t const offset )
{
uint32_t * ptr ;
unsigned char * op ;
/* Confirm we have 0xe8 0x0 0x0 0x0 0x0 */
ptr = map + offset ;
if ( * ptr ! = 0 )
return - 1 ;
op = map + offset - 1 ;
if ( * op ! = 0xe8 )
return - 1 ;
/* convert to nop */
2019-07-31 21:24:12 +03:00
if ( ulseek ( offset - 1 , SEEK_SET ) < 0 )
return - 1 ;
if ( uwrite ( ideal_nop , 5 ) < 0 )
return - 1 ;
2011-04-08 11:58:48 +04:00
return 0 ;
}
2016-10-19 02:42:00 +03:00
static unsigned char ideal_nop4_arm_le [ 4 ] = { 0x00 , 0x00 , 0xa0 , 0xe1 } ; /* mov r0, r0 */
static unsigned char ideal_nop4_arm_be [ 4 ] = { 0xe1 , 0xa0 , 0x00 , 0x00 } ; /* mov r0, r0 */
static unsigned char * ideal_nop4_arm ;
static unsigned char bl_mcount_arm_le [ 4 ] = { 0xfe , 0xff , 0xff , 0xeb } ; /* bl */
static unsigned char bl_mcount_arm_be [ 4 ] = { 0xeb , 0xff , 0xff , 0xfe } ; /* bl */
static unsigned char * bl_mcount_arm ;
static unsigned char push_arm_le [ 4 ] = { 0x04 , 0xe0 , 0x2d , 0xe5 } ; /* push {lr} */
static unsigned char push_arm_be [ 4 ] = { 0xe5 , 0x2d , 0xe0 , 0x04 } ; /* push {lr} */
static unsigned char * push_arm ;
static unsigned char ideal_nop2_thumb_le [ 2 ] = { 0x00 , 0xbf } ; /* nop */
static unsigned char ideal_nop2_thumb_be [ 2 ] = { 0xbf , 0x00 } ; /* nop */
static unsigned char * ideal_nop2_thumb ;
static unsigned char push_bl_mcount_thumb_le [ 6 ] = { 0x00 , 0xb5 , 0xff , 0xf7 , 0xfe , 0xff } ; /* push {lr}, bl */
static unsigned char push_bl_mcount_thumb_be [ 6 ] = { 0xb5 , 0x00 , 0xf7 , 0xff , 0xff , 0xfe } ; /* push {lr}, bl */
static unsigned char * push_bl_mcount_thumb ;
static int make_nop_arm ( void * map , size_t const offset )
{
char * ptr ;
int cnt = 1 ;
int nop_size ;
size_t off = offset ;
ptr = map + offset ;
if ( memcmp ( ptr , bl_mcount_arm , 4 ) = = 0 ) {
if ( memcmp ( ptr - 4 , push_arm , 4 ) = = 0 ) {
off - = 4 ;
cnt = 2 ;
}
ideal_nop = ideal_nop4_arm ;
nop_size = 4 ;
} else if ( memcmp ( ptr - 2 , push_bl_mcount_thumb , 6 ) = = 0 ) {
cnt = 3 ;
nop_size = 2 ;
off - = 2 ;
ideal_nop = ideal_nop2_thumb ;
} else
return - 1 ;
/* Convert to nop */
2019-07-31 21:24:12 +03:00
if ( ulseek ( off , SEEK_SET ) < 0 )
return - 1 ;
2016-10-19 02:42:00 +03:00
do {
2019-07-31 21:24:12 +03:00
if ( uwrite ( ideal_nop , nop_size ) < 0 )
return - 1 ;
2016-10-19 02:42:00 +03:00
} while ( - - cnt > 0 ) ;
return 0 ;
}
2015-10-30 11:31:04 +03:00
static unsigned char ideal_nop4_arm64 [ 4 ] = { 0x1f , 0x20 , 0x03 , 0xd5 } ;
static int make_nop_arm64 ( void * map , size_t const offset )
{
uint32_t * ptr ;
ptr = map + offset ;
/* bl <_mcount> is 0x94000000 before relocation */
if ( * ptr ! = 0x94000000 )
return - 1 ;
/* Convert to nop */
2019-07-31 21:24:12 +03:00
if ( ulseek ( offset , SEEK_SET ) < 0 )
return - 1 ;
if ( uwrite ( ideal_nop , 4 ) < 0 )
return - 1 ;
2015-10-30 11:31:04 +03:00
return 0 ;
}
2019-07-31 21:24:12 +03:00
static int write_file ( const char * fname )
2015-12-16 00:06:10 +03:00
{
char tmp_file [ strlen ( fname ) + 4 ] ;
size_t n ;
if ( ! file_updated )
2019-07-31 21:24:12 +03:00
return 0 ;
2015-12-16 00:06:10 +03:00
sprintf ( tmp_file , " %s.rc " , fname ) ;
/*
* After reading the entire file into memory , delete it
* and write it back , to prevent weird side effects of modifying
* an object file in place .
*/
fd_map = open ( tmp_file , O_WRONLY | O_TRUNC | O_CREAT , sb . st_mode ) ;
if ( fd_map < 0 ) {
perror ( fname ) ;
2019-07-31 21:24:12 +03:00
return - 1 ;
2015-12-16 00:06:10 +03:00
}
n = write ( fd_map , file_map , sb . st_size ) ;
if ( n ! = sb . st_size ) {
perror ( " write " ) ;
2019-07-31 21:24:12 +03:00
close ( fd_map ) ;
return - 1 ;
2015-12-16 00:06:10 +03:00
}
if ( file_append_size ) {
n = write ( fd_map , file_append , file_append_size ) ;
if ( n ! = file_append_size ) {
perror ( " write " ) ;
2019-07-31 21:24:12 +03:00
close ( fd_map ) ;
return - 1 ;
2015-12-11 15:09:03 +03:00
}
}
2015-12-16 00:06:10 +03:00
close ( fd_map ) ;
if ( rename ( tmp_file , fname ) < 0 ) {
perror ( fname ) ;
2019-07-31 21:24:12 +03:00
return - 1 ;
2015-12-16 00:06:10 +03:00
}
2019-07-31 21:24:12 +03:00
return 0 ;
2010-10-13 23:12:54 +04:00
}
/* w8rev, w8nat, ...: Handle endianness. */
static uint64_t w8rev ( uint64_t const x )
{
return ( ( 0xff & ( x > > ( 0 * 8 ) ) ) < < ( 7 * 8 ) )
| ( ( 0xff & ( x > > ( 1 * 8 ) ) ) < < ( 6 * 8 ) )
| ( ( 0xff & ( x > > ( 2 * 8 ) ) ) < < ( 5 * 8 ) )
| ( ( 0xff & ( x > > ( 3 * 8 ) ) ) < < ( 4 * 8 ) )
| ( ( 0xff & ( x > > ( 4 * 8 ) ) ) < < ( 3 * 8 ) )
| ( ( 0xff & ( x > > ( 5 * 8 ) ) ) < < ( 2 * 8 ) )
| ( ( 0xff & ( x > > ( 6 * 8 ) ) ) < < ( 1 * 8 ) )
| ( ( 0xff & ( x > > ( 7 * 8 ) ) ) < < ( 0 * 8 ) ) ;
}
static uint32_t w4rev ( uint32_t const x )
{
return ( ( 0xff & ( x > > ( 0 * 8 ) ) ) < < ( 3 * 8 ) )
| ( ( 0xff & ( x > > ( 1 * 8 ) ) ) < < ( 2 * 8 ) )
| ( ( 0xff & ( x > > ( 2 * 8 ) ) ) < < ( 1 * 8 ) )
| ( ( 0xff & ( x > > ( 3 * 8 ) ) ) < < ( 0 * 8 ) ) ;
}
static uint32_t w2rev ( uint16_t const x )
{
return ( ( 0xff & ( x > > ( 0 * 8 ) ) ) < < ( 1 * 8 ) )
| ( ( 0xff & ( x > > ( 1 * 8 ) ) ) < < ( 0 * 8 ) ) ;
}
static uint64_t w8nat ( uint64_t const x )
{
return x ;
}
static uint32_t w4nat ( uint32_t const x )
{
return x ;
}
static uint32_t w2nat ( uint16_t const x )
{
return x ;
}
static uint64_t ( * w8 ) ( uint64_t ) ;
static uint32_t ( * w ) ( uint32_t ) ;
static uint32_t ( * w2 ) ( uint16_t ) ;
/* Names of the sections that could contain calls to mcount. */
2019-07-31 21:24:13 +03:00
static int is_mcounted_section_name ( char const * const txtname )
2010-10-13 23:12:54 +04:00
{
2018-11-20 23:19:18 +03:00
return strncmp ( " .text " , txtname , 5 ) = = 0 | |
2017-03-04 00:15:39 +03:00
strcmp ( " .init.text " , txtname ) = = 0 | |
2011-04-06 21:21:17 +04:00
strcmp ( " .ref.text " , txtname ) = = 0 | |
strcmp ( " .sched.text " , txtname ) = = 0 | |
strcmp ( " .spinlock.text " , txtname ) = = 0 | |
strcmp ( " .irqentry.text " , txtname ) = = 0 | |
2016-09-29 01:22:36 +03:00
strcmp ( " .softirqentry.text " , txtname ) = = 0 | |
2011-04-06 22:10:22 +04:00
strcmp ( " .kprobes.text " , txtname ) = = 0 | |
2019-07-25 00:04:55 +03:00
strcmp ( " .cpuidle.text " , txtname ) = = 0 ;
2010-10-13 23:12:54 +04:00
}
2019-07-31 21:24:12 +03:00
static char const * already_has_rel_mcount = " success " ; /* our work here is done! */
2010-10-14 03:06:14 +04:00
/* 32 bit and 64 bit are very similar */
# include "recordmcount.h"
# define RECORD_MCOUNT_64
# include "recordmcount.h"
2010-10-13 23:12:54 +04:00
2010-10-27 14:59:07 +04:00
/* 64-bit EM_MIPS has weird ELF64_Rela.r_info.
* http : //techpubs.sgi.com/library/manuals/4000/007-4658-001/pdf/007-4658-001.pdf
* We interpret Table 29 Relocation Operation ( Elf64_Rel , Elf64_Rela ) [ p .40 ]
* to imply the order of the members ; the spec does not say so .
* typedef unsigned char Elf64_Byte ;
* fails on MIPS64 because their < elf . h > already has it !
*/
typedef uint8_t myElf64_Byte ; /* Type for a 8-bit quantity. */
union mips_r_info {
Elf64_Xword r_info ;
struct {
Elf64_Word r_sym ; /* Symbol index. */
myElf64_Byte r_ssym ; /* Special symbol. */
myElf64_Byte r_type3 ; /* Third relocation. */
myElf64_Byte r_type2 ; /* Second relocation. */
myElf64_Byte r_type ; /* First relocation. */
} r_mips ;
} ;
static uint64_t MIPS64_r_sym ( Elf64_Rel const * rp )
{
return w ( ( ( union mips_r_info ) { . r_info = rp - > r_info } ) . r_mips . r_sym ) ;
}
static void MIPS64_r_info ( Elf64_Rel * const rp , unsigned sym , unsigned type )
{
rp - > r_info = ( ( union mips_r_info ) {
. r_mips = { . r_sym = w ( sym ) , . r_type = type }
} ) . r_info ;
}
2019-07-31 21:24:13 +03:00
static int do_file ( char const * const fname )
2010-10-13 23:12:54 +04:00
{
unsigned int reltype = 0 ;
2019-07-31 21:24:16 +03:00
Elf32_Ehdr * ehdr ;
2019-07-31 21:24:12 +03:00
int rc = - 1 ;
2019-07-31 21:24:16 +03:00
ehdr = mmap_file ( fname ) ;
2019-07-31 21:24:12 +03:00
if ( ! ehdr )
goto out ;
2010-10-13 23:12:54 +04:00
w = w4nat ;
w2 = w2nat ;
w8 = w8nat ;
switch ( ehdr - > e_ident [ EI_DATA ] ) {
static unsigned int const endian = 1 ;
2011-04-06 21:32:24 +04:00
default :
2010-10-13 23:12:54 +04:00
fprintf ( stderr , " unrecognized ELF data encoding %d: %s \n " ,
ehdr - > e_ident [ EI_DATA ] , fname ) ;
2019-07-31 21:24:12 +03:00
goto out ;
2011-04-06 21:32:24 +04:00
case ELFDATA2LSB :
2011-04-06 21:21:17 +04:00
if ( * ( unsigned char const * ) & endian ! = 1 ) {
2010-10-13 23:12:54 +04:00
/* main() is big endian, file.o is little endian. */
w = w4rev ;
w2 = w2rev ;
w8 = w8rev ;
}
2016-10-19 02:42:00 +03:00
ideal_nop4_arm = ideal_nop4_arm_le ;
bl_mcount_arm = bl_mcount_arm_le ;
push_arm = push_arm_le ;
ideal_nop2_thumb = ideal_nop2_thumb_le ;
push_bl_mcount_thumb = push_bl_mcount_thumb_le ;
2011-04-06 21:32:24 +04:00
break ;
case ELFDATA2MSB :
2011-04-06 21:21:17 +04:00
if ( * ( unsigned char const * ) & endian ! = 0 ) {
2010-10-13 23:12:54 +04:00
/* main() is little endian, file.o is big endian. */
w = w4rev ;
w2 = w2rev ;
w8 = w8rev ;
}
2016-10-19 02:42:00 +03:00
ideal_nop4_arm = ideal_nop4_arm_be ;
bl_mcount_arm = bl_mcount_arm_be ;
push_arm = push_arm_be ;
ideal_nop2_thumb = ideal_nop2_thumb_be ;
push_bl_mcount_thumb = push_bl_mcount_thumb_be ;
2011-04-06 21:32:24 +04:00
break ;
2010-10-13 23:12:54 +04:00
} /* end switch */
2019-07-31 21:24:14 +03:00
if ( memcmp ( ELFMAG , ehdr - > e_ident , SELFMAG ) ! = 0 | |
w2 ( ehdr - > e_type ) ! = ET_REL | |
ehdr - > e_ident [ EI_VERSION ] ! = EV_CURRENT ) {
2010-10-13 23:12:54 +04:00
fprintf ( stderr , " unrecognized ET_REL file %s \n " , fname ) ;
2019-07-31 21:24:12 +03:00
goto out ;
2010-10-13 23:12:54 +04:00
}
2019-07-31 21:24:14 +03:00
gpfx = ' _ ' ;
2010-10-13 23:12:54 +04:00
switch ( w2 ( ehdr - > e_machine ) ) {
2011-04-06 21:32:24 +04:00
default :
2018-05-24 06:16:12 +03:00
fprintf ( stderr , " unrecognized e_machine %u %s \n " ,
2010-10-13 23:12:54 +04:00
w2 ( ehdr - > e_machine ) , fname ) ;
2019-07-31 21:24:12 +03:00
goto out ;
2011-04-08 11:58:48 +04:00
case EM_386 :
reltype = R_386_32 ;
2015-10-28 11:23:26 +03:00
rel_type_nop = R_386_NONE ;
2011-04-08 11:58:48 +04:00
make_nop = make_nop_x86 ;
ideal_nop = ideal_nop5_x86_32 ;
2011-05-10 12:10:41 +04:00
mcount_adjust_32 = - 1 ;
2019-07-31 21:24:14 +03:00
gpfx = 0 ;
break ;
case EM_ARM :
reltype = R_ARM_ABS32 ;
altmcount = " __gnu_mcount_nc " ;
make_nop = make_nop_arm ;
rel_type_nop = R_ARM_NONE ;
gpfx = 0 ;
2011-04-08 11:58:48 +04:00
break ;
2014-04-30 13:54:32 +04:00
case EM_AARCH64 :
2019-07-31 21:24:14 +03:00
reltype = R_AARCH64_ABS64 ;
make_nop = make_nop_arm64 ;
rel_type_nop = R_AARCH64_NONE ;
ideal_nop = ideal_nop4_arm64 ;
break ;
case EM_IA_64 : reltype = R_IA64_IMM64 ; break ;
case EM_MIPS : /* reltype: e_class */ break ;
case EM_PPC : reltype = R_PPC_ADDR32 ; break ;
case EM_PPC64 : reltype = R_PPC64_ADDR64 ; break ;
case EM_S390 : /* reltype: e_class */ break ;
case EM_SH : reltype = R_SH_DIR32 ; gpfx = 0 ; break ;
case EM_SPARCV9 : reltype = R_SPARC_64 ; break ;
2011-04-08 11:58:48 +04:00
case EM_X86_64 :
make_nop = make_nop_x86 ;
ideal_nop = ideal_nop5_x86_64 ;
reltype = R_X86_64_64 ;
2015-10-28 11:23:26 +03:00
rel_type_nop = R_X86_64_NONE ;
2011-05-10 12:10:41 +04:00
mcount_adjust_64 = - 1 ;
2019-07-31 21:24:14 +03:00
gpfx = 0 ;
2011-04-08 11:58:48 +04:00
break ;
2010-10-13 23:12:54 +04:00
} /* end switch */
switch ( ehdr - > e_ident [ EI_CLASS ] ) {
2011-04-06 21:32:24 +04:00
default :
2010-10-13 23:12:54 +04:00
fprintf ( stderr , " unrecognized ELF class %d %s \n " ,
ehdr - > e_ident [ EI_CLASS ] , fname ) ;
2019-07-31 21:24:12 +03:00
goto out ;
2011-04-06 21:32:24 +04:00
case ELFCLASS32 :
2011-04-06 21:21:17 +04:00
if ( w2 ( ehdr - > e_ehsize ) ! = sizeof ( Elf32_Ehdr )
| | w2 ( ehdr - > e_shentsize ) ! = sizeof ( Elf32_Shdr ) ) {
2010-10-13 23:12:54 +04:00
fprintf ( stderr ,
" unrecognized ET_REL file: %s \n " , fname ) ;
2019-07-31 21:24:12 +03:00
goto out ;
2010-10-13 23:12:54 +04:00
}
2011-04-06 21:21:17 +04:00
if ( w2 ( ehdr - > e_machine ) = = EM_MIPS ) {
2010-10-27 14:59:07 +04:00
reltype = R_MIPS_32 ;
ftrace/MIPS: Add module support for C version of recordmcount
Since MIPS modules' address space differs from the core kernel space, to access
the _mcount in the core kernel, the kernel functions in modules must use long
call (-mlong-calls): load the _mcount address into one register and jump to the
address stored by the register:
c: 3c030000 lui v1,0x0 <--------> b label
c: R_MIPS_HI16 _mcount
c: R_MIPS_NONE *ABS*
c: R_MIPS_NONE *ABS*
10: 64630000 daddiu v1,v1,0
10: R_MIPS_LO16 _mcount
10: R_MIPS_NONE *ABS*
10: R_MIPS_NONE *ABS*
14: 03e0082d move at,ra
18: 0060f809 jalr v1
label:
In the old Perl version of recordmcount, we only need to record the position of
the 1st R_MIPS_HI16 type of _mcount, and later, in ftrace_make_nop(), replace
the instruction in this position by a "b label" and in ftrace_make_call(),
replace it back.
But, the default C version of recordmcount records all of the _mcount symbols,
so, we must filter the 2nd _mcount like the Perl version of recordmcount does.
The C version of recordmcount copes with the symbols before they are linked, So
It doesn't know the type of the symbols and therefore can not filter the
symbols as the Perl version of recordmcount does. But as we can see above, the
2nd _mcount symbols of the long call alawys follows the 1st _mcount symbol of
the same long call, which means the offset from the 1st to the 2nd is fixed, it
is 0x10-0xc = 4 here, 4 is the length of the 1st load instruciton, for MIPS has
fixed length of instructions, this offset is always 4.
And as we know, the _mcount is inserted into the entry of every kernel
function, the offset between the other _mcount's is expected to be always
bigger than 4. So, to filter the 2ns _mcount symbol of the long call, we can
simply check the offset between two _mcount symbols, If it is 4, then, filter
the 2nd _mcount symbol.
To avoid touching too much code, an 'empty' function fn_is_fake_mcount() is
added for all of the archs, and the specific archs can override it via chaning
the function pointer: is_fake_mcount in do_file() with the e_machine. e.g. This
patch adds MIPS_is_fake_mcount() to override the default fn_is_fake_mcount()
pointed by is_fake_mcount.
This fn_is_fake_mcount() checks if the _mcount symbol is fake, e.g. the 2nd
_mcount symbol of the long call is fake, for there are 2 _mcount symbols mapped
to one real mcount call, so, one of them is fake and must be filtered.
This fn_is_fake_mcount() is called in sift_rel_mcount() after finding the
_mcount symbols and before adding the _mcount symbol into mrelp, so, it can
prevent the fake mcount symbol going into the last __mcount_loc table.
Signed-off-by: Wu Zhangjin <wuzhangjin@gmail.com>
LKML-Reference: <b866f0138224340a132d31861fa3f9300dee30ac.1288176026.git.wuzhangjin@gmail.com>
Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
Signed-off-by: Ralf Baechle <ralf@linux-mips.org>
2010-10-27 14:59:08 +04:00
is_fake_mcount32 = MIPS32_is_fake_mcount ;
}
2019-07-31 21:24:12 +03:00
if ( do32 ( ehdr , fname , reltype ) < 0 )
goto out ;
2011-04-06 21:32:24 +04:00
break ;
2010-10-13 23:12:54 +04:00
case ELFCLASS64 : {
Elf64_Ehdr * const ghdr = ( Elf64_Ehdr * ) ehdr ;
2011-04-06 21:21:17 +04:00
if ( w2 ( ghdr - > e_ehsize ) ! = sizeof ( Elf64_Ehdr )
| | w2 ( ghdr - > e_shentsize ) ! = sizeof ( Elf64_Shdr ) ) {
2010-10-13 23:12:54 +04:00
fprintf ( stderr ,
" unrecognized ET_REL file: %s \n " , fname ) ;
2019-07-31 21:24:12 +03:00
goto out ;
2010-10-13 23:12:54 +04:00
}
2011-05-10 12:10:43 +04:00
if ( w2 ( ghdr - > e_machine ) = = EM_S390 ) {
2010-10-13 23:12:54 +04:00
reltype = R_390_64 ;
2014-10-15 14:17:38 +04:00
mcount_adjust_64 = - 14 ;
2011-05-10 12:10:43 +04:00
}
2011-04-06 21:21:17 +04:00
if ( w2 ( ghdr - > e_machine ) = = EM_MIPS ) {
2010-10-27 14:59:07 +04:00
reltype = R_MIPS_64 ;
Elf64_r_sym = MIPS64_r_sym ;
Elf64_r_info = MIPS64_r_info ;
ftrace/MIPS: Add module support for C version of recordmcount
Since MIPS modules' address space differs from the core kernel space, to access
the _mcount in the core kernel, the kernel functions in modules must use long
call (-mlong-calls): load the _mcount address into one register and jump to the
address stored by the register:
c: 3c030000 lui v1,0x0 <--------> b label
c: R_MIPS_HI16 _mcount
c: R_MIPS_NONE *ABS*
c: R_MIPS_NONE *ABS*
10: 64630000 daddiu v1,v1,0
10: R_MIPS_LO16 _mcount
10: R_MIPS_NONE *ABS*
10: R_MIPS_NONE *ABS*
14: 03e0082d move at,ra
18: 0060f809 jalr v1
label:
In the old Perl version of recordmcount, we only need to record the position of
the 1st R_MIPS_HI16 type of _mcount, and later, in ftrace_make_nop(), replace
the instruction in this position by a "b label" and in ftrace_make_call(),
replace it back.
But, the default C version of recordmcount records all of the _mcount symbols,
so, we must filter the 2nd _mcount like the Perl version of recordmcount does.
The C version of recordmcount copes with the symbols before they are linked, So
It doesn't know the type of the symbols and therefore can not filter the
symbols as the Perl version of recordmcount does. But as we can see above, the
2nd _mcount symbols of the long call alawys follows the 1st _mcount symbol of
the same long call, which means the offset from the 1st to the 2nd is fixed, it
is 0x10-0xc = 4 here, 4 is the length of the 1st load instruciton, for MIPS has
fixed length of instructions, this offset is always 4.
And as we know, the _mcount is inserted into the entry of every kernel
function, the offset between the other _mcount's is expected to be always
bigger than 4. So, to filter the 2ns _mcount symbol of the long call, we can
simply check the offset between two _mcount symbols, If it is 4, then, filter
the 2nd _mcount symbol.
To avoid touching too much code, an 'empty' function fn_is_fake_mcount() is
added for all of the archs, and the specific archs can override it via chaning
the function pointer: is_fake_mcount in do_file() with the e_machine. e.g. This
patch adds MIPS_is_fake_mcount() to override the default fn_is_fake_mcount()
pointed by is_fake_mcount.
This fn_is_fake_mcount() checks if the _mcount symbol is fake, e.g. the 2nd
_mcount symbol of the long call is fake, for there are 2 _mcount symbols mapped
to one real mcount call, so, one of them is fake and must be filtered.
This fn_is_fake_mcount() is called in sift_rel_mcount() after finding the
_mcount symbols and before adding the _mcount symbol into mrelp, so, it can
prevent the fake mcount symbol going into the last __mcount_loc table.
Signed-off-by: Wu Zhangjin <wuzhangjin@gmail.com>
LKML-Reference: <b866f0138224340a132d31861fa3f9300dee30ac.1288176026.git.wuzhangjin@gmail.com>
Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
Signed-off-by: Ralf Baechle <ralf@linux-mips.org>
2010-10-27 14:59:08 +04:00
is_fake_mcount64 = MIPS64_is_fake_mcount ;
2010-10-27 14:59:07 +04:00
}
2019-07-31 21:24:12 +03:00
if ( do64 ( ghdr , fname , reltype ) < 0 )
goto out ;
2011-04-06 21:32:24 +04:00
break ;
}
2010-10-13 23:12:54 +04:00
} /* end switch */
2019-07-31 21:24:12 +03:00
rc = write_file ( fname ) ;
out :
2019-07-31 21:24:16 +03:00
file_append_cleanup ( ) ;
mmap_cleanup ( ) ;
2019-07-31 21:24:12 +03:00
return rc ;
2010-10-13 23:12:54 +04:00
}
2019-07-31 21:24:13 +03:00
int main ( int argc , char * argv [ ] )
2010-10-13 23:12:54 +04:00
{
2010-11-30 19:33:53 +03:00
const char ftrace [ ] = " /ftrace.o " ;
ftrace: Do not process kernel/trace/ftrace.o with C recordmcount program
The file kernel/trace/ftrace.c references the mcount() call to
convert the mcount() callers to nops. But because it references
mcount(), the mcount() address is placed in the relocation table.
The C version of recordmcount reads the relocation table of all
object files, and it will add all references to mcount to the
__mcount_loc table that is used to find the places that call mcount()
and change the call to a nop. When recordmcount finds the mcount reference
in kernel/trace/ftrace.o, it saves that location even though the code
is not a call, but references mcount as data.
On boot up, when all calls are converted to nops, the code has a safety
check to determine what op code it is actually replacing before it
replaces it. If that op code at the address does not match, then
a warning is printed and the function tracer is disabled.
The reference to mcount in ftrace.c, causes this warning to trigger,
since the reference is not a call to mcount(). The ftrace.c file is
not compiled with the -pg flag, so no calls to mcount() should be
expected.
This patch simply makes recordmcount.c skip the kernel/trace/ftrace.c
file. This was the same solution used by the perl version of
recordmcount.
Reported-by: Ingo Molnar <mingo@elte.hu>
Cc: John Reiser <jreiser@bitwagon.com>
Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
2010-10-15 19:49:47 +04:00
int ftrace_size = sizeof ( ftrace ) - 1 ;
2010-10-13 23:12:54 +04:00
int n_error = 0 ; /* gcc-4.3.0 false positive complaint */
2011-04-13 02:53:25 +04:00
int c ;
int i ;
ftrace: Do not process kernel/trace/ftrace.o with C recordmcount program
The file kernel/trace/ftrace.c references the mcount() call to
convert the mcount() callers to nops. But because it references
mcount(), the mcount() address is placed in the relocation table.
The C version of recordmcount reads the relocation table of all
object files, and it will add all references to mcount to the
__mcount_loc table that is used to find the places that call mcount()
and change the call to a nop. When recordmcount finds the mcount reference
in kernel/trace/ftrace.o, it saves that location even though the code
is not a call, but references mcount as data.
On boot up, when all calls are converted to nops, the code has a safety
check to determine what op code it is actually replacing before it
replaces it. If that op code at the address does not match, then
a warning is printed and the function tracer is disabled.
The reference to mcount in ftrace.c, causes this warning to trigger,
since the reference is not a call to mcount(). The ftrace.c file is
not compiled with the -pg flag, so no calls to mcount() should be
expected.
This patch simply makes recordmcount.c skip the kernel/trace/ftrace.c
file. This was the same solution used by the perl version of
recordmcount.
Reported-by: Ingo Molnar <mingo@elte.hu>
Cc: John Reiser <jreiser@bitwagon.com>
Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
2010-10-15 19:49:47 +04:00
2011-04-13 02:53:25 +04:00
while ( ( c = getopt ( argc , argv , " w " ) ) > = 0 ) {
switch ( c ) {
case ' w ' :
warn_on_notrace_sect = 1 ;
break ;
default :
fprintf ( stderr , " usage: recordmcount [-w] file.o... \n " ) ;
return 0 ;
}
}
if ( ( argc - optind ) < 1 ) {
fprintf ( stderr , " usage: recordmcount [-w] file.o... \n " ) ;
ftrace: Do not process kernel/trace/ftrace.o with C recordmcount program
The file kernel/trace/ftrace.c references the mcount() call to
convert the mcount() callers to nops. But because it references
mcount(), the mcount() address is placed in the relocation table.
The C version of recordmcount reads the relocation table of all
object files, and it will add all references to mcount to the
__mcount_loc table that is used to find the places that call mcount()
and change the call to a nop. When recordmcount finds the mcount reference
in kernel/trace/ftrace.o, it saves that location even though the code
is not a call, but references mcount as data.
On boot up, when all calls are converted to nops, the code has a safety
check to determine what op code it is actually replacing before it
replaces it. If that op code at the address does not match, then
a warning is printed and the function tracer is disabled.
The reference to mcount in ftrace.c, causes this warning to trigger,
since the reference is not a call to mcount(). The ftrace.c file is
not compiled with the -pg flag, so no calls to mcount() should be
expected.
This patch simply makes recordmcount.c skip the kernel/trace/ftrace.c
file. This was the same solution used by the perl version of
recordmcount.
Reported-by: Ingo Molnar <mingo@elte.hu>
Cc: John Reiser <jreiser@bitwagon.com>
Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
2010-10-15 19:49:47 +04:00
return 0 ;
}
/* Process each file in turn, allowing deep failure. */
2011-04-13 02:53:25 +04:00
for ( i = optind ; i < argc ; i + + ) {
char * file = argv [ i ] ;
ftrace: Do not process kernel/trace/ftrace.o with C recordmcount program
The file kernel/trace/ftrace.c references the mcount() call to
convert the mcount() callers to nops. But because it references
mcount(), the mcount() address is placed in the relocation table.
The C version of recordmcount reads the relocation table of all
object files, and it will add all references to mcount to the
__mcount_loc table that is used to find the places that call mcount()
and change the call to a nop. When recordmcount finds the mcount reference
in kernel/trace/ftrace.o, it saves that location even though the code
is not a call, but references mcount as data.
On boot up, when all calls are converted to nops, the code has a safety
check to determine what op code it is actually replacing before it
replaces it. If that op code at the address does not match, then
a warning is printed and the function tracer is disabled.
The reference to mcount in ftrace.c, causes this warning to trigger,
since the reference is not a call to mcount(). The ftrace.c file is
not compiled with the -pg flag, so no calls to mcount() should be
expected.
This patch simply makes recordmcount.c skip the kernel/trace/ftrace.c
file. This was the same solution used by the perl version of
recordmcount.
Reported-by: Ingo Molnar <mingo@elte.hu>
Cc: John Reiser <jreiser@bitwagon.com>
Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
2010-10-15 19:49:47 +04:00
int len ;
/*
* The file kernel / trace / ftrace . o references the mcount
* function but does not call it . Since ftrace . o should
* not be traced anyway , we just skip it .
*/
2011-04-13 02:53:25 +04:00
len = strlen ( file ) ;
ftrace: Do not process kernel/trace/ftrace.o with C recordmcount program
The file kernel/trace/ftrace.c references the mcount() call to
convert the mcount() callers to nops. But because it references
mcount(), the mcount() address is placed in the relocation table.
The C version of recordmcount reads the relocation table of all
object files, and it will add all references to mcount to the
__mcount_loc table that is used to find the places that call mcount()
and change the call to a nop. When recordmcount finds the mcount reference
in kernel/trace/ftrace.o, it saves that location even though the code
is not a call, but references mcount as data.
On boot up, when all calls are converted to nops, the code has a safety
check to determine what op code it is actually replacing before it
replaces it. If that op code at the address does not match, then
a warning is printed and the function tracer is disabled.
The reference to mcount in ftrace.c, causes this warning to trigger,
since the reference is not a call to mcount(). The ftrace.c file is
not compiled with the -pg flag, so no calls to mcount() should be
expected.
This patch simply makes recordmcount.c skip the kernel/trace/ftrace.c
file. This was the same solution used by the perl version of
recordmcount.
Reported-by: Ingo Molnar <mingo@elte.hu>
Cc: John Reiser <jreiser@bitwagon.com>
Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
2010-10-15 19:49:47 +04:00
if ( len > = ftrace_size & &
2011-04-13 02:53:25 +04:00
strcmp ( file + ( len - ftrace_size ) , ftrace ) = = 0 )
ftrace: Do not process kernel/trace/ftrace.o with C recordmcount program
The file kernel/trace/ftrace.c references the mcount() call to
convert the mcount() callers to nops. But because it references
mcount(), the mcount() address is placed in the relocation table.
The C version of recordmcount reads the relocation table of all
object files, and it will add all references to mcount to the
__mcount_loc table that is used to find the places that call mcount()
and change the call to a nop. When recordmcount finds the mcount reference
in kernel/trace/ftrace.o, it saves that location even though the code
is not a call, but references mcount as data.
On boot up, when all calls are converted to nops, the code has a safety
check to determine what op code it is actually replacing before it
replaces it. If that op code at the address does not match, then
a warning is printed and the function tracer is disabled.
The reference to mcount in ftrace.c, causes this warning to trigger,
since the reference is not a call to mcount(). The ftrace.c file is
not compiled with the -pg flag, so no calls to mcount() should be
expected.
This patch simply makes recordmcount.c skip the kernel/trace/ftrace.c
file. This was the same solution used by the perl version of
recordmcount.
Reported-by: Ingo Molnar <mingo@elte.hu>
Cc: John Reiser <jreiser@bitwagon.com>
Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
2010-10-15 19:49:47 +04:00
continue ;
2019-07-31 21:24:12 +03:00
if ( do_file ( file ) ) {
2015-12-31 02:06:41 +03:00
fprintf ( stderr , " %s: failed \n " , file ) ;
2010-10-13 23:12:54 +04:00
+ + n_error ;
2019-07-31 21:24:12 +03:00
}
2010-10-13 23:12:54 +04:00
}
return ! ! n_error ;
}