final version of exceptions cleanups/interface changes

This commit is contained in:
Stanislav Shwartsman 2009-01-21 22:09:59 +00:00
parent 0b3c7262d4
commit 29a252b26e
3 changed files with 53 additions and 72 deletions

View File

@ -1,5 +1,5 @@
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// $Id: cpu.h,v 1.559 2009-01-20 21:28:43 sshwarts Exp $ // $Id: cpu.h,v 1.560 2009-01-21 22:09:59 sshwarts Exp $
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// //
// Copyright (C) 2001 MandrakeSoft S.A. // Copyright (C) 2001 MandrakeSoft S.A.
@ -3072,12 +3072,12 @@ public: // for now...
BX_SMF const char *strseg(bx_segment_reg_t *seg); BX_SMF const char *strseg(bx_segment_reg_t *seg);
BX_SMF void interrupt(Bit8u vector, unsigned type, bx_bool push_error, BX_SMF void interrupt(Bit8u vector, unsigned type, bx_bool push_error,
Bit16u error_code); Bit16u error_code);
BX_SMF void real_mode_int(Bit8u vector, unsigned type, bx_bool push_error, BX_SMF void real_mode_int(Bit8u vector, unsigned is_INT, bx_bool push_error,
Bit16u error_code); Bit16u error_code);
BX_SMF void protected_mode_int(Bit8u vector, unsigned type, bx_bool push_error, BX_SMF void protected_mode_int(Bit8u vector, unsigned is_INT, bx_bool push_error,
Bit16u error_code); Bit16u error_code);
#if BX_SUPPORT_X86_64 #if BX_SUPPORT_X86_64
BX_SMF void long_mode_int(Bit8u vector, unsigned type, bx_bool push_error, BX_SMF void long_mode_int(Bit8u vector, unsigned is_INT, bx_bool push_error,
Bit16u error_code); Bit16u error_code);
#endif #endif
BX_SMF void exception(unsigned vector, Bit16u error_code, unsigned unused) BX_SMF void exception(unsigned vector, Bit16u error_code, unsigned unused)

View File

@ -1,5 +1,5 @@
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// $Id: exception.cc,v 1.130 2009-01-20 21:28:43 sshwarts Exp $ // $Id: exception.cc,v 1.131 2009-01-21 22:09:59 sshwarts Exp $
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// //
// Copyright (C) 2001 MandrakeSoft S.A. // Copyright (C) 2001 MandrakeSoft S.A.
@ -59,7 +59,7 @@ static const bx_bool is_exception_OK[3][3] = {
#define BX_EXCEPTION_CLASS_ABORT 2 #define BX_EXCEPTION_CLASS_ABORT 2
#if BX_SUPPORT_X86_64 #if BX_SUPPORT_X86_64
void BX_CPU_C::long_mode_int(Bit8u vector, unsigned type, bx_bool is_error_code, Bit16u error_code) void BX_CPU_C::long_mode_int(Bit8u vector, unsigned is_INT, bx_bool push_error, Bit16u error_code)
{ {
// long mode interrupt // long mode interrupt
Bit64u desctmp1, desctmp2; Bit64u desctmp1, desctmp2;
@ -67,22 +67,6 @@ void BX_CPU_C::long_mode_int(Bit8u vector, unsigned type, bx_bool is_error_code,
bx_descriptor_t gate_descriptor, cs_descriptor; bx_descriptor_t gate_descriptor, cs_descriptor;
bx_selector_t cs_selector; bx_selector_t cs_selector;
bx_bool is_INT = 0;
switch(type) {
case BX_SOFTWARE_INTERRUPT:
case BX_PRIVILEGED_SOFTWARE_INTERRUPT:
case BX_SOFTWARE_EXCEPTION:
is_INT = 1;
break;
case BX_EXTERNAL_INTERRUPT:
case BX_NMI:
case BX_HARDWARE_EXCEPTION:
break;
default:
BX_PANIC(("long_mode_int(): unknown exception type %d", type));
}
// interrupt vector must be within IDT table limits, // interrupt vector must be within IDT table limits,
// else #GP(vector number*16 + 2 + EXT) // else #GP(vector number*16 + 2 + EXT)
if ((vector*16 + 15) > BX_CPU_THIS_PTR idtr.limit) { if ((vector*16 + 15) > BX_CPU_THIS_PTR idtr.limit) {
@ -217,7 +201,7 @@ void BX_CPU_C::long_mode_int(Bit8u vector, unsigned type, bx_bool is_error_code,
write_new_stack_qword_64(RSP_for_cpl_x - 40, cs_descriptor.dpl, old_RIP); write_new_stack_qword_64(RSP_for_cpl_x - 40, cs_descriptor.dpl, old_RIP);
RSP_for_cpl_x -= 40; RSP_for_cpl_x -= 40;
if (is_error_code) { if (push_error) {
RSP_for_cpl_x -= 8; RSP_for_cpl_x -= 8;
write_new_stack_qword_64(RSP_for_cpl_x, cs_descriptor.dpl, error_code); write_new_stack_qword_64(RSP_for_cpl_x, cs_descriptor.dpl, error_code);
} }
@ -275,7 +259,7 @@ void BX_CPU_C::long_mode_int(Bit8u vector, unsigned type, bx_bool is_error_code,
write_new_stack_qword_64(RSP - 40, cs_descriptor.dpl, RIP); write_new_stack_qword_64(RSP - 40, cs_descriptor.dpl, RIP);
RSP -= 40; RSP -= 40;
if (is_error_code) { if (push_error) {
RSP -= 8; RSP -= 8;
write_new_stack_qword_64(RSP, cs_descriptor.dpl, error_code); write_new_stack_qword_64(RSP, cs_descriptor.dpl, error_code);
} }
@ -301,7 +285,7 @@ void BX_CPU_C::long_mode_int(Bit8u vector, unsigned type, bx_bool is_error_code,
} }
#endif #endif
void BX_CPU_C::protected_mode_int(Bit8u vector, unsigned type, bx_bool is_error_code, Bit16u error_code) void BX_CPU_C::protected_mode_int(Bit8u vector, unsigned is_INT, bx_bool push_error, Bit16u error_code)
{ {
// protected mode interrupt // protected mode interrupt
Bit32u dword1, dword2; Bit32u dword1, dword2;
@ -315,22 +299,6 @@ void BX_CPU_C::protected_mode_int(Bit8u vector, unsigned type, bx_bool is_error_
Bit16u gate_dest_selector; Bit16u gate_dest_selector;
Bit32u gate_dest_offset; Bit32u gate_dest_offset;
bx_bool is_INT = 0;
switch(type) {
case BX_SOFTWARE_INTERRUPT:
case BX_PRIVILEGED_SOFTWARE_INTERRUPT:
case BX_SOFTWARE_EXCEPTION:
is_INT = 1;
break;
case BX_EXTERNAL_INTERRUPT:
case BX_NMI:
case BX_HARDWARE_EXCEPTION:
break;
default:
BX_PANIC(("interrupt(): unknown exception type %d", type));
}
// interrupt vector must be within IDT table limits, // interrupt vector must be within IDT table limits,
// else #GP(vector number*8 + 2 + EXT) // else #GP(vector number*8 + 2 + EXT)
if ((vector*8 + 7) > BX_CPU_THIS_PTR idtr.limit) { if ((vector*8 + 7) > BX_CPU_THIS_PTR idtr.limit) {
@ -424,7 +392,7 @@ void BX_CPU_C::protected_mode_int(Bit8u vector, unsigned type, bx_bool is_error_
// stack limits must allow push of 2 more bytes, else #SS(0) // stack limits must allow push of 2 more bytes, else #SS(0)
// push error code onto stack // push error code onto stack
if (is_error_code) { if (push_error) {
if (tss_descriptor.type >= 9) // TSS386 if (tss_descriptor.type >= 9) // TSS386
push_32(error_code); push_32(error_code);
else else
@ -604,7 +572,7 @@ void BX_CPU_C::protected_mode_int(Bit8u vector, unsigned type, bx_bool is_error_
write_new_stack_dword_32(&new_stack, temp_ESP-20, cs_descriptor.dpl, old_EIP); write_new_stack_dword_32(&new_stack, temp_ESP-20, cs_descriptor.dpl, old_EIP);
temp_ESP -= 20; temp_ESP -= 20;
if (is_error_code) { if (push_error) {
temp_ESP -= 4; temp_ESP -= 4;
write_new_stack_dword_32(&new_stack, temp_ESP, cs_descriptor.dpl, error_code); write_new_stack_dword_32(&new_stack, temp_ESP, cs_descriptor.dpl, error_code);
} }
@ -618,7 +586,7 @@ void BX_CPU_C::protected_mode_int(Bit8u vector, unsigned type, bx_bool is_error_
write_new_stack_word_32(&new_stack, temp_ESP-10, cs_descriptor.dpl, (Bit16u) old_EIP); write_new_stack_word_32(&new_stack, temp_ESP-10, cs_descriptor.dpl, (Bit16u) old_EIP);
temp_ESP -= 10; temp_ESP -= 10;
if (is_error_code) { if (push_error) {
temp_ESP -= 2; temp_ESP -= 2;
write_new_stack_word_32(&new_stack, temp_ESP, cs_descriptor.dpl, error_code); write_new_stack_word_32(&new_stack, temp_ESP, cs_descriptor.dpl, error_code);
} }
@ -664,7 +632,7 @@ void BX_CPU_C::protected_mode_int(Bit8u vector, unsigned type, bx_bool is_error_
write_new_stack_dword_32(&new_stack, (Bit16u)(temp_SP-20), cs_descriptor.dpl, old_EIP); write_new_stack_dword_32(&new_stack, (Bit16u)(temp_SP-20), cs_descriptor.dpl, old_EIP);
temp_SP -= 20; temp_SP -= 20;
if (is_error_code) { if (push_error) {
temp_SP -= 4; temp_SP -= 4;
write_new_stack_dword_32(&new_stack, temp_SP, cs_descriptor.dpl, error_code); write_new_stack_dword_32(&new_stack, temp_SP, cs_descriptor.dpl, error_code);
} }
@ -678,7 +646,7 @@ void BX_CPU_C::protected_mode_int(Bit8u vector, unsigned type, bx_bool is_error_
write_new_stack_word_32(&new_stack, (Bit16u)(temp_SP-10), cs_descriptor.dpl, (Bit16u) old_EIP); write_new_stack_word_32(&new_stack, (Bit16u)(temp_SP-10), cs_descriptor.dpl, (Bit16u) old_EIP);
temp_SP -= 10; temp_SP -= 10;
if (is_error_code) { if (push_error) {
temp_SP -= 2; temp_SP -= 2;
write_new_stack_word_32(&new_stack, temp_SP, cs_descriptor.dpl, error_code); write_new_stack_word_32(&new_stack, temp_SP, cs_descriptor.dpl, error_code);
} }
@ -741,14 +709,14 @@ void BX_CPU_C::protected_mode_int(Bit8u vector, unsigned type, bx_bool is_error_
push_32(read_eflags()); push_32(read_eflags());
push_32(BX_CPU_THIS_PTR sregs[BX_SEG_REG_CS].selector.value); push_32(BX_CPU_THIS_PTR sregs[BX_SEG_REG_CS].selector.value);
push_32(EIP); push_32(EIP);
if (is_error_code) if (push_error)
push_32(error_code); push_32(error_code);
} }
else { // 286 gate else { // 286 gate
push_16((Bit16u) read_eflags()); push_16((Bit16u) read_eflags());
push_16(BX_CPU_THIS_PTR sregs[BX_SEG_REG_CS].selector.value); push_16(BX_CPU_THIS_PTR sregs[BX_SEG_REG_CS].selector.value);
push_16(IP); push_16(IP);
if (is_error_code) if (push_error)
push_16(error_code); push_16(error_code);
} }
@ -774,7 +742,7 @@ void BX_CPU_C::protected_mode_int(Bit8u vector, unsigned type, bx_bool is_error_
} }
} }
void BX_CPU_C::real_mode_int(Bit8u vector, unsigned type, bx_bool is_error_code, Bit16u error_code) void BX_CPU_C::real_mode_int(Bit8u vector, unsigned is_INT, bx_bool push_error, Bit16u error_code)
{ {
// real mode interrupt // real mode interrupt
Bit16u cs_selector; Bit16u cs_selector;
@ -801,7 +769,7 @@ void BX_CPU_C::real_mode_int(Bit8u vector, unsigned type, bx_bool is_error_code,
BX_CPU_THIS_PTR clear_RF(); BX_CPU_THIS_PTR clear_RF();
} }
void BX_CPU_C::interrupt(Bit8u vector, unsigned type, bx_bool is_error_code, Bit16u error_code) void BX_CPU_C::interrupt(Bit8u vector, unsigned type, bx_bool push_error, Bit16u error_code)
{ {
#if BX_DEBUGGER #if BX_DEBUGGER
BX_CPU_THIS_PTR show_flag |= Flag_intsig; BX_CPU_THIS_PTR show_flag |= Flag_intsig;
@ -813,12 +781,29 @@ void BX_CPU_C::interrupt(Bit8u vector, unsigned type, bx_bool is_error_code, Bit
bx_dbg_interrupt(BX_CPU_ID, vector, error_code); bx_dbg_interrupt(BX_CPU_ID, vector, error_code);
#endif #endif
BX_INSTR_INTERRUPT(BX_CPU_ID, vector);
invalidate_prefetch_q();
bx_bool is_INT = 0;
switch(type) {
case BX_SOFTWARE_INTERRUPT:
case BX_PRIVILEGED_SOFTWARE_INTERRUPT:
case BX_SOFTWARE_EXCEPTION:
is_INT = 1;
break;
case BX_EXTERNAL_INTERRUPT:
case BX_NMI:
case BX_HARDWARE_EXCEPTION:
break;
default:
BX_PANIC(("interrupt(): unknown exception type %d", type));
}
BX_DEBUG(("interrupt(): vector = %u, TYPE = %u, EXT = %u", BX_DEBUG(("interrupt(): vector = %u, TYPE = %u, EXT = %u",
vector, type, (unsigned) BX_CPU_THIS_PTR EXT)); vector, type, (unsigned) BX_CPU_THIS_PTR EXT));
BX_INSTR_INTERRUPT(BX_CPU_ID, vector);
invalidate_prefetch_q();
// Discard any traps and inhibits for new context; traps will // Discard any traps and inhibits for new context; traps will
// resume upon return. // resume upon return.
BX_CPU_THIS_PTR debug_trap = 0; BX_CPU_THIS_PTR debug_trap = 0;
@ -831,16 +816,17 @@ void BX_CPU_C::interrupt(Bit8u vector, unsigned type, bx_bool is_error_code, Bit
#if BX_SUPPORT_X86_64 #if BX_SUPPORT_X86_64
if (long_mode()) { if (long_mode()) {
long_mode_int(vector, type, is_error_code, error_code); long_mode_int(vector, is_INT, push_error, error_code);
return;
} }
else
#endif #endif
{
if(real_mode()) { if(real_mode()) {
real_mode_int(vector, type, is_error_code, error_code); real_mode_int(vector, is_INT, push_error, error_code);
} }
else { else {
protected_mode_int(vector, type, is_error_code, error_code); protected_mode_int(vector, is_INT, push_error, error_code);
}
} }
} }

View File

@ -1,5 +1,5 @@
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// $Id: proc_ctrl.cc,v 1.274 2009-01-19 17:43:54 sshwarts Exp $ // $Id: proc_ctrl.cc,v 1.275 2009-01-21 22:09:59 sshwarts Exp $
///////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////
// //
// Copyright (C) 2001 MandrakeSoft S.A. // Copyright (C) 2001 MandrakeSoft S.A.
@ -217,7 +217,6 @@ void BX_CPP_AttrRegparmN(1) BX_CPU_C::CLFLUSH(bxInstruction_c *i)
#endif #endif
} }
#if BX_CPU_LEVEL >= 3
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOV_DdRd(bxInstruction_c *i) void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOV_DdRd(bxInstruction_c *i)
{ {
#if BX_CPU_LEVEL >= 4 #if BX_CPU_LEVEL >= 4
@ -756,9 +755,6 @@ void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOV_RqCq(bxInstruction_c *i)
} }
#endif // #if BX_SUPPORT_X86_64 #endif // #if BX_SUPPORT_X86_64
#endif // #if BX_CPU_LEVEL >= 3
#if BX_CPU_LEVEL >= 2
void BX_CPP_AttrRegparmN(1) BX_CPU_C::LMSW_Ew(bxInstruction_c *i) void BX_CPP_AttrRegparmN(1) BX_CPU_C::LMSW_Ew(bxInstruction_c *i)
{ {
Bit16u msw; Bit16u msw;
@ -781,7 +777,7 @@ void BX_CPP_AttrRegparmN(1) BX_CPU_C::LMSW_Ew(bxInstruction_c *i)
// LMSW cannot clear PE // LMSW cannot clear PE
if (BX_CPU_THIS_PTR cr0.get_PE()) if (BX_CPU_THIS_PTR cr0.get_PE())
msw |= 0x0001; // adjust PE bit to current value of 1 msw |= 0x1; // adjust PE bit to current value of 1
msw &= 0xf; // LMSW only affects last 4 flags msw &= 0xf; // LMSW only affects last 4 flags
Bit32u cr0 = (BX_CPU_THIS_PTR cr0.get32() & 0xfffffff0) | msw; Bit32u cr0 = (BX_CPU_THIS_PTR cr0.get32() & 0xfffffff0) | msw;
@ -806,7 +802,6 @@ void BX_CPP_AttrRegparmN(1) BX_CPU_C::SMSW_EwM(bxInstruction_c *i)
/* pointer, segment address pair */ /* pointer, segment address pair */
write_virtual_word(i->seg(), eaddr, msw); write_virtual_word(i->seg(), eaddr, msw);
} }
#endif
void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOV_TdRd(bxInstruction_c *i) void BX_CPP_AttrRegparmN(1) BX_CPU_C::MOV_TdRd(bxInstruction_c *i)
{ {
@ -1164,10 +1159,10 @@ void BX_CPU_C::handleAlignmentCheck(void)
bx_bool BX_CPP_AttrRegparmN(1) BX_CPU_C::SetCR0(Bit32u val_32) bx_bool BX_CPP_AttrRegparmN(1) BX_CPU_C::SetCR0(Bit32u val_32)
{ {
bx_bool pe = val_32 & 0x01; bx_bool pe = val_32 & 0x1;
bx_bool nw = (val_32 >> 29) & 0x01; bx_bool nw = (val_32 >> 29) & 0x1;
bx_bool cd = (val_32 >> 30) & 0x01; bx_bool cd = (val_32 >> 30) & 0x1;
bx_bool pg = (val_32 >> 31) & 0x01; bx_bool pg = (val_32 >> 31) & 0x1;
if (pg && !pe) { if (pg && !pe) {
BX_ERROR(("SetCR0: GP(0) when attempt to set CR0.PG with CR0.PE cleared !")); BX_ERROR(("SetCR0: GP(0) when attempt to set CR0.PG with CR0.PE cleared !"));