mirror of
https://github.com/TomHarte/CLK.git
synced 2024-11-26 23:52:26 +00:00
672 lines
21 KiB
C++
672 lines
21 KiB
C++
//
|
||
// Instruction.hpp
|
||
// Clock Signal
|
||
//
|
||
// Created by Thomas Harte on 15/01/21.
|
||
// Copyright © 2021 Thomas Harte. All rights reserved.
|
||
//
|
||
|
||
#ifndef InstructionSets_x86_Instruction_h
|
||
#define InstructionSets_x86_Instruction_h
|
||
|
||
#include <cstddef>
|
||
#include <cstdint>
|
||
#include <type_traits>
|
||
|
||
namespace InstructionSet {
|
||
namespace x86 {
|
||
|
||
/*
|
||
Operations are documented below to establish expectations as to which
|
||
instruction fields will be meaningful for each; this is a work-in-progress
|
||
and may currently contain errors in the opcode descriptions — especially
|
||
where implicit register dependencies are afoot.
|
||
*/
|
||
enum class Operation: uint8_t {
|
||
Invalid,
|
||
|
||
//
|
||
// 8086 instructions.
|
||
//
|
||
|
||
/// ASCII adjust after addition; source will be AL and destination will be AX.
|
||
AAA,
|
||
/// ASCII adjust before division; destination will be AX and source will be a multiplier.
|
||
AAD,
|
||
/// ASCII adjust after multiplication; destination will be AX and source will be a divider.
|
||
AAM,
|
||
/// ASCII adjust after subtraction; source will be AL and destination will be AX.
|
||
AAS,
|
||
/// Decimal adjust after addition; source and destination will be AL.
|
||
DAA,
|
||
/// Decimal adjust after subtraction; source and destination will be AL.
|
||
DAS,
|
||
|
||
/// Convert byte into word; source will be AL, destination will be AH.
|
||
CBW,
|
||
/// Convert word to double word; source will be AX and destination will be DX.
|
||
CWD,
|
||
|
||
/// Escape, for a coprocessor; perform the bus cycles necessary to read the source and destination and perform a NOP.
|
||
ESC,
|
||
|
||
/// Stops the processor until the next interrupt is fired.
|
||
HLT,
|
||
/// Waits until the WAIT input is asserted; if an interrupt occurs then it is serviced but returns to the WAIT.
|
||
WAIT,
|
||
|
||
/// Add with carry; source, destination, operand and displacement will be populated appropriately.
|
||
ADC,
|
||
/// Add; source, destination, operand and displacement will be populated appropriately.
|
||
ADD,
|
||
/// Subtract with borrow; source, destination, operand and displacement will be populated appropriately.
|
||
SBB,
|
||
/// Subtract; source, destination, operand and displacement will be populated appropriately.
|
||
SUB,
|
||
/// Unsigned multiply; multiplies the source value by AX or AL, storing the result in DX:AX or AX.
|
||
MUL,
|
||
/// Single operand signed multiply; multiplies the source value by AX or AL, storing the result in DX:AX or AX.
|
||
IMUL_1,
|
||
/// Unsigned divide; divide the source value by AX or AL, storing the quotient in AL and the remainder in AH.
|
||
DIV,
|
||
/// Signed divide; divide the source value by AX or AL, storing the quotient in AL and the remainder in AH.
|
||
IDIV,
|
||
|
||
/// Increment; source, destination, operand and displacement will be populated appropriately.
|
||
INC,
|
||
/// Decrement; source, destination, operand and displacement will be populated appropriately.
|
||
DEC,
|
||
|
||
/// Reads from the port specified by source to the destination.
|
||
IN,
|
||
/// Writes from the port specified by destination from the source.
|
||
OUT,
|
||
|
||
// Various jumps; see the displacement to calculate targets.
|
||
JO, JNO, JB, JNB, JE, JNE, JBE, JNBE,
|
||
JS, JNS, JP, JNP, JL, JNL, JLE, JNLE,
|
||
|
||
/// Far call; see the segment() and offset() fields.
|
||
CALLF,
|
||
/// Displacement call; followed by a 16-bit operand providing a call offset.
|
||
CALLD,
|
||
/// Near call.
|
||
CALLN,
|
||
/// Return from interrupt.
|
||
IRET,
|
||
/// Near return; if source is not ::None then it will be an ::Immediate indicating how many additional bytes to remove from the stack.
|
||
RETF,
|
||
/// Far return; if source is not ::None then it will be an ::Immediate indicating how many additional bytes to remove from the stack.
|
||
RETN,
|
||
/// Near jump; if an operand is not ::None then it gives an absolute destination; otherwise see the displacement.
|
||
JMPN,
|
||
/// Far jump to the indicated segment and offset.
|
||
JMPF,
|
||
/// Relative jump performed only if CX = 0; see the displacement.
|
||
JPCX,
|
||
/// Generates a software interrupt of the level stated in the operand.
|
||
INT,
|
||
/// Generates a software interrupt of level 3.
|
||
INT3,
|
||
/// Generates a software interrupt of level 4 if overflow is set.
|
||
INTO,
|
||
|
||
/// Load status flags to AH.
|
||
LAHF,
|
||
/// Load status flags from AH.
|
||
SAHF,
|
||
/// Load a segment and offset from the source into DS and the destination.
|
||
LDS,
|
||
/// Load a segment and offset from the source into ES and the destination.
|
||
LES,
|
||
/// Computes the effective address of the source and loads it into the destination.
|
||
LEA,
|
||
|
||
/// Compare [bytes or words, per operation size]; source and destination implied to be DS:[SI] and ES:[DI].
|
||
CMPS,
|
||
/// Load string; reads from DS:SI into AL or AX, subject to segment override.
|
||
LODS,
|
||
/// Move string; moves a byte or word from DS:SI to ES:DI. If a segment override is provided, it overrides the the source.
|
||
MOVS,
|
||
/// Scan string; reads a byte or word from DS:SI and compares it to AL or AX.
|
||
SCAS,
|
||
/// Store string; store AL or AX to ES:DI.
|
||
STOS,
|
||
|
||
// Perform a possibly-conditional loop, decrementing CX. See the displacement.
|
||
LOOP, LOOPE, LOOPNE,
|
||
|
||
/// Loads the destination with the source.
|
||
MOV,
|
||
/// Negatives; source and destination point to the same thing, to negative.
|
||
NEG,
|
||
/// Logical NOT; source and destination point to the same thing, to negative.
|
||
NOT,
|
||
/// Logical AND; source, destination, operand and displacement will be populated appropriately.
|
||
AND,
|
||
/// Logical OR of source onto destination.
|
||
OR,
|
||
/// Logical XOR of source onto destination.
|
||
XOR,
|
||
/// NOP; no further fields.
|
||
NOP,
|
||
/// POP from the stack to destination.
|
||
POP,
|
||
/// POP from the stack to the flags register.
|
||
POPF,
|
||
/// PUSH the source to the stack.
|
||
PUSH,
|
||
/// PUSH the flags register to the stack.
|
||
PUSHF,
|
||
/// Rotate the destination left through carry the number of bits indicated by source; if the source is a register then implicitly its size is 1.
|
||
RCL,
|
||
/// Rotate the destination right through carry the number of bits indicated by source; if the source is a register then implicitly its size is 1.
|
||
RCR,
|
||
/// Rotate the destination left the number of bits indicated by source; if the source is a register then implicitly its size is 1.
|
||
ROL,
|
||
/// Rotate the destination right the number of bits indicated by source; if the source is a register then implicitly its size is 1.
|
||
ROR,
|
||
/// Arithmetic shift left the destination by the number of bits indicated by source; if the source is a register then implicitly its size is 1.
|
||
SAL,
|
||
/// Arithmetic shift right the destination by the number of bits indicated by source; if the source is a register then implicitly its size is 1.
|
||
SAR,
|
||
/// Logical shift right the destination by the number of bits indicated by source; if the source is a register then implicitly its size is 1.
|
||
SHR,
|
||
|
||
/// Clear carry flag; no source or destination provided.
|
||
CLC,
|
||
/// Clear direction flag; no source or destination provided.
|
||
CLD,
|
||
/// Clear interrupt flag; no source or destination provided.
|
||
CLI,
|
||
/// Set carry flag.
|
||
STC,
|
||
/// Set decimal flag.
|
||
STD,
|
||
/// Set interrupt flag.
|
||
STI,
|
||
/// Complement carry flag; no source or destination provided.
|
||
CMC,
|
||
|
||
/// Compare; source, destination, operand and displacement will be populated appropriately.
|
||
CMP,
|
||
/// Sets flags based on the result of a logical AND of source and destination.
|
||
TEST,
|
||
|
||
/// Exchanges the contents of the source and destination.
|
||
XCHG,
|
||
|
||
/// Load AL with DS:[AL+BX].
|
||
XLAT,
|
||
|
||
// TODO: expand detail on all operations below.
|
||
|
||
//
|
||
// 80186 additions.
|
||
//
|
||
|
||
/// Checks an array index against bounds.
|
||
BOUND,
|
||
|
||
/// Create stack frame.
|
||
ENTER,
|
||
/// Procedure exit; copies BP to SP, then pops a new BP from the stack.
|
||
LEAVE,
|
||
|
||
/// Inputs from a port, incrementing or decrementing the destination.
|
||
INS,
|
||
/// Outputs to a port, incrementing or decrementing the destination.
|
||
OUTS,
|
||
|
||
/// Pushes all general purpose registers to the stack, in the order:
|
||
/// AX, CX, DX, BX, [original] SP, BP, SI, DI.
|
||
PUSHA,
|
||
/// Pops all general purpose registers from the stack, in the reverse of
|
||
/// the PUSHA order, i.e. DI, SI, BP, [final] SP, BX, DX, CX, AX.
|
||
POPA,
|
||
|
||
//
|
||
// 80286 additions.
|
||
//
|
||
|
||
/// Adjusts requested privilege level.
|
||
ARPL,
|
||
/// Clears the task-switched flag.
|
||
CLTS,
|
||
/// Loads access rights.
|
||
LAR,
|
||
|
||
/// Loads the global descriptor table.
|
||
LGDT,
|
||
/// Loads the interrupt descriptor table.
|
||
LIDT,
|
||
/// Loads the local descriptor table.
|
||
LLDT,
|
||
/// Stores the global descriptor table.
|
||
SGDT,
|
||
/// Stores the interrupt descriptor table.
|
||
SIDT,
|
||
/// Stores the local descriptor table.
|
||
SLDT,
|
||
|
||
/// Verifies a segment for reading.
|
||
VERR,
|
||
/// Verifies a segment for writing.
|
||
VERW,
|
||
|
||
/// Loads the machine status word.
|
||
LMSW,
|
||
/// Stores the machine status word.
|
||
SMSW,
|
||
/// Loads a segment limit
|
||
LSL,
|
||
/// Loads the task register.
|
||
LTR,
|
||
/// Stores the task register.
|
||
STR,
|
||
|
||
/// Three-operand form of IMUL; multiply the immediate by the source and write to the destination.
|
||
IMUL_3,
|
||
|
||
/// Undocumented (but used); loads all registers, including internal ones.
|
||
LOADALL,
|
||
|
||
//
|
||
// 80386 additions.
|
||
//
|
||
|
||
/// Loads a pointer to FS.
|
||
LFS,
|
||
/// Loads a pointer to GS.
|
||
LGS,
|
||
/// Loads a pointer to SS.
|
||
LSS,
|
||
|
||
/// Shift left double.
|
||
SHLDimm,
|
||
SHLDCL,
|
||
/// Shift right double.
|
||
SHRDimm,
|
||
SHRDCL,
|
||
|
||
/// Bit scan forwards.
|
||
BSF,
|
||
/// Bit scan reverse.
|
||
BSR,
|
||
/// Bit test.
|
||
BT,
|
||
/// Bit test and complement.
|
||
BTC,
|
||
/// Bit test and reset.
|
||
BTR,
|
||
/// Bit test and set.
|
||
BTS,
|
||
|
||
/// Compare string double word.
|
||
CMPSD,
|
||
/// [Early 80386s only] Insert bit string.
|
||
IBTS,
|
||
|
||
/// Inputs a double word from a port, incrementing or decrementing the destination.
|
||
INSD,
|
||
|
||
/// Convert dword to qword.
|
||
CDQ,
|
||
/// Convert word to dword; AX will be expanded to fill EAX.
|
||
/// Compare and contrast to CWD which would expand AX to DX:AX.
|
||
CWDE,
|
||
|
||
/// Move with zero extension.
|
||
MOVZX,
|
||
/// Move with sign extension.
|
||
MOVSX,
|
||
|
||
/// Two-operand form of IMUL; multiply the source by the destination and write to the destination.
|
||
IMUL_2,
|
||
|
||
SETO, SETNO, SETB, SETNB, SETZ, SETNZ, SETBE, SETNBE,
|
||
SETS, SETNS, SETP, SETNP, SETL, SETNL, SETLE, SETNLE,
|
||
|
||
MOVtoCr, MOVfromCr,
|
||
MOVtoDr, MOVfromDr,
|
||
MOVtoTr, MOVfromTr,
|
||
};
|
||
|
||
enum class DataSize: uint8_t {
|
||
Byte = 0,
|
||
Word = 1,
|
||
DWord = 2,
|
||
None = 3,
|
||
};
|
||
|
||
constexpr int byte_size(DataSize size) {
|
||
return (1 << int(size)) & 7;
|
||
}
|
||
|
||
constexpr int bit_size(DataSize size) {
|
||
return (8 << int(size)) & 0x3f;
|
||
}
|
||
|
||
enum class AddressSize: uint8_t {
|
||
b16 = 0,
|
||
b32 = 1,
|
||
};
|
||
|
||
constexpr DataSize data_size(AddressSize size) {
|
||
return DataSize(int(size) + 1);
|
||
}
|
||
|
||
constexpr int byte_size(AddressSize size) {
|
||
return 2 << int(size);
|
||
}
|
||
|
||
constexpr int bit_size(AddressSize size) {
|
||
return 16 << int(size);
|
||
}
|
||
|
||
enum class Source: uint8_t {
|
||
// These are in SIB order; this matters for packing later on.
|
||
|
||
/// AL, AX or EAX depending on size.
|
||
eAX,
|
||
/// CL, CX or ECX depending on size.
|
||
eCX,
|
||
/// DL, DX or EDX depending on size.
|
||
eDX,
|
||
/// BL, BX or BDX depending on size.
|
||
eBX,
|
||
/// AH if size is 1; SP or ESP otherwise.
|
||
eSPorAH,
|
||
/// CH if size is 1; BP or EBP otherwise.
|
||
eBPorCH,
|
||
/// DH if size is 1; SI or ESI otherwise.
|
||
eSIorDH,
|
||
/// BH if size is 1; DI or EDI otherwise.
|
||
eDIorBH,
|
||
|
||
// Aliases for the dual-purpose enums.
|
||
eSP = eSPorAH, AH = eSPorAH,
|
||
eBP = eBPorCH, CH = eBPorCH,
|
||
eSI = eSIorDH, DH = eSIorDH,
|
||
eDI = eDIorBH, BH = eDIorBH,
|
||
|
||
// Selectors.
|
||
ES, CS, SS, DS, FS, GS,
|
||
|
||
/// @c None can be treated as a source that produces 0 when encountered;
|
||
/// it is semantically valid to receive it with that meaning in some contexts —
|
||
/// e.g. to indicate no index in indirect addressing.
|
||
/// It's listed here in order to allow an [optional] segment override to fit into three bits.
|
||
None,
|
||
|
||
/// The address included within this instruction should be used as the source.
|
||
DirectAddress,
|
||
// TODO: is this better eliminated in favour of an indirect
|
||
// source with a base() and index() of 0?
|
||
|
||
/// The immediate value included within this instruction should be used as the source.
|
||
Immediate,
|
||
|
||
/// The ScaleIndexBase associated with this source should be used.
|
||
Indirect = 0b11000,
|
||
// Elsewhere, as an implementation detail, the low three bits of an indirect source
|
||
// are reused; (Indirect-1) is also used as a sentinel value but is not a valid member
|
||
// of the enum and isn't exposed externally.
|
||
};
|
||
constexpr Source SourceIndirectNoBase = Source(uint8_t(Source::Indirect) - 1);
|
||
|
||
enum class Repetition: uint8_t {
|
||
None, RepE, RepNE
|
||
};
|
||
|
||
/// Provides a 32-bit-style scale, index and base; to produce the address this represents,
|
||
/// calcluate base() + (index() << scale()).
|
||
///
|
||
/// This form of indirect addressing is used to describe both 16- and 32-bit indirect addresses,
|
||
/// even though it is a superset of that supported prior to the 80386.
|
||
///
|
||
/// This class can represent only exactly what a SIB byte can — a scale of 0 to 3, a base
|
||
/// that is any one of the eight general purpose registers, and an index that is one of the seven
|
||
/// general purpose registers excluding eSP or is ::None.
|
||
///
|
||
/// It cannot natively describe a base of ::None.
|
||
class ScaleIndexBase {
|
||
public:
|
||
constexpr ScaleIndexBase() noexcept {}
|
||
constexpr ScaleIndexBase(uint8_t sib) noexcept : sib_(sib) {}
|
||
constexpr ScaleIndexBase(int scale, Source index, Source base) noexcept : sib_(uint8_t(scale << 6 | (int(index != Source::None ? index : Source::eSI) << 3) | int(base))) {}
|
||
constexpr ScaleIndexBase(Source index, Source base) noexcept : ScaleIndexBase(0, index, base) {}
|
||
constexpr explicit ScaleIndexBase(Source base) noexcept : ScaleIndexBase(0, Source::None, base) {}
|
||
|
||
/// @returns the power of two by which to multiply @c index() before adding it to @c base().
|
||
constexpr int scale() const {
|
||
return sib_ >> 6;
|
||
}
|
||
|
||
/// @returns the @c index for this address; this is guaranteed to be one of eAX, eBX, eCX, eDX, None, eBP, eSI or eDI.
|
||
constexpr Source index() const {
|
||
constexpr Source sources[] = {
|
||
Source::eAX, Source::eCX, Source::eDX, Source::eBX, Source::None, Source::eBP, Source::eSI, Source::eDI,
|
||
};
|
||
static_assert(sizeof(sources) == 8);
|
||
return sources[(sib_ >> 3) & 0x7];
|
||
}
|
||
|
||
/// @returns the @c base for this address; this is guaranteed to be one of eAX, eBX, eCX, eDX, eSP, eBP, eSI or eDI.
|
||
constexpr Source base() const {
|
||
return Source(sib_ & 0x7);
|
||
}
|
||
|
||
bool operator ==(const ScaleIndexBase &rhs) const {
|
||
// Permit either exact equality or index and base being equal
|
||
// but transposed with a scale of 1.
|
||
return
|
||
(sib_ == rhs.sib_) ||
|
||
(
|
||
!scale() && !rhs.scale() &&
|
||
rhs.index() == base() &&
|
||
rhs.base() == index()
|
||
);
|
||
}
|
||
|
||
private:
|
||
// Data is stored directly as an 80386 SIB byte.
|
||
uint8_t sib_ = 0;
|
||
};
|
||
static_assert(sizeof(ScaleIndexBase) == 1);
|
||
static_assert(alignof(ScaleIndexBase) == 1);
|
||
|
||
/// Provides the location of an operand's source or destination.
|
||
///
|
||
/// Callers should use .source() as a first point of entry. If it directly nominates a register
|
||
/// then use the register contents directly. If it indicates ::DirectAddress or ::Immediate
|
||
/// then ask the instruction for the address or immediate value that was provided in
|
||
/// the instruction.
|
||
///
|
||
/// If .source() indicates ::Indirect then use base(), index() and scale() to construct an address.
|
||
///
|
||
/// In all cases, the applicable segment is indicated by the instruction.
|
||
class DataPointer {
|
||
public:
|
||
/// Constricts a DataPointer referring to the given source; it shouldn't be ::Indirect.
|
||
constexpr DataPointer(Source source) noexcept : source_(source) {}
|
||
|
||
/// Constricts a DataPointer with a source of ::Indirect and the specified sib.
|
||
constexpr DataPointer(ScaleIndexBase sib) noexcept : sib_(sib) {}
|
||
|
||
/// Constructs a DataPointer with a source and SIB; use the source to indicate
|
||
/// whether the base field of the SIB is effective.
|
||
constexpr DataPointer(Source source, ScaleIndexBase sib) noexcept : source_(source), sib_(sib) {}
|
||
|
||
/// Constructs an indirect DataPointer referencing the given base, index and scale.
|
||
constexpr DataPointer(Source base, Source index, int scale) noexcept :
|
||
source_(base != Source::None ? Source::Indirect : SourceIndirectNoBase),
|
||
sib_(scale, index, base) {}
|
||
|
||
constexpr bool operator ==(const DataPointer &rhs) const {
|
||
// Require a SIB match only if source_ is ::Indirect or ::IndirectNoBase.
|
||
return source_ == rhs.source_ && (source_ < SourceIndirectNoBase || sib_ == rhs.sib_);
|
||
}
|
||
|
||
// TODO: determine whether conditionals below
|
||
// have introduced branching.
|
||
|
||
constexpr Source source() const {
|
||
return (source_ >= SourceIndirectNoBase) ? Source::Indirect : source_;
|
||
}
|
||
|
||
constexpr int scale() const {
|
||
return sib_.scale();
|
||
}
|
||
|
||
constexpr Source index() const {
|
||
return sib_.index();
|
||
}
|
||
|
||
constexpr Source base() const {
|
||
return (source_ <= SourceIndirectNoBase) ? Source::None : sib_.base();
|
||
}
|
||
|
||
private:
|
||
Source source_ = Source::Indirect;
|
||
ScaleIndexBase sib_;
|
||
};
|
||
|
||
template<bool is_32bit> class Instruction {
|
||
public:
|
||
Operation operation = Operation::Invalid;
|
||
|
||
bool operator ==(const Instruction &rhs) const {
|
||
return
|
||
repetition_size_ == rhs.repetition_size_ &&
|
||
sources_ == rhs.sources_ &&
|
||
displacement_ == rhs.displacement_ &&
|
||
operand_ == rhs.operand_ &&
|
||
sib_ == rhs.sib_;
|
||
}
|
||
|
||
using DisplacementT = typename std::conditional<is_32bit, int32_t, int16_t>::type;
|
||
using ImmediateT = typename std::conditional<is_32bit, uint32_t, uint16_t>::type;
|
||
using AddressT = ImmediateT;
|
||
|
||
/* Note to self — current thinking is:
|
||
|
||
First 32bits:
|
||
5 bits source;
|
||
5 bits dest;
|
||
5 bits partial SIB, combined with three low bits of source or dest if indirect;
|
||
8 bits operation;
|
||
4 bits original instruction size;
|
||
2 bits data size;
|
||
1 bit memory size;
|
||
2 bits extension flags.
|
||
|
||
Extensions (16 or 32 bit, depending on templated size):
|
||
1) reptition + segment override + lock + original instruction size (= 10 bits);
|
||
2) displacement;
|
||
3) immediate operand.
|
||
|
||
Presence or absence of extensions is dictated by:
|
||
* instruction size = 0 => the repetition, etc extension (including the real extension size); and
|
||
* the extension flags for displacement and/or immediate.
|
||
|
||
Therefore an instruction's footprint is:
|
||
* 4–8 bytes (16-bit processors);
|
||
* 4–12 bytes (32-bit processors).
|
||
|
||
I'll then implement a collection suited to packing these things based on their
|
||
packing_size(), and later iterating them.
|
||
|
||
To verify: the 8086 allows unlimited-length instructions (which I'll probably handle by
|
||
generating length-15 NOPs and not resetting parser state), the 80386 limits them to
|
||
15 bytes, but what do the processors in between do?
|
||
*/
|
||
|
||
private:
|
||
// b0, b1: a Repetition;
|
||
// b2+: operation size.
|
||
uint8_t repetition_size_ = 0;
|
||
|
||
// b0–b5: source;
|
||
// b6–b11: destination;
|
||
// b12–b14: segment override;
|
||
// b15: lock.
|
||
uint16_t sources_ = 0;
|
||
|
||
// Unpackable fields.
|
||
DisplacementT displacement_ = 0;
|
||
ImmediateT operand_ = 0; // ... or used to store a segment for far operations.
|
||
|
||
// Fields yet to be properly incorporated...
|
||
ScaleIndexBase sib_;
|
||
AddressSize address_size_ = AddressSize::b16;
|
||
|
||
public:
|
||
/// @returns The number of bytes used for meaningful content within this class. A receiver must use at least @c sizeof(Instruction) bytes
|
||
/// to store an @c Instruction but is permitted to reuse the trailing sizeof(Instruction) - packing_size() for any purpose it likes. Teleologically,
|
||
/// this allows a denser packing of instructions into containers.
|
||
size_t packing_size() const { return sizeof(*this); /* TODO */ }
|
||
|
||
DataPointer source() const { return DataPointer(Source(sources_ & 0x3f), sib_); }
|
||
DataPointer destination() const { return DataPointer(Source((sources_ >> 6) & 0x3f), sib_); }
|
||
bool lock() const { return sources_ & 0x8000; }
|
||
|
||
AddressSize address_size() const {
|
||
return AddressSize(address_size_);
|
||
}
|
||
Source data_segment() const {
|
||
const auto segment_override = Source(
|
||
int(Source::ES) +
|
||
((sources_ >> 12) & 7)
|
||
);
|
||
if(segment_override != Source::None) return segment_override;
|
||
|
||
// TODO: default source should be SS for anything touching the stack.
|
||
return Source::DS;
|
||
}
|
||
|
||
Repetition repetition() const { return Repetition(repetition_size_ & 3); }
|
||
DataSize operation_size() const { return DataSize(repetition_size_ >> 2); }
|
||
|
||
// TODO: confirm whether far call for some reason makes these 32-bit in protected mode.
|
||
uint16_t segment() const { return uint16_t(operand_); }
|
||
uint16_t offset() const { return uint16_t(displacement_); }
|
||
|
||
DisplacementT displacement() const { return displacement_; }
|
||
ImmediateT operand() const { return operand_; }
|
||
|
||
Instruction() noexcept {}
|
||
Instruction(
|
||
Operation operation,
|
||
Source source,
|
||
Source destination,
|
||
ScaleIndexBase sib,
|
||
bool lock,
|
||
AddressSize address_size,
|
||
Source segment_override,
|
||
Repetition repetition,
|
||
DataSize operation_size,
|
||
DisplacementT displacement,
|
||
ImmediateT operand) noexcept :
|
||
operation(operation),
|
||
repetition_size_(uint8_t((int(operation_size) << 2) | int(repetition))),
|
||
sources_(uint16_t(
|
||
int(source) |
|
||
(int(destination) << 6) |
|
||
((int(segment_override) & 7) << 12) |
|
||
(int(lock) << 15)
|
||
)),
|
||
displacement_(displacement),
|
||
operand_(operand),
|
||
sib_(sib),
|
||
address_size_(address_size) {}
|
||
};
|
||
|
||
// TODO: repack.
|
||
//static_assert(sizeof(Instruction) <= 8);
|
||
|
||
}
|
||
}
|
||
|
||
#endif /* InstructionSets_x86_Instruction_h */
|