From c906108c21474dfb4ed285bcc0ac6fe02cd400cc Mon Sep 17 00:00:00 2001
From: Stan Shebs <shebs@codesourcery.com>
Date: Fri, 16 Apr 1999 01:35:26 +0000
Subject: Initial creation of sourceware repository

---
 sim/common/cgen-engine.h | 473 +++++++++++++++++++++++++++++++++++++++++++++++
 1 file changed, 473 insertions(+)
 create mode 100644 sim/common/cgen-engine.h

(limited to 'sim/common/cgen-engine.h')

diff --git a/sim/common/cgen-engine.h b/sim/common/cgen-engine.h
new file mode 100644
index 0000000..9421332
--- /dev/null
+++ b/sim/common/cgen-engine.h
@@ -0,0 +1,473 @@
+/* Engine header for Cpu tools GENerated simulators.
+   Copyright (C) 1998, 1999 Free Software Foundation, Inc.
+   Contributed by Cygnus Support.
+
+This file is part of GDB, the GNU debugger.
+
+This program is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License along
+with this program; if not, write to the Free Software Foundation, Inc.,
+59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  */
+
+/* This file must be included after eng.h and before ${cpu}.h.  */
+
+/* Semantic functions come in six versions on two axes:
+   fast/full-featured, and using one of the simple/scache/compilation engines.
+   A full featured simulator is always provided.  --enable-sim-fast includes
+   support for fast execution by duplicating the semantic code but leaving
+   out all features like tracing and profiling.
+   Using the scache is selected with --enable-sim-scache.  */
+/* FIXME: --enable-sim-fast not implemented yet.  */
+/* FIXME: undecided how to handle WITH_SCACHE_PBB.  */
+
+/* There are several styles of engines, all generally supported by the
+   same code:
+
+   WITH_SCACHE && WITH_SCACHE_PBB - pseudo-basic-block scaching
+   WITH_SCACHE && !WITH_SCACHE_PBB - scaching on an insn by insn basis
+   !WITH_SCACHE - simple engine: fetch an insn, execute an insn
+
+   The !WITH_SCACHE case can also be broken up into two flavours:
+   extract the fields of the insn into an ARGBUF struct, or defer the
+   extraction to the semantic handler.  The former can be viewed as the
+   WITH_SCACHE case with a cache size of 1 (thus there's no need for a
+   WITH_EXTRACTION macro).  The WITH_SCACHE case always extracts the fields
+   into an ARGBUF struct.  */
+
+#ifndef CGEN_ENGINE_H
+#define CGEN_ENGINE_H
+
+/* Instruction field support macros.  */
+
+#define EXTRACT_MSB0_INT(val, total, start, length) \
+(((INT) (val) << ((sizeof (INT) * 8) - (total) + (start))) \
+ >> ((sizeof (INT) * 8) - (length)))
+#define EXTRACT_MSB0_UINT(val, total, start, length) \
+(((UINT) (val) << ((sizeof (UINT) * 8) - (total) + (start))) \
+ >> ((sizeof (UINT) * 8) - (length)))
+
+#define EXTRACT_LSB0_INT(val, total, start, length) \
+(((INT) (val) << ((sizeof (INT) * 8) - (start) - 1)) \
+ >> ((sizeof (INT) * 8) - (length)))
+#define EXTRACT_LSB0_UINT(val, total, start, length) \
+(((UINT) (val) << ((sizeof (UINT) * 8) - (start) - 1)) \
+ >> ((sizeof (UINT) * 8) - (length)))
+
+#if CGEN_INSN_LSB0_P
+
+#define EXTRACT_INT(val, total, start, length) \
+  EXTRACT_LSB0_INT ((val), (total), (start), (length))
+#define EXTRACT_UINT(val, total, start, length) \
+  EXTRACT_LSB0_UINT ((val), (total), (start), (length))
+
+#else
+
+#define EXTRACT_INT(val, total, start, length) \
+  EXTRACT_MSB0_INT ((val), (total), (start), (length))
+#define EXTRACT_UINT(val, total, start, length) \
+  EXTRACT_MSB0_UINT ((val), (total), (start), (length))
+
+#endif
+
+/* Semantic routines.  */
+
+/* Type of the machine generated extraction fns.  */
+/* ??? No longer used.  */
+typedef void (EXTRACT_FN) (SIM_CPU *, IADDR, CGEN_INSN_INT, ARGBUF *);
+
+/* Type of the machine generated semantic fns.  */
+
+#if WITH_SCACHE
+
+/* Instruction fields are extracted into ARGBUF before calling the
+   semantic routine.  */
+#if HAVE_PARALLEL_INSNS
+typedef SEM_PC (SEMANTIC_FN) (SIM_CPU *, SEM_ARG, PAREXEC *);
+#else
+typedef SEM_PC (SEMANTIC_FN) (SIM_CPU *, SEM_ARG);
+#endif
+
+#else
+
+/* Result of semantic routines is a status indicator (wip).  */
+typedef unsigned int SEM_STATUS;
+
+/* Instruction fields are extracted by the semantic routine.
+   ??? TODO: multi word insns.  */
+#if HAVE_PARALLEL_INSNS
+typedef SEM_STATUS (SEMANTIC_FN) (SIM_CPU *, SEM_ARG, PAREXEC *, CGEN_INSN_INT);
+#else
+typedef SEM_STATUS (SEMANTIC_FN) (SIM_CPU *, SEM_ARG, CGEN_INSN_INT);
+#endif
+
+#endif
+
+/* In the ARGBUF struct, a pointer to the semantic routine for the insn.  */
+
+union sem {
+#if ! WITH_SEM_SWITCH_FULL
+  SEMANTIC_FN *sem_full;
+#endif
+#if ! WITH_SEM_SWITCH_FAST
+  SEMANTIC_FN *sem_fast;
+#endif
+#if WITH_SEM_SWITCH_FULL || WITH_SEM_SWITCH_FAST
+#ifdef __GNUC__
+  void *sem_case;
+#else
+  int sem_case;
+#endif
+#endif
+};
+
+/* Set the appropriate semantic handler in ABUF.  */
+
+#if WITH_SEM_SWITCH_FULL
+#ifdef __GNUC__
+#define SEM_SET_FULL_CODE(abuf, idesc) \
+  do { (abuf)->semantic.sem_case = (idesc)->sem_full_lab; } while (0)
+#else 
+#define SEM_SET_FULL_CODE(abuf, idesc) \
+  do { (abuf)->semantic.sem_case = (idesc)->num; } while (0)
+#endif
+#else
+#define SEM_SET_FULL_CODE(abuf, idesc) \
+  do { (abuf)->semantic.sem_full = (idesc)->sem_full; } while (0)
+#endif
+
+#if WITH_SEM_SWITCH_FAST
+#ifdef __GNUC__
+#define SEM_SET_FAST_CODE(abuf, idesc) \
+  do { (abuf)->semantic.sem_case = (idesc)->sem_fast_lab; } while (0)
+#else 
+#define SEM_SET_FAST_CODE(abuf, idesc) \
+  do { (abuf)->semantic.sem_case = (idesc)->num; } while (0)
+#endif
+#else
+#define SEM_SET_FAST_CODE(abuf, idesc) \
+  do { (abuf)->semantic.sem_fast = (idesc)->sem_fast; } while (0)
+#endif
+
+#define SEM_SET_CODE(abuf, idesc, fast_p) \
+do { \
+  if (fast_p) \
+    SEM_SET_FAST_CODE ((abuf), (idesc)); \
+  else \
+    SEM_SET_FULL_CODE ((abuf), (idesc)); \
+} while (0)
+
+/* Return non-zero if IDESC is a conditional or unconditional CTI.  */
+
+#define IDESC_CTI_P(idesc) \
+     ((CGEN_ATTR_BOOLS (CGEN_INSN_ATTRS ((idesc)->idata)) \
+       & (CGEN_ATTR_MASK (CGEN_INSN_COND_CTI) \
+	  | CGEN_ATTR_MASK (CGEN_INSN_UNCOND_CTI))) \
+      != 0)
+
+/* Return non-zero if IDESC is a skip insn.  */
+
+#define IDESC_SKIP_P(idesc) \
+     ((CGEN_ATTR_BOOLS (CGEN_INSN_ATTRS ((idesc)->idata)) \
+       & CGEN_ATTR_MASK (CGEN_INSN_SKIP_CTI)) \
+      != 0)
+
+/* These are used so that we can compile two copies of the semantic code,
+   one with full feature support and one without that runs fast(er).  */
+#define SEM_FN_NAME(cpu,fn) XCONCAT3 (cpu,_sem_,fn)
+#define SEMF_FN_NAME(cpu,fn) XCONCAT3 (cpu,_semf_,fn)
+
+/* Return pointer to ARGBUF given ptr to SCACHE.  */
+#define SEM_ARGBUF(sem_arg) (& (sem_arg) -> argbuf)
+
+/* There are several styles of engines, all generally supported by the
+   same code:
+
+   WITH_SCACHE && WITH_SCACHE_PBB - pseudo-basic-block scaching
+   WITH_SCACHE && !WITH_SCACHE_PBB - scaching on an insn by insn basis
+   !WITH_SCACHE - simple engine: fetch an insn, execute an insn
+
+   ??? The !WITH_SCACHE case can also be broken up into two flavours:
+   extract the fields of the insn into an ARGBUF struct, or defer the
+   extraction to the semantic handler.  The WITH_SCACHE case always
+   extracts the fields into an ARGBUF struct.  */
+
+#if WITH_SCACHE
+
+#define CIA_ADDR(cia) (cia)
+
+#if WITH_SCACHE_PBB
+
+/* Return the scache pointer of the current insn.  */
+#define SEM_SEM_ARG(vpc, sc) (vpc)
+
+/* Return the virtual pc of the next insn to execute
+   (assuming this isn't a cti or the branch isn't taken).  */
+#define SEM_NEXT_VPC(sem_arg, pc, len) ((sem_arg) + 1)
+
+/* Update the instruction counter.  */
+#define PBB_UPDATE_INSN_COUNT(cpu,sc) \
+  (CPU_INSN_COUNT (cpu) += SEM_ARGBUF (sc) -> fields.chain.insn_count)
+
+/* Value for br_addr_ptr indicating branch wasn't taken.  */
+#define SEM_BRANCH_UNTAKEN ((SEM_PC *) 0)
+
+/* Value for br_addr_ptr indicating branch was taken to uncacheable
+   address (e.g. j reg).  */
+#define SEM_BRANCH_UNCACHEABLE ((SEM_PC *) 1)
+
+/* Initialize next-pbb link for SEM_BRANCH_VIA_CACHE.  */
+#define SEM_BRANCH_INIT_EXTRACT(abuf) \
+do { (abuf)->fields.cti.addr_cache = 0; } while (0)
+
+/* Do not append a `;' to invocations of this.
+   npc,npc_ptr are for communication between the cti insn and cti-chain.  */
+#define SEM_BRANCH_INIT \
+  IADDR npc = 0; /* assign a value for -Wall */ \
+  SEM_PC *npc_ptr = SEM_BRANCH_UNTAKEN;
+
+/* SEM_IN_SWITCH is defined at the top of the mainloop.c files
+   generated by genmloop.sh.  It exists so generated semantic code needn't
+   care whether it's being put in a switch or in a function.  */
+#ifdef SEM_IN_SWITCH
+#define SEM_BRANCH_FINI(pcvar) \
+do { \
+  pbb_br_npc = npc; \
+  pbb_br_npc_ptr = npc_ptr; \
+} while (0)
+#else /* 1 semantic function per instruction */
+#define SEM_BRANCH_FINI(pcvar) \
+do { \
+  CPU_PBB_BR_NPC (current_cpu) = npc; \
+  CPU_PBB_BR_NPC_PTR (current_cpu) = npc_ptr; \
+} while (0)
+#endif
+
+/* Return address of cached branch address value.  */
+#define SEM_BRANCH_ADDR_CACHE(sem_arg) \
+  (& SEM_ARGBUF (sem_arg)->fields.cti.addr_cache)
+
+#define SEM_BRANCH_VIA_CACHE(cpu, sc, newval, pcvar, cachevarptr) \
+do { \
+  npc = (newval); \
+  npc_ptr = (cachevarptr); \
+} while (0)
+
+#define SEM_BRANCH_VIA_ADDR(cpu, sc, newval, pcvar) \
+do { \
+  npc = (newval); \
+  npc_ptr = SEM_BRANCH_UNCACHEABLE; \
+} while (0)
+
+#else /* ! WITH_SCACHE_PBB */
+
+#define SEM_SEM_ARG(vpc, sc) (sc)
+
+#define SEM_NEXT_VPC(sem_arg, pc, len) ((pc) + (len))
+
+#define SEM_BRANCH_INIT_EXTRACT(abuf) do { } while (0)
+
+/* ??? May wish to move taken_p out of here and make it explicit.  */
+#define SEM_BRANCH_INIT \
+  int taken_p = 0;
+
+#ifndef TARGET_SEM_BRANCH_FINI(pcvar, taken_p)
+#define TARGET_SEM_BRANCH_FINI(pcvar, taken_p)
+#endif
+#define SEM_BRANCH_FINI(pcvar) \
+  do { TARGET_SEM_BRANCH_FINI (pcvar, taken_p); } while (0)
+
+#define SEM_BRANCH_ADDR_CACHE(sem_arg) shouldnt_be_used
+
+#define SEM_BRANCH_VIA_CACHE(cpu, sc, newval, pcvar, cachevar) \
+do { \
+  (pcvar) = (newval); \
+  taken_p = 1; \
+} while (0)
+
+#define SEM_BRANCH_VIA_ADDR(cpu, sc, newval, pcvar) \
+do { \
+  (pcvar) = (newval); \
+  taken_p = 1; \
+} while (0)
+
+#endif /* ! WITH_SCACHE_PBB */
+
+#else /* ! WITH_SCACHE */
+
+/* This is the "simple" engine case.  */
+
+#define CIA_ADDR(cia) (cia)
+
+#define SEM_SEM_ARG(vpc, sc) (sc)
+
+#define SEM_NEXT_VPC(sem_arg, pc, len) ((pc) + (len))
+
+#define SEM_BRANCH_INIT \
+  int taken_p = 0;
+
+#define SEM_BRANCH_ADDR_CACHE(sem_arg) shouldnt_be_used
+
+#define SEM_BRANCH_VIA_CACHE(cpu, abuf, newval, pcvar, cachevar) \
+do { \
+  (pcvar) = (newval); \
+  taken_p = 1; \
+} while (0)
+
+#define SEM_BRANCH_VIA_ADDR(cpu, abuf, newval, pcvar) \
+do { \
+  (pcvar) = (newval); \
+  taken_p = 1; \
+} while (0)
+
+/* Finish off branch insns.
+   The target must define TARGET_SEM_BRANCH_FINI.
+   ??? This can probably go away when define-execute is finished.  */
+#define SEM_BRANCH_FINI(pcvar, bool_attrs) \
+  do { TARGET_SEM_BRANCH_FINI ((pcvar), (bool_attrs), taken_p); } while (0)
+
+/* Finish off non-branch insns.
+   The target must define TARGET_SEM_NBRANCH_FINI.
+   ??? This can probably go away when define-execute is finished.  */
+#define SEM_NBRANCH_FINI(pcvar, bool_attrs) \
+  do { TARGET_SEM_NBRANCH_FINI ((pcvar), (bool_attrs)); } while (0)
+
+#endif /* ! WITH_SCACHE */
+
+/* Instruction information.  */
+
+/* Compile time computable instruction data.
+
+   ??? May wish to move parallel execution support into its own struct.
+   It's a fair bit of "clutter" for the "normal" case.  */
+
+struct insn_sem {
+  /* The instruction type (a number that identifies each insn over the
+     entire architecture).  */
+  CGEN_INSN_TYPE type;
+
+  /* Index in IDESC table.  */
+  int index;
+
+  /* Sanity check, at most one of these may be true.  */
+#if WITH_PARALLEL_READ && WITH_PARALLEL_WRITE
+#error "Both WITH_PARALLEL_READ && WITH_PARALLEL_WRITE can't be true."
+#endif
+
+#if WITH_PARALLEL_READ || WITH_PARALLEL_WRITE
+  /* Index in IDESC table of parallel handler.  */
+  int par_index;
+#endif
+
+#if WITH_PARALLEL_READ
+#ifndef __GNUC__
+  /* Semantic format number of pre-read handler.
+     Only used by chips that support parallel execution of several insns.
+     It is always implemented as a `switch'.  In the case of GNUC we use
+     computed gotos.  When not GNUC, this is the argument to `switch'.  */
+  int fmt;
+#endif
+#endif
+
+#if WITH_PARALLEL_WRITE
+  /* Index in IDESC table of writeback handler.
+     Only used by chips that support parallel execution of several insns.  */
+  int write_index;
+#endif
+
+  /* Routines to execute the insn.
+     The full version has all features (profiling,tracing) compiled in.
+     The fast version has none of that.  */
+#if ! WITH_SEM_SWITCH_FULL
+  SEMANTIC_FN *sem_full;
+#endif
+#if WITH_FAST && ! WITH_SEM_SWITCH_FAST
+  SEMANTIC_FN *sem_fast;
+#endif
+};
+
+/* Run-time computed instruction descriptor.  */
+
+struct idesc {
+  /* Parallel read-before-exec support.  */
+#if WITH_PARALLEL_READ
+  struct idesc *par_idesc;
+#ifdef __GNUC__
+  void *read;
+#else
+  int fmt;
+#endif
+#endif
+
+  /* Parallel write-after-exec support.  */
+#if WITH_PARALLEL_WRITE
+  /* Pointer to parallel handler if serial insn.
+     Pointer to writeback handler if parallel insn.  */
+  struct idesc *par_idesc;
+#endif
+
+#if WITH_SEM_SWITCH_FULL
+#ifdef __GNUC__
+  void *sem_full_lab;
+#else
+  /* nothing needed, switch's on `num' member */
+#endif
+#else
+  SEMANTIC_FN *sem_full;
+#endif
+
+#if WITH_SEM_SWITCH_FAST
+#ifdef __GNUC__
+  void *sem_fast_lab;
+#else
+  /* nothing needed, switch's on `num' member */
+#endif
+#else
+  SEMANTIC_FN *sem_fast;
+#endif
+
+  /* Instruction number (index in IDESC table, profile table).
+     Also used to switch on in non-gcc semantic switches.  */
+  int num;
+
+  /* instruction data (name, attributes, size, etc.) */
+  const CGEN_INSN *idata;
+
+  /* instruction attributes, copied from `idata' for speed */
+  const CGEN_INSN_ATTR_TYPE *attrs;
+
+  /* instruction length in bytes, copied from `idata' for speed */
+  int length;
+
+  /* profiling/modelling support */
+  const INSN_TIMING *timing;
+};
+
+/* Tracing/profiling.  */
+
+/* Return non-zero if a before/after handler is needed.
+   When tracing/profiling a selected range there's no need to slow
+   down simulation of the other insns (except to get more accurate data!).
+
+   ??? May wish to profile all insns if doing insn tracing, or to
+   get more accurate cycle data.
+
+   First test ANY_P so we avoid a potentially expensive HIT_P call
+   [if there are lots of address ranges].  */
+
+#define PC_IN_TRACE_RANGE_P(cpu, pc) \
+  (TRACE_ANY_P (cpu) \
+   && ADDR_RANGE_HIT_P (TRACE_RANGE (CPU_TRACE_DATA (cpu)), (pc)))
+#define PC_IN_PROFILE_RANGE_P(cpu, pc) \
+  (PROFILE_ANY_P (cpu) \
+   && ADDR_RANGE_HIT_P (PROFILE_RANGE (CPU_PROFILE_DATA (cpu)), (pc)))
+
+#endif /* CGEN_ENGINE_H */
-- 
cgit v1.1