2010-10-13 21:12:54 +02: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.
|
|
|
|
* Licensed under the GNU General Public License, version 2 (GPLv2).
|
|
|
|
*
|
|
|
|
* 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>
|
|
|
|
#include <elf.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <setjmp.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
static int fd_map; /* File descriptor for file being modified. */
|
|
|
|
static int mmap_failed; /* Boolean flag. */
|
|
|
|
static void *ehdr_curr; /* current ElfXX_Ehdr * for resource cleanup */
|
|
|
|
static char gpfx; /* prefix for global symbol name (sometimes '_') */
|
|
|
|
static struct stat sb; /* Remember .st_size, etc. */
|
|
|
|
static jmp_buf jmpenv; /* setjmp/longjmp per-file error escape */
|
2010-11-30 17:36:48 +01:00
|
|
|
static const char *altmcount; /* alternate mcount symbol name */
|
2010-10-13 21:12:54 +02:00
|
|
|
|
|
|
|
/* setjmp() return values */
|
|
|
|
enum {
|
|
|
|
SJ_SETJMP = 0, /* hardwired first return */
|
|
|
|
SJ_FAIL,
|
|
|
|
SJ_SUCCEED
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Per-file resource cleanup when multiple files. */
|
|
|
|
static void
|
|
|
|
cleanup(void)
|
|
|
|
{
|
|
|
|
if (!mmap_failed)
|
|
|
|
munmap(ehdr_curr, sb.st_size);
|
|
|
|
else
|
|
|
|
free(ehdr_curr);
|
|
|
|
close(fd_map);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __attribute__((noreturn))
|
|
|
|
fail_file(void)
|
|
|
|
{
|
|
|
|
cleanup();
|
|
|
|
longjmp(jmpenv, SJ_FAIL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __attribute__((noreturn))
|
|
|
|
succeed_file(void)
|
|
|
|
{
|
|
|
|
cleanup();
|
|
|
|
longjmp(jmpenv, SJ_SUCCEED);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ulseek, uread, ...: Check return value for errors. */
|
|
|
|
|
|
|
|
static off_t
|
|
|
|
ulseek(int const fd, off_t const offset, int const whence)
|
|
|
|
{
|
|
|
|
off_t const w = lseek(fd, offset, whence);
|
|
|
|
if ((off_t)-1 == w) {
|
|
|
|
perror("lseek");
|
|
|
|
fail_file();
|
|
|
|
}
|
|
|
|
return w;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t
|
|
|
|
uread(int const fd, void *const buf, size_t const count)
|
|
|
|
{
|
|
|
|
size_t const n = read(fd, buf, count);
|
|
|
|
if (n != count) {
|
|
|
|
perror("read");
|
|
|
|
fail_file();
|
|
|
|
}
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t
|
|
|
|
uwrite(int const fd, void const *const buf, size_t const count)
|
|
|
|
{
|
|
|
|
size_t const n = write(fd, buf, count);
|
|
|
|
if (n != count) {
|
|
|
|
perror("write");
|
|
|
|
fail_file();
|
|
|
|
}
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *
|
|
|
|
umalloc(size_t size)
|
|
|
|
{
|
|
|
|
void *const addr = malloc(size);
|
|
|
|
if (0 == addr) {
|
|
|
|
fprintf(stderr, "malloc failed: %zu bytes\n", size);
|
|
|
|
fail_file();
|
|
|
|
}
|
|
|
|
return addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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)
|
|
|
|
{
|
|
|
|
void *addr;
|
|
|
|
|
|
|
|
fd_map = open(fname, O_RDWR);
|
|
|
|
if (0 > fd_map || 0 > fstat(fd_map, &sb)) {
|
|
|
|
perror(fname);
|
|
|
|
fail_file();
|
|
|
|
}
|
|
|
|
if (!S_ISREG(sb.st_mode)) {
|
|
|
|
fprintf(stderr, "not a regular file: %s\n", fname);
|
|
|
|
fail_file();
|
|
|
|
}
|
|
|
|
addr = mmap(0, sb.st_size, PROT_READ|PROT_WRITE, MAP_PRIVATE,
|
|
|
|
fd_map, 0);
|
|
|
|
mmap_failed = 0;
|
|
|
|
if (MAP_FAILED == addr) {
|
|
|
|
mmap_failed = 1;
|
|
|
|
addr = umalloc(sb.st_size);
|
|
|
|
uread(fd_map, addr, sb.st_size);
|
|
|
|
}
|
|
|
|
return addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 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. */
|
|
|
|
static int
|
|
|
|
is_mcounted_section_name(char const *const txtname)
|
|
|
|
{
|
|
|
|
return 0 == strcmp(".text", txtname) ||
|
|
|
|
0 == strcmp(".sched.text", txtname) ||
|
|
|
|
0 == strcmp(".spinlock.text", txtname) ||
|
|
|
|
0 == strcmp(".irqentry.text", txtname) ||
|
|
|
|
0 == strcmp(".text.unlikely", txtname);
|
|
|
|
}
|
|
|
|
|
2010-10-14 01:06:14 +02:00
|
|
|
/* 32 bit and 64 bit are very similar */
|
|
|
|
#include "recordmcount.h"
|
|
|
|
#define RECORD_MCOUNT_64
|
|
|
|
#include "recordmcount.h"
|
2010-10-13 21:12:54 +02:00
|
|
|
|
2010-10-27 12:59:07 +02: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;
|
|
|
|
}
|
|
|
|
|
2010-10-13 21:12:54 +02:00
|
|
|
static void
|
|
|
|
do_file(char const *const fname)
|
|
|
|
{
|
|
|
|
Elf32_Ehdr *const ehdr = mmap_file(fname);
|
|
|
|
unsigned int reltype = 0;
|
|
|
|
|
|
|
|
ehdr_curr = ehdr;
|
|
|
|
w = w4nat;
|
|
|
|
w2 = w2nat;
|
|
|
|
w8 = w8nat;
|
|
|
|
switch (ehdr->e_ident[EI_DATA]) {
|
|
|
|
static unsigned int const endian = 1;
|
|
|
|
default: {
|
|
|
|
fprintf(stderr, "unrecognized ELF data encoding %d: %s\n",
|
|
|
|
ehdr->e_ident[EI_DATA], fname);
|
|
|
|
fail_file();
|
|
|
|
} break;
|
|
|
|
case ELFDATA2LSB: {
|
|
|
|
if (1 != *(unsigned char const *)&endian) {
|
|
|
|
/* main() is big endian, file.o is little endian. */
|
|
|
|
w = w4rev;
|
|
|
|
w2 = w2rev;
|
|
|
|
w8 = w8rev;
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
case ELFDATA2MSB: {
|
|
|
|
if (0 != *(unsigned char const *)&endian) {
|
|
|
|
/* main() is little endian, file.o is big endian. */
|
|
|
|
w = w4rev;
|
|
|
|
w2 = w2rev;
|
|
|
|
w8 = w8rev;
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
} /* end switch */
|
|
|
|
if (0 != memcmp(ELFMAG, ehdr->e_ident, SELFMAG)
|
|
|
|
|| ET_REL != w2(ehdr->e_type)
|
|
|
|
|| EV_CURRENT != ehdr->e_ident[EI_VERSION]) {
|
|
|
|
fprintf(stderr, "unrecognized ET_REL file %s\n", fname);
|
|
|
|
fail_file();
|
|
|
|
}
|
|
|
|
|
|
|
|
gpfx = 0;
|
|
|
|
switch (w2(ehdr->e_machine)) {
|
|
|
|
default: {
|
|
|
|
fprintf(stderr, "unrecognized e_machine %d %s\n",
|
|
|
|
w2(ehdr->e_machine), fname);
|
|
|
|
fail_file();
|
|
|
|
} break;
|
|
|
|
case EM_386: reltype = R_386_32; break;
|
2010-11-30 17:36:48 +01:00
|
|
|
case EM_ARM: reltype = R_ARM_ABS32;
|
|
|
|
altmcount = "__gnu_mcount_nc";
|
|
|
|
break;
|
2010-10-13 21:12:54 +02:00
|
|
|
case EM_IA_64: reltype = R_IA64_IMM64; gpfx = '_'; break;
|
2010-10-27 12:59:07 +02:00
|
|
|
case EM_MIPS: /* reltype: e_class */ gpfx = '_'; break;
|
2010-10-13 21:12:54 +02:00
|
|
|
case EM_PPC: reltype = R_PPC_ADDR32; gpfx = '_'; break;
|
|
|
|
case EM_PPC64: reltype = R_PPC64_ADDR64; gpfx = '_'; break;
|
|
|
|
case EM_S390: /* reltype: e_class */ gpfx = '_'; break;
|
|
|
|
case EM_SH: reltype = R_SH_DIR32; break;
|
|
|
|
case EM_SPARCV9: reltype = R_SPARC_64; gpfx = '_'; break;
|
|
|
|
case EM_X86_64: reltype = R_X86_64_64; break;
|
|
|
|
} /* end switch */
|
|
|
|
|
|
|
|
switch (ehdr->e_ident[EI_CLASS]) {
|
|
|
|
default: {
|
|
|
|
fprintf(stderr, "unrecognized ELF class %d %s\n",
|
|
|
|
ehdr->e_ident[EI_CLASS], fname);
|
|
|
|
fail_file();
|
|
|
|
} break;
|
|
|
|
case ELFCLASS32: {
|
|
|
|
if (sizeof(Elf32_Ehdr) != w2(ehdr->e_ehsize)
|
|
|
|
|| sizeof(Elf32_Shdr) != w2(ehdr->e_shentsize)) {
|
|
|
|
fprintf(stderr,
|
|
|
|
"unrecognized ET_REL file: %s\n", fname);
|
|
|
|
fail_file();
|
|
|
|
}
|
|
|
|
if (EM_S390 == w2(ehdr->e_machine))
|
|
|
|
reltype = R_390_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 12:59:08 +02:00
|
|
|
if (EM_MIPS == w2(ehdr->e_machine)) {
|
2010-10-27 12:59:07 +02: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 12:59:08 +02:00
|
|
|
is_fake_mcount32 = MIPS32_is_fake_mcount;
|
|
|
|
}
|
2010-10-13 21:12:54 +02:00
|
|
|
do32(ehdr, fname, reltype);
|
|
|
|
} break;
|
|
|
|
case ELFCLASS64: {
|
|
|
|
Elf64_Ehdr *const ghdr = (Elf64_Ehdr *)ehdr;
|
|
|
|
if (sizeof(Elf64_Ehdr) != w2(ghdr->e_ehsize)
|
|
|
|
|| sizeof(Elf64_Shdr) != w2(ghdr->e_shentsize)) {
|
|
|
|
fprintf(stderr,
|
|
|
|
"unrecognized ET_REL file: %s\n", fname);
|
|
|
|
fail_file();
|
|
|
|
}
|
|
|
|
if (EM_S390 == w2(ghdr->e_machine))
|
|
|
|
reltype = R_390_64;
|
2010-10-27 12:59:07 +02:00
|
|
|
if (EM_MIPS == w2(ghdr->e_machine)) {
|
|
|
|
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 12:59:08 +02:00
|
|
|
is_fake_mcount64 = MIPS64_is_fake_mcount;
|
2010-10-27 12:59:07 +02:00
|
|
|
}
|
2010-10-13 21:12:54 +02:00
|
|
|
do64(ghdr, fname, reltype);
|
|
|
|
} break;
|
|
|
|
} /* end switch */
|
|
|
|
|
|
|
|
cleanup();
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
main(int argc, char const *argv[])
|
|
|
|
{
|
2010-11-30 17:33:53 +01: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 17:49:47 +02:00
|
|
|
int ftrace_size = sizeof(ftrace) - 1;
|
2010-10-13 21:12:54 +02:00
|
|
|
int n_error = 0; /* gcc-4.3.0 false positive complaint */
|
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 17:49:47 +02:00
|
|
|
|
|
|
|
if (argc <= 1) {
|
2010-10-13 21:12:54 +02:00
|
|
|
fprintf(stderr, "usage: recordmcount 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 17:49:47 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Process each file in turn, allowing deep failure. */
|
2010-10-13 21:12:54 +02:00
|
|
|
for (--argc, ++argv; 0 < argc; --argc, ++argv) {
|
|
|
|
int const sjval = setjmp(jmpenv);
|
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 17:49:47 +02: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.
|
|
|
|
*/
|
|
|
|
len = strlen(argv[0]);
|
|
|
|
if (len >= ftrace_size &&
|
|
|
|
strcmp(argv[0] + (len - ftrace_size), ftrace) == 0)
|
|
|
|
continue;
|
|
|
|
|
2010-10-13 21:12:54 +02:00
|
|
|
switch (sjval) {
|
|
|
|
default: {
|
|
|
|
fprintf(stderr, "internal error: %s\n", argv[0]);
|
|
|
|
exit(1);
|
|
|
|
} break;
|
|
|
|
case SJ_SETJMP: { /* normal sequence */
|
|
|
|
/* Avoid problems if early cleanup() */
|
|
|
|
fd_map = -1;
|
|
|
|
ehdr_curr = NULL;
|
|
|
|
mmap_failed = 1;
|
|
|
|
do_file(argv[0]);
|
|
|
|
} break;
|
|
|
|
case SJ_FAIL: { /* error in do_file or below */
|
|
|
|
++n_error;
|
|
|
|
} break;
|
|
|
|
case SJ_SUCCEED: { /* premature success */
|
|
|
|
/* do nothing */
|
|
|
|
} break;
|
|
|
|
} /* end switch */
|
|
|
|
}
|
|
|
|
return !!n_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|