Home of the original IBM PC emulator for browsers.
[The following information is from an undated 15-page Intel document titled “Undocumented iAPX 286 Test Instruction”, pp. 4-12]
An iAPX 86/88 program using real mode addressing can be executed in protected mode with full protection between it and other programs. All segment register semantics of iAPX 86 real mode can be emulated. The address space of the real mode program can also be limited to less than 1 megabyte and be relocated anywhere in the 16 Megabyte physical address space. The following sections describe several aspects of this emulation.
iAPX 86 real mode emulation requires any segment register load instruction cause a protection exception. An error code with bits 1-0 being zero and bits 15-2 being non-zero identify a segment register load exception. Such exceptions are restartable. All instructions that do not load a segment register run at full iAPX 286 speed and with full access checks.
The exception handler must interpret the segment register load instruction to place the iAPX 86 paragraph ID and the associated protected descriptor into the LOADALL memory area. LOADALL then loads the segment register with the value used by the interrupted instruction, points the descriptor cache entry at the protected physical memory region, and restores the other segment registers.
Most segment load instructions will cause exception 13 if all LDT and GDT entries are marked with a privilege level less than the CPL of the emulated program. The CPL of the emulated program is defined by the DPL fields of the CS and SS descriptor caches.
Segment register loads using a selector value of 0000H to 0003H do not cause an exception on loading the segment register. Instead, any memory reference using the segment register will cause exception 13 with an error code of 0. No memory reference will occur. This case can be identified by checking whether DS or ES contain a value of 0-3. These exceptions are also restartable.
Limits can be enforced on the size of the emulated iAPX 86 address space. An iAPX 86 paragraph ID that is outside the defined memory area can be loaded, but the segment register can be marked invalid for memory addressing. LOADALL can be used to load the iAPX 86 paragraph ID into the segment register, but the descriptor cache entry is marked invalid. The paragraph ID can still be read without causing a protection exception.
If a selector value is loaded whose segment overruns the end of the defined physical memory area, the limit field can be set less than 65535 to prevent accesses outside the defined memory area with that segment register.
The emulated iAPX 86/88 address space can be relocated anywhere in the 16 Mbyte iAPX 286 physical address space by adding a 24-bit relocation factor to the 20-bit iAPX 86/88 physical address value associated with the iAPX 86 paragraph ID.
The LOADALL instruction allows a protected mode 80286 to provide a simulated iAPX 86/88 interrupt table to iAPX 86/88 programs. The protected mode iAPX 286 interrupt table is different from iAPX 86/88 since it must contain more information and be protected from improper use. The protected mode interrupt table cannot be addressed by the same selector-offset pairs used in iAPX 86 real address mode.
The iAPX 86/88 interrupt table is simulated by having all INT instructions cause a protection exception. Setting the DPL of all IDT gate entries to less than the CPL of the emulated program will cause exception 13 for all INT instructions. The error code will indicate an IDT vector with the EXT bit cleared. External interrupts and program exceptions will continue to use the protected IDT.
The iAPX 86/88 INT instruction can be simulated by the exception 13 handler. For INT instructions, it looks into the iAPX 86 interrupt vector table for the vector associated with the interrupt vector in the error code. After simulating the machine state save, the iAPX 86/88 program is restarted at the interrupt vector address.
Interrupt handlers for external interrupts can pass control to an iAPX 86 real mode program. Each external interrupt handler for an iAPX 86 interrupt must determine if the interrupt is for a real mode program; if so, then it emulates a real mode interrupt the same way as for the INT instruction.
Code segment writes are possible by using writable data segment descriptors for the CS cache entry. Normally the code segment is write protected. If the code segment descriptor is always marked writable, then writes using the CS prefix will work correctly.
A temporary value which does not correspond to a valid segment causes exception 13. It is possible to place that value into the program visible segment register, but mark the descriptor cache entry invalid. The invalid descriptor lets the program reference the numeric value stored in the segment register value, (i.e. MOV AX,ES) but prevents any memory reference instruction from using the segment register to address memory (i.e. MOV AX,ES:[BX]).
This feature requires an error handler to know that exception 13 with an error code which is an invalid segment selector value indicates a potential temporary value problem. The exception handler must simulate the segment load instruction to place the error code into the appropriate segment register and use LOADALL to mark the descriptor cache entry invalid. The program may then be resumed after the segment load instruction.
All I/O instructions of the iAPX 86 program can be simulated. When the IOPL (I/O privilege level) is less than the CPL of the simulated iAPX 86 program, exception 13 will occur, with an error code of 0, on IN, OUT, STI, CLI, and LOCK instructions. The exception handler can identify these instructions and emulate their actions. The iAPX 86 program can then be restarted.
The LOCK instruction prefix causes exception 13 when CPL is greater than IOPL. For most systems, the LOCK prefix could be ignored. Restarting the program after the LOCK prefix would be acceptable. In special cases, the LOCKED instruction may need to be run with a lower CPL.
A system which emulates a real mode program may also run protected mode software. If the GDT and IDT has all entries marked level 2 or less, the emulated program cannot use them if it runs at level 3. The emulated program can have a task state segment associated with it. An LDT may be present if all entries are marked level 2 or less. Normal protected mode tasks may use an LOT with entries at privilege level 3.
Interrupt handlers may use either task or interrupt/trap gates. All interrupt handlers using trap/interrupt gates must execute at privilege level 2 or less. Interrupts that use task gates may run at any privilege level.
The register save operation of the task switch or interrupt handler will work without exceptions. The iAPX 86 paragraph IDs in the segment registers can be read without a protection exception. The segment registers can be reloaded with protected selectors without a protection exception. Interrupting from a emulated iAPX 86 program does not affect interrupt latency.
Returning from an interrupt requires some checks. The return from the interrupt handler must check whether an iAPX 86 real mode program had been executing. If so, the return sequence must use the LOADALL instruction to reload all the registers rather than the normal IRET instruction.
Depending on the iAPX 86 paragraph IDs used, the IRET instruction might not cause a protection exception on returning to an emulated iAPX 86 program. The CS value of an interrupted iAPX 86 program saved on the stack or in the TSS does not correctly identify the privilege level, normally 3, of the emulated iAPX 86 real mode program. The privilege level of the interrupted program is determined by the RPL fields of the saved CS and SS selectors. If these values are the same and refer to a visible code segment, the CPU could attempt to execute the protected code segment at an incorrect address.
The interrupt handler should test whether an emulated iAPX 86 program was executing. An interrupted protected mode program can be restarted in the normal manner while an emulated program requires LOADALL.
The instruction and data addresses saved in the protected mode 80287 environment area are in a different format than from the 8087. In real mode, the 80287 environment is in the same format as the 8087. In protected mode, the 80287 environment is changed to store 32-bit virtual pointers rather than 20-bit iAPX 86/88 physical addresses.
The 80287 can be used by both normal protected mode programs and emulated iAPX 86/88 real mode programs. The 80287 operates in either real mode or protected mode. The FSETPM instruction must be executed before starting a normal protected mode program if the 80287 was in real mode. The 80287 must be reset, via the RESET pin, to reenter real mode for an emulated iAPX 86/88 program after being used by a normal protected mode program. External hardware could reset the part to reenter real-mode. The TS bit of the MSW can be used to monitor for the first ESCAPE instruction executed in a program. The exception 7 handler can then determine what mode of operation is required in the 80287.
The 20-bit physical addresses kept by the 80287 for the instruction and data pointers will reflect the paragraph ID in the program visible segment register and offset used by the ESC instruction to address memory. The descriptor cache base and limit loaded by LOADALL is used to generate physical memory addresses for data transfers.
The WAIT instructions required by the 8087 before ESC instructions can be safely executed by the 80287.
An 80286 cannot exactly emulate an 8086/88 in all possible cases. Most differences are due to the extra protection checks made in the 80286 which are not made in the 8086/88. The discrepancies listed here are minor enough that very few programs will be affected.
The PUSH SP instruction pushes a different value on the iAPX 286 than on the iAPX 86,88,186. The value pushed onto the stack by the 80286 is the value of SP before the push instruction executes. The value pushed onto the stack by the 8086/88/186 is the SP value after the push instruction executes.
Shifts and rotates on the iAPX 286 mask the count to 5 bits. The iAPX 86/88 allows all 8 bits to be used. The iAPX 186/188 also masks the shift count to 5 bits.
Exceptions 9, 12, or 13 occur during attempts to wrap-around a segment depending on the location and type of operand involved. All exception 12 cases can be emulated and the program restarted. Exception 13 or 12 that occurs for an ESC instruction occurs before the 80286 or 80287 execute the instruction, and are therefore restartable. Exception 9 cannot be restarted.
Most simple load and store instructions that violate a segment limit are restartable. The current case that cannot be restarted in general is any floating point operand reference where the second or subsequent word exceeded a segment limit. The exception 9 handler must execute FNINIT before any other WAIT or ESC instruction. The internal status of the 80287 cannot be read until it is forced idle by FNINIT. The FNINIT instruction will mark all floating point data registers as empty, set top of stack to 0, and mask all errors. The numeric instruction and data addresses stored in the 80287 will correctly point at the failing instruction. If the 80286 program interrupted by the math address error is not the program that executed the failed ESC instruction, then that program can be restarted.
Memory address space wrap-around is not directly supported. The iAPX 86/88 allow wrap around from the top of the 1 megabyte address space into the bottom of the 1 megabyte address space (i.e. address FC00:4000 is same as 0000:0000). To emulate instructions that address memory with such wrap-around, requires the segment register limit be set to cause a protection exception for addresses beyond simulated physical address 0FFFFFH, or addresses below 00000H using an expand down segment, and software emulation of the instruction to address memory at the bottom of the address space.
An 8086/88/186 program will require different amounts of time to execute instructions on the 80286. Most instructions will run faster on the 80286. Instructions which do not modify a segment register, and do not use a segment register with a zero in it will run faster on the 80286. Instructions that first access memory with a segment register containing a zero will run slower on the 80286. Instructions that load a segment register with a non-zero selector value will run slower on the 80286.
The iAPX 286 and iAPX 186 can generate the most negative number as a quotient for the IDIV instruction. The iAPX 86/88 will generate the divide error exception instead.
The iAPX 286 divide error return address will point at the divide instruction including prefixes. The registers will appear as if the instruction had not executed. The iAPX 86/88/186/188 return address will point after the divide instruction and the DX:AX or AH:AL registers may have been changed.
The numeric instruction address stored in the 80287 includes all leading prefixes before the ESC opcode. The 8087 numeric instruction address always points at the ESC opcode.
An iAPX 286/20 system does not require an interrupt controller for the ERROR signal. iAPX 86/20 systems use an interrupt controller to prioritize simultaneous interrupts and mask errors from the 8087 if servicing them must be delayed.
If the same interrupt controller is provided as in the iAPX 86 system, the input used for the 8087 ERROR signal can be grounded. Instructions that control that input of the interrupt controller become NOPs. Watch out for non-specific EOI instructions inside an 8087 error handler which may affect other interrupt inputs.
80287 errors do not normally affect an interrupt handler. As long as any program does not execute WAIT or ESC instructions, it cannot be interrupted by the 80287.
If a different interrupt system is used in the iAPX 286 system than in the iAPX 86 system, any I/O instructions to the interrupt controller may have to be emulated.
Numeric error interrupts use interrupt vector 16. Since an external interrupt controller may be used in iAPX 86,88,186 systems, another interrupt vector may have been used for numeric interrupts.
Do not perform port I/O to ports 00F8H to 00FFH. The 80287 may not operate properly if this is allowed. These I/O locations are reserved by Intel.
The interrupt enable bit of the flag word may not change when a POPF or IRET instruction attempts to change it. The IOPL field of the flag word controls whether IF can be changed. Subsequent PUSHF and INT instructions will save a value of IF which differs from the value in an 8086/8088 program.
If STI and CLI are emulated as NOPs then they will fail to change IF. Subsequent PUSHF and INT instructions will save a value of IF which differs from the value in an 8086/8088 program.
The flag word has two new fields: IOPL and NT. IOPL can not change except at level 0, but NT can be changed by IRET and POPF instructions. The IRET instruction attempts a task switch when NT is set. The back link field of the current TSS should have a 0 in it to cause exception 13, with an error code of 0, if the iAPX 86 program attempts an IRET after setting NT. The exception 13 handler may then simulate an iAPX 86 !RET operation.
The iAPX 86 address space may be limited. Programs may use some form of memory space scanner to see how much memory is available. Accesses to illegal locations are expected. The program emulator must decide what to do about illegal accesses.
The 80286 defines new instructions for undefined opcodes in the 8086/88. An 8086/88 program with an unknown bug in it that executes these undefined opcodes will work differently on an 80286.
Programs with self-modifying code may work differently on an 80286. The 80286 prefetcher can fetch more bytes ahead of the current instruction than the 8086 or 8088. A program that modifies an instruction that has already been prefetched will not see the changed instruction. Any program which jumps after modifying an instruction before executing it will correctly execute the modified instruction.
Regions of the emulated iAPX 86,88,186 address space can not be write protected. The XCHG, ADC, SBB, RCL, and RCR instructions are not restartable if their memory-based operand is in a write-protected segment.
Current iAPX 86 real mode programs can use the extended address space of the iAPX 286 in a limited manner. To address the extended memory, LOADALL must be used to load the descriptor cache with an base address beyond the normal 1 Mbyte address range. That segment register must not be changed by software, else the segment register will point back into the 1 Mbyte address space.
Two types of systems are examined: accessing a single large database in a limited manner, or splitting software into normal and extended areas. The first is the easiest to implement, while the second is more general.
Access to a large data area outside the 1 Mbyte address space could be provided by a subroutine. The subroutine scans the large data structure to locate the necessary item, then copy all data between the normal address space and the extended address space.
Interrupts must be disabled while the subroutine uses segment registers that have been set by LOADALL. The reload of segment registers inside an interrupt routine would change the actual physical address from that loaded by LOADALL before the interrupt. After all accesses in the extended area are done, interrupts may be enabled.
Returning the address of an extended data structure requires passing data through a segment register. For example, the ES register could have been changed by LOADALL to point at a data area outside the bottom megabyte of physical memory. The subroutine must not reload ES while it runs. The value stored in ES is not important since it is not related to the physical address. Interrupts must not be allowed since the interrupt routine may reload ES.
A second technique uses special paragraph IDs (i.e. FFFFH) to signal that a piece of software is running in extended mode. All interrupt handlers in the system must look when they return to the interrupted program to see if any of the segment registers contain FFFFH. If so, then that segment register points at extended memory. LOADALL must be used to load all the registers and the segment base address used last. The LOADALL memory area should contain that value left there from the previous usage. Descriptors for the other segment registers with normal paragraph IDs must be constructed before executing LOADALL.
A semaphore must be placed around software that writes into the LOADALL area such that once written into, the software can execute LOADALL without interruption.
The 80286 can alternate between real mode and protected mode. Some programs could be executed in real mode in the bottom megabyte of memory, while others execute in protected mode in the upper 15 Mbytes of memory. An external OR gate could RESET the 80286, independent of the rest of the system, to force it to enter real mode. A short routine at the power up address could redirect the software to the correct real mode program.
After executing the real mode program, LOADALL could then quickly restart the protected mode software. LOADALL can be used as a form of task switch from real mode to a protected mode task.
One operating system could service both the real and protected mode software. Any operating system call from the real mode program would cause a switch to protected mode. The protected mode software could then construct descriptors that refer to the same physical memory addresses used by the real mode paragraph IDs. After conversion, the operating system could then perform all work in protected mode.
Interrupts must be handled specially. Interrupt handlers for both real mode and protected mode must be present at all times. If an interrupt handler needs to access a data area, that data area must be addressable from both real and protected mode. The real mode interrupt table would be would be kept at location 000000H. The protected mode IDT could be anywhere. LOADALL will switch to the protected interrupt table.