212 lines
5.0 KiB
C++
212 lines
5.0 KiB
C++
/*
|
|
* Interactive disassembler (IDA).
|
|
* Zilog Z8 module
|
|
*
|
|
*/
|
|
|
|
#include "z8.hpp"
|
|
|
|
static const char *const phrases[] =
|
|
{
|
|
"F", "LT", "LE", "ULE", "OV", "MI", "Z", "C",
|
|
"T", "GE", "GT", "UGT", "NOV", "PL", "NZ", "NC"
|
|
};
|
|
|
|
//----------------------------------------------------------------------
|
|
inline void z8_t::out_reg(outctx_t &ctx, int rgnum)
|
|
{
|
|
ctx.out_register(ph.reg_names[rgnum]);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------
|
|
void z8_t::z8_header(outctx_t &ctx)
|
|
{
|
|
ctx.gen_header(GH_PRINT_PROC_ASM_AND_BYTESEX);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------
|
|
void z8_t::z8_footer(outctx_t &ctx)
|
|
{
|
|
ctx.gen_empty_line();
|
|
|
|
ctx.out_line(ash.end, COLOR_ASMDIR);
|
|
ctx.flush_outbuf(DEFAULT_INDENT);
|
|
|
|
ctx.gen_cmt_line("end of file");
|
|
}
|
|
|
|
//--------------------------------------------------------------------------
|
|
//lint -esym(1764, ctx) could be made const
|
|
//lint -esym(818, Srange) could be made const
|
|
void z8_t::z8_segstart(outctx_t &ctx, segment_t *Srange)
|
|
{
|
|
qstring sname;
|
|
get_visible_segm_name(&sname, Srange);
|
|
|
|
ctx.gen_cmt_line(COLSTR("segment %s", SCOLOR_AUTOCMT), sname.c_str());
|
|
|
|
ea_t org = ctx.insn_ea - get_segm_base(Srange);
|
|
if ( org != 0 )
|
|
{
|
|
char buf[MAX_NUMBUF];
|
|
btoa(buf, sizeof(buf), org);
|
|
ctx.gen_cmt_line("%s %s", ash.origin, buf);
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------
|
|
//lint -esym(818, seg) could be made const
|
|
void z8_t::z8_segend(outctx_t &ctx, segment_t *seg)
|
|
{
|
|
qstring sname;
|
|
get_visible_segm_name(&sname, seg);
|
|
ctx.gen_cmt_line("end of '%s'", sname.c_str());
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
void idaapi out_insn(outctx_t &ctx)
|
|
{
|
|
ctx.out_mnemonic();
|
|
|
|
ctx.out_one_operand(0);
|
|
|
|
if ( ctx.insn.Op2.type != o_void )
|
|
{
|
|
ctx.out_symbol(',');
|
|
ctx.out_char(' ');
|
|
ctx.out_one_operand(1);
|
|
}
|
|
|
|
ctx.out_immchar_cmts();
|
|
ctx.flush_outbuf();
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
bool z8_t::out_opnd(outctx_t &ctx, const op_t &x)
|
|
{
|
|
uval_t v;
|
|
|
|
z8_t &pm = *static_cast<z8_t *>(ctx.procmod);
|
|
switch ( x.type )
|
|
{
|
|
case o_imm:
|
|
ctx.out_symbol('#');
|
|
ctx.out_value(x, OOFW_IMM);
|
|
break;
|
|
|
|
case o_ind_reg:
|
|
ctx.out_symbol('@');
|
|
// fallthrough
|
|
|
|
case o_reg:
|
|
out_reg(ctx, x.reg);
|
|
break;
|
|
|
|
case o_phrase:
|
|
ctx.out_keyword(phrases[x.phrase]);
|
|
break;
|
|
|
|
case o_displ:
|
|
ctx.out_value(x, OOF_ADDR | OOFW_16);
|
|
ctx.out_symbol('(');
|
|
out_reg(ctx, x.reg);
|
|
ctx.out_symbol(')');
|
|
break;
|
|
|
|
case o_ind_mem:
|
|
ctx.out_symbol('@');
|
|
// fallthrough
|
|
|
|
case o_mem:
|
|
case o_near:
|
|
v = pm.map_addr(ctx.insn, x.addr, x.n, x.type != o_near);
|
|
if ( !ctx.out_name_expr(x, v, x.addr) )
|
|
{
|
|
const char *name = pm.find_ioport(v);
|
|
if ( name != NULL )
|
|
{
|
|
ctx.out_line(name, COLOR_IMPNAME);
|
|
}
|
|
else
|
|
{
|
|
ctx.out_value(x, OOF_ADDR | OOF_NUMBER | OOFS_NOSIGN | OOFW_16);
|
|
remember_problem(PR_NONAME, ctx.insn.ea);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case o_void:
|
|
return 0;
|
|
|
|
default:
|
|
warning("out: %a: bad optype %d", ctx.insn.ea, x.type);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------
|
|
static void out_equ(outctx_t &ctx, const char *name, const char *equ, uchar off)
|
|
{
|
|
ctx.out_line(name, COLOR_DNAME);
|
|
ctx.out_char(' ');
|
|
ctx.out_line(equ, COLOR_KEYWORD);
|
|
ctx.out_char(' ');
|
|
ctx.out_tagon(COLOR_NUMBER);
|
|
ctx.out_btoa(off);
|
|
ctx.out_tagoff(COLOR_NUMBER);
|
|
ctx.ctxflags |= CTXF_LABEL_OK;
|
|
ctx.flush_outbuf(0x80000000);
|
|
}
|
|
|
|
//--------------------------------------------------------------------------
|
|
void z8_t::z8_data(outctx_t &ctx, bool analyze_only)
|
|
{
|
|
ea_t ea = ctx.insn_ea;
|
|
segment_t *s = getseg(ea);
|
|
if ( s != NULL && s->type == SEG_IMEM )
|
|
{
|
|
qstring name;
|
|
if ( get_visible_name(&name, ea) > 0 )
|
|
out_equ(ctx, name.begin(), ash.a_equ, uint16(ea - get_segm_base(s)));
|
|
}
|
|
else
|
|
{
|
|
ctx.out_data(analyze_only);
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------------
|
|
//lint -esym(1764, ctx) could be made const
|
|
void z8_t::z8_assumes(outctx_t &ctx)
|
|
{
|
|
ea_t ea = ctx.insn_ea;
|
|
segment_t *seg = getseg(ea);
|
|
if ( (inf_get_outflags() & OFLG_GEN_ASSUME) == 0 || seg == NULL )
|
|
return;
|
|
// always show at the start of code segments
|
|
bool seg_started = (ea == seg->start_ea) && (seg->type == SEG_CODE);
|
|
|
|
sreg_range_t sra;
|
|
if ( !get_sreg_range(&sra, ea, rRp) )
|
|
return;
|
|
bool show = sra.start_ea == ea;
|
|
if ( show )
|
|
{
|
|
sreg_range_t prev_sra;
|
|
if ( get_prev_sreg_range(&prev_sra, ea, rRp) )
|
|
show = sra.val != prev_sra.val;
|
|
}
|
|
if ( seg_started || show )
|
|
{
|
|
sel_t rp = sra.val;
|
|
if ( rp == BADSEL )
|
|
rp = 0;
|
|
char num[MAX_NUMBUF];
|
|
btoa(num, sizeof(num), rp);
|
|
char nbuf[MAXSTR];
|
|
qsnprintf(nbuf, sizeof(nbuf), COLSTR(".rp %s", SCOLOR_ASMDIR), num);
|
|
ctx.flush_buf(nbuf, DEFAULT_INDENT);
|
|
}
|
|
}
|