aboutsummaryrefslogtreecommitdiff
path: root/sim/z8k/writecode.c
diff options
context:
space:
mode:
Diffstat (limited to 'sim/z8k/writecode.c')
-rw-r--r--sim/z8k/writecode.c1948
1 files changed, 0 insertions, 1948 deletions
diff --git a/sim/z8k/writecode.c b/sim/z8k/writecode.c
deleted file mode 100644
index acff7c4..0000000
--- a/sim/z8k/writecode.c
+++ /dev/null
@@ -1,1948 +0,0 @@
-/* generate instructions for Z8KSIM
-
- Copyright 1992, 1993, 2002 Free Software Foundation, Inc.
-
-This file is part of Z8KSIM
-
-Z8KSIM 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.
-
-Z8KSIM 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 Z8KZIM; if not, write to the Free Software
-Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
-
-/* This program generates the code which emulates each of the z8k
- instructions
-
- code goes into three files, tc-gen1.h, tc-gen2.h and tc-gen3.h.
- which file being made depends upon the options
-
- -1 tc-gen1.h contains the fully expanded code for some selected
- opcodes, (those in the quick.c list)
-
- -2 tc-gen2.h contains a list of pointers to functions, one for each
- opcode. It points to functions in tc-gen3.h and tc-gen1.h
- depending upon quick.c
-
- -3 tc-gen3.h contains all the opcodes in unexpanded form.
-
- -b3 tc-genb3.h same as -3 but for long pointers
-
- */
-
-/* steve chamberlain
- sac@cygnus.com */
-
-#include "config.h"
-
-#include <ansidecl.h>
-#include <stdio.h>
-#ifdef HAVE_STDLIB_H
-#include <stdlib.h>
-#endif
-#ifdef HAVE_STRING_H
-#include <string.h>
-#else
-#ifdef HAVE_STRINGS_H
-#include <strings.h>
-#endif
-#endif
-
-#define NICENAMES
-
-#define DEFINE_TABLE
-#include "../opcodes/z8k-opc.h"
-
-#define NOPS 500
-
-struct opcode_value
-{
- int n;
- struct opcode_value *next;
-};
-
-#define NICENAMES
-int BIG;
-
-static char *reg_names[] =
-{"bad", "src", "dst", "aux_a", "aux_b", "aux_r", "aux_x"};
-
-#define IS_DST(x) ((x & 0xf) == 2)
-#define IS_SRC(x) ((x & 0xf)==1)
-#define SIZE_ADDRESS (BIG ? 8 : 4) /* number of nibbles in a ptr*/
-
-static int file;
-
-static int nibs = 0;
-
-static char *current_size;
-static char *current_name;
-static char current_word0[40];
-static char current_byte0[40];
-static char current_byte1[40];
-static int indent;
-static char *p;
-static char *d;
-
-struct opcode_value *list[NOPS];
-
-static opcode_entry_type *
-lookup_inst (what)
- int what;
-{
- static short *z8k_inv_list = NULL;
- const nr_z8k_inv_list_elements = 1 << 16;
- if (z8k_inv_list == NULL)
- {
- /* Initialize the list to 0xff == -1 */
- z8k_inv_list = calloc (nr_z8k_inv_list_elements, sizeof (short));
- memset (z8k_inv_list, 0xff, nr_z8k_inv_list_elements * sizeof (short));
- }
- /* Entry empty? Fill it in. */
- if (z8k_inv_list[what] == -1)
- {
-
- int nibl_index;
- int nibl_matched;
- unsigned short instr_nibl;
- unsigned short tabl_datum, datum_class, datum_value;
- char instr_nibbles[8];
-
- opcode_entry_type *ptr = z8k_table;
-
- nibl_matched = 0;
-
- instr_nibbles[3] = (what >> 0) & 0xf;
- instr_nibbles[2] = (what >> 4) & 0xf;
- instr_nibbles[1] = (what >> 8) & 0xf;
- instr_nibbles[0] = (what >> 12) & 0xf;
-
- /* Assume it won't be found. */
- z8k_inv_list[what] = -2;
-
- while (ptr->name)
- {
- nibl_matched = 1;
- for (nibl_index = 0; nibl_index < 4 && nibl_matched; nibl_index++)
- {
- instr_nibl = instr_nibbles[nibl_index];
-
- tabl_datum = ptr->byte_info[nibl_index];
- datum_class = tabl_datum & CLASS_MASK;
- datum_value = ~CLASS_MASK & tabl_datum;
-
- switch (datum_class)
- {
- case CLASS_BIT_1OR2:
- if (datum_value != (instr_nibl & ~0x2))
- nibl_matched = 0;
- break;
-
- case CLASS_IGNORE:
- break;
- case CLASS_BIT:
- if (datum_value != instr_nibl)
- nibl_matched = 0;
- break;
- case CLASS_00II:
- if (!((~instr_nibl) & 0x4))
- nibl_matched = 0;
- break;
- case CLASS_01II:
- if (!(instr_nibl & 0x4))
- nibl_matched = 0;
- break;
- case CLASS_0CCC:
- if (!((~instr_nibl) & 0x8))
- nibl_matched = 0;
- break;
- case CLASS_1CCC:
- if (!(instr_nibl & 0x8))
- nibl_matched = 0;
- break;
- case CLASS_0DISP7:
- if (!((~instr_nibl) & 0x8))
- nibl_matched = 0;
- nibl_index += 1;
- break;
- case CLASS_1DISP7:
- if (!(instr_nibl & 0x8))
- nibl_matched = 0;
- nibl_index += 1;
- break;
- case CLASS_REGN0:
- if (instr_nibl == 0)
- nibl_matched = 0;
- break;
- default:
- break;
- }
- }
- if (nibl_matched)
- {
- z8k_inv_list[what] = ptr->idx;
- break; /* while */
- }
- ptr++;
- }
- }
- if (z8k_inv_list[what] >= 0)
- return z8k_table + z8k_inv_list[what];
- return 0;
-}
-
-static char *
-insn_4 (n)
- int n;
-{
- switch (n)
- {
- case 1:
- return "((iwords_0>>8) & 0xf)";
- case 2:
- return "((ibytes_1 >> 4) & 0xf)";
- case 3:
- return "((ibytes_1) & 0xf)";
- case 4:
- return "((ibytes_2>>4) & 0xf)";
- case 5:
- return "((ibytes_2) & 0xf)";
- case 6:
- return "((ibytes_3 >> 4) & 0xf)";
- case 7:
- return "((ibytes_3) & 0xf)";
- default:
- return "****";
- }
-}
-
-char *
-ptr_mode ()
-{
- if (BIG)
- {
- return "ptr_long";
- }
- return "word";
-}
-
-static
-char *
-ptr_size ()
-{
- if (BIG)
- return "4";
- return "2";
-}
-
-static char *
-reg_n (x)
- unsigned int x;
-{
- return reg_names[x & 0xf];
-}
-
-char *
-stack_ptr ()
-{
- return BIG ? "14" : "15";
-}
-
-char *
-mem ()
-{
-#if 0
- return BIG ? "segmem" : "unsegmem";
-#else
- return "mem";
-#endif
-}
-
-int
-match (a)
- char *a;
-{
- if (strncmp (p, a, strlen (a)) == 0)
- {
- p += strlen (a);
- return 1;
- }
- return 0;
-}
-
-static
-void
-sub (y)
- char *y;
-{
- sprintf (d, "%s", y);
- d += strlen (d);
-}
-
-static char *
-insn_16 (n)
- int n;
-{
- switch (n)
- {
- case 0:
- return "(iwords_0)";
- case 4:
- return "(iwords_1)";
- case 8:
- return "(iwords_2)";
- case 12:
- return "(iwords_3)";
- default:
- return "****";
- }
-}
-
-static
-char *
-insn_32 (n)
- int n;
-{
- switch (n)
- {
- case 0:
- return "((iwords_0<<16) | (iwords_1))";
- case 4:
- return "((iwords_1<<16) | (iwords_2))";
- case 8:
- return "((iwords_2<<16) | (iwords_3))";
- default:
- return "****";
- }
-}
-
-static char *
-size_name (x)
- int x;
-{
- switch (x)
- {
- case 8:
- return "byte";
- case 16:
- return "word";
- case 32:
- return "long";
- case 64:
- return "quad";
- }
- return "!!";
-}
-
-/*VARARGS*/
-void
-emit (string, a1, a2, a3, a4, a5)
- char *string;
- char* a1;
- char* a2;
- char* a3;
- char* a4;
- char* a5;
-{
- int indent_inc = 0;
- int indent_dec = 0;
- int i;
- char buffer[1000];
-
- d = buffer;
- p = string;
-
- while (*p)
- {
- if (match ("<fop>"))
- {
- if (BIG)
- {
- sub ("bfop");
- }
- else
- {
- sub ("sfop");
- }
- }
- else if (match ("<iptr>"))
- {
- if (BIG)
- {
- switch (nibs)
- {
- case 4:
- sub ("(((iwords_1 << 8) | (iwords_2)) & 0x7fffff)");
- break;
- default:
- sub ("fail(context,124)");
- break;
- }
- }
- else
- {
- switch (nibs)
- {
- case 4:
- sub ("iwords_1");
- break;
- default:
- sub ("fail(context,123)");
- break;
- }
- }
- }
- else if (match ("<name>"))
- {
- sub (current_name);
- }
- else if (match ("<size>"))
- {
- sub (current_size);
- }
- else if (match ("<insn_4>"))
- {
- sub (insn_4 (nibs));
- }
- else if (match ("<insn_16>"))
- {
- sub (insn_16 (nibs));
- }
- else if (match ("<insn_32>"))
- {
- sub (insn_32 (nibs));
- }
- else if (match ("iwords_0"))
- {
- sub (current_word0);
- }
- else if (match ("ibytes_0"))
- {
- sub (current_byte0);
- }
- else if (match ("<ibytes_1>"))
- {
- sub (current_byte1);
- }
- else if (match ("<next_size>"))
- {
- if (strcmp (current_size, "word") == 0)
- sub ("long");
- if (strcmp (current_size, "byte") == 0)
- sub ("word");
- else if (strcmp (current_size, "long") == 0)
- sub ("quad");
- else
- abort ();
- }
- else if (match ("<addr_type>"))
- {
- if (BIG)
- sub ("unsigned long");
- else
- sub ("unsigned short");
- }
-
- else if (match ("<c_size>"))
- {
- if (strcmp (current_size, "word") == 0)
- sub ("short");
- else if (strcmp (current_size, "byte") == 0)
- sub ("char");
- else if (strcmp (current_size, "long") == 0)
- sub ("long");
- else
- abort ();
- }
-
- else if (match ("<pc>"))
- {
- sub ("pc");
- }
- else if (match ("<mem>"))
- {
- sub (mem ());
- }
- else if (match ("<sp>"))
- {
- sub (stack_ptr ());
- }
- else if (match ("<ptr_size>"))
- {
- sub (ptr_size ());
- }
- else if (match ("<ptr_mode>"))
- {
- sub (ptr_mode ());
- }
- else if (match ("<insn_8>"))
- {
- switch (nibs)
- {
- case 2:
- sub ("(iwords_0&0xff)");
- break;
- case 4:
- sub ("(iwords_1>>8)");
- break;
- case 6:
- sub ("(iwords_1&0xff)");
- break;
- case 8:
- sub ("(iwords_2>>8)");
- break;
- case 12:
- sub ("(/* WHO */iwords_3&0xff)");
- break;
- default:
- abort ();
- }
- }
- else
- {
- if (*p == '{')
- indent_inc++;
- if (*p == '}')
- indent_dec++;
- *d++ = *p;
- p++;
- }
- }
- *d++ = 0;
- indent -= indent_dec;
- for (i = 0; i < indent; i++)
- printf ("\t");
- indent += indent_inc;
- printf (buffer, a1, a2, a3, a4, a5);
-
-}
-
-/* fetch the lvalues of the operands */
-void
-info_args (p)
- opcode_entry_type *p;
-{
- unsigned int *s;
-
- int done_one_imm8 = 0;
-
- /* int done_read = 4;*/
- s = p->byte_info;
- nibs = 0;
- while (*s)
- {
- switch (*s & CLASS_MASK)
- {
- case CLASS_BIT_1OR2:
- emit ("register unsigned int imm_src=(<insn_4>& 2)?2:1;\n");
- break;
- case CLASS_IGNORE:
- case CLASS_BIT:
- /* Just ignore these, we've already decoded this bit */
- nibs++;
- break;
- case CLASS_REGN0:
- case CLASS_REG:
- /* this nibble tells us which register to use as an arg,
- if we've already gobbled the nibble we know what to use */
- {
- int regname = *s & 0xf;
-
- emit ("register unsigned int reg_%s=<insn_4>;\n",
- reg_names[regname]);
-
- nibs++;
- }
- break;
- case CLASS_ADDRESS:
- emit ("register unsigned base_%s=<iptr>;\n", reg_n (*s));
-
- nibs += SIZE_ADDRESS;
-
- break;
- case CLASS_01II:
- case CLASS_00II:
- emit ("register unsigned int imm_src=<insn_4>&0x2;\n");
- nibs++;
- break;
- case CLASS_FLAGS:
- emit ("register unsigned int imm_src=<insn_4>;\n");
- nibs++;
-break;
- case CLASS_IMM:
- /* Work out the size of the think to fetch */
-
- {
- switch (*s & ~CLASS_MASK)
- {
- case ARG_IMM16:
- emit ("register unsigned imm_src=<insn_16>;\n");
- nibs += 4;
- break;
- case ARG_IMM32:
- emit ("register unsigned int imm_src= %s;\n", insn_32 (nibs));
- nibs += 8;
- break;
- case ARG_IMM4:
- emit ("register unsigned int imm_src=<insn_4>;\n");
- nibs++;
- break;
- case ARG_NIM4:
- emit ("register unsigned int imm_src = - <insn_4>;\n");
- nibs++;
- break;
- case ARG_IMM2:
- emit ("register unsigned int imm_src=<insn_4> & 0x2;\n");
- nibs++;
- break;
-
- case ARG_IMM4M1:
- emit ("register unsigned int imm_src=(<insn_4> + 1);\n");
- nibs++;
- break;
- case ARG_IMM_1:
- emit ("register unsigned int imm_src=1;\n");
- break;
- case ARG_IMM_2:
- emit ("register unsigned int imm_src=2;\n");
- break;
- case ARG_NIM8:
- emit ("register unsigned int imm_src=-<insn_8>;\n");
- nibs += 2;
- break;
- case ARG_IMM8:
- if (!done_one_imm8)
- {
- emit ("register unsigned int imm_src=<insn_8>;\n");
- nibs += 2;
- done_one_imm8 = 1;
- }
- break;
- default:
- emit ("register int fail%d=fail(context,1);\n", nibs);
- break;
- }
- break;
-
- case CLASS_DISP8:
- /* We can't use `(char)' since char might be unsigned.
- We can't use `(signed char)' because the compiler might be K&R.
- This seems safe, since it only assumes that bytes are 8
- bits. */
- emit ("register unsigned int oplval_dst=((ibytes_1 << (sizeof (int) * 8 - 8)) >> (sizeof (int) * 8 - 9)) + pc;\n");
-#if 0
- /* Original code: fails if characters are unsigned. */
- emit ("register unsigned int oplval_dst=(((char)ibytes_1)<<1) + pc;\n");
-#endif
- nibs += 2;
- break;
- case CLASS_CC:
- emit ("register unsigned int op_cc=<insn_4>;\n");
- nibs++;
- break;
- default:
- emit ("register int FAIL%d=fail(context,2);\n", nibs);
- break;
- }
- ;
- /* work out how to fetch the immediate value */
- }
-
- s++;
- }
-}
-
-void
-info_special (p, getdst, nostore, before, nosrc)
- opcode_entry_type *p;
- int *getdst;
- int *nostore;
- int *before;
- int *nosrc;
-{
- switch (p->opcode)
- {
- case OPC_exts:
- case OPC_extsb:
- case OPC_extsl:
- *nostore = 1;
- *nosrc = 1;
- break;
- case OPC_ldm:
- *nostore = 1;
- *nosrc = 1;
- break;
- case OPC_negb:
- case OPC_neg:
- case OPC_sla:
- case OPC_slab:
- case OPC_slal:
- case OPC_sda:
- case OPC_sdab:
- case OPC_sdal:
- case OPC_com:
- case OPC_comb:
- case OPC_adc:
- case OPC_sbc:
- case OPC_nop:
- case OPC_adcb:
- case OPC_add:
- case OPC_addb:
- case OPC_addl:
- case OPC_inc:
- case OPC_sub:
- case OPC_subb:
- case OPC_subl:
- case OPC_and:
- case OPC_andb:
- case OPC_xorb:
- case OPC_xor:
- break;
-
- case OPC_mult:
- case OPC_multl:
- case OPC_div:
- case OPC_divl:
-
- *nostore = 1;
- break;
-
- case OPC_testb:
- case OPC_test:
- case OPC_testl:
- case OPC_cp:
- case OPC_cpb:
- case OPC_cpl:
- case OPC_bit:
- *nostore = 1;
- *before = 0;
- break;
-
- case OPC_bpt:
- case OPC_jr:
- case OPC_jp:
- case OPC_ret:
- case OPC_call:
- case OPC_tcc:
- *nosrc = 1;
- *nostore = 1;
- *before = 1;
- break;
- case OPC_sc:
- *nostore = 1;
- *before = 0;
- break;
- case OPC_clrb:
- case OPC_clr:
- *before = 1;
- *nosrc = 1;
- break;
- case OPC_ldi:
- case OPC_ldib:
- case OPC_lddb:
- case OPC_ldd:
-
- *before = 1;
- *nostore = 1;
- *nosrc = 1;
- break;
- case OPC_ldk:
- case OPC_ld:
- case OPC_ldb:
- case OPC_ldl:
- *before = 1;
- *getdst = 0;
- break;
- case OPC_push:
- case OPC_pushl:
- case OPC_pop:
- case OPC_popl:
- *before = 1;
- *getdst = 0;
- break;
- case OPC_lda:
- *nosrc = 1;
- break;
- }
-}
-
-/* calculate the lvalues required for the opcode */
-void
-info_lvals (p)
- opcode_entry_type *p;
-{
- /* emit code to work out lvalues, if any */
- unsigned int *i = p->arg_info;
-
- while (*i)
- {
- current_name = reg_n (*i);
- current_size = size_name (p->type);
- switch (*i & CLASS_MASK)
- {
- case CLASS_X:
- /* address(reg) */
- emit ("register <addr_type> oplval_<name>= ((base_<name> + (short)get_word_reg(context,reg_<name>)) & 0xffff) + (base_<name> & ~0xffff);\n");
- break;
- case CLASS_IR:
- /* Indirect register */
- emit ("register int oplval_<name> = get_<ptr_mode>_reg(context,reg_<name>);\n");
- break;
- case CLASS_DA:
- emit ("register int oplval_<name>=base_<name>;\n");
- break;
- case CLASS_IMM:
- case CLASS_REG_WORD:
- case CLASS_REG_LONG:
- case CLASS_REG_BYTE:
- case CLASS_PR:
- break;
- case CLASS_BA:
- emit ("register int oplval_<name> = get_<ptr_mode>_reg(context,reg_<name>) + (short)(imm_src);\n");
- break;
- case CLASS_BX:
- emit ("register int oplval_<name> = get_<ptr_mode>_reg(context,reg_<name>)\n");
- emit (" + get_word_reg(context,reg_aux_x);\n");
- break;
- }
- i++;
- }
-}
-
-/* emit code to fetch the args from calculated lvalues */
-int allregs;
-void
-info_fetch (p, getdst)
- opcode_entry_type *p;
- int getdst;
-{
- unsigned int *i = p->arg_info;
- int had_src = 0;
-
- allregs = 1;
- while (*i)
- {
-
- current_name = reg_n (*i);
- current_size = size_name (p->type);
- switch (*i & CLASS_MASK)
- {
- case CLASS_X:
- case CLASS_IR:
- case CLASS_BA:
- case CLASS_BX:
- case CLASS_DA:
- if (!getdst && IS_DST (*i))
- break;
- emit ("register int op_<name>= get_<size>_<mem>_da(context,oplval_<name>);\n");
- allregs = 0;
- break;
- case CLASS_IMM:
- if (!had_src)
- {
- if (p->opcode == OPC_out ||
- p->opcode == OPC_outb ||
- p->opcode == OPC_sout ||
- p->opcode == OPC_soutb)
- {
- /* The imm is a dest here */
- emit ("register int op_dst = imm_src;\n");
- }
- else
- {
- emit ("register int op_src = imm_src;\n");
- }
- }
- break;
- case CLASS_REG_QUAD:
- if (!getdst && IS_DST (*i))
- break;
- had_src |= IS_SRC (*i);
- emit ("UDItype op_<name> ;\n");
-
- break;
- case CLASS_REG_WORD:
- if (!getdst && IS_DST (*i))
- break;
- had_src |= IS_SRC (*i);
- emit ("register int op_<name> = get_word_reg(context,reg_<name>);\n");
- break;
-
- case CLASS_REG_LONG:
- if (!getdst && IS_DST (*i))
- break;
- had_src |= IS_SRC (*i);
- emit ("register int op_<name> = get_long_reg(context,reg_<name>);\n");
- break;
- case CLASS_REG_BYTE:
- if (!getdst && IS_DST (*i))
- break;
- had_src |= IS_SRC (*i);
- emit ("register int op_<name> = get_byte_reg(context,reg_<name>);\n");
- break;
- }
- i++;
- }
-}
-
-static void
-normal_flags (p, s, neg)
- opcode_entry_type *p;
- char *s;
-{
- emit (" %s;\n", s);
- emit ("NORMAL_FLAGS(context,%d, tmp, op_dst, op_src,%d); \n", p->type,neg);
-}
-
-static void
-test_normal_flags (p, s, opt)
- opcode_entry_type *p;
- char *s;
- int opt;
-{
- emit (" %s;\n", s);
- if (0 && opt)
- {
- emit ("context->broken_flags = TST_FLAGS;\n");
- emit ("context->size = %d;\n", p->type);
- }
- else
- {
- emit ("TEST_NORMAL_FLAGS(context,%d, tmp); \n", p->type);
- }
-
-}
-
-static void
-optimize_normal_flags (p, s,neg)
- opcode_entry_type *p;
- char *s;
-{
- emit (" %s;\n", s);
-#if 0
- emit ("context->broken_flags = CMP_FLAGS;\n");
-#else
- emit ("NORMAL_FLAGS(context,%d, tmp, op_dst, op_src,%d); \n", p->type, neg);
-#endif
-}
-
-static
-void
-jp (p)
- opcode_entry_type *p;
-{
-
- emit ("if(op_cc == 8 || COND(context,op_cc)) pc = oplval_dst;\n");
-}
-
-static void
-jr (p)
- opcode_entry_type *p;
-{
- emit ("if(op_cc == 8 || COND(context,op_cc)) pc = oplval_dst;\n");
-}
-
-static void
-ret (p)
- opcode_entry_type *p;
-{
- emit ("if(op_cc == 8 || COND(context,op_cc))\n{\n");
- emit ("pc = get_<ptr_mode>_<mem>_ir(context,<sp>);\n");
- emit ("put_<ptr_mode>_reg(context,<sp>, get_<ptr_mode>_reg(context,<sp>) + <ptr_size>);\n");
- emit ("};\n");
-}
-
-static void
-call (p)
- opcode_entry_type *p;
-{
- emit ("put_<ptr_mode>_reg(context,<sp>,tmp = get_<ptr_mode>_reg(context,<sp>) - <ptr_size>);\n");
- emit ("put_<ptr_mode>_<mem>_da(context,tmp, pc);\n");
- emit ("pc = oplval_dst;\n");
-}
-
-static void
-push (p)
- opcode_entry_type *p;
-{
- emit ("tmp = op_src;\n");
- emit ("oplval_dst -= %d;\n", p->type / 8);
- emit ("put_<ptr_mode>_reg(context,reg_dst, oplval_dst);\n");
-}
-
-static void
-pop (p)
- opcode_entry_type *p;
-{
- emit ("tmp = op_src;\n");
- emit ("put_<ptr_mode>_reg(context,reg_src, oplval_src + %d);\n", p->type / 8);
-}
-
-static void
-ld (p)
- opcode_entry_type *p;
-{
- emit ("tmp = op_src;\n");
-}
-
-static void
-sc ()
-{
- emit ("support_call(context,imm_src);\n");
-}
-
-static void
-bpt ()
-{
- emit ("pc -=2; \n");
- emit ("context->exception = SIM_BREAKPOINT;\n");
-}
-
-static void
-ldi (p, size, inc)
- opcode_entry_type *p;
- int size;
- int inc;
-{
- int dinc = (size / 8) * inc;
-
- current_size = size_name (size);
- emit ("{ \n");
- emit ("int type = %s;\n", insn_4 (7));
- emit ("int rs = get_<ptr_mode>_reg(context,reg_src);\n");
- emit ("int rd = get_<ptr_mode>_reg(context,reg_dst);\n");
- emit ("int rr = get_word_reg(context,reg_aux_r);\n");
- emit ("do {\n");
- emit ("put_<size>_<mem>_da(context,rd, get_<size>_<mem>_da(context,rs));\n");
- emit ("rd += %d;\n", dinc);
- emit ("rs += %d;\n", dinc);
- emit ("rr --;\n");
- emit ("context->cycles += 9;\n");
- emit ("} while (!type && rr != 0 && context->exception <= 1);\n");
- emit ("if (context->exception>1) pc -=4;\n");
- emit ("put_<ptr_mode>_reg(context,reg_src, rs);\n");
- emit ("put_<ptr_mode>_reg(context,reg_dst, rd);\n");
- emit ("put_word_reg(context,reg_aux_r, rr);\n");
- emit ("}\n");
-
-}
-
-static void
-shift (p, arith)
- opcode_entry_type *p;
- int arith;
-{
-
- /* We can't use `(char)' since char might be unsigned.
- We can't use `(signed char)' because the compiler might be K&R.
- This seems safe, since it only assumes that bytes are 8 bits. */
- emit ("op_src = (op_src << (sizeof (int) * 8 - 8)) >> (sizeof (int) * 8 - 8);\n");
-#if 0
- /* Original code: fails if characters are unsigned. */
- emit ("op_src = (char)op_src;\n");
-#endif
- emit ("if (op_src < 0) \n");
- emit ("{\n");
- emit ("op_src = -op_src;\n");
- emit ("op_dst = (%s <c_size>)op_dst;\n", arith ? "" : "unsigned");
- emit ("tmp = (%s op_dst) >> op_src;\n", arith ? "" : "(unsigned)");
- emit ("context->carry = op_dst >> (op_src-1);\n", p->type);
- emit ("}\n");
- emit ("else\n");
- emit ("{\n");
- emit ("tmp = op_dst << op_src;\n");
- emit ("context->carry = op_dst >> (%d - op_src);\n", p->type);
- emit ("}\n");
- emit ("context->zero = (<c_size>)tmp == 0;\n");
- emit ("context->sign = (int)((<c_size>)tmp) < 0;\n");
- emit ("context->overflow = ((int)tmp < 0) != ((int)op_dst < 0);\n");
- emit ("context->cycles += 3*op_src;\n");
- emit ("context->broken_flags = 0;\n");
-
-}
-
-static void
-rotate (p, through_carry, size, left)
- opcode_entry_type *p;
- int through_carry;
- int size;
- int left;
-{
-
- if (!left)
- {
- emit ("while (op_src--) {\n");
- emit ("int rotbit;\n");
- emit ("rotbit = op_dst & 1;\n");
- emit ("op_dst = ((unsigned)op_dst) >> 1;\n");
-
- if (through_carry)
- {
- emit ("op_dst |= context->carry << %d;\n", size - 1);
- }
- else
- {
- emit ("op_dst |= rotbit << %d;\n", size - 1);
- }
- emit ("context->carry = rotbit;\n");
- emit ("}\n");
- }
- else
- {
- emit ("while (op_src--) {\n");
- emit ("int rotbit;\n");
-
- emit ("rotbit = (op_dst >> (%d))&1;\n", size - 1);
- emit ("op_dst <<=1;\n");
- if (through_carry)
- {
- emit ("if (context->carry) op_dst |=1;\n");
- }
- else
- {
- emit ("if (rotbit) op_dst |= 1;\n");
- }
- emit ("context->carry = rotbit;\n");
- emit ("}\n");
- }
- emit ("tmp = (<c_size>)op_dst;\n");
- emit ("context->zero = tmp == 0;\n");
- emit ("context->sign = (int)tmp < 0;\n");
- emit ("context->overflow = ((int)tmp < 0) != ((int)op_dst < 0);\n");
- emit ("context->cycles += 3*op_src;\n");
- emit ("context->broken_flags = 0;\n");
-
-}
-
-static void
-adiv (p)
- opcode_entry_type *p;
-{
- emit ("if (op_src==0)\n");
- emit ("{\n");
- emit ("context->exception = SIM_DIV_ZERO;\n");
- emit ("}\n");
- emit ("else\n");
- emit ("{\n");
-
- if (p->type == 32)
- {
- emit ("op_dst.low = (int)get_long_reg(context,reg_dst+2);\n");
- emit ("op_dst.high = (int)get_long_reg(context,reg_dst+0);\n");
-#ifdef __GNUC__
- emit ("tmp = (((long long)op_dst.high << 32) + (op_dst.low)) / (int)op_src;\n");
-#else
- emit ("tmp = (long)op_dst.low / (int)op_src;\n");
-#endif
- emit ("put_long_reg(context,reg_dst+2, tmp);\n");
-#ifdef __GNUC__
- emit ("put_long_reg(context,reg_dst, (((long long)op_dst.high << 32) + (op_dst.low)) %% (int)op_src);\n");
-#else
- emit ("put_long_reg(context,reg_dst, (int)op_dst.low %% (int)op_src);\n");
-#endif
-
- emit ("context->zero = op_src == 0 || (op_dst.low==0 && op_dst.high==0);\n");
- }
- else
- {
- emit ("tmp = (long)op_dst / (short)op_src;\n");
- emit ("put_word_reg(context,reg_dst+1, tmp);\n");
- emit ("put_word_reg(context,reg_dst, (long) op_dst %% (short)op_src);\n");
- emit ("context->zero = op_src == 0 || op_dst==0;\n");
- }
-
- emit ("context->sign = (int)tmp < 0;\n");
- emit ("context->overflow =(tmp & 0x%x) != 0;\n",
- ~((1 << (p->type)) - 1));
- emit ("context->carry = (tmp & 0x%x) != 0;\n",
- ~(1 << (p->type)));
-
- emit ("}\n");
-}
-
-static void
-dobit (p)
-opcode_entry_type *p;
-{
- emit("context->zero = (op_dst & (1<<op_src))==0;\n");
- emit("context->broken_flags = 0;\n");
-}
-static void
-doset (p, v)
-opcode_entry_type*p;
-int v;
-{
- if (v)
- emit (" tmp = op_dst | (1<< op_src);\n");
- else
- emit (" tmp = op_dst & ~(1<< op_src);\n");
-}
-
-static void
-mult (p)
- opcode_entry_type *p;
-{
-
- if (p->type == 32)
- {
- emit ("op_dst.low = get_long_reg(context,reg_dst+2);\n");
- emit ("tmp = op_dst.low * op_src;\n");
- emit ("put_long_reg(context,reg_dst+2, tmp);\n");
- emit ("put_long_reg(context,reg_dst, 0);\n");
- }
- else
- {
- emit ("op_dst = get_word_reg(context,reg_dst+1);\n");
- emit ("tmp = op_dst * op_src;\n");
- emit ("put_long_reg(context,reg_dst, tmp);\n");
- }
-
- emit ("context->sign = (int)tmp < 0;\n");
- emit ("context->overflow =0;\n");
- emit ("context->carry = (tmp & 0x%x) != 0;\n", ~((1 << (p->type)) - 1));
- emit ("context->zero = tmp == 0;\n");
-
-}
-
-static void
-exts (p)
- opcode_entry_type *p;
-{
- /* Fetch the ls part of the src */
- current_size = size_name (p->type * 2);
-
- if (p->type == 32)
- {
- emit ("tmp= get_long_reg(context,reg_dst+2);\n");
- emit ("if (tmp & (1<<%d)) {\n", p->type - 1);
- emit ("put_long_reg(context,reg_dst, 0xffffffff);\n");
- emit ("}\n");
- emit ("else\n");
- emit ("{\n");
- emit ("put_long_reg(context,reg_dst, 0);\n");
- emit ("}\n");
- }
- else
- {
- emit ("tmp= get_<size>_reg(context,reg_dst);\n");
- emit ("if (tmp & (1<<%d)) {\n", p->type - 1);
- emit ("tmp |= 0x%x;\n", ~((1 << p->type) - 1));
- emit ("}\n");
- emit ("else\n");
- emit ("{\n");
-
- emit ("tmp &= 0x%x;\n", ((1 << p->type) - 1));
- emit ("}\n");
- emit ("put_<size>_reg(context,reg_dst, tmp);\n");
- }
-}
-doflag(on)
-int on;
-{
- /* Load up the flags */
- emit(" COND (context, 0x0b);\n");
-
- if (on)
- emit ("{ int on =1;\n ");
- else
- emit ("{ int on =0;\n ");
-
- emit ("if (imm_src & 1)\n");
- emit ("PSW_OVERFLOW = on;\n");
-
- emit ("if (imm_src & 2)\n");
- emit ("PSW_SIGN = on;\n");
-
- emit ("if (imm_src & 4)\n");
- emit ("PSW_ZERO = on;\n");
-
- emit ("if (imm_src & 8)\n");
- emit ("PSW_CARRY = on;\n");
- emit("}\n");
-
-
-}
-/* emit code to perform operation */
-void
-info_docode (p)
- opcode_entry_type *p;
-{
- switch (p->opcode)
- {
- case OPC_clr:
- case OPC_clrb:
- emit ("tmp = 0;\n");
- break;
- case OPC_ex:
- case OPC_exb:
-
- emit ("tmp = op_src; \n");
- if (allregs)
- {
- emit ("put_<size>_reg(context,reg_src, op_dst);\n");
- }
- else
- {
- emit ("put_<size>_mem_da(context, oplval_src, op_dst);\n");
- }
- break;
- case OPC_adc:
- case OPC_adcb:
- normal_flags (p, "op_src += COND(context,7);tmp = op_dst + op_src ;",0);
- break;
- case OPC_sbc:
- normal_flags (p, "op_src += COND(context,7);tmp = op_dst - op_src ;",1);
- break;
- case OPC_nop:
- break;
- case OPC_com:
- case OPC_comb:
- test_normal_flags (p, "tmp = ~ op_dst", 1);
- break;
- case OPC_and:
- case OPC_andb:
- test_normal_flags (p, "tmp = op_dst & op_src", 1);
- break;
- case OPC_xor:
- case OPC_xorb:
- test_normal_flags (p, "tmp = op_dst ^ op_src", 1);
- break;
- case OPC_or:
- case OPC_orb:
- test_normal_flags (p, "tmp = op_dst | op_src", 1);
- break;
- case OPC_sla:
- case OPC_slab:
- case OPC_slal:
- case OPC_sda:
- case OPC_sdab:
- case OPC_sdal:
- shift (p, 1);
- break;
-
- case OPC_sll:
- case OPC_sllb:
- case OPC_slll:
- case OPC_sdl:
- case OPC_sdlb:
- case OPC_sdll:
- shift (p, 0);
- break;
- case OPC_rl:
- rotate (p, 0, 16, 1);
- break;
- case OPC_rlb:
- rotate (p, 0, 8, 1);
- break;
- case OPC_rr:
- rotate (p, 0, 16, 0);
- break;
- case OPC_rrb:
- rotate (p, 0, 8, 0);
- break;
- case OPC_rrc:
- rotate (p, 1, 16, 0);
- break;
- case OPC_rrcb:
- rotate (p, 1, 8, 0);
- break;
- case OPC_rlc:
- rotate (p, 1, 16, 1);
- break;
- case OPC_rlcb:
- rotate (p, 1, 8, 1);
- break;
-
- case OPC_extsb:
- case OPC_exts:
- case OPC_extsl:
- exts (p);
- break;
- case OPC_add:
- case OPC_addb:
- case OPC_addl:
- case OPC_inc:
- case OPC_incb:
- optimize_normal_flags (p, "tmp = op_dst + op_src",0);
- break;
- case OPC_testb:
- case OPC_test:
- case OPC_testl:
- test_normal_flags (p, "tmp = op_dst", 0);
- break;
- case OPC_cp:
- case OPC_cpb:
- case OPC_cpl:
- normal_flags (p, "tmp = op_dst - op_src",1);
- break;
- case OPC_negb:
- case OPC_neg:
- emit ("{\n");
- emit ("int op_src = -op_dst;\n");
- emit ("op_dst = 0;\n");
- optimize_normal_flags (p, "tmp = op_dst + op_src;\n",1);
- emit ("}");
- break;
-
- case OPC_sub:
- case OPC_subb:
- case OPC_subl:
- case OPC_dec:
- case OPC_decb:
- optimize_normal_flags (p, "tmp = op_dst - op_src",1);
- break;
- case OPC_bpt:
- bpt ();
- break;
- case OPC_jr:
- jr (p);
- break;
- case OPC_sc:
- sc ();
- break;
- case OPC_jp:
- jp (p);
- break;
- case OPC_ret:
- ret (p);
- break;
- case OPC_call:
- call (p);
- break;
- case OPC_tcc:
- case OPC_tccb:
- emit ("if(op_cc == 8 || COND(context,op_cc)) put_word_reg(context,reg_dst, 1);\n");
- break;
- case OPC_lda:
- emit ("tmp = oplval_src; \n");
- /*(((oplval_src) & 0xff0000) << 8) | (oplval_src & 0xffff); \n");*/
- break;
- case OPC_ldk:
- case OPC_ld:
-
- case OPC_ldb:
- case OPC_ldl:
- ld (p);
- break;
- case OPC_ldib:
- ldi (p, 8, 1);
- break;
- case OPC_ldi:
- ldi (p, 16, 1);
- break;
-
- case OPC_lddb:
- ldi (p, 8, -1);
- break;
- case OPC_ldd:
- ldi (p, 16, -1);
- break;
-
- case OPC_push:
- case OPC_pushl:
- push (p);
- break;
-
- case OPC_div:
- case OPC_divl:
- adiv (p);
- break;
- case OPC_mult:
- case OPC_multl:
- mult (p);
- break;
- case OPC_pop:
- case OPC_popl:
- pop (p);
- break;
- case OPC_set:
- doset (p,1);
- break;
- case OPC_res:
- doset (p,0);
- break;
- case OPC_bit:
- dobit(p);
- break;
- case OPC_resflg:
- doflag(0);
- break;
- case OPC_setflg:
- doflag(1);
- break;
- default:
-
- emit ("tmp = fail(context,%d);\n", p->opcode);
- break;
- }
-}
-
-/* emit code to store result in calculated lvalue */
-
-void
-info_store (p)
- opcode_entry_type *p;
-{
- unsigned int *i = p->arg_info;
-
- while (*i)
- {
- current_name = reg_n (*i);
- current_size = size_name (p->type);
-
- if (IS_DST (*i))
- {
- switch (*i & CLASS_MASK)
- {
- case CLASS_PR:
- emit ("put_<ptr_mode>_reg(context,reg_<name>, tmp);\n");
- break;
- case CLASS_REG_LONG:
- case CLASS_REG_WORD:
- case CLASS_REG_BYTE:
-
- emit ("put_<size>_reg(context,reg_<name>,tmp);\n");
- break;
- case CLASS_X:
- case CLASS_IR:
- case CLASS_DA:
- case CLASS_BX:
- case CLASS_BA:
-
- emit ("put_<size>_<mem>_da(context,oplval_<name>, tmp);\n");
- break;
- case CLASS_IMM:
- break;
- default:
- emit ("abort(); ");
- break;
- }
-
- }
- i++;
- }
-}
-
-static
-void
-mangle (p, shortcut, value)
- opcode_entry_type *p;
- int shortcut;
- int value;
-{
- int nostore = 0;
- int extra;
- int getdst = 1;
- int before = 0;
- int nosrc = 0;
-
- emit ("/\052 %s \052/\n", p->nicename);
- if (shortcut)
- {
- emit ("int <fop>_%04x(context,pc)\n", value);
- }
- else
- {
- emit ("int <fop>_%d(context,pc,iwords0)\n", p->idx);
- emit ("int iwords0;\n");
- }
- emit ("sim_state_type *context;\n");
- emit ("int pc;\n");
- emit ("{\n");
- emit ("register unsigned int tmp;\n");
- if (shortcut)
- {
- emit ("register unsigned int iwords0 = 0x%x;\n", value);
- }
-
- /* work out how much bigger this opcode could be because it's large
- model */
- if (BIG)
- {
- int i;
-
- extra = 0;
- for (i = 0; i < 4; i++)
- {
- if ((p->arg_info[i] & CLASS_MASK) == CLASS_DA
- || (p->arg_info[i] & CLASS_MASK) == CLASS_X)
- extra += 2;
- }
- }
- else
- {
- extra = 0;
- }
- printf (" /* Length %d */ \n", p->length + extra);
- switch (p->length + extra)
- {
- case 2:
- emit ("pc += 2\n;");
- break;
- case 4:
- emit ("register unsigned int iwords1 = get_word_mem_da(context,pc+2);\n");
- emit ("pc += 4;\n");
- break;
- case 6:
-
- emit ("register unsigned int iwords1 = get_word_mem_da(context,pc+2);\n");
- emit ("register unsigned int iwords2 = get_word_mem_da(context,pc+4);\n");
- emit ("pc += 6;\n");
- break;
- case 8:
- emit ("register unsigned int iwords1 = get_word_mem_da(context,pc+2);\n");
- emit ("register unsigned int iwords2 = get_word_mem_da(context,pc+4);\n");
- emit ("register unsigned int iwords3 = get_word_mem_da(context,pc+6);\n");
- emit ("pc += 8;\n");
- break;
- default:
- break;
-
- }
- emit ("context->cycles += %d;\n", p->cycles);
-
- emit ("{\n");
- info_args (p);
- info_special (p, &getdst, &nostore, &before, &nosrc);
-
- info_lvals (p);
- if (!nosrc)
- {
- info_fetch (p, getdst);
- }
-
- if (before)
- {
- info_docode (p);
- }
- else
- {
- info_docode (p);
- }
- if (!nostore)
- info_store (p);
- emit ("}\n");
- emit ("return pc;\n");
- emit ("}\n");
-}
-
-void
-static
-one_instruction (i)
- int i;
-{
- /* find the table entry */
- opcode_entry_type *p = z8k_table + i;
-
- if (!p)
- return;
- mangle (p, 0, 0);
-}
-
-void
-add_to_list (ptr, value)
- struct opcode_value **ptr;
- int value;
-{
- struct opcode_value *prev;
-
- prev = *ptr;
- *ptr = (struct opcode_value *) malloc (sizeof (struct opcode_value));
-
- (*ptr)->n = value;
- (*ptr)->next = prev;
-}
-
-void
-build_list (i)
- int i;
-{
- opcode_entry_type *p = lookup_inst (i);
-
- if (!p)
- return;
- add_to_list (&list[p->idx], i);
-}
-
-int
-main (ac, av)
- int ac;
- char **av;
-{
- int i;
- int needcomma = 0;
-
- for (i = 1; i < ac; i++)
- {
- if (strcmp (av[i], "-1") == 0)
- file = 1;
- if (strcmp (av[i], "-2") == 0)
- file = 2;
- if (strcmp (av[i], "-3") == 0)
- file = 3;
- if (strcmp (av[i], "-b3") == 0)
- {
- file = 3;
- BIG = 1;
- }
-
- }
-
- /* First work out which opcodes use which bit patterns,
- build a list of all matching bit pattens */
- for (i = 0; i < 1 << 16; i++)
- {
- build_list (i);
- }
-#if DUMP_LIST
- for (i = 0; i < NOPS; i++)
- {
- struct opcode_value *p;
-
- printf ("%d,", i);
- p = list[i];
- while (p)
- {
- printf (" %04x,", p->n);
- p = p->next;
- }
- printf ("-1\n");
- }
-
-#endif
-
- if (file == 1)
- {
- extern int quick[];
-
- /* Do the shortcuts */
- printf (" /* SHORTCUTS */\n");
- for (i = 0; quick[i]; i++)
- {
- int t = quick[i];
-
- mangle (lookup_inst (t), 1, t);
- }
- }
- if (file == 3)
- {
- printf (" /* NOT -SHORTCUTS */\n");
- for (i = 0; i < NOPS; i++)
- {
- if (list[i])
- {
- one_instruction (i);
- }
- else
- {
- emit ("int <fop>_%d(context,pc)\n", i);
- printf ("sim_state_type *context;\n");
- printf ("int pc;\n");
- emit ("{ <fop>_bad1();return pc; }\n");
- }
- }
- emit ("int <fop>_bad() ;\n");
-
- /* Write the jump table */
- emit ("int (*(<fop>_table[]))() = {");
- needcomma = 0;
- for (i = 0; i < NOPS; i++)
- {
- if (needcomma)
- printf (",");
- emit ("<fop>_%d\n", i);
- needcomma = 1;
- }
- emit ("};\n");
- }
-
- if (file == 2)
- {
- extern int quick[];
- /* Static - since it's too be to be automatic on the apollo */
- static int big[64 * 1024];
-
- for (i = 0; i < 64 * 1024; i++)
- big[i] = 0;
-
- for (i = 0; quick[i]; i++)
- {
-#if 0
-
- printf ("extern int <fop>_%04x();\n", quick[i]);
-#endif
-
- big[quick[i]] = 1;
- }
-
- for (i = 0; i < NOPS; i++)
- {
-#if 0
- printf ("extern int fop_%d();\n", i);
-#endif
- }
-#if 0
- printf ("extern int fop_bad();\n");
-#endif
- printf ("struct op_info op_info_table[] = {\n");
- for (i = 0; i < 1 << 16; i++)
- {
- opcode_entry_type *p = lookup_inst (i);
-
- if (needcomma)
- printf (",");
-#if 0
- if (big[i])
- {
- printf ("<fop>_%04x", i);
- }
- else
-#endif
- if (p != NULL)
- {
- printf ("%d", p->idx);
- }
- else
- printf ("400");
- if (p != NULL)
- {
- printf (" /* %04x %s */\n", i, p->nicename);
- }
- else
- {
- printf ("\n");
- }
- needcomma = 1;
- }
- printf ("};\n");
-
- }
- return 0;
-}
-
-char *
-insn_ptr (n)
- int n;
-{
- if (BIG)
- {
- abort ();
- }
-
- switch (n)
- {
- case 4:
- return "iwords_1";
- default:
- return "fail(context,123)";
- }
-}
-
-/* work out if the opcode only wants lvalues */
-int
-lvalue (p)
- opcode_entry_type *p;
-{
- switch (p->opcode)
- {
- case OPC_lda:
- return 1;
- case OPC_call:
- case OPC_jp:
- return 1;
- default:
- return 0;
- }
-}
-
-int
-info_len_in_words (o)
- opcode_entry_type *o;
-{
- unsigned int *p = o->byte_info;
- int nibs = 0;
-
- while (*p)
- {
- switch (*p & CLASS_MASK)
- {
- case CLASS_IGNORE:
- case CLASS_BIT:
- case CLASS_REGN0:
- case CLASS_REG:
- case CLASS_01II:
- case CLASS_00II:
- nibs++;
- break;
- case CLASS_ADDRESS:
- nibs += SIZE_ADDRESS;
- break;
- case CLASS_IMM:
- switch (*p & ~CLASS_MASK)
- {
- case ARG_IMM16:
- nibs += 4;
- break;
- case ARG_IMM32:
- nibs += 8;
- break;
- case ARG_IMM2:
- case ARG_IMM4:
- case ARG_NIM4:
- case ARG_IMM4M1:
- case ARG_IMM_1:
- case ARG_IMM_2:
- case ARG_IMMNMINUS1:
- nibs++;
- break;
- case ARG_NIM8:
-
- case ARG_IMM8:
- nibs += 2;
- break;
- default:
- abort ();
- }
- break;
- case CLASS_DISP:
- switch (*p & ~CLASS_MASK)
- {
- case ARG_DISP16:
- nibs += 4;
- break;
- case ARG_DISP12:
- nibs += 3;
- break;
- case ARG_DISP8:
- nibs += 2;
- break;
- default:
- abort ();
- }
- break;
- case CLASS_0DISP7:
- case CLASS_1DISP7:
- case CLASS_DISP8:
- nibs += 2;
- break;
- case CLASS_BIT_1OR2:
- case CLASS_0CCC:
- case CLASS_1CCC:
- case CLASS_CC:
- nibs++;
- break;
- default:
- emit ("don't know %x\n", *p);
- }
- p++;
- }
-
- return nibs / 4; /* return umber of words */
-}