OpenCores
URL https://opencores.org/ocsvn/openrisc/openrisc/trunk

Subversion Repositories openrisc

[/] [openrisc/] [trunk/] [or1ksim/] [mprofiler.c] - Rev 37

Go to most recent revision | Compare with Previous | Blame | View Log

/* mprofiler.c -- memory profiling utility
 
   Copyright (C) 2002 Marko Mlinar, markom@opencores.org
   Copyright (C) 1999 Damjan Lampret, lampret@opencores.org
   Copyright (C) 2008 Embecosm Limited
 
   Contributor Jeremy Bennett <jeremy.bennett@embecosm.com>
 
   This file is part of Or1ksim, the OpenRISC 1000 Architectural Simulator.
 
   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, see <http://www.gnu.org/licenses/>.  */
 
/* This program is commented throughout in a fashion suitable for processing
   with Doxygen. */
 
 
/* Autoconf and/or portability configuration */
#include "config.h"
#include "port.h"
 
/* System includes */
#include <stdlib.h>
#include <regex.h>
 
/* Package includes */
#include "mprofiler.h"
#include "sim-config.h"
#include "argtable2.h"
#include "support/profile.h"
 
/* output modes */
#define MODE_DETAIL     0
#define MODE_PRETTY     1
#define MODE_ACCESS     2
#define MODE_WIDTH      3
 
/* Input buffer size */
#define BUF_SIZE        256
 
/* HASH */
#define HASH_SIZE       0x10000
#define HASH_FUNC(x)    ((x) & 0xffff)
 
/*! Hash table data structure */
static struct memory_hash
{
  struct memory_hash *next;
  oraddr_t            addr;
  unsigned long       cnt[3];		/* Various counters */
} *hash[HASH_SIZE];
 
/*! Groups size -- how much addresses should be joined together */
static int  group_bits = 2;
 
/*! Start address */
static oraddr_t  start_addr = 0;
 
/*! End address */
static oraddr_t  end_addr = 0xffffffff;
 
/* File to read from */
static FILE *fprof = 0;
 
static void
hash_add (oraddr_t addr, int index)
{
  struct memory_hash *h = hash[HASH_FUNC (addr)];
  while (h && h->addr != addr)
    h = h->next;
 
  if (!h)
    {
      h = (struct memory_hash *) malloc (sizeof (struct memory_hash));
      h->next = hash[HASH_FUNC (addr)];
      hash[HASH_FUNC (addr)] = h;
      h->addr = addr;
      h->cnt[0] = h->cnt[1] = h->cnt[2] = 0;
    }
  h->cnt[index]++;
}
 
 
static unsigned long
hash_get (oraddr_t addr, int index)
{
  struct memory_hash *h = hash[HASH_FUNC (addr)];
  while (h && h->addr != addr)
    h = h->next;
 
  if (!h)
    return 0;
  return h->cnt[index];
}
 
static void
init ()
{
  int i;
  for (i = 0; i < HASH_SIZE; i++)
    hash[i] = NULL;
}
 
static void
read_file (FILE * f, int mode)
{
  struct mprofentry_struct buf[BUF_SIZE];
  int num_read;
  do
    {
      int i;
      num_read = fread (buf, sizeof (struct mprofentry_struct), BUF_SIZE, f);
      for (i = 0; i < num_read; i++)
	if (buf[i].addr >= start_addr && buf[i].addr <= end_addr)
	  {
	    int index;
	    unsigned t = buf[i].type;
	    if (t > 64)
	      {
		PRINTF ("!");
		t = 0;
	      }
	    if (mode == MODE_WIDTH)
	      t >>= 3;
	    else
	      t &= 0x7;
 
	    switch (t)
	      {
	      case 1:
		index = 0;
		break;
	      case 2:
		index = 1;
		break;
	      case 4:
		index = 2;
		break;
	      default:
		index = 0;
		PRINTF ("!!!!");
		break;
	      }
	    hash_add (buf[i].addr >> group_bits, index);
	  }
    }
  while (num_read > 0);
}
 
static int
nbits (unsigned long a)
{
  int cnt = 0;
  int b = a;
  if (!a)
    return 0;
 
  while (a)
    a >>= 1, cnt++;
  if (cnt > 1 && ((b >> (cnt - 2)) & 1))
    cnt = cnt * 2 + 1;
  else
    cnt *= 2;
 
  return cnt - 1;
}
 
static void
printout (int mode)
{
  oraddr_t addr = start_addr & ~((1 << group_bits) - 1);
  PRINTF ("start = %" PRIxADDR " (%" PRIxADDR "); end = %" PRIxADDR
	  "; group_bits = %08x\n", start_addr, addr, end_addr,
	  (1 << group_bits) - 1);
  for (; addr <= end_addr; addr += (1 << group_bits))
    {
      int i;
      unsigned long a = hash_get (addr >> group_bits, 0);
      unsigned long b = hash_get (addr >> group_bits, 1);
      unsigned long c = hash_get (addr >> group_bits, 2);
      PRINTF ("%" PRIxADDR ":", addr);
      switch (mode)
	{
	case MODE_DETAIL:
	  if (a)
	    PRINTF (" %10li R", a);
	  else
	    PRINTF ("            R");
	  if (b)
	    PRINTF (" %10li W", b);
	  else
	    PRINTF ("            W");
	  if (c)
	    PRINTF (" %10li F", c);
	  else
	    PRINTF ("            F");
	  break;
	case MODE_ACCESS:
	  PRINTF (" %10li", a + b + c);
	  break;
	case MODE_PRETTY:
	  PRINTF (" %10li ", a + b + c);
	  for (i = 0; i < nbits (a + b + c); i++)
	    PRINTF ("#");
	  break;
	case MODE_WIDTH:
	  if (a)
	    PRINTF (" %10li B", a);
	  else
	    PRINTF ("            B");
	  if (b)
	    PRINTF (" %10li H", b);
	  else
	    PRINTF ("            H");
	  if (c)
	    PRINTF (" %10li W", c);
	  else
	    PRINTF ("            W");
	  break;
	}
      PRINTF ("\n");
      if (addr >= addr + (1 << group_bits))
	break;			/* Overflow? */
    }
}
 
/*---------------------------------------------------------------------------*/
/*! Parse the arguments for the profiling utility
 
    Updated by Jeremy Bennett to use argtable2. Also has an option just to
    print help, for use with the CLI.
 
    @param[in] argc       Number of command args
    @param[in] argv       Vector of the command args
    @param[in] just_help  If 1 (true), ignore argc & argv and just print out
                          the help message without parsing args
 
    @return  0 on success, 1 on failure                                      */
/*---------------------------------------------------------------------------*/
int
main_mprofiler (int argc, char *argv[], int just_help)
{
  struct arg_lit *vercop;
  struct arg_lit *help;
  struct arg_rex *mode_arg;
  struct arg_int *group;
  struct arg_file *prof_file;
  struct arg_int *from;
  struct arg_int *to;
  struct arg_end *end;
 
  void *argtab[8];
  int nerrors;
 
  int mode = MODE_DETAIL;
 
  /* Specify each argument, with fallback values */
  vercop = arg_lit0 ("v", "version", "version and copyright notice");
  help = arg_lit0 ("h", "help", "print this help message");
  mode_arg = arg_rex0 ("m", "mode",
		       "(detailed|d)|(pretty|p)|(access|a)|"
		       "(width|w)", "<m>", REG_ICASE,
		       "Output mode (detailed, pretty, access " "or width)");
  mode_arg->sval[0] = "detailed";
  group = arg_int0 ("g", "group", "<n>",
		    "group 2^n bits successive addresses " "together");
  group->ival[0] = 0;
  prof_file = arg_file0 ("f", "filename", "<file>",
			 "data file to analyse (default " "sim.mprofile)");
  prof_file->filename[0] = "sim.mprofile";
  from = arg_int1 (NULL, NULL, "<from>", "start address");
  to = arg_int1 (NULL, NULL, "<to>", "end address");
  end = arg_end (20);
 
  /* Set up the argument table */
  argtab[0] = vercop;
  argtab[1] = help;
  argtab[2] = mode_arg;
  argtab[3] = group;
  argtab[4] = prof_file;
  argtab[5] = from;
  argtab[6] = to;
  argtab[7] = end;
 
  /* If we are just asked for a help message, then we don't parse the
     args. This is used to implement the help function from the CLI. */
  if (just_help)
    {
      printf ("mprofile");
      arg_print_syntax (stdout, argtab, "\n");
      arg_print_glossary (stdout, argtab, "  %-25s %s\n");
 
      arg_freetable (argtab, sizeof (argtab) / sizeof (argtab[0]));
      return 0;
    }
 
  /* Parse */
  nerrors = arg_parse (argc, argv, argtab);
 
  /* Special case here is if help or version is specified, we ignore any other
     errors and just print the help or version information and then give up. */
  if (vercop->count > 0)
    {
      PRINTF ("OpenRISC 1000 Memory Profiling Utility, version %s\n",
	      PACKAGE_VERSION);
 
      arg_freetable (argtab, sizeof (argtab) / sizeof (argtab[0]));
      return 0;
    }
 
  if (help->count > 0)
    {
      printf ("Usage: %s ", argv[0]);
      arg_print_syntax (stdout, argtab, "\n");
      arg_print_glossary (stdout, argtab, "  %-25s %s\n");
 
      arg_freetable (argtab, sizeof (argtab) / sizeof (argtab[0]));
      return 0;
    }
 
  /* Deal with any errors */
  if (0 != nerrors)
    {
      arg_print_errors (stderr, end, "mprofile");
      fprintf (stderr, "Usage: %s ", argv[0]);
      arg_print_syntaxv (stderr, argtab, "\n");
 
      arg_freetable (argtab, sizeof (argtab) / sizeof (argtab[0]));
      return 1;
    }
 
  /* If version or help is requested, that is all that is printed out */
  /* Sort out the mode */
  if ((0 == strcmp (mode_arg->sval[0], "detail")) ||
      (0 == strcmp (mode_arg->sval[0], "d")))
    {
      mode = MODE_DETAIL;
    }
  else if ((0 == strcmp (mode_arg->sval[0], "pretty")) ||
	   (0 == strcmp (mode_arg->sval[0], "p")))
    {
      mode = MODE_DETAIL;
    }
  else if ((0 == strcmp (mode_arg->sval[0], "access")) ||
	   (0 == strcmp (mode_arg->sval[0], "a")))
    {
      mode = MODE_DETAIL;
    }
  else if ((0 == strcmp (mode_arg->sval[0], "width")) ||
	   (0 == strcmp (mode_arg->sval[0], "w")))
    {
      mode = MODE_DETAIL;
    }
  else
    {
      fprintf (stderr, "Impossible mode: %s\n", mode_arg->sval[0]);
    }
 
  /* Group bits */
  group_bits = group->ival[0];
 
  /* Start and end addresses */
  start_addr = from->ival[0];
  end_addr = to->ival[0];
 
  /* Get the profile */
  fprof = fopen (prof_file->filename[0], "rm");
 
  if (!fprof)
    {
      fprintf (stderr, "Cannot open profile file: %s\n",
	       prof_file->filename[0]);
 
      arg_freetable (argtab, sizeof (argtab) / sizeof (argtab[0]));
      return 1;
    }
 
  /* Finished with the args */
  arg_freetable (argtab, sizeof (argtab) / sizeof (argtab[0]));
 
  init ();
  read_file (fprof, mode);
  fclose (fprof);
  printout (mode);
  return 0;
 
}	/* main_mprofiler () */
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.