Decoding functionality for Bochs disassembler.
Fixed 'step over' debugger command using bx_dbg_read_linear method. Small debugger fix in cpu.cc
This commit is contained in:
parent
f2de4e5d70
commit
5a65e1065e
@ -1,5 +1,5 @@
|
|||||||
/////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////
|
||||||
// $Id: dbg_main.cc,v 1.48 2006-02-02 22:33:32 sshwarts Exp $
|
// $Id: dbg_main.cc,v 1.49 2006-02-05 19:48:26 sshwarts Exp $
|
||||||
/////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////
|
||||||
//
|
//
|
||||||
// Copyright (C) 2001 MandrakeSoft S.A.
|
// Copyright (C) 2001 MandrakeSoft S.A.
|
||||||
@ -577,31 +577,31 @@ void bx_dbg_timebp_command(bx_bool absolute, Bit64u time)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (timebp_timer >= 0) {
|
if (timebp_timer >= 0) {
|
||||||
if (timebp_queue_size == 0 || abs_time < timebp_queue[0]) {
|
if (timebp_queue_size == 0 || abs_time < timebp_queue[0]) {
|
||||||
/* first in queue */
|
/* first in queue */
|
||||||
for (int i = timebp_queue_size; i >= 0; i--)
|
for (int i = timebp_queue_size; i >= 0; i--)
|
||||||
timebp_queue[i+1] = timebp_queue[i];
|
timebp_queue[i+1] = timebp_queue[i];
|
||||||
timebp_queue[0] = abs_time;
|
timebp_queue[0] = abs_time;
|
||||||
timebp_queue_size++;
|
timebp_queue_size++;
|
||||||
bx_pc_system.activate_timer_ticks(timebp_timer, diff, 1);
|
bx_pc_system.activate_timer_ticks(timebp_timer, diff, 1);
|
||||||
} else {
|
} else {
|
||||||
/* not first, insert at suitable place */
|
/* not first, insert at suitable place */
|
||||||
for (int i = 1; i < timebp_queue_size; i++) {
|
for (int i = 1; i < timebp_queue_size; i++) {
|
||||||
if (timebp_queue[i] == abs_time) {
|
if (timebp_queue[i] == abs_time) {
|
||||||
dbg_printf("Time breakpoint not inserted (duplicate)\n");
|
dbg_printf("Time breakpoint not inserted (duplicate)\n");
|
||||||
return;
|
return;
|
||||||
} else if (abs_time < timebp_queue[i]) {
|
} else if (abs_time < timebp_queue[i]) {
|
||||||
for (int j = timebp_queue_size; j >= i; j++)
|
for (int j = timebp_queue_size; j >= i; j++)
|
||||||
timebp_queue[j+1] = timebp_queue[j];
|
timebp_queue[j+1] = timebp_queue[j];
|
||||||
timebp_queue[i] = abs_time;
|
timebp_queue[i] = abs_time;
|
||||||
goto inserted;
|
goto inserted;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
/* last */
|
||||||
/* last */
|
timebp_queue[timebp_queue_size] = abs_time;
|
||||||
timebp_queue[timebp_queue_size] = abs_time;
|
|
||||||
inserted:
|
inserted:
|
||||||
timebp_queue_size++;
|
timebp_queue_size++;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
timebp_queue_size = 1;
|
timebp_queue_size = 1;
|
||||||
timebp_queue[0] = abs_time;
|
timebp_queue[0] = abs_time;
|
||||||
@ -2614,8 +2614,8 @@ void bx_dbg_print_descriptor (unsigned char desc[8], int verbose)
|
|||||||
dbg_printf("P=present=%d\n", present);
|
dbg_printf("P=present=%d\n", present);
|
||||||
#endif
|
#endif
|
||||||
/* brief output */
|
/* brief output */
|
||||||
// 32-bit trap gate, target=0010:c0108ec4, DPL=0, present=1
|
// 32-bit trap gate, target=0010:c0108ec4, DPL=0, present=1
|
||||||
// code segment, base=0000:00cfffff, length=0xffff
|
// code segment, base=0000:00cfffff, length=0xffff
|
||||||
if (s) {
|
if (s) {
|
||||||
// either a code or a data segment. bit 11 (type file MSB) then says
|
// either a code or a data segment. bit 11 (type file MSB) then says
|
||||||
// 0=data segment, 1=code seg
|
// 0=data segment, 1=code seg
|
||||||
@ -3417,30 +3417,19 @@ Bit32u bx_dbg_get_laddr(Bit16u sel, Bit32u ofs)
|
|||||||
|
|
||||||
void bx_dbg_step_over_command ()
|
void bx_dbg_step_over_command ()
|
||||||
{
|
{
|
||||||
Bit8u *fetchPtr;
|
bx_address Laddr = BX_CPU(which_cpu)->guard_found.laddr;
|
||||||
bxInstruction_c iStorage BX_CPP_AlignN (32);
|
|
||||||
bxInstruction_c *i = &iStorage;
|
|
||||||
bx_address Laddr = BX_CPU (dbg_cpu)->get_segment_base(BX_SEG_REG_CS) +
|
|
||||||
BX_CPU (dbg_cpu)->get_ip ();
|
|
||||||
Bit32u Paddr;
|
|
||||||
bx_bool paddr_valid;
|
|
||||||
unsigned remainingInPage;
|
|
||||||
|
|
||||||
BX_CPU (dbg_cpu)->dbg_xlate_linear2phy (Laddr, &Paddr, &paddr_valid);
|
if (! bx_dbg_read_linear(dbg_cpu, Laddr, 16, bx_disasm_ibuf))
|
||||||
|
{
|
||||||
if(!paddr_valid) {
|
|
||||||
dbg_printf("bx_dbg_step_over_command:: Invalid physical address\n");
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
fetchPtr = BX_CPU (dbg_cpu)->mem->getHostMemAddr (BX_CPU(dbg_cpu), Paddr, BX_READ);
|
|
||||||
unsigned ret = BX_CPU (dbg_cpu)->fetchDecode (fetchPtr, i, 15);
|
|
||||||
remainingInPage = BX_CPU(dbg_cpu)->eipPageWindowSize -
|
|
||||||
(BX_CPU(dbg_cpu)->dword.eip + BX_CPU(dbg_cpu)->eipPageBias);
|
|
||||||
|
|
||||||
if (ret == 0)
|
x86_insn insn = bx_disassemble.decode(BX_CPU(which_cpu)->guard_found.is_32bit_code,
|
||||||
BX_CPU (dbg_cpu)->boundaryFetch (fetchPtr, remainingInPage, i);
|
BX_CPU(which_cpu)->guard_found.is_64bit_code,
|
||||||
|
BX_CPU(which_cpu)->get_segment_base(BX_SEG_REG_CS),
|
||||||
|
BX_CPU(which_cpu)->guard_found.eip, bx_disasm_ibuf, bx_disasm_tbuf);
|
||||||
|
|
||||||
unsigned b1 = i->b1 ();
|
unsigned b1 = insn.b1;
|
||||||
|
|
||||||
switch(b1) {
|
switch(b1) {
|
||||||
// Jcc short
|
// Jcc short
|
||||||
@ -3503,7 +3492,7 @@ void bx_dbg_step_over_command ()
|
|||||||
return;
|
return;
|
||||||
// jmp absolute indirect
|
// jmp absolute indirect
|
||||||
case 0xFF:
|
case 0xFF:
|
||||||
switch (i->nnn ()) {
|
switch (insn.nnn) {
|
||||||
// near
|
// near
|
||||||
case 4:
|
case 4:
|
||||||
// far
|
// far
|
||||||
@ -3514,7 +3503,7 @@ void bx_dbg_step_over_command ()
|
|||||||
}
|
}
|
||||||
|
|
||||||
// calls, ints, loops and so on
|
// calls, ints, loops and so on
|
||||||
int BpId = bx_dbg_lbreakpoint_command (bkStepOver, Laddr + i->ilen ());
|
int BpId = bx_dbg_lbreakpoint_command (bkStepOver, Laddr + insn.ilen);
|
||||||
if (BpId == -1) {
|
if (BpId == -1) {
|
||||||
dbg_printf("bx_dbg_step_over_command:: Failed to set lbreakpoint !\n");
|
dbg_printf("bx_dbg_step_over_command:: Failed to set lbreakpoint !\n");
|
||||||
return;
|
return;
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////
|
||||||
// $Id: cpu.cc,v 1.127 2006-02-01 18:12:08 sshwarts Exp $
|
// $Id: cpu.cc,v 1.128 2006-02-05 19:48:28 sshwarts Exp $
|
||||||
/////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////
|
||||||
//
|
//
|
||||||
// Copyright (C) 2001 MandrakeSoft S.A.
|
// Copyright (C) 2001 MandrakeSoft S.A.
|
||||||
@ -995,6 +995,7 @@ bx_bool BX_CPU_C::dbg_is_end_instr_bpoint(Bit16u cs, bx_address eip, bx_address
|
|||||||
BX_CPU_THIS_PTR guard_found.eip = eip;
|
BX_CPU_THIS_PTR guard_found.eip = eip;
|
||||||
BX_CPU_THIS_PTR guard_found.laddr = laddr;
|
BX_CPU_THIS_PTR guard_found.laddr = laddr;
|
||||||
BX_CPU_THIS_PTR guard_found.is_32bit_code = is_32;
|
BX_CPU_THIS_PTR guard_found.is_32bit_code = is_32;
|
||||||
|
BX_CPU_THIS_PTR guard_found.is_64bit_code = is_64;
|
||||||
BX_CPU_THIS_PTR guard_found.guard_found = BX_DBG_GUARD_ICOUNT;
|
BX_CPU_THIS_PTR guard_found.guard_found = BX_DBG_GUARD_ICOUNT;
|
||||||
return(1);
|
return(1);
|
||||||
}
|
}
|
||||||
|
@ -49,27 +49,17 @@ static const unsigned char instruction_has_modrm[512] = {
|
|||||||
/* 0 1 2 3 4 5 6 7 8 9 a b c d e f */
|
/* 0 1 2 3 4 5 6 7 8 9 a b c d e f */
|
||||||
};
|
};
|
||||||
|
|
||||||
unsigned disassembler::disasm16(bx_address base, bx_address ip, Bit8u *instr, char *disbuf)
|
unsigned disassembler::disasm(bx_bool is_32, bx_bool is_64, bx_address base, bx_address ip, const Bit8u *instr, char *disbuf)
|
||||||
{
|
{
|
||||||
return disasm(0, 0, base, ip, instr, disbuf);
|
x86_insn insn = decode(is_32, is_64, base, ip, instr, disbuf);
|
||||||
|
return insn.ilen;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned disassembler::disasm32(bx_address base, bx_address ip, Bit8u *instr, char *disbuf)
|
x86_insn disassembler::decode(bx_bool is_32, bx_bool is_64, bx_address base, bx_address ip, const Bit8u *instr, char *disbuf)
|
||||||
{
|
|
||||||
return disasm(1, 0, base, ip, instr, disbuf);
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned disassembler::disasm64(bx_address base, bx_address ip, Bit8u *instr, char *disbuf)
|
|
||||||
{
|
|
||||||
return disasm(1, 1, base, ip, instr, disbuf);
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned disassembler::disasm(bx_bool is_32, bx_bool is_64, bx_address base, bx_address ip, Bit8u *instr, char *disbuf)
|
|
||||||
{
|
{
|
||||||
x86_insn insn(is_32, is_64);
|
x86_insn insn(is_32, is_64);
|
||||||
Bit8u *instruction_begin = instruction = instr;
|
const Bit8u *instruction_begin = instruction = instr;
|
||||||
resolve_modrm = NULL;
|
resolve_modrm = NULL;
|
||||||
unsigned n_prefixes = 0;
|
|
||||||
|
|
||||||
db_eip = ip;
|
db_eip = ip;
|
||||||
db_base = base; // cs linear base (base for PM & cs<<4 for RM & VM)
|
db_base = base; // cs linear base (base for PM & cs<<4 for RM & VM)
|
||||||
@ -86,7 +76,7 @@ unsigned disassembler::disasm(bx_bool is_32, bx_bool is_64, bx_address base, bx_
|
|||||||
for(;;)
|
for(;;)
|
||||||
{
|
{
|
||||||
insn.b1 = fetch_byte();
|
insn.b1 = fetch_byte();
|
||||||
n_prefixes++;
|
insn.prefixes++;
|
||||||
|
|
||||||
switch(insn.b1) {
|
switch(insn.b1) {
|
||||||
case 0xf3: // rep
|
case 0xf3: // rep
|
||||||
@ -166,7 +156,7 @@ unsigned disassembler::disasm(bx_bool is_32, bx_bool is_64, bx_address base, bx_
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
n_prefixes--;
|
insn.prefixes--;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -208,11 +198,14 @@ unsigned disassembler::disasm(bx_bool is_32, bx_bool is_64, bx_address base, bx_
|
|||||||
|
|
||||||
case _GRPSSE:
|
case _GRPSSE:
|
||||||
{
|
{
|
||||||
if(sse_prefix) n_prefixes--;
|
if(sse_prefix) insn.prefixes--;
|
||||||
/* For SSE opcodes, look into another 4 entries table
|
/* For SSE opcodes, look into another 4 entries table
|
||||||
with the opcode prefixes (NONE, 0x66, 0xF2, 0xF3) */
|
with the opcode prefixes (NONE, 0x66, 0xF2, 0xF3) */
|
||||||
int op = sse_prefix_index[sse_prefix];
|
int op = sse_prefix_index[sse_prefix];
|
||||||
if (op < 0) return 0;
|
if (op < 0) {
|
||||||
|
printf("disassembler panic - too many sse prefixes !\n");
|
||||||
|
return x86_insn(is_32, is_64);
|
||||||
|
}
|
||||||
entry = &(OPCODE_TABLE(entry)[op]);
|
entry = &(OPCODE_TABLE(entry)[op]);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@ -241,7 +234,7 @@ unsigned disassembler::disasm(bx_bool is_32, bx_bool is_64, bx_address base, bx_
|
|||||||
|
|
||||||
default:
|
default:
|
||||||
printf("Internal disassembler error - unknown attribute !\n");
|
printf("Internal disassembler error - unknown attribute !\n");
|
||||||
return 0;
|
return x86_insn(is_32, is_64);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* get additional attributes from group table */
|
/* get additional attributes from group table */
|
||||||
@ -254,7 +247,7 @@ unsigned disassembler::disasm(bx_bool is_32, bx_bool is_64, bx_address base, bx_
|
|||||||
unsigned branch_hint = 0;
|
unsigned branch_hint = 0;
|
||||||
|
|
||||||
// print prefixes
|
// print prefixes
|
||||||
for(unsigned i=0;i<n_prefixes;i++)
|
for(unsigned i=0;i<insn.prefixes;i++)
|
||||||
{
|
{
|
||||||
Bit8u prefix_byte = *(instr+i);
|
Bit8u prefix_byte = *(instr+i);
|
||||||
|
|
||||||
@ -294,7 +287,9 @@ unsigned disassembler::disasm(bx_bool is_32, bx_bool is_64, bx_address base, bx_
|
|||||||
dis_sprintf(", taken");
|
dis_sprintf(", taken");
|
||||||
}
|
}
|
||||||
|
|
||||||
return(instruction - instruction_begin);
|
insn.ilen = (unsigned)(instruction - instruction_begin);
|
||||||
|
|
||||||
|
return insn;
|
||||||
}
|
}
|
||||||
|
|
||||||
void disassembler::dis_sprintf(char *fmt, ...)
|
void disassembler::dis_sprintf(char *fmt, ...)
|
||||||
|
@ -121,7 +121,8 @@ public:
|
|||||||
Bit8u extend8b;
|
Bit8u extend8b;
|
||||||
Bit8u rex_r, rex_x, rex_b;
|
Bit8u rex_r, rex_x, rex_b;
|
||||||
Bit8u seg_override;
|
Bit8u seg_override;
|
||||||
unsigned b1;
|
unsigned b1, prefixes;
|
||||||
|
unsigned ilen;
|
||||||
|
|
||||||
Bit8u modrm, mod, nnn, rm;
|
Bit8u modrm, mod, nnn, rm;
|
||||||
Bit8u sib, scale, index, base;
|
Bit8u sib, scale, index, base;
|
||||||
@ -152,6 +153,8 @@ BX_CPP_INLINE x86_insn::x86_insn(bx_bool is32, bx_bool is64)
|
|||||||
extend8b = 0;
|
extend8b = 0;
|
||||||
rex_r = rex_b = rex_x = 0;
|
rex_r = rex_b = rex_x = 0;
|
||||||
seg_override = NO_SEG_OVERRIDE;
|
seg_override = NO_SEG_OVERRIDE;
|
||||||
|
prefixes = 0;
|
||||||
|
ilen = 0;
|
||||||
b1 = 0;
|
b1 = 0;
|
||||||
|
|
||||||
modrm = mod = nnn = rm = 0;
|
modrm = mod = nnn = rm = 0;
|
||||||
@ -162,11 +165,28 @@ BX_CPP_INLINE x86_insn::x86_insn(bx_bool is32, bx_bool is64)
|
|||||||
class disassembler {
|
class disassembler {
|
||||||
public:
|
public:
|
||||||
disassembler() { set_syntax_intel(); }
|
disassembler() { set_syntax_intel(); }
|
||||||
unsigned disasm16(bx_address base, bx_address ip, Bit8u *instr, char *disbuf);
|
|
||||||
unsigned disasm32(bx_address base, bx_address ip, Bit8u *instr, char *disbuf);
|
|
||||||
unsigned disasm64(bx_address base, bx_address ip, Bit8u *instr, char *disbuf);
|
|
||||||
|
|
||||||
unsigned disasm(bx_bool is_32, bx_bool is_64, bx_address base, bx_address ip, Bit8u *instr, char *disbuf);
|
unsigned disasm(bx_bool is_32, bx_bool is_64, bx_address base, bx_address ip, const Bit8u *instr, char *disbuf);
|
||||||
|
|
||||||
|
unsigned disasm16(bx_address base, bx_address ip, const Bit8u *instr, char *disbuf)
|
||||||
|
{ return disasm(0, 0, base, ip, instr, disbuf); }
|
||||||
|
|
||||||
|
unsigned disasm32(bx_address base, bx_address ip, const Bit8u *instr, char *disbuf)
|
||||||
|
{ return disasm(1, 0, base, ip, instr, disbuf); }
|
||||||
|
|
||||||
|
unsigned disasm64(bx_address base, bx_address ip, const Bit8u *instr, char *disbuf)
|
||||||
|
{ return disasm(1, 1, base, ip, instr, disbuf); }
|
||||||
|
|
||||||
|
x86_insn decode(bx_bool is_32, bx_bool is_64, bx_address base, bx_address ip, const Bit8u *instr, char *disbuf);
|
||||||
|
|
||||||
|
x86_insn decode16(bx_address base, bx_address ip, const Bit8u *instr, char *disbuf)
|
||||||
|
{ return decode(0, 0, base, ip, instr, disbuf); }
|
||||||
|
|
||||||
|
x86_insn decode32(bx_address base, bx_address ip, const Bit8u *instr, char *disbuf)
|
||||||
|
{ return decode(1, 0, base, ip, instr, disbuf); }
|
||||||
|
|
||||||
|
x86_insn decode64(bx_address base, bx_address ip, const Bit8u *instr, char *disbuf)
|
||||||
|
{ return decode(1, 1, base, ip, instr, disbuf); }
|
||||||
|
|
||||||
void set_syntax_intel();
|
void set_syntax_intel();
|
||||||
void set_syntax_att ();
|
void set_syntax_att ();
|
||||||
@ -195,7 +215,7 @@ private:
|
|||||||
|
|
||||||
bx_address db_eip, db_base;
|
bx_address db_eip, db_base;
|
||||||
|
|
||||||
Bit8u *instruction; // for fetching of next byte of instruction
|
const Bit8u *instruction; // for fetching of next byte of instruction
|
||||||
|
|
||||||
char *disbufptr;
|
char *disbufptr;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user