diff options
author | Oxore <oxore@protonmail.com> | 2023-05-01 12:37:48 +0300 |
---|---|---|
committer | Oxore <oxore@protonmail.com> | 2023-05-01 12:37:48 +0300 |
commit | 666b07b0d61d10834533fcbdd870080d9ad1948d (patch) | |
tree | bd2ad841ec741527ed75edca5075a5220835db9c /disasm.cpp | |
parent | 2a4c7440706e57887c0d92397dc5aafce0a3932e (diff) |
Fix MOVEM boundary check
Diffstat (limited to 'disasm.cpp')
-rw-r--r-- | disasm.cpp | 21 |
1 files changed, 9 insertions, 12 deletions
@@ -385,14 +385,18 @@ static void disasm_jsr_jmp( static void disasm_movem( DisasmNode& node, uint16_t instr, const DataBuffer &code, const Settings &s) { + if (node.offset + kInstructionSizeStepBytes >= code.occupied_size) { + // Not enough space for regmask + return disasm_verbatim(node, instr, code, s); + } + const unsigned regmask = GetU16BE(code.buffer + node.offset + kInstructionSizeStepBytes); + if (regmask == 0) { + // This is just not representable: at least one register must be specified + return disasm_verbatim(node, instr, code, s); + } const auto dir = static_cast<MoveDirection>((instr >> 10) & 1); const auto opsize = static_cast<OpSize>(((instr >> 6) & 1) + 1); const char suffix = suffix_from_opsize(opsize); - // Although it would be much mode logical to fetch register mask first, - // since it goes right next after the instruction, but fetching addressing - // mode register first provides us the ultimate boundary check with early - // return, so we don't have to check for node.occupied_size when fetching - // regmask after this. const auto a = AddrModeArg::Fetch( node.offset + kInstructionSizeStepBytes * 2, code, instr, suffix); switch (a.mode) { @@ -426,13 +430,6 @@ static void disasm_movem( case AddrMode::kImmediate: // 4ebc / 4efc return disasm_verbatim(node, instr, code, s); } - // Make sure that regmask is fetched after AddrModeArg has done boundary - // check. - const unsigned regmask = GetU16BE(code.buffer + node.offset + kInstructionSizeStepBytes); - if (regmask == 0) { - // This is just not representable: at least one register must be specified - return disasm_verbatim(node, instr, code, s); - } node.size = kInstructionSizeStepBytes * 2 + a.Size(); snprintf(node.mnemonic, kMnemonicBufferSize, "movem%c", suffix); char regmask_str[48]{}; |