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

Subversion Repositories openrisc

[/] [openrisc/] [trunk/] [gnu-stable/] [binutils-2.20.1/] [opcodes/] [cgen-opc.c] - Diff between revs 816 and 818

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

Rev 816 Rev 818
/* CGEN generic opcode support.
/* CGEN generic opcode support.
 
 
   Copyright 1996, 1997, 1998, 1999, 2000, 2001, 2003, 2005, 2007, 2009
   Copyright 1996, 1997, 1998, 1999, 2000, 2001, 2003, 2005, 2007, 2009
   Free Software Foundation, Inc.
   Free Software Foundation, Inc.
 
 
   This file is part of libopcodes.
   This file is part of libopcodes.
 
 
   This library is free software; you can redistribute it and/or modify
   This library 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, or (at your option)
   the Free Software Foundation; either version 3, or (at your option)
   any later version.
   any later version.
 
 
   It is distributed in the hope that it will be useful, but WITHOUT
   It is distributed in the hope that it will be useful, but WITHOUT
   ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
   ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
   or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public
   or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public
   License for more details.
   License for more details.
 
 
   You should have received a copy of the GNU General Public License along
   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.,
   with this program; if not, write to the Free Software Foundation, Inc.,
   51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA.  */
   51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA.  */
 
 
#include "alloca-conf.h"
#include "alloca-conf.h"
#include "sysdep.h"
#include "sysdep.h"
#include <stdio.h>
#include <stdio.h>
#include "ansidecl.h"
#include "ansidecl.h"
#include "libiberty.h"
#include "libiberty.h"
#include "safe-ctype.h"
#include "safe-ctype.h"
#include "bfd.h"
#include "bfd.h"
#include "symcat.h"
#include "symcat.h"
#include "opcode/cgen.h"
#include "opcode/cgen.h"
 
 
static unsigned int hash_keyword_name
static unsigned int hash_keyword_name
  (const CGEN_KEYWORD *, const char *, int);
  (const CGEN_KEYWORD *, const char *, int);
static unsigned int hash_keyword_value
static unsigned int hash_keyword_value
  (const CGEN_KEYWORD *, unsigned int);
  (const CGEN_KEYWORD *, unsigned int);
static void build_keyword_hash_tables
static void build_keyword_hash_tables
  (CGEN_KEYWORD *);
  (CGEN_KEYWORD *);
 
 
/* Return number of hash table entries to use for N elements.  */
/* Return number of hash table entries to use for N elements.  */
#define KEYWORD_HASH_SIZE(n) ((n) <= 31 ? 17 : 31)
#define KEYWORD_HASH_SIZE(n) ((n) <= 31 ? 17 : 31)
 
 
/* Look up *NAMEP in the keyword table KT.
/* Look up *NAMEP in the keyword table KT.
   The result is the keyword entry or NULL if not found.  */
   The result is the keyword entry or NULL if not found.  */
 
 
const CGEN_KEYWORD_ENTRY *
const CGEN_KEYWORD_ENTRY *
cgen_keyword_lookup_name (CGEN_KEYWORD *kt, const char *name)
cgen_keyword_lookup_name (CGEN_KEYWORD *kt, const char *name)
{
{
  const CGEN_KEYWORD_ENTRY *ke;
  const CGEN_KEYWORD_ENTRY *ke;
  const char *p,*n;
  const char *p,*n;
 
 
  if (kt->name_hash_table == NULL)
  if (kt->name_hash_table == NULL)
    build_keyword_hash_tables (kt);
    build_keyword_hash_tables (kt);
 
 
  ke = kt->name_hash_table[hash_keyword_name (kt, name, 0)];
  ke = kt->name_hash_table[hash_keyword_name (kt, name, 0)];
 
 
  /* We do case insensitive comparisons.
  /* We do case insensitive comparisons.
     If that ever becomes a problem, add an attribute that denotes
     If that ever becomes a problem, add an attribute that denotes
     "do case sensitive comparisons".  */
     "do case sensitive comparisons".  */
 
 
  while (ke != NULL)
  while (ke != NULL)
    {
    {
      n = name;
      n = name;
      p = ke->name;
      p = ke->name;
 
 
      while (*p
      while (*p
             && (*p == *n
             && (*p == *n
                 || (ISALPHA (*p) && (TOLOWER (*p) == TOLOWER (*n)))))
                 || (ISALPHA (*p) && (TOLOWER (*p) == TOLOWER (*n)))))
        ++n, ++p;
        ++n, ++p;
 
 
      if (!*p && !*n)
      if (!*p && !*n)
        return ke;
        return ke;
 
 
      ke = ke->next_name;
      ke = ke->next_name;
    }
    }
 
 
  if (kt->null_entry)
  if (kt->null_entry)
    return kt->null_entry;
    return kt->null_entry;
  return NULL;
  return NULL;
}
}
 
 
/* Look up VALUE in the keyword table KT.
/* Look up VALUE in the keyword table KT.
   The result is the keyword entry or NULL if not found.  */
   The result is the keyword entry or NULL if not found.  */
 
 
const CGEN_KEYWORD_ENTRY *
const CGEN_KEYWORD_ENTRY *
cgen_keyword_lookup_value (CGEN_KEYWORD *kt, int value)
cgen_keyword_lookup_value (CGEN_KEYWORD *kt, int value)
{
{
  const CGEN_KEYWORD_ENTRY *ke;
  const CGEN_KEYWORD_ENTRY *ke;
 
 
  if (kt->name_hash_table == NULL)
  if (kt->name_hash_table == NULL)
    build_keyword_hash_tables (kt);
    build_keyword_hash_tables (kt);
 
 
  ke = kt->value_hash_table[hash_keyword_value (kt, value)];
  ke = kt->value_hash_table[hash_keyword_value (kt, value)];
 
 
  while (ke != NULL)
  while (ke != NULL)
    {
    {
      if (value == ke->value)
      if (value == ke->value)
        return ke;
        return ke;
      ke = ke->next_value;
      ke = ke->next_value;
    }
    }
 
 
  return NULL;
  return NULL;
}
}
 
 
/* Add an entry to a keyword table.  */
/* Add an entry to a keyword table.  */
 
 
void
void
cgen_keyword_add (CGEN_KEYWORD *kt, CGEN_KEYWORD_ENTRY *ke)
cgen_keyword_add (CGEN_KEYWORD *kt, CGEN_KEYWORD_ENTRY *ke)
{
{
  unsigned int hash;
  unsigned int hash;
  size_t i;
  size_t i;
 
 
  if (kt->name_hash_table == NULL)
  if (kt->name_hash_table == NULL)
    build_keyword_hash_tables (kt);
    build_keyword_hash_tables (kt);
 
 
  hash = hash_keyword_name (kt, ke->name, 0);
  hash = hash_keyword_name (kt, ke->name, 0);
  ke->next_name = kt->name_hash_table[hash];
  ke->next_name = kt->name_hash_table[hash];
  kt->name_hash_table[hash] = ke;
  kt->name_hash_table[hash] = ke;
 
 
  hash = hash_keyword_value (kt, ke->value);
  hash = hash_keyword_value (kt, ke->value);
  ke->next_value = kt->value_hash_table[hash];
  ke->next_value = kt->value_hash_table[hash];
  kt->value_hash_table[hash] = ke;
  kt->value_hash_table[hash] = ke;
 
 
  if (ke->name[0] == 0)
  if (ke->name[0] == 0)
    kt->null_entry = ke;
    kt->null_entry = ke;
 
 
  for (i = 1; i < strlen (ke->name); i++)
  for (i = 1; i < strlen (ke->name); i++)
    if (! ISALNUM (ke->name[i])
    if (! ISALNUM (ke->name[i])
        && ! strchr (kt->nonalpha_chars, ke->name[i]))
        && ! strchr (kt->nonalpha_chars, ke->name[i]))
      {
      {
        size_t idx = strlen (kt->nonalpha_chars);
        size_t idx = strlen (kt->nonalpha_chars);
 
 
        /* If you hit this limit, please don't just
        /* If you hit this limit, please don't just
           increase the size of the field, instead
           increase the size of the field, instead
           look for a better algorithm.  */
           look for a better algorithm.  */
        if (idx >= sizeof (kt->nonalpha_chars) - 1)
        if (idx >= sizeof (kt->nonalpha_chars) - 1)
          abort ();
          abort ();
        kt->nonalpha_chars[idx] = ke->name[i];
        kt->nonalpha_chars[idx] = ke->name[i];
        kt->nonalpha_chars[idx+1] = 0;
        kt->nonalpha_chars[idx+1] = 0;
      }
      }
}
}
 
 
/* FIXME: Need function to return count of keywords.  */
/* FIXME: Need function to return count of keywords.  */
 
 
/* Initialize a keyword table search.
/* Initialize a keyword table search.
   SPEC is a specification of what to search for.
   SPEC is a specification of what to search for.
   A value of NULL means to find every keyword.
   A value of NULL means to find every keyword.
   Currently NULL is the only acceptable value [further specification
   Currently NULL is the only acceptable value [further specification
   deferred].
   deferred].
   The result is an opaque data item used to record the search status.
   The result is an opaque data item used to record the search status.
   It is passed to each call to cgen_keyword_search_next.  */
   It is passed to each call to cgen_keyword_search_next.  */
 
 
CGEN_KEYWORD_SEARCH
CGEN_KEYWORD_SEARCH
cgen_keyword_search_init (CGEN_KEYWORD *kt, const char *spec)
cgen_keyword_search_init (CGEN_KEYWORD *kt, const char *spec)
{
{
  CGEN_KEYWORD_SEARCH search;
  CGEN_KEYWORD_SEARCH search;
 
 
  /* FIXME: Need to specify format of params.  */
  /* FIXME: Need to specify format of params.  */
  if (spec != NULL)
  if (spec != NULL)
    abort ();
    abort ();
 
 
  if (kt->name_hash_table == NULL)
  if (kt->name_hash_table == NULL)
    build_keyword_hash_tables (kt);
    build_keyword_hash_tables (kt);
 
 
  search.table = kt;
  search.table = kt;
  search.spec = spec;
  search.spec = spec;
  search.current_hash = 0;
  search.current_hash = 0;
  search.current_entry = NULL;
  search.current_entry = NULL;
  return search;
  return search;
}
}
 
 
/* Return the next keyword specified by SEARCH.
/* Return the next keyword specified by SEARCH.
   The result is the next entry or NULL if there are no more.  */
   The result is the next entry or NULL if there are no more.  */
 
 
const CGEN_KEYWORD_ENTRY *
const CGEN_KEYWORD_ENTRY *
cgen_keyword_search_next (CGEN_KEYWORD_SEARCH *search)
cgen_keyword_search_next (CGEN_KEYWORD_SEARCH *search)
{
{
  /* Has search finished?  */
  /* Has search finished?  */
  if (search->current_hash == search->table->hash_table_size)
  if (search->current_hash == search->table->hash_table_size)
    return NULL;
    return NULL;
 
 
  /* Search in progress?  */
  /* Search in progress?  */
  if (search->current_entry != NULL
  if (search->current_entry != NULL
      /* Anything left on this hash chain?  */
      /* Anything left on this hash chain?  */
      && search->current_entry->next_name != NULL)
      && search->current_entry->next_name != NULL)
    {
    {
      search->current_entry = search->current_entry->next_name;
      search->current_entry = search->current_entry->next_name;
      return search->current_entry;
      return search->current_entry;
    }
    }
 
 
  /* Move to next hash chain [unless we haven't started yet].  */
  /* Move to next hash chain [unless we haven't started yet].  */
  if (search->current_entry != NULL)
  if (search->current_entry != NULL)
    ++search->current_hash;
    ++search->current_hash;
 
 
  while (search->current_hash < search->table->hash_table_size)
  while (search->current_hash < search->table->hash_table_size)
    {
    {
      search->current_entry = search->table->name_hash_table[search->current_hash];
      search->current_entry = search->table->name_hash_table[search->current_hash];
      if (search->current_entry != NULL)
      if (search->current_entry != NULL)
        return search->current_entry;
        return search->current_entry;
      ++search->current_hash;
      ++search->current_hash;
    }
    }
 
 
  return NULL;
  return NULL;
}
}
 
 
/* Return first entry in hash chain for NAME.
/* Return first entry in hash chain for NAME.
   If CASE_SENSITIVE_P is non-zero, return a case sensitive hash.  */
   If CASE_SENSITIVE_P is non-zero, return a case sensitive hash.  */
 
 
static unsigned int
static unsigned int
hash_keyword_name (const CGEN_KEYWORD *kt,
hash_keyword_name (const CGEN_KEYWORD *kt,
                   const char *name,
                   const char *name,
                   int case_sensitive_p)
                   int case_sensitive_p)
{
{
  unsigned int hash;
  unsigned int hash;
 
 
  if (case_sensitive_p)
  if (case_sensitive_p)
    for (hash = 0; *name; ++name)
    for (hash = 0; *name; ++name)
      hash = (hash * 97) + (unsigned char) *name;
      hash = (hash * 97) + (unsigned char) *name;
  else
  else
    for (hash = 0; *name; ++name)
    for (hash = 0; *name; ++name)
      hash = (hash * 97) + (unsigned char) TOLOWER (*name);
      hash = (hash * 97) + (unsigned char) TOLOWER (*name);
  return hash % kt->hash_table_size;
  return hash % kt->hash_table_size;
}
}
 
 
/* Return first entry in hash chain for VALUE.  */
/* Return first entry in hash chain for VALUE.  */
 
 
static unsigned int
static unsigned int
hash_keyword_value (const CGEN_KEYWORD *kt, unsigned int value)
hash_keyword_value (const CGEN_KEYWORD *kt, unsigned int value)
{
{
  return value % kt->hash_table_size;
  return value % kt->hash_table_size;
}
}
 
 
/* Build a keyword table's hash tables.
/* Build a keyword table's hash tables.
   We probably needn't build the value hash table for the assembler when
   We probably needn't build the value hash table for the assembler when
   we're using the disassembler, but we keep things simple.  */
   we're using the disassembler, but we keep things simple.  */
 
 
static void
static void
build_keyword_hash_tables (CGEN_KEYWORD *kt)
build_keyword_hash_tables (CGEN_KEYWORD *kt)
{
{
  int i;
  int i;
  /* Use the number of compiled in entries as an estimate for the
  /* Use the number of compiled in entries as an estimate for the
     typical sized table [not too many added at runtime].  */
     typical sized table [not too many added at runtime].  */
  unsigned int size = KEYWORD_HASH_SIZE (kt->num_init_entries);
  unsigned int size = KEYWORD_HASH_SIZE (kt->num_init_entries);
 
 
  kt->hash_table_size = size;
  kt->hash_table_size = size;
  kt->name_hash_table = (CGEN_KEYWORD_ENTRY **)
  kt->name_hash_table = (CGEN_KEYWORD_ENTRY **)
    xmalloc (size * sizeof (CGEN_KEYWORD_ENTRY *));
    xmalloc (size * sizeof (CGEN_KEYWORD_ENTRY *));
  memset (kt->name_hash_table, 0, size * sizeof (CGEN_KEYWORD_ENTRY *));
  memset (kt->name_hash_table, 0, size * sizeof (CGEN_KEYWORD_ENTRY *));
  kt->value_hash_table = (CGEN_KEYWORD_ENTRY **)
  kt->value_hash_table = (CGEN_KEYWORD_ENTRY **)
    xmalloc (size * sizeof (CGEN_KEYWORD_ENTRY *));
    xmalloc (size * sizeof (CGEN_KEYWORD_ENTRY *));
  memset (kt->value_hash_table, 0, size * sizeof (CGEN_KEYWORD_ENTRY *));
  memset (kt->value_hash_table, 0, size * sizeof (CGEN_KEYWORD_ENTRY *));
 
 
  /* The table is scanned backwards as we want keywords appearing earlier to
  /* The table is scanned backwards as we want keywords appearing earlier to
     be prefered over later ones.  */
     be prefered over later ones.  */
  for (i = kt->num_init_entries - 1; i >= 0; --i)
  for (i = kt->num_init_entries - 1; i >= 0; --i)
    cgen_keyword_add (kt, &kt->init_entries[i]);
    cgen_keyword_add (kt, &kt->init_entries[i]);
}
}


/* Hardware support.  */
/* Hardware support.  */
 
 
/* Lookup a hardware element by its name.
/* Lookup a hardware element by its name.
   Returns NULL if NAME is not supported by the currently selected
   Returns NULL if NAME is not supported by the currently selected
   mach/isa.  */
   mach/isa.  */
 
 
const CGEN_HW_ENTRY *
const CGEN_HW_ENTRY *
cgen_hw_lookup_by_name (CGEN_CPU_DESC cd, const char *name)
cgen_hw_lookup_by_name (CGEN_CPU_DESC cd, const char *name)
{
{
  unsigned int i;
  unsigned int i;
  const CGEN_HW_ENTRY **hw = cd->hw_table.entries;
  const CGEN_HW_ENTRY **hw = cd->hw_table.entries;
 
 
  for (i = 0; i < cd->hw_table.num_entries; ++i)
  for (i = 0; i < cd->hw_table.num_entries; ++i)
    if (hw[i] && strcmp (name, hw[i]->name) == 0)
    if (hw[i] && strcmp (name, hw[i]->name) == 0)
      return hw[i];
      return hw[i];
 
 
  return NULL;
  return NULL;
}
}
 
 
/* Lookup a hardware element by its number.
/* Lookup a hardware element by its number.
   Hardware elements are enumerated, however it may be possible to add some
   Hardware elements are enumerated, however it may be possible to add some
   at runtime, thus HWNUM is not an enum type but rather an int.
   at runtime, thus HWNUM is not an enum type but rather an int.
   Returns NULL if HWNUM is not supported by the currently selected mach.  */
   Returns NULL if HWNUM is not supported by the currently selected mach.  */
 
 
const CGEN_HW_ENTRY *
const CGEN_HW_ENTRY *
cgen_hw_lookup_by_num (CGEN_CPU_DESC cd, unsigned int hwnum)
cgen_hw_lookup_by_num (CGEN_CPU_DESC cd, unsigned int hwnum)
{
{
  unsigned int i;
  unsigned int i;
  const CGEN_HW_ENTRY **hw = cd->hw_table.entries;
  const CGEN_HW_ENTRY **hw = cd->hw_table.entries;
 
 
  /* ??? This can be speeded up.  */
  /* ??? This can be speeded up.  */
  for (i = 0; i < cd->hw_table.num_entries; ++i)
  for (i = 0; i < cd->hw_table.num_entries; ++i)
    if (hw[i] && hwnum == hw[i]->type)
    if (hw[i] && hwnum == hw[i]->type)
      return hw[i];
      return hw[i];
 
 
  return NULL;
  return NULL;
}
}


/* Operand support.  */
/* Operand support.  */
 
 
/* Lookup an operand by its name.
/* Lookup an operand by its name.
   Returns NULL if NAME is not supported by the currently selected
   Returns NULL if NAME is not supported by the currently selected
   mach/isa.  */
   mach/isa.  */
 
 
const CGEN_OPERAND *
const CGEN_OPERAND *
cgen_operand_lookup_by_name (CGEN_CPU_DESC cd, const char *name)
cgen_operand_lookup_by_name (CGEN_CPU_DESC cd, const char *name)
{
{
  unsigned int i;
  unsigned int i;
  const CGEN_OPERAND **op = cd->operand_table.entries;
  const CGEN_OPERAND **op = cd->operand_table.entries;
 
 
  for (i = 0; i < cd->operand_table.num_entries; ++i)
  for (i = 0; i < cd->operand_table.num_entries; ++i)
    if (op[i] && strcmp (name, op[i]->name) == 0)
    if (op[i] && strcmp (name, op[i]->name) == 0)
      return op[i];
      return op[i];
 
 
  return NULL;
  return NULL;
}
}
 
 
/* Lookup an operand by its number.
/* Lookup an operand by its number.
   Operands are enumerated, however it may be possible to add some
   Operands are enumerated, however it may be possible to add some
   at runtime, thus OPNUM is not an enum type but rather an int.
   at runtime, thus OPNUM is not an enum type but rather an int.
   Returns NULL if OPNUM is not supported by the currently selected
   Returns NULL if OPNUM is not supported by the currently selected
   mach/isa.  */
   mach/isa.  */
 
 
const CGEN_OPERAND *
const CGEN_OPERAND *
cgen_operand_lookup_by_num (CGEN_CPU_DESC cd, int opnum)
cgen_operand_lookup_by_num (CGEN_CPU_DESC cd, int opnum)
{
{
  return cd->operand_table.entries[opnum];
  return cd->operand_table.entries[opnum];
}
}


/* Instruction support.  */
/* Instruction support.  */
 
 
/* Return number of instructions.  This includes any added at runtime.  */
/* Return number of instructions.  This includes any added at runtime.  */
 
 
int
int
cgen_insn_count (CGEN_CPU_DESC cd)
cgen_insn_count (CGEN_CPU_DESC cd)
{
{
  int count = cd->insn_table.num_init_entries;
  int count = cd->insn_table.num_init_entries;
  CGEN_INSN_LIST *rt_insns = cd->insn_table.new_entries;
  CGEN_INSN_LIST *rt_insns = cd->insn_table.new_entries;
 
 
  for ( ; rt_insns != NULL; rt_insns = rt_insns->next)
  for ( ; rt_insns != NULL; rt_insns = rt_insns->next)
    ++count;
    ++count;
 
 
  return count;
  return count;
}
}
 
 
/* Return number of macro-instructions.
/* Return number of macro-instructions.
   This includes any added at runtime.  */
   This includes any added at runtime.  */
 
 
int
int
cgen_macro_insn_count (CGEN_CPU_DESC cd)
cgen_macro_insn_count (CGEN_CPU_DESC cd)
{
{
  int count = cd->macro_insn_table.num_init_entries;
  int count = cd->macro_insn_table.num_init_entries;
  CGEN_INSN_LIST *rt_insns = cd->macro_insn_table.new_entries;
  CGEN_INSN_LIST *rt_insns = cd->macro_insn_table.new_entries;
 
 
  for ( ; rt_insns != NULL; rt_insns = rt_insns->next)
  for ( ; rt_insns != NULL; rt_insns = rt_insns->next)
    ++count;
    ++count;
 
 
  return count;
  return count;
}
}
 
 
/* Cover function to read and properly byteswap an insn value.  */
/* Cover function to read and properly byteswap an insn value.  */
 
 
CGEN_INSN_INT
CGEN_INSN_INT
cgen_get_insn_value (CGEN_CPU_DESC cd, unsigned char *buf, int length)
cgen_get_insn_value (CGEN_CPU_DESC cd, unsigned char *buf, int length)
{
{
  int big_p = (cd->insn_endian == CGEN_ENDIAN_BIG);
  int big_p = (cd->insn_endian == CGEN_ENDIAN_BIG);
  int insn_chunk_bitsize = cd->insn_chunk_bitsize;
  int insn_chunk_bitsize = cd->insn_chunk_bitsize;
  CGEN_INSN_INT value = 0;
  CGEN_INSN_INT value = 0;
 
 
  if (insn_chunk_bitsize != 0 && insn_chunk_bitsize < length)
  if (insn_chunk_bitsize != 0 && insn_chunk_bitsize < length)
    {
    {
      /* We need to divide up the incoming value into insn_chunk_bitsize-length
      /* We need to divide up the incoming value into insn_chunk_bitsize-length
         segments, and endian-convert them, one at a time. */
         segments, and endian-convert them, one at a time. */
      int i;
      int i;
 
 
      /* Enforce divisibility. */
      /* Enforce divisibility. */
      if ((length % insn_chunk_bitsize) != 0)
      if ((length % insn_chunk_bitsize) != 0)
        abort ();
        abort ();
 
 
      for (i = 0; i < length; i += insn_chunk_bitsize) /* NB: i == bits */
      for (i = 0; i < length; i += insn_chunk_bitsize) /* NB: i == bits */
        {
        {
          int index;
          int index;
          bfd_vma this_value;
          bfd_vma this_value;
          index = i; /* NB: not dependent on endianness; opposite of cgen_put_insn_value! */
          index = i; /* NB: not dependent on endianness; opposite of cgen_put_insn_value! */
          this_value = bfd_get_bits (& buf[index / 8], insn_chunk_bitsize, big_p);
          this_value = bfd_get_bits (& buf[index / 8], insn_chunk_bitsize, big_p);
          value = (value << insn_chunk_bitsize) | this_value;
          value = (value << insn_chunk_bitsize) | this_value;
        }
        }
    }
    }
  else
  else
    {
    {
      value = bfd_get_bits (buf, length, cd->insn_endian == CGEN_ENDIAN_BIG);
      value = bfd_get_bits (buf, length, cd->insn_endian == CGEN_ENDIAN_BIG);
    }
    }
 
 
  return value;
  return value;
}
}
 
 
/* Cover function to store an insn value properly byteswapped.  */
/* Cover function to store an insn value properly byteswapped.  */
 
 
void
void
cgen_put_insn_value (CGEN_CPU_DESC cd,
cgen_put_insn_value (CGEN_CPU_DESC cd,
                     unsigned char *buf,
                     unsigned char *buf,
                     int length,
                     int length,
                     CGEN_INSN_INT value)
                     CGEN_INSN_INT value)
{
{
  int big_p = (cd->insn_endian == CGEN_ENDIAN_BIG);
  int big_p = (cd->insn_endian == CGEN_ENDIAN_BIG);
  int insn_chunk_bitsize = cd->insn_chunk_bitsize;
  int insn_chunk_bitsize = cd->insn_chunk_bitsize;
 
 
  if (insn_chunk_bitsize != 0 && insn_chunk_bitsize < length)
  if (insn_chunk_bitsize != 0 && insn_chunk_bitsize < length)
    {
    {
      /* We need to divide up the incoming value into insn_chunk_bitsize-length
      /* We need to divide up the incoming value into insn_chunk_bitsize-length
         segments, and endian-convert them, one at a time. */
         segments, and endian-convert them, one at a time. */
      int i;
      int i;
 
 
      /* Enforce divisibility. */
      /* Enforce divisibility. */
      if ((length % insn_chunk_bitsize) != 0)
      if ((length % insn_chunk_bitsize) != 0)
        abort ();
        abort ();
 
 
      for (i = 0; i < length; i += insn_chunk_bitsize) /* NB: i == bits */
      for (i = 0; i < length; i += insn_chunk_bitsize) /* NB: i == bits */
        {
        {
          int index;
          int index;
          index = (length - insn_chunk_bitsize - i); /* NB: not dependent on endianness! */
          index = (length - insn_chunk_bitsize - i); /* NB: not dependent on endianness! */
          bfd_put_bits ((bfd_vma) value, & buf[index / 8], insn_chunk_bitsize, big_p);
          bfd_put_bits ((bfd_vma) value, & buf[index / 8], insn_chunk_bitsize, big_p);
          value >>= insn_chunk_bitsize;
          value >>= insn_chunk_bitsize;
        }
        }
    }
    }
  else
  else
    {
    {
      bfd_put_bits ((bfd_vma) value, buf, length, big_p);
      bfd_put_bits ((bfd_vma) value, buf, length, big_p);
    }
    }
}
}


/* Look up instruction INSN_*_VALUE and extract its fields.
/* Look up instruction INSN_*_VALUE and extract its fields.
   INSN_INT_VALUE is used if CGEN_INT_INSN_P.
   INSN_INT_VALUE is used if CGEN_INT_INSN_P.
   Otherwise INSN_BYTES_VALUE is used.
   Otherwise INSN_BYTES_VALUE is used.
   INSN, if non-null, is the insn table entry.
   INSN, if non-null, is the insn table entry.
   Otherwise INSN_*_VALUE is examined to compute it.
   Otherwise INSN_*_VALUE is examined to compute it.
   LENGTH is the bit length of INSN_*_VALUE if known, otherwise 0.
   LENGTH is the bit length of INSN_*_VALUE if known, otherwise 0.
   0 is only valid if `insn == NULL && ! CGEN_INT_INSN_P'.
   0 is only valid if `insn == NULL && ! CGEN_INT_INSN_P'.
   If INSN != NULL, LENGTH must be valid.
   If INSN != NULL, LENGTH must be valid.
   ALIAS_P is non-zero if alias insns are to be included in the search.
   ALIAS_P is non-zero if alias insns are to be included in the search.
 
 
   The result is a pointer to the insn table entry, or NULL if the instruction
   The result is a pointer to the insn table entry, or NULL if the instruction
   wasn't recognized.  */
   wasn't recognized.  */
 
 
/* ??? Will need to be revisited for VLIW architectures.  */
/* ??? Will need to be revisited for VLIW architectures.  */
 
 
const CGEN_INSN *
const CGEN_INSN *
cgen_lookup_insn (CGEN_CPU_DESC cd,
cgen_lookup_insn (CGEN_CPU_DESC cd,
                  const CGEN_INSN *insn,
                  const CGEN_INSN *insn,
                  CGEN_INSN_INT insn_int_value,
                  CGEN_INSN_INT insn_int_value,
                  /* ??? CGEN_INSN_BYTES would be a nice type name to use here.  */
                  /* ??? CGEN_INSN_BYTES would be a nice type name to use here.  */
                  unsigned char *insn_bytes_value,
                  unsigned char *insn_bytes_value,
                  int length,
                  int length,
                  CGEN_FIELDS *fields,
                  CGEN_FIELDS *fields,
                  int alias_p)
                  int alias_p)
{
{
  unsigned char *buf;
  unsigned char *buf;
  CGEN_INSN_INT base_insn;
  CGEN_INSN_INT base_insn;
  CGEN_EXTRACT_INFO ex_info;
  CGEN_EXTRACT_INFO ex_info;
  CGEN_EXTRACT_INFO *info;
  CGEN_EXTRACT_INFO *info;
 
 
  if (cd->int_insn_p)
  if (cd->int_insn_p)
    {
    {
      info = NULL;
      info = NULL;
      buf = (unsigned char *) alloca (cd->max_insn_bitsize / 8);
      buf = (unsigned char *) alloca (cd->max_insn_bitsize / 8);
      cgen_put_insn_value (cd, buf, length, insn_int_value);
      cgen_put_insn_value (cd, buf, length, insn_int_value);
      base_insn = insn_int_value;
      base_insn = insn_int_value;
    }
    }
  else
  else
    {
    {
      info = &ex_info;
      info = &ex_info;
      ex_info.dis_info = NULL;
      ex_info.dis_info = NULL;
      ex_info.insn_bytes = insn_bytes_value;
      ex_info.insn_bytes = insn_bytes_value;
      ex_info.valid = -1;
      ex_info.valid = -1;
      buf = insn_bytes_value;
      buf = insn_bytes_value;
      base_insn = cgen_get_insn_value (cd, buf, length);
      base_insn = cgen_get_insn_value (cd, buf, length);
    }
    }
 
 
  if (!insn)
  if (!insn)
    {
    {
      const CGEN_INSN_LIST *insn_list;
      const CGEN_INSN_LIST *insn_list;
 
 
      /* The instructions are stored in hash lists.
      /* The instructions are stored in hash lists.
         Pick the first one and keep trying until we find the right one.  */
         Pick the first one and keep trying until we find the right one.  */
 
 
      insn_list = cgen_dis_lookup_insn (cd, (char *) buf, base_insn);
      insn_list = cgen_dis_lookup_insn (cd, (char *) buf, base_insn);
      while (insn_list != NULL)
      while (insn_list != NULL)
        {
        {
          insn = insn_list->insn;
          insn = insn_list->insn;
 
 
          if (alias_p
          if (alias_p
              /* FIXME: Ensure ALIAS attribute always has same index.  */
              /* FIXME: Ensure ALIAS attribute always has same index.  */
              || ! CGEN_INSN_ATTR_VALUE (insn, CGEN_INSN_ALIAS))
              || ! CGEN_INSN_ATTR_VALUE (insn, CGEN_INSN_ALIAS))
            {
            {
              /* Basic bit mask must be correct.  */
              /* Basic bit mask must be correct.  */
              /* ??? May wish to allow target to defer this check until the
              /* ??? May wish to allow target to defer this check until the
                 extract handler.  */
                 extract handler.  */
              if ((base_insn & CGEN_INSN_BASE_MASK (insn))
              if ((base_insn & CGEN_INSN_BASE_MASK (insn))
                  == CGEN_INSN_BASE_VALUE (insn))
                  == CGEN_INSN_BASE_VALUE (insn))
                {
                {
                  /* ??? 0 is passed for `pc' */
                  /* ??? 0 is passed for `pc' */
                  int elength = CGEN_EXTRACT_FN (cd, insn)
                  int elength = CGEN_EXTRACT_FN (cd, insn)
                    (cd, insn, info, base_insn, fields, (bfd_vma) 0);
                    (cd, insn, info, base_insn, fields, (bfd_vma) 0);
                  if (elength > 0)
                  if (elength > 0)
                    {
                    {
                      /* sanity check */
                      /* sanity check */
                      if (length != 0 && length != elength)
                      if (length != 0 && length != elength)
                        abort ();
                        abort ();
                      return insn;
                      return insn;
                    }
                    }
                }
                }
            }
            }
 
 
          insn_list = insn_list->next;
          insn_list = insn_list->next;
        }
        }
    }
    }
  else
  else
    {
    {
      /* Sanity check: can't pass an alias insn if ! alias_p.  */
      /* Sanity check: can't pass an alias insn if ! alias_p.  */
      if (! alias_p
      if (! alias_p
          && CGEN_INSN_ATTR_VALUE (insn, CGEN_INSN_ALIAS))
          && CGEN_INSN_ATTR_VALUE (insn, CGEN_INSN_ALIAS))
        abort ();
        abort ();
      /* Sanity check: length must be correct.  */
      /* Sanity check: length must be correct.  */
      if (length != CGEN_INSN_BITSIZE (insn))
      if (length != CGEN_INSN_BITSIZE (insn))
        abort ();
        abort ();
 
 
      /* ??? 0 is passed for `pc' */
      /* ??? 0 is passed for `pc' */
      length = CGEN_EXTRACT_FN (cd, insn)
      length = CGEN_EXTRACT_FN (cd, insn)
        (cd, insn, info, base_insn, fields, (bfd_vma) 0);
        (cd, insn, info, base_insn, fields, (bfd_vma) 0);
      /* Sanity check: must succeed.
      /* Sanity check: must succeed.
         Could relax this later if it ever proves useful.  */
         Could relax this later if it ever proves useful.  */
      if (length == 0)
      if (length == 0)
        abort ();
        abort ();
      return insn;
      return insn;
    }
    }
 
 
  return NULL;
  return NULL;
}
}
 
 
/* Fill in the operand instances used by INSN whose operands are FIELDS.
/* Fill in the operand instances used by INSN whose operands are FIELDS.
   INDICES is a pointer to a buffer of MAX_OPERAND_INSTANCES ints to be filled
   INDICES is a pointer to a buffer of MAX_OPERAND_INSTANCES ints to be filled
   in.  */
   in.  */
 
 
void
void
cgen_get_insn_operands (CGEN_CPU_DESC cd,
cgen_get_insn_operands (CGEN_CPU_DESC cd,
                        const CGEN_INSN *insn,
                        const CGEN_INSN *insn,
                        const CGEN_FIELDS *fields,
                        const CGEN_FIELDS *fields,
                        int *indices)
                        int *indices)
{
{
  const CGEN_OPINST *opinst;
  const CGEN_OPINST *opinst;
  int i;
  int i;
 
 
  if (insn->opinst == NULL)
  if (insn->opinst == NULL)
    abort ();
    abort ();
  for (i = 0, opinst = insn->opinst; opinst->type != CGEN_OPINST_END; ++i, ++opinst)
  for (i = 0, opinst = insn->opinst; opinst->type != CGEN_OPINST_END; ++i, ++opinst)
    {
    {
      enum cgen_operand_type op_type = opinst->op_type;
      enum cgen_operand_type op_type = opinst->op_type;
      if (op_type == CGEN_OPERAND_NIL)
      if (op_type == CGEN_OPERAND_NIL)
        indices[i] = opinst->index;
        indices[i] = opinst->index;
      else
      else
        indices[i] = (*cd->get_int_operand) (cd, op_type, fields);
        indices[i] = (*cd->get_int_operand) (cd, op_type, fields);
    }
    }
}
}
 
 
/* Cover function to cgen_get_insn_operands when either INSN or FIELDS
/* Cover function to cgen_get_insn_operands when either INSN or FIELDS
   isn't known.
   isn't known.
   The INSN, INSN_*_VALUE, and LENGTH arguments are passed to
   The INSN, INSN_*_VALUE, and LENGTH arguments are passed to
   cgen_lookup_insn unchanged.
   cgen_lookup_insn unchanged.
   INSN_INT_VALUE is used if CGEN_INT_INSN_P.
   INSN_INT_VALUE is used if CGEN_INT_INSN_P.
   Otherwise INSN_BYTES_VALUE is used.
   Otherwise INSN_BYTES_VALUE is used.
 
 
   The result is the insn table entry or NULL if the instruction wasn't
   The result is the insn table entry or NULL if the instruction wasn't
   recognized.  */
   recognized.  */
 
 
const CGEN_INSN *
const CGEN_INSN *
cgen_lookup_get_insn_operands (CGEN_CPU_DESC cd,
cgen_lookup_get_insn_operands (CGEN_CPU_DESC cd,
                               const CGEN_INSN *insn,
                               const CGEN_INSN *insn,
                               CGEN_INSN_INT insn_int_value,
                               CGEN_INSN_INT insn_int_value,
                               /* ??? CGEN_INSN_BYTES would be a nice type name to use here.  */
                               /* ??? CGEN_INSN_BYTES would be a nice type name to use here.  */
                               unsigned char *insn_bytes_value,
                               unsigned char *insn_bytes_value,
                               int length,
                               int length,
                               int *indices,
                               int *indices,
                               CGEN_FIELDS *fields)
                               CGEN_FIELDS *fields)
{
{
  /* Pass non-zero for ALIAS_P only if INSN != NULL.
  /* Pass non-zero for ALIAS_P only if INSN != NULL.
     If INSN == NULL, we want a real insn.  */
     If INSN == NULL, we want a real insn.  */
  insn = cgen_lookup_insn (cd, insn, insn_int_value, insn_bytes_value,
  insn = cgen_lookup_insn (cd, insn, insn_int_value, insn_bytes_value,
                           length, fields, insn != NULL);
                           length, fields, insn != NULL);
  if (! insn)
  if (! insn)
    return NULL;
    return NULL;
 
 
  cgen_get_insn_operands (cd, insn, fields, indices);
  cgen_get_insn_operands (cd, insn, fields, indices);
  return insn;
  return insn;
}
}
 
 
/* Allow signed overflow of instruction fields.  */
/* Allow signed overflow of instruction fields.  */
void
void
cgen_set_signed_overflow_ok (CGEN_CPU_DESC cd)
cgen_set_signed_overflow_ok (CGEN_CPU_DESC cd)
{
{
  cd->signed_overflow_ok_p = 1;
  cd->signed_overflow_ok_p = 1;
}
}
 
 
/* Generate an error message if a signed field in an instruction overflows.  */
/* Generate an error message if a signed field in an instruction overflows.  */
void
void
cgen_clear_signed_overflow_ok (CGEN_CPU_DESC cd)
cgen_clear_signed_overflow_ok (CGEN_CPU_DESC cd)
{
{
  cd->signed_overflow_ok_p = 0;
  cd->signed_overflow_ok_p = 0;
}
}
 
 
/* Will an error message be generated if a signed field in an instruction overflows ? */
/* Will an error message be generated if a signed field in an instruction overflows ? */
unsigned int
unsigned int
cgen_signed_overflow_ok_p (CGEN_CPU_DESC cd)
cgen_signed_overflow_ok_p (CGEN_CPU_DESC cd)
{
{
  return cd->signed_overflow_ok_p;
  return cd->signed_overflow_ok_p;
}
}
 
 

powered by: WebSVN 2.1.0

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