/*
* Z80 disassembler for Z80-CPU simulator
*
* Copyright (C) 1989-92 by Udo Munk
*
* This module of the Z80-CPU simulator must not be modified by a user,
* see license agreement!
*
* History:
* 06-DEC-89 Development on TARGON/35 with AT&T Unix System V.3
* 07-APR-92 forget to implement Op-Codes LD A,R and LD R,A, added
* 25-JUN-92 comments in english
*/

#include <stdio.h>

/*
*      Forward declarations
*/
int opout(), nout(), iout(), rout(), nnout(), inout(), cbop(), edop(), ddfd();

/*
*      Op-code tables
*/
struct opt {
       int (*fun) ();
       char *text;
};

static struct opt optab[256] = {
       { opout,  "NOP"                 },      /* 0x00 */
       { nnout,  "LD\tBC,"             },      /* 0x01 */
       { opout,  "LD\t(BC),A"          },      /* 0x02 */
       { opout,  "INC\tBC"             },      /* 0x03 */
       { opout,  "INC\tB"              },      /* 0x04 */
       { opout,  "DEC\tB"              },      /* 0x05 */
       { nout,   "LD\tB,"              },      /* 0x06 */
       { opout,  "RLCA"                },      /* 0x07 */
       { opout,  "EX\tAF,AF'"          },      /* 0x08 */
       { opout,  "ADD\tHL,BC"          },      /* 0x09 */
       { opout,  "LD\tA,(BC)"          },      /* 0x0a */
       { opout,  "DEC\tBC"             },      /* 0x0b */
       { opout,  "INC\tC"              },      /* 0x0c */
       { opout,  "DEC\tC"              },      /* 0x0d */
       { nout,   "LD\tC,"              },      /* 0x0e */
       { opout,  "RRCA"                },      /* 0x0f */
       { rout,   "DJNZ\t"              },      /* 0x10 */
       { nnout,  "LD\tDE,"             },      /* 0x11 */
       { opout,  "LD\t(DE),A"          },      /* 0x12 */
       { opout,  "INC\tDE"             },      /* 0x13 */
       { opout,  "INC\tD"              },      /* 0x14 */
       { opout,  "DEC\tD"              },      /* 0x15 */
       { nout,   "LD\tD,"              },      /* 0x16 */
       { opout,  "RLA"                 },      /* 0x17 */
       { rout,   "JR\t"                },      /* 0x18 */
       { opout,  "ADD\tHL,DE"          },      /* 0x19 */
       { opout,  "LD\tA,(DE)"          },      /* 0x1a */
       { opout,  "DEC\tDE"             },      /* 0x1b */
       { opout,  "INC\tE"              },      /* 0x1c */
       { opout,  "DEC\tE"              },      /* 0x1d */
       { nout,   "LD\tE,"              },      /* 0x1e */
       { opout,  "RRA"                 },      /* 0x1f */
       { rout,   "JR\tNZ,"             },      /* 0x20 */
       { nnout,  "LD\tHL,"             },      /* 0x21 */
       { inout,  "LD\t(%04x),HL"       },      /* 0x22 */
       { opout,  "INC\tHL"             },      /* 0x23 */
       { opout,  "INC\tH"              },      /* 0x24 */
       { opout,  "DEC\tH"              },      /* 0x25 */
       { nout,   "LD\tH,"              },      /* 0x26 */
       { opout,  "DAA"                 },      /* 0x27 */
       { rout,   "JR\tZ,"              },      /* 0x28 */
       { opout,  "ADD\tHL,HL"          },      /* 0x29 */
       { inout,  "LD\tHL,(%04x)"       },      /* 0x2a */
       { opout,  "DEC\tHL"             },      /* 0x2b */
       { opout,  "INC\tL"              },      /* 0x2c */
       { opout,  "DEC\tL"              },      /* 0x2d */
       { nout,   "LD\tL,"              },      /* 0x2e */
       { opout,  "CPL"                 },      /* 0x2f */
       { rout,   "JR\tNC,"             },      /* 0x30 */
       { nnout,  "LD\tSP,"             },      /* 0x31 */
       { inout,  "LD\t(%04x),A"        },      /* 0x32 */
       { opout,  "INC\tSP"             },      /* 0x33 */
       { opout,  "INC\t(HL)"           },      /* 0x34 */
       { opout,  "DEC\t(HL)"           },      /* 0x35 */
       { nout,   "LD\t(HL),"           },      /* 0x36 */
       { opout,  "SCF"                 },      /* 0x37 */
       { rout,   "JR\tC,"              },      /* 0x38 */
       { opout,  "ADD\tHL,SP"          },      /* 0x39 */
       { inout,  "LD\tA,(%04x)"        },      /* 0x3a */
       { opout,  "DEC\tSP"             },      /* 0x3b */
       { opout,  "INC\tA"              },      /* 0x3c */
       { opout,  "DEC\tA"              },      /* 0x3d */
       { nout,   "LD\tA,"              },      /* 0x3e */
       { opout,  "CCF"                 },      /* 0x3f */
       { opout,  "LD\tB,B"             },      /* 0x40 */
       { opout,  "LD\tB,C"             },      /* 0x41 */
       { opout,  "LD\tB,D"             },      /* 0x42 */
       { opout,  "LD\tB,E"             },      /* 0x43 */
       { opout,  "LD\tB,H"             },      /* 0x44 */
       { opout,  "LD\tB,L"             },      /* 0x45 */
       { opout,  "LD\tB,(HL)"          },      /* 0x46 */
       { opout,  "LD\tB,A"             },      /* 0x47 */
       { opout,  "LD\tC,B"             },      /* 0x48 */
       { opout,  "LD\tC,C"             },      /* 0x49 */
       { opout,  "LD\tC,D"             },      /* 0x4a */
       { opout,  "LD\tC,E"             },      /* 0x4b */
       { opout,  "LD\tC,H"             },      /* 0x4c */
       { opout,  "LD\tC,L"             },      /* 0x4d */
       { opout,  "LD\tC,(HL)"          },      /* 0x4e */
       { opout,  "LD\tC,A"             },      /* 0x4f */
       { opout,  "LD\tD,B"             },      /* 0x50 */
       { opout,  "LD\tD,C"             },      /* 0x51 */
       { opout,  "LD\tD,D"             },      /* 0x52 */
       { opout,  "LD\tD,E"             },      /* 0x53 */
       { opout,  "LD\tD,H"             },      /* 0x54 */
       { opout,  "LD\tD,L"             },      /* 0x55 */
       { opout,  "LD\tD,(HL)"          },      /* 0x56 */
       { opout,  "LD\tD,A"             },      /* 0x57 */
       { opout,  "LD\tE,B"             },      /* 0x58 */
       { opout,  "LD\tE,C"             },      /* 0x59 */
       { opout,  "LD\tE,D"             },      /* 0x5a */
       { opout,  "LD\tE,E"             },      /* 0x5b */
       { opout,  "LD\tE,H"             },      /* 0x5c */
       { opout,  "LD\tE,L"             },      /* 0x5d */
       { opout,  "LD\tE,(HL)"          },      /* 0x5e */
       { opout,  "LD\tE,A"             },      /* 0x5f */
       { opout,  "LD\tH,B"             },      /* 0x60 */
       { opout,  "LD\tH,C"             },      /* 0x61 */
       { opout,  "LD\tH,D"             },      /* 0x62 */
       { opout,  "LD\tH,E"             },      /* 0x63 */
       { opout,  "LD\tH,H"             },      /* 0x64 */
       { opout,  "LD\tH,L"             },      /* 0x65 */
       { opout,  "LD\tH,(HL)"          },      /* 0x66 */
       { opout,  "LD\tH,A"             },      /* 0x67 */
       { opout,  "LD\tL,B"             },      /* 0x68 */
       { opout,  "LD\tL,C"             },      /* 0x69 */
       { opout,  "LD\tL,D"             },      /* 0x6a */
       { opout,  "LD\tL,E"             },      /* 0x6b */
       { opout,  "LD\tL,H"             },      /* 0x6c */
       { opout,  "LD\tL,L"             },      /* 0x6d */
       { opout,  "LD\tL,(HL)"          },      /* 0x6e */
       { opout,  "LD\tL,A"             },      /* 0x6f */
       { opout,  "LD\t(HL),B"          },      /* 0x70 */
       { opout,  "LD\t(HL),C"          },      /* 0x71 */
       { opout,  "LD\t(HL),D"          },      /* 0x72 */
       { opout,  "LD\t(HL),E"          },      /* 0x73 */
       { opout,  "LD\t(HL),H"          },      /* 0x74 */
       { opout,  "LD\t(HL),L"          },      /* 0x75 */
       { opout,  "HALT"                },      /* 0x76 */
       { opout,  "LD\t(HL),A"          },      /* 0x77 */
       { opout,  "LD\tA,B"             },      /* 0x78 */
       { opout,  "LD\tA,C"             },      /* 0x79 */
       { opout,  "LD\tA,D"             },      /* 0x7a */
       { opout,  "LD\tA,E"             },      /* 0x7b */
       { opout,  "LD\tA,H"             },      /* 0x7c */
       { opout,  "LD\tA,L"             },      /* 0x7d */
       { opout,  "LD\tA,(HL)"          },      /* 0x7e */
       { opout,  "LD\tA,A"             },      /* 0x7f */
       { opout,  "ADD\tA,B"            },      /* 0x80 */
       { opout,  "ADD\tA,C"            },      /* 0x81 */
       { opout,  "ADD\tA,D"            },      /* 0x82 */
       { opout,  "ADD\tA,E"            },      /* 0x83 */
       { opout,  "ADD\tA,H"            },      /* 0x84 */
       { opout,  "ADD\tA,L"            },      /* 0x85 */
       { opout,  "ADD\tA,(HL)"         },      /* 0x86 */
       { opout,  "ADD\tA,A"            },      /* 0x87 */
       { opout,  "ADC\tA,B"            },      /* 0x88 */
       { opout,  "ADC\tA,C"            },      /* 0x89 */
       { opout,  "ADC\tA,D"            },      /* 0x8a */
       { opout,  "ADC\tA,E"            },      /* 0x8b */
       { opout,  "ADC\tA,H"            },      /* 0x8c */
       { opout,  "ADC\tA,L"            },      /* 0x8d */
       { opout,  "ADC\tA,(HL)"         },      /* 0x8e */
       { opout,  "ADC\tA,A"            },      /* 0x8f */
       { opout,  "SUB\tB"              },      /* 0x90 */
       { opout,  "SUB\tC"              },      /* 0x91 */
       { opout,  "SUB\tD"              },      /* 0x92 */
       { opout,  "SUB\tE"              },      /* 0x93 */
       { opout,  "SUB\tH"              },      /* 0x94 */
       { opout,  "SUB\tL"              },      /* 0x95 */
       { opout,  "SUB\t(HL)"           },      /* 0x96 */
       { opout,  "SUB\tA"              },      /* 0x97 */
       { opout,  "SBC\tA,B"            },      /* 0x98 */
       { opout,  "SBC\tA,C"            },      /* 0x99 */
       { opout,  "SBC\tA,D"            },      /* 0x9a */
       { opout,  "SBC\tA,E"            },      /* 0x9b */
       { opout,  "SBC\tA,H"            },      /* 0x9c */
       { opout,  "SBC\tA,L"            },      /* 0x9d */
       { opout,  "SBC\tA,(HL)"         },      /* 0x9e */
       { opout,  "SBC\tA,A"            },      /* 0x9f */
       { opout,  "AND\tB"              },      /* 0xa0 */
       { opout,  "AND\tC"              },      /* 0xa1 */
       { opout,  "AND\tD"              },      /* 0xa2 */
       { opout,  "AND\tE"              },      /* 0xa3 */
       { opout,  "AND\tH"              },      /* 0xa4 */
       { opout,  "AND\tL"              },      /* 0xa5 */
       { opout,  "AND\t(HL)"           },      /* 0xa6 */
       { opout,  "AND\tA"              },      /* 0xa7 */
       { opout,  "XOR\tB"              },      /* 0xa8 */
       { opout,  "XOR\tC"              },      /* 0xa9 */
       { opout,  "XOR\tD"              },      /* 0xaa */
       { opout,  "XOR\tE"              },      /* 0xab */
       { opout,  "XOR\tH"              },      /* 0xac */
       { opout,  "XOR\tL"              },      /* 0xad */
       { opout,  "XOR\t(HL)"           },      /* 0xae */
       { opout,  "XOR\tA"              },      /* 0xaf */
       { opout,  "OR\tB"               },      /* 0xb0 */
       { opout,  "OR\tC"               },      /* 0xb1 */
       { opout,  "OR\tD"               },      /* 0xb2 */
       { opout,  "OR\tE"               },      /* 0xb3 */
       { opout,  "OR\tH"               },      /* 0xb4 */
       { opout,  "OR\tL"               },      /* 0xb5 */
       { opout,  "OR\t(HL)"            },      /* 0xb6 */
       { opout,  "OR\tA"               },      /* 0xb7 */
       { opout,  "CP\tB"               },      /* 0xb8 */
       { opout,  "CP\tC"               },      /* 0xb9 */
       { opout,  "CP\tD"               },      /* 0xba */
       { opout,  "CP\tE"               },      /* 0xbb */
       { opout,  "CP\tH"               },      /* 0xbc */
       { opout,  "CP\tL"               },      /* 0xbd */
       { opout,  "CP\t(HL)"            },      /* 0xbe */
       { opout,  "CP\tA"               },      /* 0xbf */
       { opout,  "RET\tNZ"             },      /* 0xc0 */
       { opout,  "POP\tBC"             },      /* 0xc1 */
       { nnout,  "JP\tNZ,"             },      /* 0xc2 */
       { nnout,  "JP\t"                },      /* 0xc3 */
       { nnout,  "CALL\tNZ,"           },      /* 0xc4 */
       { opout,  "PUSH\tBC"            },      /* 0xc5 */
       { nout,   "ADD\tA,"             },      /* 0xc6 */
       { opout,  "RST\t0"              },      /* 0xc7 */
       { opout,  "RET\tZ"              },      /* 0xc8 */
       { opout,  "RET"                 },      /* 0xc9 */
       { nnout,  "JP\tZ,"              },      /* 0xca */
       { cbop,   ""                    },      /* 0xcb */
       { nnout,  "CALL\tZ,"            },      /* 0xcc */
       { nnout,  "CALL\t"              },      /* 0xcd */
       { nout,   "ADC\tA,"             },      /* 0xce */
       { opout,  "RST\t8"              },      /* 0xcf */
       { opout,  "RET\tNC"             },      /* 0xd0 */
       { opout,  "POP\tDE"             },      /* 0xd1 */
       { nnout,  "JP\tNC,"             },      /* 0xd2 */
       { iout,   "OUT\t(%02x),A"       },      /* 0xd3 */
       { nnout,  "CALL\tNC,"           },      /* 0xd4 */
       { opout,  "PUSH\tDE"            },      /* 0xd5 */
       { nout,   "SUB\t"               },      /* 0xd6 */
       { opout,  "RST\t10"             },      /* 0xd7 */
       { opout,  "RET\tC"              },      /* 0xd8 */
       { opout,  "EXX"                 },      /* 0xd9 */
       { nnout,  "JP\tC,"              },      /* 0xda */
       { iout,   "IN\tA,(%02x)"        },      /* 0xdb */
       { nnout,  "CALL\tC,"            },      /* 0xdc */
       { ddfd,   ""                    },      /* 0xdd */
       { nout,   "SBC\tA,"             },      /* 0xde */
       { opout,  "RST\t18"             },      /* 0xdf */
       { opout,  "RET\tPO"             },      /* 0xe0 */
       { opout,  "POP\tHL"             },      /* 0xe1 */
       { nnout,  "JP\tPO,"             },      /* 0xe2 */
       { opout,  "EX\t(SP),HL"         },      /* 0xe3 */
       { nnout,  "CALL\tPO,"           },      /* 0xe4 */
       { opout,  "PUSH\tHL"            },      /* 0xe5 */
       { nout,   "AND\t"               },      /* 0xe6 */
       { opout,  "RST\t20"             },      /* 0xe7 */
       { opout,  "RET\tPE"             },      /* 0xe8 */
       { opout,  "JP\t(HL)"            },      /* 0xe9 */
       { nnout,  "JP\tPE,"             },      /* 0xea */
       { opout,  "EX\tDE,HL"           },      /* 0xeb */
       { nnout,  "CALL\tPE,"           },      /* 0xec */
       { edop,   ""                    },      /* 0xed */
       { nout,   "XOR\t"               },      /* 0xee */
       { opout,  "RST\t28"             },      /* 0xef */
       { opout,  "RET\tP"              },      /* 0xf0 */
       { opout,  "POP\tAF"             },      /* 0xf1 */
       { nnout,  "JP\tP,"              },      /* 0xf2 */
       { opout,  "DI"                  },      /* 0xf3 */
       { nnout,  "CALL\tP,"            },      /* 0xf4 */
       { opout,  "PUSH\tAF"            },      /* 0xf5 */
       { nout,   "OR\t"                },      /* 0xf6 */
       { opout,  "RST\t30"             },      /* 0xf7 */
       { opout,  "RET\tM"              },      /* 0xf8 */
       { opout,  "LD\tSP,HL"           },      /* 0xf9 */
       { nnout,  "JP\tM,"              },      /* 0xfa */
       { opout,  "EI"                  },      /* 0xfb */
       { nnout,  "CALL\tM,"            },      /* 0xfc */
       { ddfd,   ""                    },      /* 0xfd */
       { nout,   "CP\t"                },      /* 0xfe */
       { opout,  "RST\t38"             }       /* 0xff */
};

static int addr;
static char *unkown = "???";
static char *reg[] = { "B", "C", "D", "E", "H", "L", "(HL)", "A" };
static char *regix = "IX";
static char *regiy = "IY";

/*
*      The function disass() is the only global function of
*      this module. The first argument is a pointer to a
*      unsigned char pointer, which points to the op-code
*      to disassemble. The output of the disassembly goes
*      to stdout, terminated by a newline. After the
*      disassembly the pointer to the op-code will be
*      increased by the size of the op-code, so that
*      disass() can be called again.
*      The secound argument is the (Z80) address of the
*      op-code to disassemble. It is used to calculate the
*      destination address of relative jumps.
*/
disass(p, adr)
register unsigned char **p;
int adr;
{
       register int len;

       addr = adr;
       len = (*optab[**p].fun) (optab[**p].text, p);
       *p += len;
}

/*
*      disassemble 1 byte op-codes
*/
static opout(s, p)
register char *s, **p;
{
       puts(s);
       return(1);
}

/*
*      disassemble 2 byte op-codes of type "Op n"
*/
static nout(s, p)
register char *s;
unsigned char **p;
{
       printf("%s%02x\n", s, *(*p + 1));
       return(2);
}

/*
*      disassemble 2 byte op-codes with indirect addressing
*/
static iout(s, p)
register char *s;
unsigned char **p;
{
       printf(s, *(*p + 1));
       putchar('\n');
       return(2);
}

/*
*      disassemble 2 byte op-codes with relative addressing
*/
static rout(s, p)
register char *s;
char **p;
{
       printf("%s%04x\n", s, addr + *(*p + 1) + 2);
       return(2);
}

/*
*      disassemble 3 byte op-codes of type "Op nn"
*/
static nnout(s, p)
register char *s;
unsigned char **p;
{
       register int i;

       i = *(*p + 1) + (*(*p + 2) << 8);
       printf("%s%04x\n", s, i);
       return(3);
}

/*
*      disassemble 3 byte op-codes with indirect addressing
*/
static inout(s, p)
register char *s;
unsigned char **p;
{
       register int i;

       i = *(*p + 1) + (*(*p + 2) << 8);
       printf(s, i);
       putchar('\n');
       return(3);
}

/*
*      disassemble multi byte op-codes with prefix 0xcb
*/
static cbop(s, p)
register char *s;
unsigned char **p;
{
       register int b2;

       b2 = *(*p + 1);
       if (b2 >= 0x00 && b2 <= 0x07) {
               printf("RLC\t");
               printf("%s\n", reg[b2 & 7]);
               return(2);
       }
       if (b2 >= 0x08 && b2 <= 0x0f) {
               printf("RRC\t");
               printf("%s\n", reg[b2 & 7]);
               return(2);
       }
       if (b2 >= 0x10 && b2 <= 0x17) {
               printf("RL\t");
               printf("%s\n", reg[b2 & 7]);
               return(2);
       }
       if (b2 >= 0x18 && b2 <= 0x1f) {
               printf("RR\t");
               printf("%s\n", reg[b2 & 7]);
               return(2);
       }
       if (b2 >= 0x20 && b2 <= 0x27) {
               printf("SLA\t");
               printf("%s\n", reg[b2 & 7]);
               return(2);
       }
       if (b2 >= 0x28 && b2 <= 0x2f) {
               printf("SRA\t");
               printf("%s\n", reg[b2 & 7]);
               return(2);
       }
       if (b2 >= 0x38 && b2 <= 0x3f) {
               printf("SRL\t");
               printf("%s\n", reg[b2 & 7]);
               return(2);
       }
       if (b2 >= 0x40 && b2 <= 0x7f) {
               printf("BIT\t");
               printf("%c,", ((b2 >> 3) & 7) + '0');
               printf("%s\n", reg[b2 & 7]);
               return(2);
       }
       if (b2 >= 0x80 && b2 <= 0xbf) {
               printf("RES\t");
               printf("%c,", ((b2 >> 3) & 7) + '0');
               printf("%s\n", reg[b2 & 7]);
               return(2);
       }
       if (b2 >= 0xc0) {
               printf("SET\t");
               printf("%c,", ((b2 >> 3) & 7) + '0');
               printf("%s\n", reg[b2 & 7]);
               return(2);
       }
       puts(unkown);
       return(2);
}

/*
*      disassemble multi byte op-codes with prefix 0xed
*/
static edop(s, p)
register char *s;
unsigned char **p;
{
       register int b2, i;
       int len = 2;

       b2 = *(*p + 1);
       switch (b2) {
       case 0x40:
               puts("IN\tB,(C)");
               break;
       case 0x41:
               puts("OUT\t(C),B");
               break;
       case 0x42:
               puts("SBC\tHL,BC");
               break;
       case 0x43:
               i = *(*p + 2) + (*(*p + 3) << 8);
               printf("LD\t(%04x),BC\n", i);
               len = 4;
               break;
       case 0x44:
               puts("NEG");
               break;
       case 0x45:
               puts("RETN");
               break;
       case 0x46:
               puts("IM\t0");
               break;
       case 0x47:
               puts("LD\tI,A");
               break;
       case 0x48:
               puts("IN\tC,(C)");
               break;
       case 0x49:
               puts("OUT\t(C),C");
               break;
       case 0x4a:
               puts("ADC\tHL,BC");
               break;
       case 0x4b:
               i = *(*p + 2) + (*(*p + 3) << 8);
               printf("LD\tBC,(%04x)\n", i);
               len = 4;
               break;
       case 0x4d:
               puts("RETI");
               break;
       case 0x4f:
               puts("LD\tR,A");
               break;
       case 0x50:
               puts("IN\tD,(C)");
               break;
       case 0x51:
               puts("OUT\t(C),D");
               break;
       case 0x52:
               puts("SBC\tHL,DE");
               break;
       case 0x53:
               i = *(*p + 2) + (*(*p + 3) << 8);
               printf("LD\t(%04x),DE\n", i);
               len = 4;
               break;
       case 0x56:
               puts("IM\t1");
               break;
       case 0x57:
               puts("LD\tA,I");
               break;
       case 0x58:
               puts("IN\tE,(C)");
               break;
       case 0x59:
               puts("OUT\t(C),E");
               break;
       case 0x5a:
               puts("ADC\tHL,DE");
               break;
       case 0x5b:
               i = *(*p + 2) + (*(*p + 3) << 8);
               printf("LD\tDE,(%04x)\n", i);
               len = 4;
               break;
       case 0x5e:
               puts("IM\t2");
               break;
       case 0x5f:
               puts("LD\tA,R");
               break;
       case 0x60:
               puts("IN\tH,(C)");
               break;
       case 0x61:
               puts("OUT\t(C),H");
               break;
       case 0x62:
               puts("SBC\tHL,HL");
               break;
       case 0x67:
               puts("RRD");
               break;
       case 0x68:
               puts("IN\tL,(C)");
               break;
       case 0x69:
               puts("OUT\t(C),L");
               break;
       case 0x6a:
               puts("ADC\tHL,HL");
               break;
       case 0x6f:
               puts("RLD");
               break;
       case 0x72:
               puts("SBC\tHL,SP");
               break;
       case 0x73:
               i = *(*p + 2) + (*(*p + 3) << 8);
               printf("LD\t(%04x),SP\n", i);
               len = 4;
               break;
       case 0x78:
               puts("IN\tA,(C)");
               break;
       case 0x79:
               puts("OUT\t(C),A");
               break;
       case 0x7a:
               puts("ADC\tHL,SP");
               break;
       case 0x7b:
               i = *(*p + 2) + (*(*p + 3) << 8);
               printf("LD\tSP,(%04x)\n", i);
               len = 4;
               break;
       case 0xa0:
               puts("LDI");
               break;
       case 0xa1:
               puts("CPI");
               break;
       case 0xa2:
               puts("INI");
               break;
       case 0xa3:
               puts("OUTI");
               break;
       case 0xa8:
               puts("LDD");
               break;
       case 0xa9:
               puts("CPD");
               break;
       case 0xaa:
               puts("IND");
               break;
       case 0xab:
               puts("OUTD");
               break;
       case 0xb0:
               puts("LDIR");
               break;
       case 0xb1:
               puts("CPIR");
               break;
       case 0xb2:
               puts("INIR");
               break;
       case 0xb3:
               puts("OTIR");
               break;
       case 0xb8:
               puts("LDDR");
               break;
       case 0xb9:
               puts("CPDR");
               break;
       case 0xba:
               puts("INDR");
               break;
       case 0xbb:
               puts("OTDR");
               break;
       default:
               puts(unkown);
       }
       return(len);
}

/*
*      disassemble multi byte op-codes with prefix 0xdd and 0xfd
*/
static ddfd(s, p)
register char *s;
unsigned char **p;
{
       register int b2;
       register char *ireg;
       int len = 3;

       if (**p == 0xdd)
               ireg = regix;
       else
               ireg = regiy;
       b2 = *(*p + 1);
       if (b2 >= 0x70 && b2 <= 0x77) {
               printf("LD\t(%s+%02x),%s\n", ireg, *(*p + 2), reg[b2 & 7]);
               return(3);
       }
       switch (b2) {
       case 0x09:
               printf("ADD\t%s,BC\n", ireg);
               len = 2;
               break;
       case 0x19:
               printf("ADD\t%s,DE\n", ireg);
               len = 2;
               break;
       case 0x21:
               printf("LD\t%s,%04x\n", ireg, *(*p + 2) +
(*(*p   + 3) << 8));
               len = 4;
               break;
       case 0x22:
               printf("LD\t(%04x),%s\n", *(*p + 2) + (*(*p + 3) << 8), ireg);
               len = 4;
               break;
       case 0x23:
               printf("INC\t%s\n", ireg);
               len = 2;
               break;
       case 0x29:
               if (**p == 0xdd)
                       printf("ADD\tIX,IX\n");
               else
                       printf("ADD\tIY,IY\n");
               len = 2;
               break;
       case 0x2a:
               printf("LD\t%s,(%04x)\n", ireg, *(*p + 2) + (*(*p + 3) << 8));
               len = 4;
               break;
       case 0x2b:
               printf("DEC\t%s\n", ireg);
               len = 2;
               break;
       case 0x34:
               printf("INC\t(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0x35:
               printf("DEC\t(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0x36:
               printf("LD\t(%s+%02x),%02x\n", ireg, *(*p + 2), *(*p + 3));
               len = 4;
               break;
       case 0x39:
               printf("ADD\t%s,SP\n", ireg);
               len = 2;
               break;
       case 0x46:
               printf("LD\tB,(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0x4e:
               printf("LD\tC,(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0x56:
               printf("LD\tD,(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0x5e:
               printf("LD\tE,(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0x66:
               printf("LD\tH,(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0x6e:
               printf("LD\tL,(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0x7e:
               printf("LD\tA,(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0x86:
               printf("ADD\tA,(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0x8e:
               printf("ADC\tA,(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0x96:
               printf("SUB\t(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0x9e:
               printf("SBC\tA,(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0xa6:
               printf("AND\t(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0xae:
               printf("XOR\t(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0xb6:
               printf("OR\t(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0xbe:
               printf("CP\t(%s+%02x)\n", ireg, *(*p + 2));
               break;
       case 0xcb:
               switch (*(*p + 3)) {
               case 0x06:
                       printf("RLC\t(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x0e:
                       printf("RRC\t(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x16:
                       printf("RL\t(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x1e:
                       printf("RR\t(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x26:
                       printf("SLA\t(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x2e:
                       printf("SRA\t(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x3e:
                       printf("SRL\t(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x46:
                       printf("BIT\t0,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x4e:
                       printf("BIT\t1,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x56:
                       printf("BIT\t2,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x5e:
                       printf("BIT\t3,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x66:
                       printf("BIT\t4,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x6e:
                       printf("BIT\t5,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x76:
                       printf("BIT\t6,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x7e:
                       printf("BIT\t7,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x86:
                       printf("RES\t0,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x8e:
                       printf("RES\t1,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x96:
                       printf("RES\t2,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0x9e:
                       printf("RES\t3,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0xa6:
                       printf("RES\t4,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0xae:
                       printf("RES\t5,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0xb6:
                       printf("RES\t6,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0xbe:
                       printf("RES\t7,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0xc6:
                       printf("SET\t0,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0xce:
                       printf("SET\t1,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0xd6:
                       printf("SET\t2,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0xde:
                       printf("SET\t3,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0xe6:
                       printf("SET\t4,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0xee:
                       printf("SET\t5,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0xf6:
                       printf("SET\t6,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               case 0xfe:
                       printf("SET\t7,(%s+%02x)\n", ireg, *(*p + 2));
                       break;
               default:
                       puts(unkown);
               }
               len = 4;
               break;
       case 0xe1:
               printf("POP\t%s\n", ireg);
               len = 2;
               break;
       case 0xe3:
               printf("EX\t(SP),%s\n", ireg);
               len = 2;
               break;
       case 0xe5:
               printf("PUSH\t%s\n", ireg);
               len = 2;
               break;
       case 0xe9:
               printf("JP\t(%s)\n", ireg);
               len = 2;
               break;
       case 0xf9:
               printf("LD\tSP,%s\n", ireg);
               len = 2;
               break;
       default:
               puts(unkown);
       }
       return(len);
}