Files
bfd
binutils
config
cpu
elfcpp
etc
gas
gdb
gold
gprof
po
.gdbinit
.gitignore
ChangeLog
ChangeLog-2004
ChangeLog-2005
ChangeLog-2006
ChangeLog-2007
ChangeLog-2008
ChangeLog-2009
ChangeLog-2010
ChangeLog-2011
ChangeLog-2012
ChangeLog-2013
ChangeLog-2014
ChangeLog-2015
ChangeLog-2016
ChangeLog-2017
ChangeLog-9203
MAINTAINERS
Makefile.am
Makefile.in
README
TEST
TODO
aarch64.c
aclocal.m4
alpha.c
basic_blocks.c
basic_blocks.h
bb_exit_func.c
bbconv.pl
bsd_callg_bl.m
call_graph.c
call_graph.h
cg_arcs.c
cg_arcs.h
cg_dfn.c
cg_dfn.h
cg_print.c
cg_print.h
configure
configure.ac
corefile.c
corefile.h
dep-in.sed
fdl.texi
flat_bl.m
fsf_callg_bl.m
gconfig.in
gen-c-prog.awk
gmon.h
gmon_io.c
gmon_io.h
gmon_out.h
gprof.c
gprof.h
gprof.texi
hertz.c
hertz.h
hist.c
hist.h
i386.c
mips.c
search_list.c
search_list.h
source.c
source.h
sparc.c
stamp-h.in
sym_ids.c
sym_ids.h
symtab.c
symtab.h
utils.c
utils.h
vax.c
include
intl
ld
libdecnumber
libiberty
opcodes
readline
sim
texinfo
zlib
.cvsignore
.gitattributes
.gitignore
COPYING
COPYING.LIB
COPYING.LIBGLOSS
COPYING.NEWLIB
COPYING3
COPYING3.LIB
ChangeLog
MAINTAINERS
Makefile.def
Makefile.in
Makefile.tpl
README
README-maintainer-mode
compile
config-ml.in
config.guess
config.rpath
config.sub
configure
configure.ac
depcomp
djunpack.bat
install-sh
libtool.m4
ltgcc.m4
ltmain.sh
ltoptions.m4
ltsugar.m4
ltversion.m4
lt~obsolete.m4
makefile.vms
missing
mkdep
mkinstalldirs
move-if-change
setup.com
src-release.sh
symlink-tree
ylwrap
binutils-gdb/gprof/call_graph.c

131 lines
3.9 KiB
C
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/* call_graph.c - Create call graphs.
Copyright (C) 1999-2018 Free Software Foundation, Inc.
This file is part of GNU Binutils.
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 3 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., 51 Franklin Street - Fifth Floor, Boston, MA
02110-1301, USA. */
#include "gprof.h"
#include "search_list.h"
#include "source.h"
#include "symtab.h"
#include "cg_arcs.h"
#include "call_graph.h"
#include "corefile.h"
#include "gmon_io.h"
#include "gmon_out.h"
#include "sym_ids.h"
void
cg_tally (bfd_vma from_pc, bfd_vma self_pc, unsigned long count)
{
Sym *parent;
Sym *child;
parent = sym_lookup (&symtab, from_pc);
child = sym_lookup (&symtab, self_pc);
if (child == NULL || parent == NULL)
return;
/* If we're doing line-by-line profiling, both the parent and the
child will probably point to line symbols instead of function
symbols. For the parent this is fine, since this identifies the
line number in the calling routing, but the child should always
point to a function entry point, so we back up in the symbol
table until we find it.
For normal profiling, is_func will be set on all symbols, so this
code will do nothing. */
while (child >= symtab.base && ! child->is_func)
--child;
if (child < symtab.base)
return;
/* Keep arc if it is on INCL_ARCS table or if the INCL_ARCS table
is empty and it is not in the EXCL_ARCS table. */
if (sym_id_arc_is_present (&syms[INCL_ARCS], parent, child)
|| (syms[INCL_ARCS].len == 0
&& !sym_id_arc_is_present (&syms[EXCL_ARCS], parent, child)))
{
child->ncalls += count;
DBG (TALLYDEBUG,
printf (_("[cg_tally] arc from %s to %s traversed %lu times\n"),
parent->name, child->name, count));
arc_add (parent, child, count);
}
}
/* Read a record from file IFP describing an arc in the function
call-graph and the count of how many times the arc has been
traversed. FILENAME is the name of file IFP and is provided
for formatting error-messages only. */
void
cg_read_rec (FILE *ifp, const char *filename)
{
bfd_vma from_pc, self_pc;
unsigned int count;
if (gmon_io_read_vma (ifp, &from_pc)
|| gmon_io_read_vma (ifp, &self_pc)
|| gmon_io_read_32 (ifp, &count))
{
fprintf (stderr, _("%s: %s: unexpected end of file\n"),
whoami, filename);
done (1);
}
DBG (SAMPLEDEBUG,
printf ("[cg_read_rec] frompc 0x%lx selfpc 0x%lx count %lu\n",
(unsigned long) from_pc, (unsigned long) self_pc,
(unsigned long) count));
/* Add this arc: */
cg_tally (from_pc, self_pc, count);
}
/* Write all the arcs in the call-graph to file OFP. FILENAME is
the name of OFP and is provided for formatting error-messages
only. */
void
cg_write_arcs (FILE *ofp, const char *filename)
{
Arc *arc;
Sym *sym;
for (sym = symtab.base; sym < symtab.limit; sym++)
{
for (arc = sym->cg.children; arc; arc = arc->next_child)
{
if (gmon_io_write_8 (ofp, GMON_TAG_CG_ARC)
|| gmon_io_write_vma (ofp, arc->parent->addr)
|| gmon_io_write_vma (ofp, arc->child->addr)
|| gmon_io_write_32 (ofp, arc->count))
{
perror (filename);
done (1);
}
DBG (SAMPLEDEBUG,
printf ("[cg_write_arcs] frompc 0x%lx selfpc 0x%lx count %lu\n",
(unsigned long) arc->parent->addr,
(unsigned long) arc->child->addr, arc->count));
}
}
}