diff --git a/InstructionSets/x86/DataPointerResolver.hpp b/InstructionSets/x86/DataPointerResolver.hpp index a532fda16..028984e37 100644 --- a/InstructionSets/x86/DataPointerResolver.hpp +++ b/InstructionSets/x86/DataPointerResolver.hpp @@ -277,9 +277,9 @@ template void DataPointerResolver(instruction.data_segment(), instruction.displacement()); + value = memory.template read(instruction.segment_override(), instruction.displacement()); } break; case Source::Immediate: @@ -295,13 +295,13 @@ template void DataPointerResolver( \ - instruction.data_segment(), \ + instruction.segment_override(), \ address \ ); \ } \ diff --git a/InstructionSets/x86/Implementation/PerformImplementation.hpp b/InstructionSets/x86/Implementation/PerformImplementation.hpp index 4f13b707d..15243c06f 100644 --- a/InstructionSets/x86/Implementation/PerformImplementation.hpp +++ b/InstructionSets/x86/Implementation/PerformImplementation.hpp @@ -281,7 +281,7 @@ template < // If execution has reached here then a memory fetch is required. // Do it and exit. - segment = Source::DS; // TODO. + segment = source.segment(instruction.segment_override()); fetched_data = original_data = memory.template read(segment, address); return fetched_data; }; diff --git a/InstructionSets/x86/Instruction.cpp b/InstructionSets/x86/Instruction.cpp index c247a4f11..fd0a9b8d5 100644 --- a/InstructionSets/x86/Instruction.cpp +++ b/InstructionSets/x86/Instruction.cpp @@ -383,7 +383,7 @@ std::string InstructionSet::x86::to_string( stream << InstructionSet::x86::to_string(operation_size) << ' '; } - Source segment = instruction.data_segment(); + Source segment = instruction.segment_override(); if(segment == Source::None) { segment = pointer.default_segment(); if(segment == Source::None) { diff --git a/InstructionSets/x86/Instruction.hpp b/InstructionSets/x86/Instruction.hpp index a22796244..3bc2b14a4 100644 --- a/InstructionSets/x86/Instruction.hpp +++ b/InstructionSets/x86/Instruction.hpp @@ -625,6 +625,13 @@ class DataPointer { } } + constexpr Source segment(Source segment_override) const { + // TODO: remove conditionaluty here. + if(segment_override != Source::None) return segment_override; + if(const auto segment = default_segment(); segment != Source::None) return segment; + return Source::DS; + } + template constexpr Source base() const { if constexpr (obscure_indirectNoBase) { return (source_ <= Source::IndirectNoBase) ? Source::None : sib_.base(); @@ -766,7 +773,7 @@ template class Instruction { /// On x86 a segment override cannot modify the segment used as a destination in string instructions, /// or that used by stack instructions, but this function does not spend the time necessary to provide /// the correct default for those. - Source data_segment() const { + Source segment_override() const { if(!has_length_extension()) return Source::None; return Source( int(Source::ES) + diff --git a/OSBindings/Mac/Clock SignalTests/x86DecoderTests.mm b/OSBindings/Mac/Clock SignalTests/x86DecoderTests.mm index 7544cf22a..26bf5b2d2 100644 --- a/OSBindings/Mac/Clock SignalTests/x86DecoderTests.mm +++ b/OSBindings/Mac/Clock SignalTests/x86DecoderTests.mm @@ -411,7 +411,7 @@ decode(const std::initializer_list &stream, bool set_32_bit = false) { // add DWORD PTR [edi-0x42],0x9f683aa9 // lock jp 0xfffffff0 (from 0000000e) test(instructions[0], DataSize::DWord, Operation::INC, Source::eDX); - XCTAssertEqual(instructions[0].data_segment(), Source::CS); + XCTAssertEqual(instructions[0].segment_override(), Source::CS); test(instructions[1], DataSize::Byte, Operation::OR, Source::Immediate, Source::eAX, 0x9); test(instructions[2], DataSize::DWord, Operation::ADD, Source::Immediate, ScaleIndexBase(Source::eDI), 0x9f683aa9, -0x42); test(instructions[3], Operation::JP, 0, -30); @@ -422,7 +422,7 @@ decode(const std::initializer_list &stream, bool set_32_bit = false) { // stos BYTE PTR es:[edi],al // pusha test(instructions[4], DataSize::Byte, Operation::MOV, Source::Immediate, Source::AH, 0xc1); - XCTAssertEqual(instructions[4].data_segment(), Source::DS); + XCTAssertEqual(instructions[4].segment_override(), Source::DS); test(instructions[5], DataSize::Word, Operation::POP, Source::None, Source::DS); test(instructions[6], DataSize::Byte, Operation::STOS); test(instructions[7], Operation::PUSHA); @@ -465,7 +465,7 @@ decode(const std::initializer_list &stream, bool set_32_bit = false) { test(instructions[21], DataSize::Byte, Operation::XOR, Source::Immediate, Source::eAX, 0x45); test(instructions[22], DataSize::DWord, Operation::LDS, ScaleIndexBase(Source::eCX), Source::eDX); test(instructions[23], DataSize::Byte, Operation::MOV, Source::eAX, Source::DirectAddress, 0xe4dba6d3); - XCTAssertEqual(instructions[23].data_segment(), Source::DS); + XCTAssertEqual(instructions[23].segment_override(), Source::DS); // pop ds // movs DWORD PTR es:[edi],DWORD PTR ds:[esi]