ftrace: handle weak symbol functions

During tests and checks, I've discovered that there were failures to
convert mcount callers into nops. Looking deeper into these failures,
code that was attempted to be changed was not an mcount caller.
The current code only updates if the code being changed is what it expects,
but I still investigate any time there is a failure.

What was happening is that a weak symbol was being used as a reference
for other mcount callers. That weak symbol was also referenced elsewhere
so the offsets were using the strong symbol and not the function symbol
that it was referenced from.

This patch changes the setting up of the mcount_loc section to search
for a global function that is not weak. It will pick a local over a weak
but if only a weak is found in a section, a warning is printed and the
mcount location is not recorded (just to be safe).

Signed-off-by: Steven Rostedt <srostedt@redhat.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
This commit is contained in:
Steven Rostedt 2008-08-20 10:07:35 -04:00 committed by Ingo Molnar
parent d74fcd1e4e
commit 8feff1cacc
1 changed files with 86 additions and 20 deletions

View File

@ -119,17 +119,19 @@ $mv = "mv" if ((length $mv) == 0);
#print STDERR "running: $P '$arch' '$objdump' '$objcopy' '$cc' '$ld' " . #print STDERR "running: $P '$arch' '$objdump' '$objcopy' '$cc' '$ld' " .
# "'$nm' '$rm' '$mv' '$inputfile'\n"; # "'$nm' '$rm' '$mv' '$inputfile'\n";
my %locals; my %locals; # List of local (static) functions
my %convert; my %weak; # List of weak functions
my %convert; # List of local functions used that needs conversion
my $type; my $type;
my $section_regex; # Find the start of a section my $section_regex; # Find the start of a section
my $function_regex; # Find the name of a function (return func name) my $function_regex; # Find the name of a function
# (return offset and func name)
my $mcount_regex; # Find the call site to mcount (return offset) my $mcount_regex; # Find the call site to mcount (return offset)
if ($arch eq "x86_64") { if ($arch eq "x86_64") {
$section_regex = "Disassembly of section"; $section_regex = "Disassembly of section";
$function_regex = "<(.*?)>:"; $function_regex = "^([0-9a-fA-F]+)\\s+<(.*?)>:";
$mcount_regex = "^\\s*([0-9a-fA-F]+):.*\\smcount([+-]0x[0-9a-zA-Z]+)?\$"; $mcount_regex = "^\\s*([0-9a-fA-F]+):.*\\smcount([+-]0x[0-9a-zA-Z]+)?\$";
$type = ".quad"; $type = ".quad";
@ -141,7 +143,7 @@ if ($arch eq "x86_64") {
} elsif ($arch eq "i386") { } elsif ($arch eq "i386") {
$section_regex = "Disassembly of section"; $section_regex = "Disassembly of section";
$function_regex = "<(.*?)>:"; $function_regex = "^([0-9a-fA-F]+)\\s+<(.*?)>:";
$mcount_regex = "^\\s*([0-9a-fA-F]+):.*\\smcount\$"; $mcount_regex = "^\\s*([0-9a-fA-F]+):.*\\smcount\$";
$type = ".long"; $type = ".long";
@ -158,7 +160,6 @@ if ($arch eq "x86_64") {
my $text_found = 0; my $text_found = 0;
my $read_function = 0; my $read_function = 0;
my $opened = 0; my $opened = 0;
my $text = "";
my $mcount_section = "__mcount_loc"; my $mcount_section = "__mcount_loc";
my $dirname; my $dirname;
@ -186,46 +187,111 @@ my $mcount_s = $dirname . "/.tmp_mc_" . $prefix . ".s";
my $mcount_o = $dirname . "/.tmp_mc_" . $prefix . ".o"; my $mcount_o = $dirname . "/.tmp_mc_" . $prefix . ".o";
# #
# Step 1: find all the local symbols (static functions). # Step 1: find all the local (static functions) and weak symbols.
# 't' is local, 'w/W' is weak (we never use a weak function)
# #
open (IN, "$nm $inputfile|") || die "error running $nm"; open (IN, "$nm $inputfile|") || die "error running $nm";
while (<IN>) { while (<IN>) {
if (/^[0-9a-fA-F]+\s+t\s+(\S+)/) { if (/^[0-9a-fA-F]+\s+t\s+(\S+)/) {
$locals{$1} = 1; $locals{$1} = 1;
} elsif (/^[0-9a-fA-F]+\s+([wW])\s+(\S+)/) {
$weak{$2} = $1;
} }
} }
close(IN); close(IN);
my @offsets; # Array of offsets of mcount callers
my $ref_func; # reference function to use for offsets
my $offset = 0; # offset of ref_func to section beginning
##
# update_funcs - print out the current mcount callers
#
# Go through the list of offsets to callers and write them to
# the output file in a format that can be read by an assembler.
#
sub update_funcs
{
return if ($#offsets < 0);
defined($ref_func) || die "No function to reference";
# A section only had a weak function, to represent it.
# Unfortunately, a weak function may be overwritten by another
# function of the same name, making all these offsets incorrect.
# To be safe, we simply print a warning and bail.
if (defined $weak{$ref_func}) {
print STDERR
"$inputfile: WARNING: referencing weak function" .
" $ref_func for mcount\n";
return;
}
# is this function static? If so, note this fact.
if (defined $locals{$ref_func}) {
$convert{$ref_func} = 1;
}
# Loop through all the mcount caller offsets and print a reference
# to the caller based from the ref_func.
for (my $i=0; $i <= $#offsets; $i++) {
if (!$opened) {
open(FILE, ">$mcount_s") || die "can't create $mcount_s\n";
$opened = 1;
print FILE "\t.section $mcount_section,\"a\",\@progbits\n";
}
printf FILE "\t%s %s + %d\n", $type, $ref_func, $offsets[$i] - $offset;
}
}
# #
# Step 2: find the sections and mcount call sites # Step 2: find the sections and mcount call sites
# #
open(IN, "$objdump -dr $inputfile|") || die "error running $objdump"; open(IN, "$objdump -dr $inputfile|") || die "error running $objdump";
my $text;
while (<IN>) { while (<IN>) {
# is it a section? # is it a section?
if (/$section_regex/) { if (/$section_regex/) {
$read_function = 1; $read_function = 1;
# print out any recorded offsets
update_funcs() if ($text_found);
# reset all markers and arrays
$text_found = 0; $text_found = 0;
undef($ref_func);
undef(@offsets);
# section found, now is this a start of a function? # section found, now is this a start of a function?
} elsif ($read_function && /$function_regex/) { } elsif ($read_function && /$function_regex/) {
$read_function = 0;
$text_found = 1; $text_found = 1;
$text = $1; $offset = hex $1;
# is this function static? If so, note this fact. $text = $2;
if (defined $locals{$text}) {
$convert{$text} = 1; # if this is either a local function or a weak function
# keep looking for functions that are global that
# we can use safely.
if (!defined($locals{$text}) && !defined($weak{$text})) {
$ref_func = $text;
$read_function = 0;
} else {
# if we already have a function, and this is weak, skip it
if (!defined($ref_func) || !defined($weak{$text})) {
$ref_func = $text;
}
} }
# is this a call site to mcount? If so, print the offset from the section }
} elsif ($text_found && /$mcount_regex/) {
if (!$opened) { # is this a call site to mcount? If so, record it to print later
open(FILE, ">$mcount_s") || die "can't create $mcount_s\n"; if ($text_found && /$mcount_regex/) {
$opened = 1; $offsets[$#offsets + 1] = hex $1;
print FILE "\t.section $mcount_section,\"a\",\@progbits\n";
}
print FILE "\t$type $text + 0x$1\n";
} }
} }
# dump out anymore offsets that may have been found
update_funcs() if ($text_found);
# If we did not find any mcount callers, we are done (do nothing). # If we did not find any mcount callers, we are done (do nothing).
if (!$opened) { if (!$opened) {
exit(0); exit(0);