2001-10-03 17:10:38 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
2005-10-16 01:01:36 +04:00
|
|
|
// $Id: protect_ctrl.cc,v 1.44 2005-10-15 21:01:36 sshwarts Exp $
|
2001-10-03 17:10:38 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2001-04-10 06:20:02 +04:00
|
|
|
// Copyright (C) 2001 MandrakeSoft S.A.
|
2001-04-10 05:04:59 +04:00
|
|
|
//
|
|
|
|
// MandrakeSoft S.A.
|
|
|
|
// 43, rue d'Aboukir
|
|
|
|
// 75002 Paris - France
|
|
|
|
// http://www.linux-mandrake.com/
|
|
|
|
// http://www.mandrakesoft.com/
|
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2 of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This library 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
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
|
|
|
|
|
|
|
|
|
2001-05-24 22:46:34 +04:00
|
|
|
#define NEED_CPU_REG_SHORTCUTS 1
|
2001-04-10 05:04:59 +04:00
|
|
|
#include "bochs.h"
|
merge in BRANCH-io-cleanup.
To see the commit logs for this use either cvsweb or
cvs update -r BRANCH-io-cleanup and then 'cvs log' the various files.
In general this provides a generic interface for logging.
logfunctions:: is a class that is inherited by some classes, and also
. allocated as a standalone global called 'genlog'. All logging uses
. one of the ::info(), ::error(), ::ldebug(), ::panic() methods of this
. class through 'BX_INFO(), BX_ERROR(), BX_DEBUG(), BX_PANIC()' macros
. respectively.
.
. An example usage:
. BX_INFO(("Hello, World!\n"));
iofunctions:: is a class that is allocated once by default, and assigned
as the iofunction of each logfunctions instance. It is this class that
maintains the file descriptor and other output related code, at this
point using vfprintf(). At some future point, someone may choose to
write a gui 'console' for bochs to which messages would be redirected
simply by assigning a different iofunction class to the various logfunctions
objects.
More cleanup is coming, but this works for now. If you want to see alot
of debugging output, in main.cc, change onoff[LOGLEV_DEBUG]=0 to =1.
Comments, bugs, flames, to me: todd@fries.net
2001-05-15 18:49:57 +04:00
|
|
|
#define LOG_THIS BX_CPU_THIS_PTR
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2005-08-21 22:23:36 +04:00
|
|
|
#if BX_CPU_LEVEL >= 2
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2005-08-21 22:23:36 +04:00
|
|
|
void BX_CPU_C::ARPL_EwGw(bxInstruction_c *i)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
Bit16u op2_16, op1_16;
|
|
|
|
|
|
|
|
if (protected_mode()) {
|
|
|
|
/* op1_16 is a register or memory reference */
|
2002-09-20 07:52:59 +04:00
|
|
|
if (i->modC0()) {
|
2002-09-18 02:50:53 +04:00
|
|
|
op1_16 = BX_READ_16BIT_REG(i->rm());
|
2005-01-28 23:50:48 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
|
|
|
/* pointer, segment address pair */
|
2002-09-18 09:36:48 +04:00
|
|
|
read_RMW_virtual_word(i->seg(), RMAddr(i), &op1_16);
|
2005-01-28 23:50:48 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-09-18 02:50:53 +04:00
|
|
|
op2_16 = BX_READ_16BIT_REG(i->nnn());
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((op1_16 & 0x03) < (op2_16 & 0x03)) {
|
2001-04-10 05:04:59 +04:00
|
|
|
op1_16 = (op1_16 & 0xfffc) | (op2_16 & 0x03);
|
|
|
|
/* now write back to destination */
|
2002-09-20 07:52:59 +04:00
|
|
|
if (i->modC0()) {
|
2002-09-18 09:36:48 +04:00
|
|
|
if (i->os32L()) {
|
2001-04-10 05:04:59 +04:00
|
|
|
// if 32bit opsize, then 0xff3f is or'd into
|
|
|
|
// upper 16bits of register
|
2005-01-28 23:50:48 +03:00
|
|
|
Bit32u op1_32 = BX_READ_32BIT_REG(i->rm());
|
2001-04-10 05:04:59 +04:00
|
|
|
op1_32 = (op1_32 & 0xffff0000) | op1_16;
|
|
|
|
op1_32 |= 0xff3f0000;
|
2002-09-18 02:50:53 +04:00
|
|
|
BX_WRITE_32BIT_REGZ(i->rm(), op1_32);
|
2005-01-28 23:50:48 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
2002-09-18 02:50:53 +04:00
|
|
|
BX_WRITE_16BIT_REG(i->rm(), op1_16);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2005-01-28 23:50:48 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
2002-10-25 22:26:29 +04:00
|
|
|
Write_RMW_virtual_word(op1_16);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2005-01-28 23:50:48 +03:00
|
|
|
set_ZF(1);
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
|
|
|
set_ZF(0);
|
|
|
|
}
|
2005-01-28 23:50:48 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
2003-10-24 22:34:16 +04:00
|
|
|
BX_DEBUG(("ARPL: not recognized in real or virtual-8086 mode"));
|
2001-04-10 05:04:59 +04:00
|
|
|
UndefinedOpcode(i);
|
2005-01-28 23:50:48 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2005-08-21 22:23:36 +04:00
|
|
|
#endif
|
|
|
|
|
|
|
|
void BX_CPU_C::LAR_GvEw(bxInstruction_c *i)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
/* for 16 bit operand size mode */
|
|
|
|
Bit16u raw_selector;
|
|
|
|
bx_descriptor_t descriptor;
|
2003-10-05 00:22:24 +04:00
|
|
|
bx_selector_t selector;
|
2001-04-10 05:04:59 +04:00
|
|
|
Bit32u dword1, dword2;
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (real_mode() || v8086_mode()) {
|
|
|
|
BX_INFO(("LAR: not recognized in real or virtual-8086 mode"));
|
2001-04-10 05:04:59 +04:00
|
|
|
UndefinedOpcode(i);
|
|
|
|
return;
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-09-20 07:52:59 +04:00
|
|
|
if (i->modC0()) {
|
2002-09-18 02:50:53 +04:00
|
|
|
raw_selector = BX_READ_16BIT_REG(i->rm());
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
|
|
|
/* pointer, segment address pair */
|
2002-09-18 09:36:48 +04:00
|
|
|
read_virtual_word(i->seg(), RMAddr(i), &raw_selector);
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
/* if selector null, clear ZF and done */
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((raw_selector & 0xfffc) == 0) {
|
2005-10-16 01:01:36 +04:00
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
parse_selector(raw_selector, &selector);
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (!fetch_raw_descriptor2(&selector, &dword1, &dword2)) {
|
2001-04-10 05:04:59 +04:00
|
|
|
/* not within descriptor table */
|
2005-10-16 01:01:36 +04:00
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
parse_descriptor(dword1, dword2, &descriptor);
|
|
|
|
|
|
|
|
if (descriptor.valid==0) {
|
2005-10-16 01:01:36 +04:00
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
/* if source selector is visible at CPL & RPL,
|
|
|
|
* within the descriptor table, and of type accepted by LAR instruction,
|
|
|
|
* then load register with segment limit and set ZF
|
|
|
|
*/
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (descriptor.segment) { /* normal segment */
|
|
|
|
if (descriptor.u.segment.executable && descriptor.u.segment.c_ed) {
|
2001-04-10 05:04:59 +04:00
|
|
|
/* ignore DPL for conforming segments */
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((descriptor.dpl<CPL) || (descriptor.dpl<selector.rpl)) {
|
2005-10-16 01:01:36 +04:00
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
|
|
|
}
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2005-10-16 01:01:36 +04:00
|
|
|
assert_ZF();
|
2002-09-18 09:36:48 +04:00
|
|
|
if (i->os32L()) {
|
2001-04-10 05:04:59 +04:00
|
|
|
/* masked by 00FxFF00, where x is undefined */
|
2002-09-18 02:50:53 +04:00
|
|
|
BX_WRITE_32BIT_REGZ(i->nnn(), dword2 & 0x00ffff00);
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
2002-09-18 02:50:53 +04:00
|
|
|
BX_WRITE_16BIT_REG(i->nnn(), dword2 & 0xff00);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2005-03-13 21:20:26 +03:00
|
|
|
return;
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else { /* system or gate segment */
|
2003-10-05 00:22:24 +04:00
|
|
|
switch (descriptor.type) {
|
2005-02-27 20:41:45 +03:00
|
|
|
case BX_SYS_SEGMENT_AVAIL_286_TSS:
|
|
|
|
case BX_SYS_SEGMENT_LDT:
|
|
|
|
case BX_SYS_SEGMENT_BUSY_286_TSS:
|
|
|
|
case BX_286_CALL_GATE:
|
|
|
|
case BX_TASK_GATE:
|
2001-04-10 05:04:59 +04:00
|
|
|
#if BX_CPU_LEVEL >= 3
|
2005-02-27 20:41:45 +03:00
|
|
|
case BX_SYS_SEGMENT_AVAIL_386_TSS:
|
|
|
|
case BX_SYS_SEGMENT_BUSY_386_TSS:
|
|
|
|
case BX_386_CALL_GATE:
|
2001-04-10 05:04:59 +04:00
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
default: /* rest not accepted types to LAR */
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_DEBUG(("lar(): not accepted type"));
|
2005-10-16 01:01:36 +04:00
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-02-27 20:41:45 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((descriptor.dpl<CPL) || (descriptor.dpl<selector.rpl)) {
|
2005-10-16 01:01:36 +04:00
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-02-27 20:41:45 +03:00
|
|
|
}
|
2005-10-16 01:01:36 +04:00
|
|
|
assert_ZF();
|
2002-09-18 09:36:48 +04:00
|
|
|
if (i->os32L()) {
|
2003-10-05 00:22:24 +04:00
|
|
|
/* masked by 00FxFF00, where x is undefined ? */
|
2002-09-18 02:50:53 +04:00
|
|
|
BX_WRITE_32BIT_REGZ(i->nnn(), dword2 & 0x00ffff00);
|
2005-02-27 20:41:45 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
2002-09-18 02:50:53 +04:00
|
|
|
BX_WRITE_16BIT_REG(i->nnn(), dword2 & 0xff00);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2005-02-27 20:41:45 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2005-08-21 22:23:36 +04:00
|
|
|
void BX_CPU_C::LSL_GvEw(bxInstruction_c *i)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
/* for 16 bit operand size mode */
|
|
|
|
Bit16u raw_selector;
|
|
|
|
Bit32u limit32;
|
2003-10-05 00:22:24 +04:00
|
|
|
bx_selector_t selector;
|
2001-04-10 05:04:59 +04:00
|
|
|
Bit32u dword1, dword2;
|
|
|
|
Bit32u descriptor_dpl;
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (real_mode() || v8086_mode()) {
|
|
|
|
BX_INFO(("LSL: not recognized in real or virtual-8086 mode"));
|
2001-04-10 05:04:59 +04:00
|
|
|
UndefinedOpcode(i);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-09-20 07:52:59 +04:00
|
|
|
if (i->modC0()) {
|
2002-09-18 02:50:53 +04:00
|
|
|
raw_selector = BX_READ_16BIT_REG(i->rm());
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
|
|
|
/* pointer, segment address pair */
|
2002-09-18 09:36:48 +04:00
|
|
|
read_virtual_word(i->seg(), RMAddr(i), &raw_selector);
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
/* if selector null, clear ZF and done */
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((raw_selector & 0xfffc) == 0) {
|
2005-10-16 01:01:36 +04:00
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
parse_selector(raw_selector, &selector);
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (!fetch_raw_descriptor2(&selector, &dword1, &dword2)) {
|
2001-04-10 05:04:59 +04:00
|
|
|
/* not within descriptor table */
|
2005-10-16 01:01:36 +04:00
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
descriptor_dpl = (dword2 >> 13) & 0x03;
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((dword2 & 0x00001000) == 0) { // system segment
|
|
|
|
Bit32u type = (dword2 >> 8) & 0x0000000f;
|
2001-04-10 05:04:59 +04:00
|
|
|
switch (type) {
|
2005-02-27 20:41:45 +03:00
|
|
|
case BX_SYS_SEGMENT_AVAIL_286_TSS:
|
|
|
|
case BX_SYS_SEGMENT_BUSY_286_TSS:
|
|
|
|
case BX_SYS_SEGMENT_LDT:
|
|
|
|
case BX_SYS_SEGMENT_AVAIL_386_TSS:
|
|
|
|
case BX_SYS_SEGMENT_BUSY_386_TSS:
|
2001-04-10 05:04:59 +04:00
|
|
|
limit32 = (dword1 & 0x0000ffff) | (dword2 & 0x000f0000);
|
2003-10-05 00:22:24 +04:00
|
|
|
if (dword2 & 0x00800000)
|
2001-04-10 05:04:59 +04:00
|
|
|
limit32 = (limit32 << 12) | 0x00000fff;
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((descriptor_dpl<CPL) || (descriptor_dpl<selector.rpl)) {
|
2005-10-16 01:01:36 +04:00
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-02-27 20:41:45 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
goto lsl_ok;
|
|
|
|
break;
|
|
|
|
default:
|
2005-10-16 01:01:36 +04:00
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
|
|
|
}
|
2005-02-27 20:41:45 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else { // data & code segment
|
|
|
|
limit32 = (dword1 & 0x0000ffff) | (dword2 & 0x000f0000);
|
2003-10-05 00:22:24 +04:00
|
|
|
if (dword2 & 0x00800000)
|
2001-04-10 05:04:59 +04:00
|
|
|
limit32 = (limit32 << 12) | 0x00000fff;
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((dword2 & 0x00000c00) == 0x00000c00) {
|
2001-04-10 05:04:59 +04:00
|
|
|
// conforming code segment, no check done
|
|
|
|
goto lsl_ok;
|
2005-02-27 20:41:45 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((descriptor_dpl<CPL) || (descriptor_dpl<selector.rpl)) {
|
2005-10-16 01:01:36 +04:00
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
|
|
|
}
|
2005-02-27 20:41:45 +03:00
|
|
|
goto lsl_ok;
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
lsl_ok:
|
|
|
|
/* all checks pass, limit32 is now byte granular, write to op1 */
|
2005-10-16 01:01:36 +04:00
|
|
|
assert_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-09-18 09:36:48 +04:00
|
|
|
if (i->os32L())
|
2002-09-18 02:50:53 +04:00
|
|
|
BX_WRITE_32BIT_REGZ(i->nnn(), limit32)
|
2001-04-10 05:04:59 +04:00
|
|
|
else
|
|
|
|
// chop off upper 16 bits
|
2002-09-18 02:50:53 +04:00
|
|
|
BX_WRITE_16BIT_REG(i->nnn(), (Bit16u) limit32)
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2005-08-21 22:23:36 +04:00
|
|
|
#if BX_CPU_LEVEL >= 2
|
|
|
|
|
|
|
|
void BX_CPU_C::SLDT_Ew(bxInstruction_c *i)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2003-10-05 00:22:24 +04:00
|
|
|
if (real_mode() || v8086_mode()) {
|
|
|
|
BX_INFO(("SLDT: not recognized in real or virtual-8086 mode"));
|
2001-04-10 05:04:59 +04:00
|
|
|
UndefinedOpcode(i);
|
2005-01-28 23:50:48 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
2003-10-05 00:22:24 +04:00
|
|
|
Bit16u val16 = BX_CPU_THIS_PTR ldtr.selector.value;
|
2002-09-20 07:52:59 +04:00
|
|
|
if (i->modC0()) {
|
2002-09-18 02:50:53 +04:00
|
|
|
BX_WRITE_16BIT_REG(i->rm(), val16);
|
2005-01-28 23:50:48 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
2002-09-18 09:36:48 +04:00
|
|
|
write_virtual_word(i->seg(), RMAddr(i), &val16);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2005-01-28 23:50:48 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2005-08-21 22:23:36 +04:00
|
|
|
#endif
|
|
|
|
|
|
|
|
void BX_CPU_C::STR_Ew(bxInstruction_c *i)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2003-10-05 00:22:24 +04:00
|
|
|
if (real_mode() || v8086_mode()) {
|
|
|
|
BX_INFO(("STR: not recognized in real or virtual-8086 mode"));
|
2001-04-10 05:04:59 +04:00
|
|
|
UndefinedOpcode(i);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
2003-10-05 00:22:24 +04:00
|
|
|
Bit16u val16 = BX_CPU_THIS_PTR tr.selector.value;
|
2002-09-20 07:52:59 +04:00
|
|
|
if (i->modC0()) {
|
2002-09-18 02:50:53 +04:00
|
|
|
BX_WRITE_16BIT_REG(i->rm(), val16);
|
2005-01-28 23:50:48 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
2002-09-18 09:36:48 +04:00
|
|
|
write_virtual_word(i->seg(), RMAddr(i), &val16);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2005-01-28 23:50:48 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2005-08-21 22:23:36 +04:00
|
|
|
#if BX_CPU_LEVEL >= 2
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2005-08-21 22:23:36 +04:00
|
|
|
void BX_CPU_C::LLDT_Ew(bxInstruction_c *i)
|
|
|
|
{
|
2003-10-05 00:22:24 +04:00
|
|
|
if (real_mode() || v8086_mode()) {
|
|
|
|
BX_INFO(("LLDT: not recognized in real or virtual-8086 mode"));
|
2001-04-10 05:04:59 +04:00
|
|
|
UndefinedOpcode(i);
|
2004-10-16 23:34:17 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
invalidate_prefetch_q();
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
/* protected mode */
|
|
|
|
bx_descriptor_t descriptor;
|
|
|
|
bx_selector_t selector;
|
|
|
|
Bit16u raw_selector;
|
|
|
|
Bit32u dword1, dword2;
|
|
|
|
|
|
|
|
/* #GP(0) if the current privilege level is not 0 */
|
|
|
|
if (CPL != 0) {
|
2005-03-13 21:20:26 +03:00
|
|
|
BX_INFO(("LLDT: The current priveledge level is not 0"));
|
|
|
|
exception(BX_GP_EXCEPTION, 0, 0);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (i->modC0()) {
|
2005-03-13 21:20:26 +03:00
|
|
|
raw_selector = BX_READ_16BIT_REG(i->rm());
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
|
|
|
else {
|
2002-09-18 09:36:48 +04:00
|
|
|
read_virtual_word(i->seg(), RMAddr(i), &raw_selector);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
/* if selector is NULL, invalidate and done */
|
|
|
|
if ((raw_selector & 0xfffc) == 0) {
|
2005-03-13 21:20:26 +03:00
|
|
|
BX_CPU_THIS_PTR ldtr.selector.value = raw_selector;
|
|
|
|
BX_CPU_THIS_PTR ldtr.cache.valid = 0;
|
|
|
|
return;
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
/* parse fields in selector */
|
|
|
|
parse_selector(raw_selector, &selector);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
// #GP(selector) if the selector operand does not point into GDT
|
|
|
|
if (selector.ti != 0) {
|
2005-03-13 21:20:26 +03:00
|
|
|
BX_ERROR(("LLDT: selector.ti != 0"));
|
|
|
|
exception(BX_GP_EXCEPTION, raw_selector & 0xfffc, 0);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((selector.index*8 + 7) > BX_CPU_THIS_PTR gdtr.limit) {
|
2005-03-13 21:20:26 +03:00
|
|
|
BX_PANIC(("LLDT: GDT: index > limit"));
|
|
|
|
exception(BX_GP_EXCEPTION, raw_selector & 0xfffc, 0);
|
|
|
|
return;
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
access_linear(BX_CPU_THIS_PTR gdtr.base + selector.index*8, 4, 0,
|
2001-04-10 05:04:59 +04:00
|
|
|
BX_READ, &dword1);
|
2003-10-05 00:22:24 +04:00
|
|
|
access_linear(BX_CPU_THIS_PTR gdtr.base + selector.index*8 + 4, 4, 0,
|
2001-04-10 05:04:59 +04:00
|
|
|
BX_READ, &dword2);
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
parse_descriptor(dword1, dword2, &descriptor);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
/* if selector doesn't point to an LDT descriptor #GP(selector) */
|
|
|
|
if (descriptor.valid == 0 || descriptor.segment ||
|
2005-03-05 00:03:22 +03:00
|
|
|
descriptor.type != BX_SYS_SEGMENT_LDT)
|
2003-10-05 00:22:24 +04:00
|
|
|
{
|
2005-03-13 21:20:26 +03:00
|
|
|
BX_ERROR(("LLDT: doesn't point to an LDT descriptor!"));
|
|
|
|
exception(BX_GP_EXCEPTION, raw_selector & 0xfffc, 0);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
/* #NP(selector) if LDT descriptor is not present */
|
2005-07-11 00:32:32 +04:00
|
|
|
if (! IS_PRESENT(descriptor)) {
|
2005-03-13 21:20:26 +03:00
|
|
|
BX_ERROR(("LLDT: LDT descriptor not present!"));
|
|
|
|
exception(BX_NP_EXCEPTION, raw_selector & 0xfffc, 0);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
BX_CPU_THIS_PTR ldtr.selector = selector;
|
|
|
|
BX_CPU_THIS_PTR ldtr.cache = descriptor;
|
|
|
|
BX_CPU_THIS_PTR ldtr.cache.valid = 1;
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
void BX_CPU_C::LTR_Ew(bxInstruction_c *i)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2003-10-05 00:22:24 +04:00
|
|
|
if (real_mode() || v8086_mode()) {
|
|
|
|
BX_INFO(("LTR: not recognized in real or virtual-8086 mode"));
|
|
|
|
UndefinedOpcode(i);
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
// protected mode
|
2001-04-10 05:04:59 +04:00
|
|
|
invalidate_prefetch_q();
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
bx_descriptor_t descriptor;
|
|
|
|
bx_selector_t selector;
|
|
|
|
Bit16u raw_selector;
|
|
|
|
Bit32u dword1, dword2;
|
2002-10-11 05:11:11 +04:00
|
|
|
#if BX_SUPPORT_X86_64
|
2003-10-05 00:22:24 +04:00
|
|
|
Bit32u dword3;
|
2002-10-11 05:11:11 +04:00
|
|
|
#endif
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
/* #GP(0) if the current privilege level is not 0 */
|
|
|
|
if (CPL != 0) {
|
2005-06-22 22:13:45 +04:00
|
|
|
BX_ERROR(("LTR: The current priveledge level is not 0"));
|
2005-03-13 21:20:26 +03:00
|
|
|
exception(BX_GP_EXCEPTION, 0, 0);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (i->modC0()) {
|
2005-03-13 21:20:26 +03:00
|
|
|
raw_selector = BX_READ_16BIT_REG(i->rm());
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
|
|
|
else {
|
2005-03-13 21:20:26 +03:00
|
|
|
read_virtual_word(i->seg(), RMAddr(i), &raw_selector);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
/* if selector is NULL, invalidate and done */
|
2005-10-02 19:26:51 +04:00
|
|
|
if ((raw_selector & BX_SELECTOR_RPL_MASK) == 0) {
|
2005-10-01 11:47:00 +04:00
|
|
|
BX_ERROR(("LTR: loading with NULL selector!"));
|
|
|
|
exception(BX_GP_EXCEPTION, 0, 0);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
/* parse fields in selector, then check for null selector */
|
|
|
|
parse_selector(raw_selector, &selector);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (selector.ti) {
|
2005-06-22 22:13:45 +04:00
|
|
|
BX_ERROR(("LTR: selector.ti != 0"));
|
|
|
|
exception(BX_GP_EXCEPTION, raw_selector & 0xfffc, 0);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
/* fetch 2 dwords of descriptor; call handles out of limits checks */
|
|
|
|
fetch_raw_descriptor(&selector, &dword1, &dword2, BX_GP_EXCEPTION);
|
|
|
|
parse_descriptor(dword1, dword2, &descriptor);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-10-08 18:43:18 +04:00
|
|
|
#if BX_SUPPORT_X86_64
|
2003-10-05 00:22:24 +04:00
|
|
|
if (BX_CPU_THIS_PTR cpu_mode == BX_MODE_LONG_64) {
|
2005-03-13 21:20:26 +03:00
|
|
|
// set upper 32 bits of tss base
|
2005-06-22 22:13:45 +04:00
|
|
|
access_linear(BX_CPU_THIS_PTR gdtr.base + selector.index*8 + 8, 4, 0, BX_READ, &dword3);
|
2005-03-13 21:20:26 +03:00
|
|
|
descriptor.u.tss386.base |= ((Bit64u)dword3 << 32);
|
2005-06-22 22:13:45 +04:00
|
|
|
BX_INFO(("64 bit tss base = 0x%08x%08x",
|
|
|
|
(Bit32u)(descriptor.u.tss386.base >> 32),
|
|
|
|
(Bit32u) descriptor.u.tss386.base));
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2002-10-08 18:43:18 +04:00
|
|
|
#endif
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
/* #GP(selector) if object is not a TSS or is already busy */
|
|
|
|
if (descriptor.valid==0 || descriptor.segment ||
|
2005-06-22 22:13:45 +04:00
|
|
|
(descriptor.type!=BX_SYS_SEGMENT_AVAIL_286_TSS &&
|
|
|
|
descriptor.type!=BX_SYS_SEGMENT_AVAIL_386_TSS))
|
2003-10-05 00:22:24 +04:00
|
|
|
{
|
2005-06-22 22:13:45 +04:00
|
|
|
BX_ERROR(("LTR: doesn't point to an available TSS descriptor!"));
|
|
|
|
exception(BX_GP_EXCEPTION, raw_selector & 0xfffc, 0);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
/* #NP(selector) if TSS descriptor is not present */
|
2005-07-11 00:32:32 +04:00
|
|
|
if (! IS_PRESENT(descriptor)) {
|
2005-06-22 22:13:45 +04:00
|
|
|
BX_ERROR(("LTR: LDT descriptor not present!"));
|
|
|
|
exception(BX_NP_EXCEPTION, raw_selector & 0xfffc, 0);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
|
|
|
|
2005-06-22 22:13:45 +04:00
|
|
|
/*
|
|
|
|
// the real hardware CPU allow loading of tss with limit < minimum
|
2003-10-05 00:22:24 +04:00
|
|
|
if (descriptor.type==1 && descriptor.u.tss286.limit<43) {
|
2005-03-13 21:20:26 +03:00
|
|
|
BX_PANIC(("LTR:286TSS: loading tr.limit < 43"));
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
|
|
|
else if (descriptor.type==9 && descriptor.u.tss386.limit_scaled<103) {
|
2005-03-13 21:20:26 +03:00
|
|
|
BX_PANIC(("LTR:386TSS: loading tr.limit < 103"));
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2005-06-22 22:13:45 +04:00
|
|
|
*/
|
2003-10-05 00:22:24 +04:00
|
|
|
|
|
|
|
BX_CPU_THIS_PTR tr.selector = selector;
|
|
|
|
BX_CPU_THIS_PTR tr.cache = descriptor;
|
|
|
|
BX_CPU_THIS_PTR tr.cache.valid = 1;
|
|
|
|
// tr.cache.type should not have busy bit, or it would not get
|
|
|
|
// through the conditions above.
|
|
|
|
BX_ASSERT((BX_CPU_THIS_PTR tr.cache.type & 2) == 0);
|
|
|
|
|
|
|
|
/* mark as busy */
|
|
|
|
dword2 |= 0x00000200; /* set busy bit */
|
|
|
|
access_linear(BX_CPU_THIS_PTR gdtr.base + selector.index*8 + 4, 4, 0,
|
2001-04-10 05:04:59 +04:00
|
|
|
BX_WRITE, &dword2);
|
|
|
|
}
|
|
|
|
|
2005-08-21 22:23:36 +04:00
|
|
|
#endif
|
|
|
|
|
2005-01-28 23:50:48 +03:00
|
|
|
void BX_CPU_C::VERR_Ew(bxInstruction_c *i)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
/* for 16 bit operand size mode */
|
|
|
|
Bit16u raw_selector;
|
|
|
|
bx_descriptor_t descriptor;
|
2004-10-16 23:34:17 +04:00
|
|
|
bx_selector_t selector;
|
2001-04-10 05:04:59 +04:00
|
|
|
Bit32u dword1, dword2;
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (real_mode() || v8086_mode()) {
|
|
|
|
BX_INFO(("VERR: not recognized in real or virtual-8086 mode"));
|
2001-04-10 05:04:59 +04:00
|
|
|
UndefinedOpcode(i);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-09-20 07:52:59 +04:00
|
|
|
if (i->modC0()) {
|
2002-09-18 02:50:53 +04:00
|
|
|
raw_selector = BX_READ_16BIT_REG(i->rm());
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
|
|
|
/* pointer, segment address pair */
|
2002-09-18 09:36:48 +04:00
|
|
|
read_virtual_word(i->seg(), RMAddr(i), &raw_selector);
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
/* if selector null, clear ZF and done */
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((raw_selector & 0xfffc) == 0) {
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERR: null selector"));
|
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
/* if source selector is visible at CPL & RPL,
|
|
|
|
* within the descriptor table, and of type accepted by VERR instruction,
|
|
|
|
* then load register with segment limit and set ZF */
|
|
|
|
parse_selector(raw_selector, &selector);
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (!fetch_raw_descriptor2(&selector, &dword1, &dword2)) {
|
2001-04-10 05:04:59 +04:00
|
|
|
/* not within descriptor table */
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERR: not within descriptor table"));
|
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
parse_descriptor(dword1, dword2, &descriptor);
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (descriptor.segment==0) { /* system or gate descriptor */
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERR: system descriptor"));
|
|
|
|
clear_ZF(); /* inaccessible */
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (descriptor.valid==0) {
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERR: valid bit cleared"));
|
|
|
|
clear_ZF(); /* inaccessible */
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
/* normal data/code segment */
|
2003-10-05 00:22:24 +04:00
|
|
|
if (descriptor.u.segment.executable) { /* code segment */
|
2001-04-10 05:04:59 +04:00
|
|
|
/* ignore DPL for readable conforming segments */
|
2005-03-13 21:20:26 +03:00
|
|
|
if (descriptor.u.segment.c_ed && descriptor.u.segment.r_w) {
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERR: conforming code, OK"));
|
|
|
|
assert_ZF(); /* accessible */
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2003-10-05 00:22:24 +04:00
|
|
|
if (descriptor.u.segment.r_w==0) {
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERR: code not readable"));
|
|
|
|
clear_ZF (); /* inaccessible */
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
/* readable, non-conforming code segment */
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((descriptor.dpl<CPL) || (descriptor.dpl<selector.rpl)) {
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERR: non-conforming code not withing priv level"));
|
|
|
|
clear_ZF (); /* inaccessible */
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
|
|
|
}
|
2005-03-13 21:20:26 +03:00
|
|
|
|
2005-10-16 01:01:36 +04:00
|
|
|
assert_ZF(); /* accessible */
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else { /* data segment */
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((descriptor.dpl<CPL) || (descriptor.dpl<selector.rpl)) {
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERR: data seg not withing priv level"));
|
|
|
|
clear_ZF(); /* not accessible */
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
|
|
|
}
|
2005-10-16 01:01:36 +04:00
|
|
|
assert_ZF(); /* accessible */
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2005-01-28 23:50:48 +03:00
|
|
|
void BX_CPU_C::VERW_Ew(bxInstruction_c *i)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
/* for 16 bit operand size mode */
|
|
|
|
Bit16u raw_selector;
|
|
|
|
bx_descriptor_t descriptor;
|
2003-10-05 00:22:24 +04:00
|
|
|
bx_selector_t selector;
|
2001-04-10 05:04:59 +04:00
|
|
|
Bit32u dword1, dword2;
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (real_mode() || v8086_mode()) {
|
|
|
|
BX_INFO(("VERW: not recognized in real or virtual-8086 mode"));
|
2001-04-10 05:04:59 +04:00
|
|
|
UndefinedOpcode(i);
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-09-20 07:52:59 +04:00
|
|
|
if (i->modC0()) {
|
2002-09-18 02:50:53 +04:00
|
|
|
raw_selector = BX_READ_16BIT_REG(i->rm());
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else {
|
|
|
|
/* pointer, segment address pair */
|
2002-09-18 09:36:48 +04:00
|
|
|
read_virtual_word(i->seg(), RMAddr(i), &raw_selector);
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
/* if selector null, clear ZF and done */
|
2003-10-05 00:22:24 +04:00
|
|
|
if ((raw_selector & 0xfffc) == 0) {
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERW: null selector"));
|
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
/* if source selector is visible at CPL & RPL,
|
|
|
|
* within the descriptor table, and of type accepted by VERW instruction,
|
|
|
|
* then load register with segment limit and set ZF */
|
|
|
|
parse_selector(raw_selector, &selector);
|
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (!fetch_raw_descriptor2(&selector, &dword1, &dword2)) {
|
2001-04-10 05:04:59 +04:00
|
|
|
/* not within descriptor table */
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERW: not within descriptor table"));
|
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
parse_descriptor(dword1, dword2, &descriptor);
|
|
|
|
|
|
|
|
/* rule out system segments & code segments */
|
2003-10-05 00:22:24 +04:00
|
|
|
if (descriptor.segment==0 || descriptor.u.segment.executable) {
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERW: system seg or code"));
|
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2003-10-05 00:22:24 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:22:24 +04:00
|
|
|
if (descriptor.valid==0) {
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERW: valid bit cleared"));
|
|
|
|
clear_ZF();
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
/* data segment */
|
2003-10-05 00:22:24 +04:00
|
|
|
if (descriptor.u.segment.r_w) { /* writable */
|
|
|
|
if ((descriptor.dpl<CPL) || (descriptor.dpl<selector.rpl)) {
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERW: writable data seg not within priv level"));
|
|
|
|
clear_ZF(); /* not accessible */
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2005-10-16 01:01:36 +04:00
|
|
|
assert_ZF(); /* accessible */
|
2001-04-10 05:04:59 +04:00
|
|
|
return;
|
2005-03-13 21:20:26 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_DEBUG(("VERW: data seg not writable"));
|
|
|
|
clear_ZF(); /* not accessible */
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2005-08-21 22:23:36 +04:00
|
|
|
#if BX_CPU_LEVEL >= 2
|
|
|
|
|
2005-01-28 23:50:48 +03:00
|
|
|
void BX_CPU_C::SGDT_Ms(bxInstruction_c *i)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
/* op1 is a register or memory reference */
|
2002-09-20 07:52:59 +04:00
|
|
|
if (i->modC0()) {
|
2001-04-10 05:04:59 +04:00
|
|
|
/* undefined opcode exception */
|
2004-04-17 21:10:58 +04:00
|
|
|
BX_INFO(("SGDT_Ms: use of register is undefined opcode."));
|
2001-04-10 05:04:59 +04:00
|
|
|
UndefinedOpcode(i);
|
2004-10-16 23:34:17 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2004-10-16 23:34:17 +04:00
|
|
|
#if BX_SUPPORT_X86_64
|
|
|
|
if (BX_CPU_THIS_PTR cpu_mode == BX_MODE_LONG_64)
|
|
|
|
{
|
2005-01-28 23:50:48 +03:00
|
|
|
Bit16u limit_16 = BX_CPU_THIS_PTR gdtr.limit;
|
|
|
|
Bit64u base_64 = BX_CPU_THIS_PTR gdtr.base;
|
2002-09-14 01:08:54 +04:00
|
|
|
|
2002-09-18 09:36:48 +04:00
|
|
|
write_virtual_word(i->seg(), RMAddr(i), &limit_16);
|
|
|
|
write_virtual_qword(i->seg(), RMAddr(i)+2, &base_64);
|
2004-10-16 23:34:17 +04:00
|
|
|
}
|
2002-09-14 01:08:54 +04:00
|
|
|
else
|
2004-10-16 23:34:17 +04:00
|
|
|
#endif
|
|
|
|
{
|
2005-01-28 23:50:48 +03:00
|
|
|
Bit16u limit_16 = BX_CPU_THIS_PTR gdtr.limit;
|
|
|
|
Bit32u base_32 = BX_CPU_THIS_PTR gdtr.base;
|
2001-04-10 05:04:59 +04:00
|
|
|
#if BX_CPU_LEVEL == 2
|
2002-09-14 01:08:54 +04:00
|
|
|
base_32 |= 0xff000000; /* ??? */
|
2001-04-10 05:04:59 +04:00
|
|
|
#else /* 386+ */
|
2002-09-14 01:08:54 +04:00
|
|
|
/* 32bit processors always write 32bits of base */
|
2001-04-10 05:04:59 +04:00
|
|
|
#endif
|
2002-09-18 09:36:48 +04:00
|
|
|
write_virtual_word(i->seg(), RMAddr(i), &limit_16);
|
|
|
|
write_virtual_dword(i->seg(), RMAddr(i)+2, &base_32);
|
2004-10-16 23:34:17 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2005-01-28 23:50:48 +03:00
|
|
|
void BX_CPU_C::SIDT_Ms(bxInstruction_c *i)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
#if BX_CPU_LEVEL < 2
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("SIDT_Ms: not supported on 8086!"));
|
2004-10-16 23:34:17 +04:00
|
|
|
UndefinedOpcode(i);
|
2001-04-10 05:04:59 +04:00
|
|
|
#else
|
|
|
|
|
|
|
|
/* op1 is a register or memory reference */
|
2002-09-20 07:52:59 +04:00
|
|
|
if (i->modC0()) {
|
2001-04-10 05:04:59 +04:00
|
|
|
/* undefined opcode exception */
|
2004-04-17 21:10:58 +04:00
|
|
|
BX_INFO(("SIDT: use of register is undefined opcode."));
|
2001-04-10 05:04:59 +04:00
|
|
|
UndefinedOpcode(i);
|
2004-10-16 23:34:17 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2004-10-16 23:34:17 +04:00
|
|
|
#if BX_SUPPORT_X86_64
|
|
|
|
if (BX_CPU_THIS_PTR cpu_mode == BX_MODE_LONG_64)
|
|
|
|
{
|
2005-01-28 23:50:48 +03:00
|
|
|
Bit16u limit_16 = BX_CPU_THIS_PTR idtr.limit;
|
|
|
|
Bit64u base_64 = BX_CPU_THIS_PTR idtr.base;
|
2002-09-14 01:08:54 +04:00
|
|
|
|
2002-09-18 09:36:48 +04:00
|
|
|
write_virtual_word(i->seg(), RMAddr(i), &limit_16);
|
|
|
|
write_virtual_qword(i->seg(), RMAddr(i)+2, &base_64);
|
2004-10-16 23:34:17 +04:00
|
|
|
}
|
2002-09-14 01:08:54 +04:00
|
|
|
else
|
2004-10-16 23:34:17 +04:00
|
|
|
#endif
|
|
|
|
{
|
2005-01-28 23:50:48 +03:00
|
|
|
Bit16u limit_16 = BX_CPU_THIS_PTR idtr.limit;
|
|
|
|
Bit32u base_32 = BX_CPU_THIS_PTR idtr.base;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
#if BX_CPU_LEVEL == 2
|
2002-09-14 01:08:54 +04:00
|
|
|
base_32 |= 0xff000000;
|
2001-04-10 05:04:59 +04:00
|
|
|
#else /* 386+ */
|
2004-10-16 23:34:17 +04:00
|
|
|
/* regardless of operand size, all 32bits of base are stored */
|
2001-04-10 05:04:59 +04:00
|
|
|
#endif
|
|
|
|
|
2002-09-18 09:36:48 +04:00
|
|
|
write_virtual_word(i->seg(), RMAddr(i), &limit_16);
|
|
|
|
write_virtual_dword(i->seg(), RMAddr(i)+2, &base_32);
|
2004-10-16 23:34:17 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2005-01-28 23:50:48 +03:00
|
|
|
void BX_CPU_C::LGDT_Ms(bxInstruction_c *i)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2003-10-05 00:48:13 +04:00
|
|
|
if (v8086_mode()) {
|
|
|
|
BX_INFO(("LGDT: not recognized in virtual-8086 mode"));
|
|
|
|
exception(BX_GP_EXCEPTION, 0, 0);
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
invalidate_prefetch_q();
|
|
|
|
|
2003-10-05 00:48:13 +04:00
|
|
|
if (!real_mode() && CPL!=0) {
|
|
|
|
BX_INFO(("LGDT: CPL!=0 in protected mode"));
|
2001-04-10 05:04:59 +04:00
|
|
|
exception(BX_GP_EXCEPTION, 0, 0);
|
2003-10-05 00:48:13 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:48:13 +04:00
|
|
|
/* operand might be a register or memory reference */
|
2002-09-20 07:52:59 +04:00
|
|
|
if (i->modC0()) {
|
2003-10-05 00:48:13 +04:00
|
|
|
BX_INFO(("LGDT: must be memory reference"));
|
2001-04-10 05:04:59 +04:00
|
|
|
UndefinedOpcode(i);
|
2003-10-05 00:48:13 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
#if BX_CPU_LEVEL >= 3
|
2004-10-16 23:34:17 +04:00
|
|
|
#if BX_SUPPORT_X86_64
|
|
|
|
if (BX_CPU_THIS_PTR cpu_mode == BX_MODE_LONG_64)
|
|
|
|
{
|
2002-09-14 01:08:54 +04:00
|
|
|
Bit16u limit_16;
|
|
|
|
Bit64u base_64;
|
|
|
|
|
2002-09-18 09:36:48 +04:00
|
|
|
read_virtual_word(i->seg(), RMAddr(i), &limit_16);
|
|
|
|
read_virtual_qword(i->seg(), RMAddr(i) + 2, &base_64);
|
2002-09-14 01:08:54 +04:00
|
|
|
|
|
|
|
BX_CPU_THIS_PTR gdtr.limit = limit_16;
|
|
|
|
BX_CPU_THIS_PTR gdtr.base = base_64;
|
2004-10-16 23:34:17 +04:00
|
|
|
}
|
2002-09-14 01:08:54 +04:00
|
|
|
else
|
2004-10-16 23:34:17 +04:00
|
|
|
#endif
|
2002-09-18 09:36:48 +04:00
|
|
|
if (i->os32L()) {
|
2001-04-10 05:04:59 +04:00
|
|
|
Bit16u limit_16;
|
|
|
|
Bit32u base0_31;
|
|
|
|
|
2002-09-18 09:36:48 +04:00
|
|
|
read_virtual_word(i->seg(), RMAddr(i), &limit_16);
|
|
|
|
read_virtual_dword(i->seg(), RMAddr(i) + 2, &base0_31);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
BX_CPU_THIS_PTR gdtr.limit = limit_16;
|
|
|
|
BX_CPU_THIS_PTR gdtr.base = base0_31;
|
2005-01-28 23:50:48 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else
|
|
|
|
#endif
|
2004-10-16 23:34:17 +04:00
|
|
|
{
|
2001-04-10 05:04:59 +04:00
|
|
|
Bit16u limit_16, base0_15;
|
|
|
|
Bit8u base16_23;
|
|
|
|
|
2002-09-18 09:36:48 +04:00
|
|
|
read_virtual_word(i->seg(), RMAddr(i), &limit_16);
|
|
|
|
read_virtual_word(i->seg(), RMAddr(i) + 2, &base0_15);
|
|
|
|
read_virtual_byte(i->seg(), RMAddr(i) + 4, &base16_23);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
/* ignore high 8 bits */
|
|
|
|
BX_CPU_THIS_PTR gdtr.limit = limit_16;
|
|
|
|
BX_CPU_THIS_PTR gdtr.base = (base16_23 << 16) | base0_15;
|
2004-10-16 23:34:17 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2005-01-28 23:50:48 +03:00
|
|
|
void BX_CPU_C::LIDT_Ms(bxInstruction_c *i)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
Bit16u limit_16;
|
|
|
|
Bit32u base_32;
|
|
|
|
|
2003-10-05 00:48:13 +04:00
|
|
|
if (v8086_mode()) {
|
|
|
|
BX_INFO(("LIDT: not recognized in virtual-8086 mode"));
|
|
|
|
exception(BX_GP_EXCEPTION, 0, 0);
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
invalidate_prefetch_q();
|
|
|
|
|
2003-10-05 00:48:13 +04:00
|
|
|
if (!real_mode() && CPL!=0) {
|
2005-10-16 01:01:36 +04:00
|
|
|
BX_ERROR(("LIDT: CPL!=0 in protected mode"));
|
2003-10-05 00:48:13 +04:00
|
|
|
exception(BX_GP_EXCEPTION, 0, 0);
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-05 00:48:13 +04:00
|
|
|
/* operand might be a register or memory reference */
|
2002-09-20 07:52:59 +04:00
|
|
|
if (i->modC0()) {
|
2003-10-05 00:48:13 +04:00
|
|
|
BX_INFO(("LIDT: must be memory reference"));
|
2001-04-10 05:04:59 +04:00
|
|
|
UndefinedOpcode(i);
|
2003-10-05 00:48:13 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-09-14 01:08:54 +04:00
|
|
|
#if BX_CPU_LEVEL >= 3
|
2004-10-16 23:34:17 +04:00
|
|
|
#if BX_SUPPORT_X86_64
|
|
|
|
if (BX_CPU_THIS_PTR cpu_mode == BX_MODE_LONG_64)
|
|
|
|
{
|
2002-09-14 01:08:54 +04:00
|
|
|
Bit64u base_64;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-09-18 09:36:48 +04:00
|
|
|
read_virtual_word(i->seg(), RMAddr(i), &limit_16);
|
|
|
|
read_virtual_qword(i->seg(), RMAddr(i) + 2, &base_64);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-09-14 01:08:54 +04:00
|
|
|
BX_CPU_THIS_PTR idtr.limit = limit_16;
|
|
|
|
BX_CPU_THIS_PTR idtr.base = base_64;
|
2004-10-16 23:34:17 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
if (i->os32L()) {
|
2002-09-18 09:36:48 +04:00
|
|
|
read_virtual_word(i->seg(), RMAddr(i), &limit_16);
|
|
|
|
read_virtual_dword(i->seg(), RMAddr(i) + 2, &base_32);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-09-14 01:08:54 +04:00
|
|
|
BX_CPU_THIS_PTR idtr.limit = limit_16;
|
2001-04-10 05:04:59 +04:00
|
|
|
BX_CPU_THIS_PTR idtr.base = base_32;
|
2004-10-16 23:34:17 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
else
|
|
|
|
#endif
|
2004-10-16 23:34:17 +04:00
|
|
|
{
|
2002-09-18 09:36:48 +04:00
|
|
|
read_virtual_word(i->seg(), RMAddr(i), &limit_16);
|
|
|
|
read_virtual_dword(i->seg(), RMAddr(i) + 2, &base_32);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-09-14 01:08:54 +04:00
|
|
|
BX_CPU_THIS_PTR idtr.limit = limit_16;
|
|
|
|
BX_CPU_THIS_PTR idtr.base = base_32 & 0x00ffffff; /* ignore upper 8 bits */
|
2004-10-16 23:34:17 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2005-08-21 22:23:36 +04:00
|
|
|
|
|
|
|
#endif
|