OpenCores
URL https://opencores.org/ocsvn/openrisc_2011-10-31/openrisc_2011-10-31/trunk

Subversion Repositories openrisc_2011-10-31

[/] [openrisc/] [tags/] [gnu-src/] [gdb-6.8/] [pre-binutils-2.20.1-sync/] [sim/] [igen/] [gen-icache.c] - Diff between revs 157 and 223

Go to most recent revision | Only display areas with differences | Details | Blame | View Log

Rev 157 Rev 223
/* The IGEN simulator generator for GDB, the GNU Debugger.
/* The IGEN simulator generator for GDB, the GNU Debugger.
 
 
   Copyright 2002, 2007, 2008 Free Software Foundation, Inc.
   Copyright 2002, 2007, 2008 Free Software Foundation, Inc.
 
 
   Contributed by Andrew Cagney.
   Contributed by Andrew Cagney.
 
 
   This file is part of GDB.
   This file is part of GDB.
 
 
   This program is free software; you can redistribute it and/or modify
   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
   it under the terms of the GNU General Public License as published by
   the Free Software Foundation; either version 3 of the License, or
   the Free Software Foundation; either version 3 of the License, or
   (at your option) any later version.
   (at your option) any later version.
 
 
   This program is distributed in the hope that it will be useful,
   This program is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.
   GNU General Public License for more details.
 
 
   You should have received a copy of the GNU General Public License
   You should have received a copy of the GNU General Public License
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
 
 
 
 
#include "misc.h"
#include "misc.h"
#include "lf.h"
#include "lf.h"
#include "table.h"
#include "table.h"
#include "filter.h"
#include "filter.h"
#include "igen.h"
#include "igen.h"
 
 
#include "ld-insn.h"
#include "ld-insn.h"
#include "ld-decode.h"
#include "ld-decode.h"
 
 
#include "gen.h"
#include "gen.h"
 
 
#include "gen-semantics.h"
#include "gen-semantics.h"
#include "gen-idecode.h"
#include "gen-idecode.h"
#include "gen-icache.h"
#include "gen-icache.h"
 
 
 
 
 
 
static void
static void
print_icache_function_header (lf *file,
print_icache_function_header (lf *file,
                              const char *basename,
                              const char *basename,
                              const char *format_name,
                              const char *format_name,
                              opcode_bits *expanded_bits,
                              opcode_bits *expanded_bits,
                              int is_function_definition,
                              int is_function_definition,
                              int nr_prefetched_words)
                              int nr_prefetched_words)
{
{
  lf_printf (file, "\n");
  lf_printf (file, "\n");
  lf_print__function_type_function (file, print_icache_function_type,
  lf_print__function_type_function (file, print_icache_function_type,
                                    "EXTERN_ICACHE", " ");
                                    "EXTERN_ICACHE", " ");
  print_function_name (file,
  print_function_name (file,
                       basename, format_name, NULL,
                       basename, format_name, NULL,
                       expanded_bits, function_name_prefix_icache);
                       expanded_bits, function_name_prefix_icache);
  lf_printf (file, "\n(");
  lf_printf (file, "\n(");
  print_icache_function_formal (file, nr_prefetched_words);
  print_icache_function_formal (file, nr_prefetched_words);
  lf_printf (file, ")");
  lf_printf (file, ")");
  if (!is_function_definition)
  if (!is_function_definition)
    lf_printf (file, ";");
    lf_printf (file, ";");
  lf_printf (file, "\n");
  lf_printf (file, "\n");
}
}
 
 
 
 
void
void
print_icache_declaration (lf *file,
print_icache_declaration (lf *file,
                          insn_entry * insn,
                          insn_entry * insn,
                          opcode_bits *expanded_bits,
                          opcode_bits *expanded_bits,
                          insn_opcodes *opcodes, int nr_prefetched_words)
                          insn_opcodes *opcodes, int nr_prefetched_words)
{
{
  print_icache_function_header (file,
  print_icache_function_header (file,
                                insn->name,
                                insn->name,
                                insn->format_name,
                                insn->format_name,
                                expanded_bits,
                                expanded_bits,
                                0 /* is not function definition */ ,
                                0 /* is not function definition */ ,
                                nr_prefetched_words);
                                nr_prefetched_words);
}
}
 
 
 
 
 
 
static void
static void
print_icache_extraction (lf *file,
print_icache_extraction (lf *file,
                         const char *format_name,
                         const char *format_name,
                         cache_entry_type cache_type,
                         cache_entry_type cache_type,
                         const char *entry_name,
                         const char *entry_name,
                         const char *entry_type,
                         const char *entry_type,
                         const char *entry_expression,
                         const char *entry_expression,
                         char *single_insn_field,
                         char *single_insn_field,
                         line_ref *line,
                         line_ref *line,
                         insn_field_entry *cur_field,
                         insn_field_entry *cur_field,
                         opcode_bits *expanded_bits,
                         opcode_bits *expanded_bits,
                         icache_decl_type what_to_declare,
                         icache_decl_type what_to_declare,
                         icache_body_type what_to_do)
                         icache_body_type what_to_do)
{
{
  const char *expression;
  const char *expression;
  opcode_bits *bits;
  opcode_bits *bits;
  char *reason;
  char *reason;
  ASSERT (format_name != NULL);
  ASSERT (format_name != NULL);
  ASSERT (entry_name != NULL);
  ASSERT (entry_name != NULL);
 
 
  /* figure out exactly what should be going on here */
  /* figure out exactly what should be going on here */
  switch (cache_type)
  switch (cache_type)
    {
    {
    case scratch_value:
    case scratch_value:
      if ((what_to_do & put_values_in_icache)
      if ((what_to_do & put_values_in_icache)
          || what_to_do == do_not_use_icache)
          || what_to_do == do_not_use_icache)
        {
        {
          reason = "scratch";
          reason = "scratch";
          what_to_do = do_not_use_icache;
          what_to_do = do_not_use_icache;
        }
        }
      else
      else
        return;
        return;
      break;
      break;
    case compute_value:
    case compute_value:
      if ((what_to_do & get_values_from_icache)
      if ((what_to_do & get_values_from_icache)
          || what_to_do == do_not_use_icache)
          || what_to_do == do_not_use_icache)
        {
        {
          reason = "compute";
          reason = "compute";
          what_to_do = do_not_use_icache;
          what_to_do = do_not_use_icache;
        }
        }
      else
      else
        return;
        return;
      break;
      break;
    case cache_value:
    case cache_value:
      if ((what_to_declare != undef_variables)
      if ((what_to_declare != undef_variables)
          || !(what_to_do & put_values_in_icache))
          || !(what_to_do & put_values_in_icache))
        {
        {
          reason = "cache";
          reason = "cache";
          what_to_declare = ((what_to_do & put_values_in_icache)
          what_to_declare = ((what_to_do & put_values_in_icache)
                             ? declare_variables : what_to_declare);
                             ? declare_variables : what_to_declare);
        }
        }
      else
      else
        return;
        return;
      break;
      break;
    default:
    default:
      abort ();                 /* Bad switch.  */
      abort ();                 /* Bad switch.  */
    }
    }
 
 
  /* For the type, default to a simple unsigned */
  /* For the type, default to a simple unsigned */
  if (entry_type == NULL || strlen (entry_type) == 0)
  if (entry_type == NULL || strlen (entry_type) == 0)
    entry_type = "unsigned";
    entry_type = "unsigned";
 
 
  /* look through the set of expanded sub fields to see if this field
  /* look through the set of expanded sub fields to see if this field
     has been given a constant value */
     has been given a constant value */
  for (bits = expanded_bits; bits != NULL; bits = bits->next)
  for (bits = expanded_bits; bits != NULL; bits = bits->next)
    {
    {
      if (bits->field == cur_field)
      if (bits->field == cur_field)
        break;
        break;
    }
    }
 
 
  /* Define a storage area for the cache element */
  /* Define a storage area for the cache element */
  switch (what_to_declare)
  switch (what_to_declare)
    {
    {
    case undef_variables:
    case undef_variables:
      /* We've finished with the #define value - destory it */
      /* We've finished with the #define value - destory it */
      lf_indent_suppress (file);
      lf_indent_suppress (file);
      lf_printf (file, "#undef %s\n", entry_name);
      lf_printf (file, "#undef %s\n", entry_name);
      return;
      return;
    case define_variables:
    case define_variables:
      /* Using direct access for this entry, clear any prior
      /* Using direct access for this entry, clear any prior
         definition, then define it */
         definition, then define it */
      lf_indent_suppress (file);
      lf_indent_suppress (file);
      lf_printf (file, "#undef %s\n", entry_name);
      lf_printf (file, "#undef %s\n", entry_name);
      /* Don't type cast pointer types! */
      /* Don't type cast pointer types! */
      lf_indent_suppress (file);
      lf_indent_suppress (file);
      if (strchr (entry_type, '*') != NULL)
      if (strchr (entry_type, '*') != NULL)
        lf_printf (file, "#define %s (", entry_name);
        lf_printf (file, "#define %s (", entry_name);
      else
      else
        lf_printf (file, "#define %s ((%s) ", entry_name, entry_type);
        lf_printf (file, "#define %s ((%s) ", entry_name, entry_type);
      break;
      break;
    case declare_variables:
    case declare_variables:
      /* using variables to define the value */
      /* using variables to define the value */
      if (line != NULL)
      if (line != NULL)
        lf_print__line_ref (file, line);
        lf_print__line_ref (file, line);
      lf_printf (file, "%s const %s UNUSED = ", entry_type, entry_name);
      lf_printf (file, "%s const %s UNUSED = ", entry_type, entry_name);
      break;
      break;
    }
    }
 
 
 
 
  /* define a value for that storage area as determined by what is in
  /* define a value for that storage area as determined by what is in
     the cache */
     the cache */
  if (bits != NULL
  if (bits != NULL
      && single_insn_field != NULL
      && single_insn_field != NULL
      && strcmp (entry_name, single_insn_field) == 0
      && strcmp (entry_name, single_insn_field) == 0
      && strcmp (entry_name, cur_field->val_string) == 0
      && strcmp (entry_name, cur_field->val_string) == 0
      && ((bits->opcode->is_boolean && bits->value == 0)
      && ((bits->opcode->is_boolean && bits->value == 0)
          || (!bits->opcode->is_boolean)))
          || (!bits->opcode->is_boolean)))
    {
    {
      /* The cache rule is specifying what to do with a simple
      /* The cache rule is specifying what to do with a simple
         instruction field.
         instruction field.
 
 
         Because of instruction expansion, the field is either a
         Because of instruction expansion, the field is either a
         constant value or equal to the specified constant (boolean
         constant value or equal to the specified constant (boolean
         comparison). (The latter indicated by bits->value == 0).
         comparison). (The latter indicated by bits->value == 0).
 
 
         The case of a field not being equal to the specified boolean
         The case of a field not being equal to the specified boolean
         value is handled later. */
         value is handled later. */
      expression = "constant field";
      expression = "constant field";
      ASSERT (bits->field == cur_field);
      ASSERT (bits->field == cur_field);
      if (bits->opcode->is_boolean)
      if (bits->opcode->is_boolean)
        {
        {
          ASSERT (bits->value == 0);
          ASSERT (bits->value == 0);
          lf_printf (file, "%d", bits->opcode->boolean_constant);
          lf_printf (file, "%d", bits->opcode->boolean_constant);
        }
        }
      else if (bits->opcode->last < bits->field->last)
      else if (bits->opcode->last < bits->field->last)
        {
        {
          lf_printf (file, "%d",
          lf_printf (file, "%d",
                     bits->value << (bits->field->last - bits->opcode->last));
                     bits->value << (bits->field->last - bits->opcode->last));
        }
        }
      else
      else
        {
        {
          lf_printf (file, "%d", bits->value);
          lf_printf (file, "%d", bits->value);
        }
        }
    }
    }
  else if (bits != NULL
  else if (bits != NULL
           && single_insn_field != NULL
           && single_insn_field != NULL
           && strncmp (entry_name,
           && strncmp (entry_name,
                       single_insn_field,
                       single_insn_field,
                       strlen (single_insn_field)) == 0
                       strlen (single_insn_field)) == 0
           && strncmp (entry_name + strlen (single_insn_field),
           && strncmp (entry_name + strlen (single_insn_field),
                       "_is_",
                       "_is_",
                       strlen ("_is_")) == 0
                       strlen ("_is_")) == 0
           && ((bits->opcode->is_boolean
           && ((bits->opcode->is_boolean
                && ((unsigned)
                && ((unsigned)
                    atol (entry_name + strlen (single_insn_field) +
                    atol (entry_name + strlen (single_insn_field) +
                          strlen ("_is_")) == bits->opcode->boolean_constant))
                          strlen ("_is_")) == bits->opcode->boolean_constant))
               || (!bits->opcode->is_boolean)))
               || (!bits->opcode->is_boolean)))
    {
    {
      /* The cache rule defines an entry for the comparison between a
      /* The cache rule defines an entry for the comparison between a
         single instruction field and a constant.  The value of the
         single instruction field and a constant.  The value of the
         comparison in someway matches that of the opcode field that
         comparison in someway matches that of the opcode field that
         was made constant through expansion. */
         was made constant through expansion. */
      expression = "constant compare";
      expression = "constant compare";
      if (bits->opcode->is_boolean)
      if (bits->opcode->is_boolean)
        {
        {
          lf_printf (file, "%d /* %s == %d */",
          lf_printf (file, "%d /* %s == %d */",
                     bits->value == 0,
                     bits->value == 0,
                     single_insn_field, bits->opcode->boolean_constant);
                     single_insn_field, bits->opcode->boolean_constant);
        }
        }
      else if (bits->opcode->last < bits->field->last)
      else if (bits->opcode->last < bits->field->last)
        {
        {
          lf_printf (file, "%d /* %s == %d */",
          lf_printf (file, "%d /* %s == %d */",
                     (atol
                     (atol
                      (entry_name + strlen (single_insn_field) +
                      (entry_name + strlen (single_insn_field) +
                       strlen ("_is_")) ==
                       strlen ("_is_")) ==
                      (bits->
                      (bits->
                       value << (bits->field->last - bits->opcode->last))),
                       value << (bits->field->last - bits->opcode->last))),
                     single_insn_field,
                     single_insn_field,
                     (bits->
                     (bits->
                      value << (bits->field->last - bits->opcode->last)));
                      value << (bits->field->last - bits->opcode->last)));
        }
        }
      else
      else
        {
        {
          lf_printf (file, "%d /* %s == %d */",
          lf_printf (file, "%d /* %s == %d */",
                     (atol
                     (atol
                      (entry_name + strlen (single_insn_field) +
                      (entry_name + strlen (single_insn_field) +
                       strlen ("_is_")) == bits->value), single_insn_field,
                       strlen ("_is_")) == bits->value), single_insn_field,
                     bits->value);
                     bits->value);
        }
        }
    }
    }
  else
  else
    {
    {
      /* put the field in the local variable, possibly also enter it
      /* put the field in the local variable, possibly also enter it
         into the cache */
         into the cache */
      expression = "extraction";
      expression = "extraction";
      /* handle the cache */
      /* handle the cache */
      if ((what_to_do & get_values_from_icache)
      if ((what_to_do & get_values_from_icache)
          || (what_to_do & put_values_in_icache))
          || (what_to_do & put_values_in_icache))
        {
        {
          lf_printf (file, "cache_entry->crack.%s.%s",
          lf_printf (file, "cache_entry->crack.%s.%s",
                     format_name, entry_name);
                     format_name, entry_name);
          if (what_to_do & put_values_in_icache)        /* also put it in the cache? */
          if (what_to_do & put_values_in_icache)        /* also put it in the cache? */
            {
            {
              lf_printf (file, " = ");
              lf_printf (file, " = ");
            }
            }
        }
        }
      if ((what_to_do & put_values_in_icache)
      if ((what_to_do & put_values_in_icache)
          || what_to_do == do_not_use_icache)
          || what_to_do == do_not_use_icache)
        {
        {
          if (cur_field != NULL)
          if (cur_field != NULL)
            {
            {
              if (entry_expression != NULL && strlen (entry_expression) > 0)
              if (entry_expression != NULL && strlen (entry_expression) > 0)
                error (line,
                error (line,
                       "Instruction field entry with nonempty expression\n");
                       "Instruction field entry with nonempty expression\n");
              if (cur_field->first == 0
              if (cur_field->first == 0
                  && cur_field->last == options.insn_bit_size - 1)
                  && cur_field->last == options.insn_bit_size - 1)
                lf_printf (file, "(instruction_%d)", cur_field->word_nr);
                lf_printf (file, "(instruction_%d)", cur_field->word_nr);
              else if (cur_field->last == options.insn_bit_size - 1)
              else if (cur_field->last == options.insn_bit_size - 1)
                lf_printf (file, "MASKED%d (instruction_%d, %d, %d)",
                lf_printf (file, "MASKED%d (instruction_%d, %d, %d)",
                           options.insn_bit_size,
                           options.insn_bit_size,
                           cur_field->word_nr,
                           cur_field->word_nr,
                           i2target (options.hi_bit_nr, cur_field->first),
                           i2target (options.hi_bit_nr, cur_field->first),
                           i2target (options.hi_bit_nr, cur_field->last));
                           i2target (options.hi_bit_nr, cur_field->last));
              else
              else
                lf_printf (file, "EXTRACTED%d (instruction_%d, %d, %d)",
                lf_printf (file, "EXTRACTED%d (instruction_%d, %d, %d)",
                           options.insn_bit_size,
                           options.insn_bit_size,
                           cur_field->word_nr,
                           cur_field->word_nr,
                           i2target (options.hi_bit_nr, cur_field->first),
                           i2target (options.hi_bit_nr, cur_field->first),
                           i2target (options.hi_bit_nr, cur_field->last));
                           i2target (options.hi_bit_nr, cur_field->last));
            }
            }
          else
          else
            {
            {
              lf_printf (file, "%s", entry_expression);
              lf_printf (file, "%s", entry_expression);
            }
            }
        }
        }
    }
    }
 
 
  switch (what_to_declare)
  switch (what_to_declare)
    {
    {
    case define_variables:
    case define_variables:
      lf_printf (file, ")");
      lf_printf (file, ")");
      break;
      break;
    case undef_variables:
    case undef_variables:
      break;
      break;
    case declare_variables:
    case declare_variables:
      lf_printf (file, ";");
      lf_printf (file, ";");
      break;
      break;
    }
    }
 
 
  ASSERT (reason != NULL && expression != NULL);
  ASSERT (reason != NULL && expression != NULL);
  lf_printf (file, " /* %s - %s */\n", reason, expression);
  lf_printf (file, " /* %s - %s */\n", reason, expression);
}
}
 
 
 
 
void
void
print_icache_body (lf *file,
print_icache_body (lf *file,
                   insn_entry * instruction,
                   insn_entry * instruction,
                   opcode_bits *expanded_bits,
                   opcode_bits *expanded_bits,
                   cache_entry *cache_rules,
                   cache_entry *cache_rules,
                   icache_decl_type what_to_declare,
                   icache_decl_type what_to_declare,
                   icache_body_type what_to_do, int nr_prefetched_words)
                   icache_body_type what_to_do, int nr_prefetched_words)
{
{
  /* extract instruction fields */
  /* extract instruction fields */
  lf_printf (file, "/* Extraction: %s\n", instruction->name);
  lf_printf (file, "/* Extraction: %s\n", instruction->name);
  lf_printf (file, "     ");
  lf_printf (file, "     ");
  switch (what_to_declare)
  switch (what_to_declare)
    {
    {
    case define_variables:
    case define_variables:
      lf_printf (file, "#define");
      lf_printf (file, "#define");
      break;
      break;
    case declare_variables:
    case declare_variables:
      lf_printf (file, "declare");
      lf_printf (file, "declare");
      break;
      break;
    case undef_variables:
    case undef_variables:
      lf_printf (file, "#undef");
      lf_printf (file, "#undef");
      break;
      break;
    }
    }
  lf_printf (file, " ");
  lf_printf (file, " ");
  switch (what_to_do)
  switch (what_to_do)
    {
    {
    case get_values_from_icache:
    case get_values_from_icache:
      lf_printf (file, "get-values-from-icache");
      lf_printf (file, "get-values-from-icache");
      break;
      break;
    case put_values_in_icache:
    case put_values_in_icache:
      lf_printf (file, "put-values-in-icache");
      lf_printf (file, "put-values-in-icache");
      break;
      break;
    case both_values_and_icache:
    case both_values_and_icache:
      lf_printf (file, "get-values-from-icache|put-values-in-icache");
      lf_printf (file, "get-values-from-icache|put-values-in-icache");
      break;
      break;
    case do_not_use_icache:
    case do_not_use_icache:
      lf_printf (file, "do-not-use-icache");
      lf_printf (file, "do-not-use-icache");
      break;
      break;
    }
    }
  lf_printf (file, "\n     ");
  lf_printf (file, "\n     ");
  print_insn_words (file, instruction);
  print_insn_words (file, instruction);
  lf_printf (file, " */\n");
  lf_printf (file, " */\n");
 
 
  /* pass zero - fetch from memory any missing instructions.
  /* pass zero - fetch from memory any missing instructions.
 
 
     Some of the instructions will have already been fetched (in the
     Some of the instructions will have already been fetched (in the
     instruction array), others will still need fetching. */
     instruction array), others will still need fetching. */
  switch (what_to_do)
  switch (what_to_do)
    {
    {
    case get_values_from_icache:
    case get_values_from_icache:
      break;
      break;
    case put_values_in_icache:
    case put_values_in_icache:
    case both_values_and_icache:
    case both_values_and_icache:
    case do_not_use_icache:
    case do_not_use_icache:
      {
      {
        int word_nr;
        int word_nr;
        switch (what_to_declare)
        switch (what_to_declare)
          {
          {
          case undef_variables:
          case undef_variables:
            break;
            break;
          case define_variables:
          case define_variables:
          case declare_variables:
          case declare_variables:
            for (word_nr = nr_prefetched_words;
            for (word_nr = nr_prefetched_words;
                 word_nr < instruction->nr_words; word_nr++)
                 word_nr < instruction->nr_words; word_nr++)
              {
              {
                /* FIXME - should be using print_icache_extraction? */
                /* FIXME - should be using print_icache_extraction? */
                lf_printf (file,
                lf_printf (file,
                           "%sinstruction_word instruction_%d UNUSED = ",
                           "%sinstruction_word instruction_%d UNUSED = ",
                           options.module.global.prefix.l, word_nr);
                           options.module.global.prefix.l, word_nr);
                lf_printf (file, "IMEM%d_IMMED (cia, %d)",
                lf_printf (file, "IMEM%d_IMMED (cia, %d)",
                           options.insn_bit_size, word_nr);
                           options.insn_bit_size, word_nr);
                lf_printf (file, ";\n");
                lf_printf (file, ";\n");
              }
              }
          }
          }
      }
      }
    }
    }
 
 
  /* if putting the instruction words in the cache, define references
  /* if putting the instruction words in the cache, define references
     for them */
     for them */
  if (options.gen.insn_in_icache)
  if (options.gen.insn_in_icache)
    {
    {
      /* FIXME: is the instruction_word type correct? */
      /* FIXME: is the instruction_word type correct? */
      print_icache_extraction (file, instruction->format_name, cache_value, "insn",     /* name */
      print_icache_extraction (file, instruction->format_name, cache_value, "insn",     /* name */
                               "instruction_word",      /* type */
                               "instruction_word",      /* type */
                               "instruction",   /* expression */
                               "instruction",   /* expression */
                               NULL,    /* origin */
                               NULL,    /* origin */
                               NULL,    /* line */
                               NULL,    /* line */
                               NULL, NULL, what_to_declare, what_to_do);
                               NULL, NULL, what_to_declare, what_to_do);
    }
    }
  lf_printf (file, "\n");
  lf_printf (file, "\n");
 
 
  /* pass one - process instruction fields.
  /* pass one - process instruction fields.
 
 
     If there is no cache rule, the default is to enter the field into
     If there is no cache rule, the default is to enter the field into
     the cache */
     the cache */
  {
  {
    insn_word_entry *word;
    insn_word_entry *word;
    for (word = instruction->words; word != NULL; word = word->next)
    for (word = instruction->words; word != NULL; word = word->next)
      {
      {
        insn_field_entry *cur_field;
        insn_field_entry *cur_field;
        for (cur_field = word->first;
        for (cur_field = word->first;
             cur_field->first < options.insn_bit_size;
             cur_field->first < options.insn_bit_size;
             cur_field = cur_field->next)
             cur_field = cur_field->next)
          {
          {
            if (cur_field->type == insn_field_string)
            if (cur_field->type == insn_field_string)
              {
              {
                cache_entry *cache_rule;
                cache_entry *cache_rule;
                cache_entry_type value_type = cache_value;
                cache_entry_type value_type = cache_value;
                line_ref *value_line = instruction->line;
                line_ref *value_line = instruction->line;
                /* check the cache table to see if it contains a rule
                /* check the cache table to see if it contains a rule
                   overriding the default cache action for an
                   overriding the default cache action for an
                   instruction field */
                   instruction field */
                for (cache_rule = cache_rules;
                for (cache_rule = cache_rules;
                     cache_rule != NULL; cache_rule = cache_rule->next)
                     cache_rule != NULL; cache_rule = cache_rule->next)
                  {
                  {
                    if (filter_is_subset (instruction->field_names,
                    if (filter_is_subset (instruction->field_names,
                                          cache_rule->original_fields)
                                          cache_rule->original_fields)
                        && strcmp (cache_rule->name,
                        && strcmp (cache_rule->name,
                                   cur_field->val_string) == 0)
                                   cur_field->val_string) == 0)
                      {
                      {
                        value_type = cache_rule->entry_type;
                        value_type = cache_rule->entry_type;
                        value_line = cache_rule->line;
                        value_line = cache_rule->line;
                        if (value_type == compute_value)
                        if (value_type == compute_value)
                          {
                          {
                            options.warning (cache_rule->line,
                            options.warning (cache_rule->line,
                                             "instruction field of type `compute' changed to `cache'\n");
                                             "instruction field of type `compute' changed to `cache'\n");
                            cache_rule->entry_type = cache_value;
                            cache_rule->entry_type = cache_value;
                          }
                          }
                        break;
                        break;
                      }
                      }
                  }
                  }
                /* Define an entry for the field within the
                /* Define an entry for the field within the
                   instruction */
                   instruction */
                print_icache_extraction (file, instruction->format_name, value_type, cur_field->val_string,     /* name */
                print_icache_extraction (file, instruction->format_name, value_type, cur_field->val_string,     /* name */
                                         NULL,  /* type */
                                         NULL,  /* type */
                                         NULL,  /* expression */
                                         NULL,  /* expression */
                                         cur_field->val_string, /* insn field */
                                         cur_field->val_string, /* insn field */
                                         value_line,
                                         value_line,
                                         cur_field,
                                         cur_field,
                                         expanded_bits,
                                         expanded_bits,
                                         what_to_declare, what_to_do);
                                         what_to_declare, what_to_do);
              }
              }
          }
          }
      }
      }
  }
  }
 
 
  /* pass two - any cache fields not processed above */
  /* pass two - any cache fields not processed above */
  {
  {
    cache_entry *cache_rule;
    cache_entry *cache_rule;
    for (cache_rule = cache_rules;
    for (cache_rule = cache_rules;
         cache_rule != NULL; cache_rule = cache_rule->next)
         cache_rule != NULL; cache_rule = cache_rule->next)
      {
      {
        if (filter_is_subset (instruction->field_names,
        if (filter_is_subset (instruction->field_names,
                              cache_rule->original_fields)
                              cache_rule->original_fields)
            && !filter_is_member (instruction->field_names, cache_rule->name))
            && !filter_is_member (instruction->field_names, cache_rule->name))
          {
          {
            char *single_field =
            char *single_field =
              filter_next (cache_rule->original_fields, "");
              filter_next (cache_rule->original_fields, "");
            if (filter_next (cache_rule->original_fields, single_field) !=
            if (filter_next (cache_rule->original_fields, single_field) !=
                NULL)
                NULL)
              single_field = NULL;
              single_field = NULL;
            print_icache_extraction (file, instruction->format_name, cache_rule->entry_type, cache_rule->name, cache_rule->type, cache_rule->expression, single_field, cache_rule->line, NULL,  /* cur_field */
            print_icache_extraction (file, instruction->format_name, cache_rule->entry_type, cache_rule->name, cache_rule->type, cache_rule->expression, single_field, cache_rule->line, NULL,  /* cur_field */
                                     expanded_bits,
                                     expanded_bits,
                                     what_to_declare, what_to_do);
                                     what_to_declare, what_to_do);
          }
          }
      }
      }
  }
  }
 
 
  lf_print__internal_ref (file);
  lf_print__internal_ref (file);
}
}
 
 
 
 
 
 
typedef struct _form_fields form_fields;
typedef struct _form_fields form_fields;
struct _form_fields
struct _form_fields
{
{
  char *name;
  char *name;
  filter *fields;
  filter *fields;
  form_fields *next;
  form_fields *next;
};
};
 
 
static form_fields *
static form_fields *
insn_table_cache_fields (insn_table *isa)
insn_table_cache_fields (insn_table *isa)
{
{
  form_fields *forms = NULL;
  form_fields *forms = NULL;
  insn_entry *insn;
  insn_entry *insn;
  for (insn = isa->insns; insn != NULL; insn = insn->next)
  for (insn = isa->insns; insn != NULL; insn = insn->next)
    {
    {
      form_fields **form = &forms;
      form_fields **form = &forms;
      while (1)
      while (1)
        {
        {
          if (*form == NULL)
          if (*form == NULL)
            {
            {
              /* new format name, add it */
              /* new format name, add it */
              form_fields *new_form = ZALLOC (form_fields);
              form_fields *new_form = ZALLOC (form_fields);
              new_form->name = insn->format_name;
              new_form->name = insn->format_name;
              filter_add (&new_form->fields, insn->field_names);
              filter_add (&new_form->fields, insn->field_names);
              *form = new_form;
              *form = new_form;
              break;
              break;
            }
            }
          else if (strcmp ((*form)->name, insn->format_name) == 0)
          else if (strcmp ((*form)->name, insn->format_name) == 0)
            {
            {
              /* already present, add field names to the existing list */
              /* already present, add field names to the existing list */
              filter_add (&(*form)->fields, insn->field_names);
              filter_add (&(*form)->fields, insn->field_names);
              break;
              break;
            }
            }
          form = &(*form)->next;
          form = &(*form)->next;
        }
        }
    }
    }
  return forms;
  return forms;
}
}
 
 
 
 
 
 
extern void
extern void
print_icache_struct (lf *file, insn_table *isa, cache_entry *cache_rules)
print_icache_struct (lf *file, insn_table *isa, cache_entry *cache_rules)
{
{
  /* Create a list of all the different instruction formats with their
  /* Create a list of all the different instruction formats with their
     corresponding field names. */
     corresponding field names. */
  form_fields *formats = insn_table_cache_fields (isa);
  form_fields *formats = insn_table_cache_fields (isa);
 
 
  lf_printf (file, "\n");
  lf_printf (file, "\n");
  lf_printf (file, "#define WITH_%sIDECODE_CACHE_SIZE %d\n",
  lf_printf (file, "#define WITH_%sIDECODE_CACHE_SIZE %d\n",
             options.module.global.prefix.u,
             options.module.global.prefix.u,
             (options.gen.icache ? options.gen.icache_size : 0));
             (options.gen.icache ? options.gen.icache_size : 0));
  lf_printf (file, "\n");
  lf_printf (file, "\n");
 
 
  /* create an instruction cache if being used */
  /* create an instruction cache if being used */
  if (options.gen.icache)
  if (options.gen.icache)
    {
    {
      lf_printf (file, "typedef struct _%sidecode_cache {\n",
      lf_printf (file, "typedef struct _%sidecode_cache {\n",
                 options.module.global.prefix.l);
                 options.module.global.prefix.l);
      lf_indent (file, +2);
      lf_indent (file, +2);
      {
      {
        form_fields *format;
        form_fields *format;
        lf_printf (file, "unsigned_word address;\n");
        lf_printf (file, "unsigned_word address;\n");
        lf_printf (file, "void *semantic;\n");
        lf_printf (file, "void *semantic;\n");
        lf_printf (file, "union {\n");
        lf_printf (file, "union {\n");
        lf_indent (file, +2);
        lf_indent (file, +2);
        for (format = formats; format != NULL; format = format->next)
        for (format = formats; format != NULL; format = format->next)
          {
          {
            lf_printf (file, "struct {\n");
            lf_printf (file, "struct {\n");
            lf_indent (file, +2);
            lf_indent (file, +2);
            {
            {
              cache_entry *cache_rule;
              cache_entry *cache_rule;
              char *field;
              char *field;
              /* space for any instruction words */
              /* space for any instruction words */
              if (options.gen.insn_in_icache)
              if (options.gen.insn_in_icache)
                lf_printf (file, "instruction_word insn[%d];\n",
                lf_printf (file, "instruction_word insn[%d];\n",
                           isa->max_nr_words);
                           isa->max_nr_words);
              /* define an entry for any applicable cache rules */
              /* define an entry for any applicable cache rules */
              for (cache_rule = cache_rules;
              for (cache_rule = cache_rules;
                   cache_rule != NULL; cache_rule = cache_rule->next)
                   cache_rule != NULL; cache_rule = cache_rule->next)
                {
                {
                  /* nb - sort of correct - should really check against
                  /* nb - sort of correct - should really check against
                     individual instructions */
                     individual instructions */
                  if (filter_is_subset
                  if (filter_is_subset
                      (format->fields, cache_rule->original_fields))
                      (format->fields, cache_rule->original_fields))
                    {
                    {
                      char *memb;
                      char *memb;
                      lf_printf (file, "%s %s;",
                      lf_printf (file, "%s %s;",
                                 (cache_rule->type == NULL
                                 (cache_rule->type == NULL
                                  ? "unsigned"
                                  ? "unsigned"
                                  : cache_rule->type), cache_rule->name);
                                  : cache_rule->type), cache_rule->name);
                      lf_printf (file, " /*");
                      lf_printf (file, " /*");
                      for (memb =
                      for (memb =
                           filter_next (cache_rule->original_fields, "");
                           filter_next (cache_rule->original_fields, "");
                           memb != NULL;
                           memb != NULL;
                           memb =
                           memb =
                           filter_next (cache_rule->original_fields, memb))
                           filter_next (cache_rule->original_fields, memb))
                        {
                        {
                          lf_printf (file, " %s", memb);
                          lf_printf (file, " %s", memb);
                        }
                        }
                      lf_printf (file, " */\n");
                      lf_printf (file, " */\n");
                    }
                    }
                }
                }
              /* define an entry for any fields not covered by a cache rule */
              /* define an entry for any fields not covered by a cache rule */
              for (field = filter_next (format->fields, "");
              for (field = filter_next (format->fields, "");
                   field != NULL; field = filter_next (format->fields, field))
                   field != NULL; field = filter_next (format->fields, field))
                {
                {
                  cache_entry *cache_rule;
                  cache_entry *cache_rule;
                  int found_rule = 0;
                  int found_rule = 0;
                  for (cache_rule = cache_rules;
                  for (cache_rule = cache_rules;
                       cache_rule != NULL; cache_rule = cache_rule->next)
                       cache_rule != NULL; cache_rule = cache_rule->next)
                    {
                    {
                      if (strcmp (cache_rule->name, field) == 0)
                      if (strcmp (cache_rule->name, field) == 0)
                        {
                        {
                          found_rule = 1;
                          found_rule = 1;
                          break;
                          break;
                        }
                        }
                    }
                    }
                  if (!found_rule)
                  if (!found_rule)
                    lf_printf (file, "unsigned %s; /* default */\n", field);
                    lf_printf (file, "unsigned %s; /* default */\n", field);
                }
                }
            }
            }
            lf_indent (file, -2);
            lf_indent (file, -2);
            lf_printf (file, "} %s;\n", format->name);
            lf_printf (file, "} %s;\n", format->name);
          }
          }
        lf_indent (file, -2);
        lf_indent (file, -2);
        lf_printf (file, "} crack;\n");
        lf_printf (file, "} crack;\n");
      }
      }
      lf_indent (file, -2);
      lf_indent (file, -2);
      lf_printf (file, "} %sidecode_cache;\n",
      lf_printf (file, "} %sidecode_cache;\n",
                 options.module.global.prefix.l);
                 options.module.global.prefix.l);
    }
    }
  else
  else
    {
    {
      /* alernativly, since no cache, emit a dummy definition for
      /* alernativly, since no cache, emit a dummy definition for
         idecode_cache so that code refering to the type can still compile */
         idecode_cache so that code refering to the type can still compile */
      lf_printf (file, "typedef void %sidecode_cache;\n",
      lf_printf (file, "typedef void %sidecode_cache;\n",
                 options.module.global.prefix.l);
                 options.module.global.prefix.l);
    }
    }
  lf_printf (file, "\n");
  lf_printf (file, "\n");
}
}
 
 
 
 
 
 
static void
static void
print_icache_function (lf *file,
print_icache_function (lf *file,
                       insn_entry * instruction,
                       insn_entry * instruction,
                       opcode_bits *expanded_bits,
                       opcode_bits *expanded_bits,
                       insn_opcodes *opcodes,
                       insn_opcodes *opcodes,
                       cache_entry *cache_rules, int nr_prefetched_words)
                       cache_entry *cache_rules, int nr_prefetched_words)
{
{
  int indent;
  int indent;
 
 
  /* generate code to enter decoded instruction into the icache */
  /* generate code to enter decoded instruction into the icache */
  lf_printf (file, "\n");
  lf_printf (file, "\n");
  lf_print__function_type_function (file, print_icache_function_type,
  lf_print__function_type_function (file, print_icache_function_type,
                                    "EXTERN_ICACHE", "\n");
                                    "EXTERN_ICACHE", "\n");
  indent = print_function_name (file,
  indent = print_function_name (file,
                                instruction->name,
                                instruction->name,
                                instruction->format_name,
                                instruction->format_name,
                                NULL,
                                NULL,
                                expanded_bits, function_name_prefix_icache);
                                expanded_bits, function_name_prefix_icache);
  indent += lf_printf (file, " ");
  indent += lf_printf (file, " ");
  lf_indent (file, +indent);
  lf_indent (file, +indent);
  lf_printf (file, "(");
  lf_printf (file, "(");
  print_icache_function_formal (file, nr_prefetched_words);
  print_icache_function_formal (file, nr_prefetched_words);
  lf_printf (file, ")\n");
  lf_printf (file, ")\n");
  lf_indent (file, -indent);
  lf_indent (file, -indent);
 
 
  /* function header */
  /* function header */
  lf_printf (file, "{\n");
  lf_printf (file, "{\n");
  lf_indent (file, +2);
  lf_indent (file, +2);
 
 
  print_my_defines (file,
  print_my_defines (file,
                    instruction->name,
                    instruction->name,
                    instruction->format_name, expanded_bits);
                    instruction->format_name, expanded_bits);
  print_itrace (file, instruction, 1 /*putting-value-in-cache */ );
  print_itrace (file, instruction, 1 /*putting-value-in-cache */ );
 
 
  print_idecode_validate (file, instruction, opcodes);
  print_idecode_validate (file, instruction, opcodes);
 
 
  lf_printf (file, "\n");
  lf_printf (file, "\n");
  lf_printf (file, "{\n");
  lf_printf (file, "{\n");
  lf_indent (file, +2);
  lf_indent (file, +2);
  if (options.gen.semantic_icache)
  if (options.gen.semantic_icache)
    lf_printf (file, "unsigned_word nia;\n");
    lf_printf (file, "unsigned_word nia;\n");
  print_icache_body (file,
  print_icache_body (file,
                     instruction,
                     instruction,
                     expanded_bits,
                     expanded_bits,
                     cache_rules,
                     cache_rules,
                     (options.gen.direct_access
                     (options.gen.direct_access
                      ? define_variables
                      ? define_variables
                      : declare_variables),
                      : declare_variables),
                     (options.gen.semantic_icache
                     (options.gen.semantic_icache
                      ? both_values_and_icache
                      ? both_values_and_icache
                      : put_values_in_icache), nr_prefetched_words);
                      : put_values_in_icache), nr_prefetched_words);
 
 
  lf_printf (file, "\n");
  lf_printf (file, "\n");
  lf_printf (file, "cache_entry->address = cia;\n");
  lf_printf (file, "cache_entry->address = cia;\n");
  lf_printf (file, "cache_entry->semantic = ");
  lf_printf (file, "cache_entry->semantic = ");
  print_function_name (file,
  print_function_name (file,
                       instruction->name,
                       instruction->name,
                       instruction->format_name,
                       instruction->format_name,
                       NULL, expanded_bits, function_name_prefix_semantics);
                       NULL, expanded_bits, function_name_prefix_semantics);
  lf_printf (file, ";\n");
  lf_printf (file, ";\n");
  lf_printf (file, "\n");
  lf_printf (file, "\n");
 
 
  if (options.gen.semantic_icache)
  if (options.gen.semantic_icache)
    {
    {
      lf_printf (file, "/* semantic routine */\n");
      lf_printf (file, "/* semantic routine */\n");
      print_semantic_body (file, instruction, expanded_bits, opcodes);
      print_semantic_body (file, instruction, expanded_bits, opcodes);
      lf_printf (file, "return nia;\n");
      lf_printf (file, "return nia;\n");
    }
    }
 
 
  if (!options.gen.semantic_icache)
  if (!options.gen.semantic_icache)
    {
    {
      lf_printf (file, "/* return the function proper */\n");
      lf_printf (file, "/* return the function proper */\n");
      lf_printf (file, "return ");
      lf_printf (file, "return ");
      print_function_name (file,
      print_function_name (file,
                           instruction->name,
                           instruction->name,
                           instruction->format_name,
                           instruction->format_name,
                           NULL,
                           NULL,
                           expanded_bits, function_name_prefix_semantics);
                           expanded_bits, function_name_prefix_semantics);
      lf_printf (file, ";\n");
      lf_printf (file, ";\n");
    }
    }
 
 
  if (options.gen.direct_access)
  if (options.gen.direct_access)
    {
    {
      print_icache_body (file,
      print_icache_body (file,
                         instruction,
                         instruction,
                         expanded_bits,
                         expanded_bits,
                         cache_rules,
                         cache_rules,
                         undef_variables,
                         undef_variables,
                         (options.gen.semantic_icache
                         (options.gen.semantic_icache
                          ? both_values_and_icache
                          ? both_values_and_icache
                          : put_values_in_icache), nr_prefetched_words);
                          : put_values_in_icache), nr_prefetched_words);
    }
    }
 
 
  lf_indent (file, -2);
  lf_indent (file, -2);
  lf_printf (file, "}\n");
  lf_printf (file, "}\n");
  lf_indent (file, -2);
  lf_indent (file, -2);
  lf_printf (file, "}\n");
  lf_printf (file, "}\n");
}
}
 
 
 
 
void
void
print_icache_definition (lf *file,
print_icache_definition (lf *file,
                         insn_entry * insn,
                         insn_entry * insn,
                         opcode_bits *expanded_bits,
                         opcode_bits *expanded_bits,
                         insn_opcodes *opcodes,
                         insn_opcodes *opcodes,
                         cache_entry *cache_rules, int nr_prefetched_words)
                         cache_entry *cache_rules, int nr_prefetched_words)
{
{
  print_icache_function (file,
  print_icache_function (file,
                         insn,
                         insn,
                         expanded_bits,
                         expanded_bits,
                         opcodes, cache_rules, nr_prefetched_words);
                         opcodes, cache_rules, nr_prefetched_words);
}
}
 
 
 
 
 
 
void
void
print_icache_internal_function_declaration (lf *file,
print_icache_internal_function_declaration (lf *file,
                                            function_entry * function,
                                            function_entry * function,
                                            void *data)
                                            void *data)
{
{
  ASSERT (options.gen.icache);
  ASSERT (options.gen.icache);
  if (function->is_internal)
  if (function->is_internal)
    {
    {
      lf_printf (file, "\n");
      lf_printf (file, "\n");
      lf_print__function_type_function (file, print_icache_function_type,
      lf_print__function_type_function (file, print_icache_function_type,
                                        "INLINE_ICACHE", "\n");
                                        "INLINE_ICACHE", "\n");
      print_function_name (file,
      print_function_name (file,
                           function->name,
                           function->name,
                           NULL, NULL, NULL, function_name_prefix_icache);
                           NULL, NULL, NULL, function_name_prefix_icache);
      lf_printf (file, "\n(");
      lf_printf (file, "\n(");
      print_icache_function_formal (file, 0);
      print_icache_function_formal (file, 0);
      lf_printf (file, ");\n");
      lf_printf (file, ");\n");
    }
    }
}
}
 
 
 
 
void
void
print_icache_internal_function_definition (lf *file,
print_icache_internal_function_definition (lf *file,
                                           function_entry * function,
                                           function_entry * function,
                                           void *data)
                                           void *data)
{
{
  ASSERT (options.gen.icache);
  ASSERT (options.gen.icache);
  if (function->is_internal)
  if (function->is_internal)
    {
    {
      lf_printf (file, "\n");
      lf_printf (file, "\n");
      lf_print__function_type_function (file, print_icache_function_type,
      lf_print__function_type_function (file, print_icache_function_type,
                                        "INLINE_ICACHE", "\n");
                                        "INLINE_ICACHE", "\n");
      print_function_name (file,
      print_function_name (file,
                           function->name,
                           function->name,
                           NULL, NULL, NULL, function_name_prefix_icache);
                           NULL, NULL, NULL, function_name_prefix_icache);
      lf_printf (file, "\n(");
      lf_printf (file, "\n(");
      print_icache_function_formal (file, 0);
      print_icache_function_formal (file, 0);
      lf_printf (file, ")\n");
      lf_printf (file, ")\n");
      lf_printf (file, "{\n");
      lf_printf (file, "{\n");
      lf_indent (file, +2);
      lf_indent (file, +2);
      lf_printf (file, "/* semantic routine */\n");
      lf_printf (file, "/* semantic routine */\n");
      if (options.gen.semantic_icache)
      if (options.gen.semantic_icache)
        {
        {
          lf_print__line_ref (file, function->code->line);
          lf_print__line_ref (file, function->code->line);
          table_print_code (file, function->code);
          table_print_code (file, function->code);
          lf_printf (file,
          lf_printf (file,
                     "error (\"Internal function must longjump\\n\");\n");
                     "error (\"Internal function must longjump\\n\");\n");
          lf_printf (file, "return 0;\n");
          lf_printf (file, "return 0;\n");
        }
        }
      else
      else
        {
        {
          lf_printf (file, "return ");
          lf_printf (file, "return ");
          print_function_name (file,
          print_function_name (file,
                               function->name,
                               function->name,
                               NULL,
                               NULL,
                               NULL, NULL, function_name_prefix_semantics);
                               NULL, NULL, function_name_prefix_semantics);
          lf_printf (file, ";\n");
          lf_printf (file, ";\n");
        }
        }
 
 
      lf_print__internal_ref (file);
      lf_print__internal_ref (file);
      lf_indent (file, -2);
      lf_indent (file, -2);
      lf_printf (file, "}\n");
      lf_printf (file, "}\n");
    }
    }
}
}
 
 

powered by: WebSVN 2.1.0

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