Blame gprof/i386.c

Packit Service 72eb06
/*
Packit Service 72eb06
 * Copyright (c) 1983, 1993, 2001
Packit Service 72eb06
 *      The Regents of the University of California.  All rights reserved.
Packit Service 72eb06
 *
Packit Service 72eb06
 * Redistribution and use in source and binary forms, with or without
Packit Service 72eb06
 * modification, are permitted provided that the following conditions
Packit Service 72eb06
 * are met:
Packit Service 72eb06
 * 1. Redistributions of source code must retain the above copyright
Packit Service 72eb06
 *    notice, this list of conditions and the following disclaimer.
Packit Service 72eb06
 * 2. Redistributions in binary form must reproduce the above copyright
Packit Service 72eb06
 *    notice, this list of conditions and the following disclaimer in the
Packit Service 72eb06
 *    documentation and/or other materials provided with the distribution.
Packit Service 72eb06
 * 3. Neither the name of the University nor the names of its contributors
Packit Service 72eb06
 *    may be used to endorse or promote products derived from this software
Packit Service 72eb06
 *    without specific prior written permission.
Packit Service 72eb06
 *
Packit Service 72eb06
 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
Packit Service 72eb06
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
Packit Service 72eb06
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
Packit Service 72eb06
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
Packit Service 72eb06
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
Packit Service 72eb06
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
Packit Service 72eb06
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
Packit Service 72eb06
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
Packit Service 72eb06
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
Packit Service 72eb06
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
Packit Service 72eb06
 * SUCH DAMAGE.
Packit Service 72eb06
 */
Packit Service 72eb06
#include "gprof.h"
Packit Service 72eb06
#include "search_list.h"
Packit Service 72eb06
#include "source.h"
Packit Service 72eb06
#include "symtab.h"
Packit Service 72eb06
#include "cg_arcs.h"
Packit Service 72eb06
#include "corefile.h"
Packit Service 72eb06
#include "hist.h"
Packit Service 72eb06
Packit Service 72eb06
static int i386_iscall (unsigned char *);
Packit Service 72eb06
void i386_find_call (Sym *, bfd_vma, bfd_vma);
Packit Service 72eb06
Packit Service 72eb06
static int
Packit Service 72eb06
i386_iscall (unsigned char *ip)
Packit Service 72eb06
{
Packit Service 72eb06
  if (*ip == 0xe8)
Packit Service 72eb06
    return 1;
Packit Service 72eb06
  return 0;
Packit Service 72eb06
}
Packit Service 72eb06
Packit Service 72eb06
Packit Service 72eb06
void
Packit Service 72eb06
i386_find_call (Sym *parent, bfd_vma p_lowpc, bfd_vma p_highpc)
Packit Service 72eb06
{
Packit Service 72eb06
  unsigned char *instructp;
Packit Service 72eb06
  Sym *child;
Packit Service 72eb06
  bfd_vma pc, destpc;
Packit Service 72eb06
Packit Service 72eb06
  DBG (CALLDEBUG, printf ("[findcall] %s: 0x%lx to 0x%lx\n",
Packit Service 72eb06
			  parent->name, (unsigned long) p_lowpc,
Packit Service 72eb06
			  (unsigned long) p_highpc));
Packit Service 72eb06
Packit Service 72eb06
  for (pc = p_lowpc; pc < p_highpc; ++pc)
Packit Service 72eb06
    {
Packit Service 72eb06
      instructp = (unsigned char *) core_text_space + pc - core_text_sect->vma;
Packit Service 72eb06
      if (i386_iscall (instructp))
Packit Service 72eb06
	{
Packit Service 72eb06
	  DBG (CALLDEBUG,
Packit Service 72eb06
	       printf ("[findcall]\t0x%lx:call", (unsigned long) pc));
Packit Service 72eb06
	  /*
Packit Service 72eb06
	   *  regular pc relative addressing
Packit Service 72eb06
	   *    check that this is the address of
Packit Service 72eb06
	   *    a function.
Packit Service 72eb06
	   */
Packit Service 72eb06
Packit Service 72eb06
	  destpc = bfd_get_32 (core_bfd, instructp + 1) + pc + 5;
Packit Service 72eb06
	  if (hist_check_address (destpc))
Packit Service 72eb06
	    {
Packit Service 72eb06
	      child = sym_lookup (&symtab, destpc);
Packit Service 72eb06
	      if (child && child->addr == destpc)
Packit Service 72eb06
		{
Packit Service 72eb06
		  /*
Packit Service 72eb06
		   *      a hit
Packit Service 72eb06
		   */
Packit Service 72eb06
		  DBG (CALLDEBUG,
Packit Service 72eb06
		       printf ("\tdestpc 0x%lx (%s)\n",
Packit Service 72eb06
			       (unsigned long) destpc, child->name));
Packit Service 72eb06
		  arc_add (parent, child, (unsigned long) 0);
Packit Service 72eb06
		  instructp += 4;	/* call is a 5 byte instruction */
Packit Service 72eb06
		  continue;
Packit Service 72eb06
		}
Packit Service 72eb06
	    }
Packit Service 72eb06
	  /*
Packit Service 72eb06
	   *  else:
Packit Service 72eb06
	   *    it looked like a callf, but it:
Packit Service 72eb06
	   *      a) wasn't actually a callf, or
Packit Service 72eb06
	   *      b) didn't point to a known function in the symtab, or
Packit Service 72eb06
	   *      c) something funny is going on.
Packit Service 72eb06
	   */
Packit Service 72eb06
	  DBG (CALLDEBUG, printf ("\tbut it's a botch\n"));
Packit Service 72eb06
	}
Packit Service 72eb06
    }
Packit Service 72eb06
}