2022-04-11 19:00:55 +00:00
|
|
|
|
//
|
|
|
|
|
// Instruction.hpp
|
|
|
|
|
// Clock Signal
|
|
|
|
|
//
|
|
|
|
|
// Created by Thomas Harte on 10/04/2022.
|
|
|
|
|
// Copyright © 2022 Thomas Harte. All rights reserved.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
#ifndef InstructionSets_68k_Instruction_hpp
|
|
|
|
|
#define InstructionSets_68k_Instruction_hpp
|
|
|
|
|
|
2022-04-15 13:40:37 +00:00
|
|
|
|
#include "Model.hpp"
|
2022-04-11 19:00:55 +00:00
|
|
|
|
|
2022-05-03 15:09:57 +00:00
|
|
|
|
#include <cassert>
|
|
|
|
|
#include <cstdint>
|
|
|
|
|
|
2022-04-11 19:00:55 +00:00
|
|
|
|
namespace InstructionSet {
|
|
|
|
|
namespace M68k {
|
|
|
|
|
|
|
|
|
|
enum class Operation: uint8_t {
|
|
|
|
|
Undefined,
|
|
|
|
|
|
2022-04-12 11:49:08 +00:00
|
|
|
|
NOP,
|
|
|
|
|
|
2022-04-11 19:00:55 +00:00
|
|
|
|
ABCD, SBCD, NBCD,
|
|
|
|
|
|
|
|
|
|
ADDb, ADDw, ADDl,
|
|
|
|
|
ADDAw, ADDAl,
|
|
|
|
|
ADDXb, ADDXw, ADDXl,
|
|
|
|
|
|
|
|
|
|
SUBb, SUBw, SUBl,
|
|
|
|
|
SUBAw, SUBAl,
|
|
|
|
|
SUBXb, SUBXw, SUBXl,
|
|
|
|
|
|
2022-04-13 13:29:12 +00:00
|
|
|
|
MOVEb, MOVEw, MOVEl,
|
2022-04-11 19:00:55 +00:00
|
|
|
|
MOVEAw, MOVEAl,
|
2022-04-19 23:36:21 +00:00
|
|
|
|
LEA, PEA,
|
2022-04-11 19:00:55 +00:00
|
|
|
|
|
|
|
|
|
MOVEtoSR, MOVEfromSR,
|
|
|
|
|
MOVEtoCCR,
|
2022-04-12 11:49:08 +00:00
|
|
|
|
MOVEtoUSP, MOVEfromUSP,
|
2022-04-11 19:00:55 +00:00
|
|
|
|
|
|
|
|
|
ORItoSR, ORItoCCR,
|
|
|
|
|
ANDItoSR, ANDItoCCR,
|
|
|
|
|
EORItoSR, EORItoCCR,
|
|
|
|
|
|
2022-04-15 19:33:54 +00:00
|
|
|
|
BTST, BCLR,
|
|
|
|
|
BCHG, BSET,
|
|
|
|
|
|
2022-04-11 19:00:55 +00:00
|
|
|
|
CMPb, CMPw, CMPl,
|
2022-04-12 11:49:08 +00:00
|
|
|
|
CMPAw, CMPAl,
|
2022-04-11 19:00:55 +00:00
|
|
|
|
TSTb, TSTw, TSTl,
|
|
|
|
|
|
2022-04-18 12:55:46 +00:00
|
|
|
|
JMP,
|
|
|
|
|
JSR, RTS,
|
2022-04-11 19:00:55 +00:00
|
|
|
|
DBcc,
|
|
|
|
|
Scc,
|
|
|
|
|
|
2022-04-26 16:30:14 +00:00
|
|
|
|
Bccb, Bccw, Bccl,
|
|
|
|
|
BSRb, BSRw, BSRl,
|
2022-04-18 12:55:46 +00:00
|
|
|
|
|
2022-04-11 19:00:55 +00:00
|
|
|
|
CLRb, CLRw, CLRl,
|
|
|
|
|
NEGXb, NEGXw, NEGXl,
|
|
|
|
|
NEGb, NEGw, NEGl,
|
|
|
|
|
|
|
|
|
|
ASLb, ASLw, ASLl, ASLm,
|
|
|
|
|
ASRb, ASRw, ASRl, ASRm,
|
|
|
|
|
LSLb, LSLw, LSLl, LSLm,
|
|
|
|
|
LSRb, LSRw, LSRl, LSRm,
|
|
|
|
|
ROLb, ROLw, ROLl, ROLm,
|
|
|
|
|
RORb, RORw, RORl, RORm,
|
|
|
|
|
ROXLb, ROXLw, ROXLl, ROXLm,
|
|
|
|
|
ROXRb, ROXRw, ROXRl, ROXRm,
|
|
|
|
|
|
2022-05-06 13:45:06 +00:00
|
|
|
|
MOVEMtoRl, MOVEMtoRw,
|
|
|
|
|
MOVEMtoMl, MOVEMtoMw,
|
|
|
|
|
|
2022-04-12 20:17:30 +00:00
|
|
|
|
MOVEPl, MOVEPw,
|
2022-04-11 19:00:55 +00:00
|
|
|
|
|
|
|
|
|
ANDb, ANDw, ANDl,
|
|
|
|
|
EORb, EORw, EORl,
|
|
|
|
|
NOTb, NOTw, NOTl,
|
|
|
|
|
ORb, ORw, ORl,
|
|
|
|
|
|
|
|
|
|
MULU, MULS,
|
|
|
|
|
DIVU, DIVS,
|
|
|
|
|
|
2022-04-12 11:49:08 +00:00
|
|
|
|
RTE, RTR,
|
2022-04-11 19:00:55 +00:00
|
|
|
|
|
|
|
|
|
TRAP, TRAPV,
|
|
|
|
|
CHK,
|
|
|
|
|
|
|
|
|
|
EXG, SWAP,
|
|
|
|
|
|
|
|
|
|
TAS,
|
|
|
|
|
|
|
|
|
|
EXTbtow, EXTwtol,
|
|
|
|
|
|
2022-04-15 13:40:37 +00:00
|
|
|
|
LINKw, UNLINK,
|
2022-04-11 19:00:55 +00:00
|
|
|
|
|
2022-04-12 11:49:08 +00:00
|
|
|
|
STOP, RESET,
|
2022-04-12 20:17:30 +00:00
|
|
|
|
|
|
|
|
|
Max = RESET
|
2022-04-11 19:00:55 +00:00
|
|
|
|
};
|
|
|
|
|
|
2022-04-15 13:40:37 +00:00
|
|
|
|
template <Model model>
|
|
|
|
|
constexpr bool requires_supervisor(Operation op) {
|
|
|
|
|
switch(op) {
|
2022-04-26 00:05:45 +00:00
|
|
|
|
case Operation::MOVEfromSR:
|
|
|
|
|
if constexpr (model == Model::M68000) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
[[fallthrough]];
|
|
|
|
|
case Operation::ORItoSR: case Operation::ANDItoSR:
|
|
|
|
|
case Operation::EORItoSR: case Operation::RTE:
|
|
|
|
|
case Operation::RESET: case Operation::STOP:
|
|
|
|
|
case Operation::MOVEtoUSP: case Operation::MOVEfromUSP:
|
2022-04-15 13:40:37 +00:00
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-26 16:30:14 +00:00
|
|
|
|
enum class DataSize {
|
|
|
|
|
Byte = 0,
|
|
|
|
|
Word = 1,
|
|
|
|
|
LongWord = 2,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/// Classifies operations by the size of their memory accesses, if any.
|
2022-05-05 22:51:29 +00:00
|
|
|
|
///
|
|
|
|
|
/// For any operations that don't fit the neat model of reading one or two operands,
|
|
|
|
|
/// then writing zero or one, the size determines the data size of the operands only,
|
|
|
|
|
/// not any other accesses.
|
2022-05-06 13:22:38 +00:00
|
|
|
|
constexpr DataSize operand_size(Operation operation) {
|
2022-04-13 13:29:12 +00:00
|
|
|
|
switch(operation) {
|
2022-04-26 16:30:14 +00:00
|
|
|
|
// These are given a value arbitrarily, to
|
|
|
|
|
// complete the switch statement.
|
|
|
|
|
case Operation::Undefined:
|
|
|
|
|
case Operation::NOP:
|
|
|
|
|
case Operation::STOP:
|
|
|
|
|
case Operation::RESET:
|
|
|
|
|
case Operation::RTE: case Operation::RTR:
|
|
|
|
|
case Operation::TRAP:
|
|
|
|
|
case Operation::TRAPV:
|
|
|
|
|
|
|
|
|
|
case Operation::ABCD: case Operation::SBCD:
|
|
|
|
|
case Operation::NBCD:
|
2022-04-13 13:29:12 +00:00
|
|
|
|
case Operation::ADDb: case Operation::ADDXb:
|
|
|
|
|
case Operation::SUBb: case Operation::SUBXb:
|
2022-04-26 16:30:14 +00:00
|
|
|
|
case Operation::MOVEb:
|
2022-04-15 13:40:37 +00:00
|
|
|
|
case Operation::ORItoCCR:
|
|
|
|
|
case Operation::ANDItoCCR:
|
|
|
|
|
case Operation::EORItoCCR:
|
2022-04-26 16:30:14 +00:00
|
|
|
|
case Operation::BTST: case Operation::BCLR:
|
|
|
|
|
case Operation::BCHG: case Operation::BSET:
|
|
|
|
|
case Operation::CMPb: case Operation::TSTb:
|
|
|
|
|
case Operation::Bccb: case Operation::BSRb:
|
|
|
|
|
case Operation::CLRb:
|
2022-05-03 18:40:51 +00:00
|
|
|
|
case Operation::Scc:
|
2022-04-26 16:30:14 +00:00
|
|
|
|
case Operation::NEGXb: case Operation::NEGb:
|
|
|
|
|
case Operation::ASLb: case Operation::ASRb:
|
|
|
|
|
case Operation::LSLb: case Operation::LSRb:
|
|
|
|
|
case Operation::ROLb: case Operation::RORb:
|
|
|
|
|
case Operation::ROXLb: case Operation::ROXRb:
|
|
|
|
|
case Operation::ANDb: case Operation::EORb:
|
|
|
|
|
case Operation::NOTb: case Operation::ORb:
|
|
|
|
|
case Operation::TAS:
|
|
|
|
|
return DataSize::Byte;
|
|
|
|
|
|
|
|
|
|
case Operation::ADDw: case Operation::ADDAw:
|
|
|
|
|
case Operation::ADDXw: case Operation::SUBw:
|
|
|
|
|
case Operation::SUBAw: case Operation::SUBXw:
|
|
|
|
|
case Operation::MOVEw: case Operation::MOVEAw:
|
2022-04-15 13:40:37 +00:00
|
|
|
|
case Operation::ORItoSR:
|
|
|
|
|
case Operation::ANDItoSR:
|
|
|
|
|
case Operation::EORItoSR:
|
2022-04-26 16:30:14 +00:00
|
|
|
|
case Operation::MOVEtoSR:
|
|
|
|
|
case Operation::MOVEfromSR:
|
|
|
|
|
case Operation::MOVEtoCCR:
|
|
|
|
|
case Operation::CMPw: case Operation::CMPAw:
|
|
|
|
|
case Operation::TSTw:
|
2022-05-03 18:40:51 +00:00
|
|
|
|
case Operation::DBcc:
|
2022-04-26 16:30:14 +00:00
|
|
|
|
case Operation::Bccw: case Operation::BSRw:
|
|
|
|
|
case Operation::CLRw:
|
|
|
|
|
case Operation::NEGXw: case Operation::NEGw:
|
|
|
|
|
case Operation::ASLw: case Operation::ASLm:
|
|
|
|
|
case Operation::ASRw: case Operation::ASRm:
|
|
|
|
|
case Operation::LSLw: case Operation::LSLm:
|
|
|
|
|
case Operation::LSRw: case Operation::LSRm:
|
|
|
|
|
case Operation::ROLw: case Operation::ROLm:
|
|
|
|
|
case Operation::RORw: case Operation::RORm:
|
|
|
|
|
case Operation::ROXLw: case Operation::ROXLm:
|
|
|
|
|
case Operation::ROXRw: case Operation::ROXRm:
|
2022-05-06 13:45:06 +00:00
|
|
|
|
case Operation::MOVEMtoRw:
|
|
|
|
|
case Operation::MOVEMtoRl:
|
|
|
|
|
case Operation::MOVEMtoMw:
|
|
|
|
|
case Operation::MOVEMtoMl:
|
2022-04-26 16:30:14 +00:00
|
|
|
|
case Operation::MOVEPw:
|
|
|
|
|
case Operation::ANDw: case Operation::EORw:
|
|
|
|
|
case Operation::NOTw: case Operation::ORw:
|
|
|
|
|
case Operation::DIVU: case Operation::DIVS:
|
|
|
|
|
case Operation::MULU: case Operation::MULS:
|
|
|
|
|
case Operation::EXTbtow:
|
|
|
|
|
case Operation::LINKw:
|
2022-05-08 01:16:44 +00:00
|
|
|
|
case Operation::CHK:
|
2022-04-26 16:30:14 +00:00
|
|
|
|
return DataSize::Word;
|
|
|
|
|
|
|
|
|
|
case Operation::ADDl: case Operation::ADDAl:
|
|
|
|
|
case Operation::ADDXl: case Operation::SUBl:
|
|
|
|
|
case Operation::SUBAl: case Operation::SUBXl:
|
|
|
|
|
case Operation::MOVEl: case Operation::MOVEAl:
|
|
|
|
|
case Operation::LEA: case Operation::PEA:
|
|
|
|
|
case Operation::EXG: case Operation::SWAP:
|
|
|
|
|
case Operation::MOVEtoUSP:
|
|
|
|
|
case Operation::MOVEfromUSP:
|
|
|
|
|
case Operation::CMPl: case Operation::CMPAl:
|
|
|
|
|
case Operation::TSTl:
|
|
|
|
|
case Operation::JMP: case Operation::JSR:
|
|
|
|
|
case Operation::RTS:
|
|
|
|
|
case Operation::Bccl: case Operation::BSRl:
|
|
|
|
|
case Operation::CLRl:
|
|
|
|
|
case Operation::NEGXl: case Operation::NEGl:
|
|
|
|
|
case Operation::ASLl: case Operation::ASRl:
|
|
|
|
|
case Operation::LSLl: case Operation::LSRl:
|
|
|
|
|
case Operation::ROLl: case Operation::RORl:
|
|
|
|
|
case Operation::ROXLl: case Operation::ROXRl:
|
|
|
|
|
case Operation::MOVEPl:
|
|
|
|
|
case Operation::ANDl: case Operation::EORl:
|
|
|
|
|
case Operation::NOTl: case Operation::ORl:
|
|
|
|
|
case Operation::EXTwtol:
|
|
|
|
|
case Operation::UNLINK:
|
|
|
|
|
return DataSize::LongWord;
|
2022-04-13 13:29:12 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-05-03 15:09:57 +00:00
|
|
|
|
template <Operation t_op = Operation::Undefined>
|
|
|
|
|
constexpr uint32_t quick(uint16_t instruction, Operation r_op = Operation::Undefined) {
|
|
|
|
|
switch((t_op != Operation::Undefined) ? t_op : r_op) {
|
2022-04-21 20:05:00 +00:00
|
|
|
|
case Operation::Bccb:
|
2022-04-21 20:13:06 +00:00
|
|
|
|
case Operation::BSRb:
|
2022-04-26 16:30:14 +00:00
|
|
|
|
case Operation::MOVEl: return uint32_t(int8_t(instruction));
|
|
|
|
|
case Operation::TRAP: return uint32_t(instruction & 15);
|
2022-04-21 20:05:00 +00:00
|
|
|
|
default: {
|
2022-04-26 16:30:14 +00:00
|
|
|
|
uint32_t value = (instruction >> 9) & 7;
|
2022-04-21 20:05:00 +00:00
|
|
|
|
value |= (value - 1)&8;
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-05-03 15:09:57 +00:00
|
|
|
|
static constexpr uint8_t FetchOp1 = (1 << 0);
|
|
|
|
|
static constexpr uint8_t FetchOp2 = (1 << 1);
|
|
|
|
|
static constexpr uint8_t StoreOp1 = (1 << 2);
|
|
|
|
|
static constexpr uint8_t StoreOp2 = (1 << 3);
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
|
Provides a bitfield with a value in the range 0–15 indicating which of the provided operation's
|
|
|
|
|
operands are accessed via standard fetch and store cycles.
|
2022-04-22 14:06:39 +00:00
|
|
|
|
|
2022-05-03 15:09:57 +00:00
|
|
|
|
Unusual bus sequences, such as TAS or MOVEM, are not described here.
|
|
|
|
|
*/
|
|
|
|
|
template <Model model, Operation t_operation = Operation::Undefined> uint8_t operand_flags(Operation r_operation = Operation::Undefined) {
|
|
|
|
|
switch((t_operation != Operation::Undefined) ? t_operation : r_operation) {
|
2022-04-22 14:06:39 +00:00
|
|
|
|
default:
|
2022-05-03 15:09:57 +00:00
|
|
|
|
assert(false);
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// No operands are fetched or stored.
|
2022-05-03 19:49:55 +00:00
|
|
|
|
// (which means that source and destination will appear as their effective addresses)
|
2022-05-03 15:09:57 +00:00
|
|
|
|
//
|
2022-05-04 00:45:36 +00:00
|
|
|
|
case Operation::PEA:
|
2022-05-06 13:45:06 +00:00
|
|
|
|
case Operation::JMP: case Operation::JSR:
|
|
|
|
|
case Operation::MOVEPw: case Operation::MOVEPl:
|
|
|
|
|
case Operation::MOVEMtoMw: case Operation::MOVEMtoMl:
|
|
|
|
|
case Operation::MOVEMtoRw: case Operation::MOVEMtoRl:
|
2022-05-06 16:18:56 +00:00
|
|
|
|
case Operation::TAS:
|
2022-05-06 16:30:49 +00:00
|
|
|
|
case Operation::RTR: case Operation::RTS: case Operation::RTE:
|
2022-05-03 15:09:57 +00:00
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// Single-operand read.
|
|
|
|
|
//
|
|
|
|
|
case Operation::MOVEtoSR: case Operation::MOVEtoCCR: case Operation::MOVEtoUSP:
|
|
|
|
|
case Operation::ORItoSR: case Operation::ORItoCCR:
|
|
|
|
|
case Operation::ANDItoSR: case Operation::ANDItoCCR:
|
|
|
|
|
case Operation::EORItoSR: case Operation::EORItoCCR:
|
2022-05-03 18:45:49 +00:00
|
|
|
|
case Operation::Bccb: case Operation::Bccw: case Operation::Bccl:
|
2022-05-03 19:32:54 +00:00
|
|
|
|
case Operation::BSRb: case Operation::BSRw: case Operation::BSRl:
|
2022-05-06 16:18:56 +00:00
|
|
|
|
case Operation::TSTb: case Operation::TSTw: case Operation::TSTl:
|
2022-05-03 15:09:57 +00:00
|
|
|
|
return FetchOp1;
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// Single-operand write.
|
|
|
|
|
//
|
|
|
|
|
case Operation::MOVEfromSR: case Operation::MOVEfromUSP:
|
2022-05-03 18:23:57 +00:00
|
|
|
|
case Operation::Scc:
|
2022-05-03 15:09:57 +00:00
|
|
|
|
return StoreOp1;
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// Single-operand read-modify-write.
|
|
|
|
|
//
|
|
|
|
|
case Operation::NBCD:
|
2022-05-03 19:40:04 +00:00
|
|
|
|
case Operation::NOTb: case Operation::NOTw: case Operation::NOTl:
|
|
|
|
|
case Operation::NEGb: case Operation::NEGw: case Operation::NEGl:
|
|
|
|
|
case Operation::NEGXb: case Operation::NEGXw: case Operation::NEGXl:
|
2022-05-03 15:09:57 +00:00
|
|
|
|
case Operation::EXTbtow: case Operation::EXTwtol:
|
2022-05-03 19:49:55 +00:00
|
|
|
|
case Operation::SWAP:
|
2022-05-04 12:41:55 +00:00
|
|
|
|
case Operation::UNLINK:
|
2022-05-04 23:44:59 +00:00
|
|
|
|
case Operation::ASLm: case Operation::ASRm:
|
|
|
|
|
case Operation::LSLm: case Operation::LSRm:
|
|
|
|
|
case Operation::ROLm: case Operation::RORm:
|
|
|
|
|
case Operation::ROXLm: case Operation::ROXRm:
|
2022-05-03 15:09:57 +00:00
|
|
|
|
return FetchOp1 | StoreOp1;
|
|
|
|
|
|
2022-05-03 19:40:04 +00:00
|
|
|
|
//
|
|
|
|
|
// CLR, which is model-dependent.
|
|
|
|
|
//
|
|
|
|
|
case Operation::CLRb: case Operation::CLRw: case Operation::CLRl:
|
|
|
|
|
if constexpr (model == Model::M68000) {
|
|
|
|
|
return FetchOp1 | StoreOp1;
|
|
|
|
|
} else {
|
|
|
|
|
return StoreOp1;
|
|
|
|
|
}
|
|
|
|
|
|
2022-05-03 15:09:57 +00:00
|
|
|
|
//
|
|
|
|
|
// Two-operand; read both.
|
|
|
|
|
//
|
|
|
|
|
case Operation::CMPb: case Operation::CMPw: case Operation::CMPl:
|
|
|
|
|
case Operation::CMPAw: case Operation::CMPAl:
|
2022-05-03 19:40:04 +00:00
|
|
|
|
case Operation::CHK:
|
2022-05-05 16:32:15 +00:00
|
|
|
|
case Operation::BTST:
|
2022-05-07 12:15:26 +00:00
|
|
|
|
case Operation::LINKw:
|
2022-05-03 15:09:57 +00:00
|
|
|
|
return FetchOp1 | FetchOp2;
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// Two-operand; read source, write dest.
|
|
|
|
|
//
|
|
|
|
|
case Operation::MOVEb: case Operation::MOVEw: case Operation::MOVEl:
|
|
|
|
|
case Operation::MOVEAw: case Operation::MOVEAl:
|
|
|
|
|
return FetchOp1 | StoreOp2;
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// Two-operand; read both, write dest.
|
|
|
|
|
//
|
|
|
|
|
case Operation::ABCD: case Operation::SBCD:
|
|
|
|
|
case Operation::ADDb: case Operation::ADDw: case Operation::ADDl:
|
|
|
|
|
case Operation::ADDAw: case Operation::ADDAl:
|
|
|
|
|
case Operation::ADDXb: case Operation::ADDXw: case Operation::ADDXl:
|
|
|
|
|
case Operation::SUBb: case Operation::SUBw: case Operation::SUBl:
|
|
|
|
|
case Operation::SUBAw: case Operation::SUBAl:
|
|
|
|
|
case Operation::SUBXb: case Operation::SUBXw: case Operation::SUBXl:
|
|
|
|
|
case Operation::ORb: case Operation::ORw: case Operation::ORl:
|
|
|
|
|
case Operation::ANDb: case Operation::ANDw: case Operation::ANDl:
|
|
|
|
|
case Operation::EORb: case Operation::EORw: case Operation::EORl:
|
2022-05-03 19:49:55 +00:00
|
|
|
|
case Operation::DIVU: case Operation::DIVS:
|
|
|
|
|
case Operation::MULU: case Operation::MULS:
|
2022-05-04 23:44:59 +00:00
|
|
|
|
case Operation::ASLb: case Operation::ASLw: case Operation::ASLl:
|
|
|
|
|
case Operation::ASRb: case Operation::ASRw: case Operation::ASRl:
|
|
|
|
|
case Operation::LSLb: case Operation::LSLw: case Operation::LSLl:
|
|
|
|
|
case Operation::LSRb: case Operation::LSRw: case Operation::LSRl:
|
|
|
|
|
case Operation::ROLb: case Operation::ROLw: case Operation::ROLl:
|
|
|
|
|
case Operation::RORb: case Operation::RORw: case Operation::RORl:
|
|
|
|
|
case Operation::ROXLb: case Operation::ROXLw: case Operation::ROXLl:
|
|
|
|
|
case Operation::ROXRb: case Operation::ROXRw: case Operation::ROXRl:
|
2022-05-05 16:32:15 +00:00
|
|
|
|
case Operation::BCHG:
|
2022-05-05 00:57:22 +00:00
|
|
|
|
case Operation::BCLR: case Operation::BSET:
|
2022-05-03 15:09:57 +00:00
|
|
|
|
return FetchOp1 | FetchOp2 | StoreOp2;
|
|
|
|
|
|
2022-05-03 18:40:51 +00:00
|
|
|
|
//
|
|
|
|
|
// Two-operand; read both, write source.
|
|
|
|
|
//
|
|
|
|
|
case Operation::DBcc:
|
|
|
|
|
return FetchOp1 | FetchOp2 | StoreOp1;
|
|
|
|
|
|
2022-05-03 15:09:57 +00:00
|
|
|
|
//
|
|
|
|
|
// Two-operand; read both, write both.
|
|
|
|
|
//
|
|
|
|
|
case Operation::EXG:
|
|
|
|
|
return FetchOp1 | FetchOp2 | StoreOp1 | StoreOp2;
|
2022-05-04 00:45:36 +00:00
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// Two-operand; just write destination.
|
|
|
|
|
//
|
|
|
|
|
case Operation::LEA:
|
|
|
|
|
return StoreOp2;
|
2022-04-21 20:05:00 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-30 13:00:47 +00:00
|
|
|
|
enum class Condition {
|
|
|
|
|
True = 0x00, False = 0x01,
|
|
|
|
|
High = 0x02, LowOrSame = 0x03,
|
|
|
|
|
CarryClear = 0x04, CarrySet = 0x05,
|
|
|
|
|
NotEqual = 0x06, Equal = 0x07,
|
|
|
|
|
OverflowClear = 0x08, OverflowSet = 0x09,
|
|
|
|
|
Positive = 0x0a, Negative = 0x0b,
|
|
|
|
|
GreaterThanOrEqual = 0x0c, LessThan = 0x0d,
|
|
|
|
|
GreaterThan = 0x0e, LessThanOrEqual = 0x0f,
|
|
|
|
|
};
|
|
|
|
|
|
2022-04-11 19:00:55 +00:00
|
|
|
|
/// Indicates the addressing mode applicable to an operand.
|
|
|
|
|
///
|
|
|
|
|
/// Implementation notes:
|
|
|
|
|
///
|
|
|
|
|
/// Those entries starting 0b00 or 0b01 are mapped as per the 68000's native encoding;
|
|
|
|
|
/// those starting 0b00 are those which are indicated directly by a mode field and those starting
|
|
|
|
|
/// 0b01 are those which are indicated by a register field given a mode of 0b111. The only minor
|
|
|
|
|
/// exception is AddressRegisterDirect, which exists on a 68000 but isn't specifiable by a
|
|
|
|
|
/// mode and register, it's contextual based on the instruction.
|
|
|
|
|
///
|
|
|
|
|
/// Those modes starting in 0b10 are the various extended addressing modes introduced as
|
|
|
|
|
/// of the 68020, which can be detected only after interpreting an extension word. At the
|
|
|
|
|
/// Preinstruction stage:
|
|
|
|
|
///
|
|
|
|
|
/// * AddressRegisterIndirectWithIndexBaseDisplacement, MemoryIndirectPostindexed
|
|
|
|
|
/// and MemoryIndirectPreindexed will have been partially decoded as
|
|
|
|
|
/// AddressRegisterIndirectWithIndex8bitDisplacement; and
|
|
|
|
|
/// * ProgramCounterIndirectWithIndexBaseDisplacement,
|
|
|
|
|
/// ProgramCounterMemoryIndirectPostindexed and
|
|
|
|
|
/// ProgramCounterMemoryIndirectPreindexed will have been partially decoded
|
|
|
|
|
/// as ProgramCounterIndirectWithIndex8bitDisplacement.
|
|
|
|
|
enum class AddressingMode: uint8_t {
|
|
|
|
|
/// No adddressing mode; this operand doesn't exist.
|
2022-04-24 14:43:06 +00:00
|
|
|
|
None = 0b01'101,
|
2022-04-11 19:00:55 +00:00
|
|
|
|
|
|
|
|
|
/// Dn
|
|
|
|
|
DataRegisterDirect = 0b00'000,
|
|
|
|
|
|
|
|
|
|
/// An
|
2022-04-12 18:54:11 +00:00
|
|
|
|
AddressRegisterDirect = 0b00'001,
|
2022-04-11 19:00:55 +00:00
|
|
|
|
/// (An)
|
|
|
|
|
AddressRegisterIndirect = 0b00'010,
|
|
|
|
|
/// (An)+
|
|
|
|
|
AddressRegisterIndirectWithPostincrement = 0b00'011,
|
|
|
|
|
/// -(An)
|
|
|
|
|
AddressRegisterIndirectWithPredecrement = 0b00'100,
|
|
|
|
|
/// (d16, An)
|
|
|
|
|
AddressRegisterIndirectWithDisplacement = 0b00'101,
|
|
|
|
|
/// (d8, An, Xn)
|
|
|
|
|
AddressRegisterIndirectWithIndex8bitDisplacement = 0b00'110,
|
2022-05-01 19:10:54 +00:00
|
|
|
|
/// (bd, An, Xn) [68020+]
|
2022-04-11 19:00:55 +00:00
|
|
|
|
AddressRegisterIndirectWithIndexBaseDisplacement = 0b10'000,
|
|
|
|
|
|
2022-05-01 19:10:54 +00:00
|
|
|
|
/// ([bd, An, Xn], od) [68020+]
|
2022-04-11 19:00:55 +00:00
|
|
|
|
MemoryIndirectPostindexed = 0b10'001,
|
2022-05-01 19:10:54 +00:00
|
|
|
|
/// ([bd, An], Xn, od) [68020+]
|
2022-04-11 19:00:55 +00:00
|
|
|
|
MemoryIndirectPreindexed = 0b10'010,
|
|
|
|
|
|
|
|
|
|
/// (d16, PC)
|
|
|
|
|
ProgramCounterIndirectWithDisplacement = 0b01'010,
|
|
|
|
|
/// (d8, PC, Xn)
|
|
|
|
|
ProgramCounterIndirectWithIndex8bitDisplacement = 0b01'011,
|
2022-05-01 19:10:54 +00:00
|
|
|
|
/// (bd, PC, Xn) [68020+]
|
2022-04-11 19:00:55 +00:00
|
|
|
|
ProgramCounterIndirectWithIndexBaseDisplacement = 0b10'011,
|
2022-05-01 19:10:54 +00:00
|
|
|
|
/// ([bd, PC, Xn], od) [68020+]
|
2022-04-11 19:00:55 +00:00
|
|
|
|
ProgramCounterMemoryIndirectPostindexed = 0b10'100,
|
2022-05-01 19:10:54 +00:00
|
|
|
|
/// ([bc, PC], Xn, od) [68020+]
|
2022-04-11 19:00:55 +00:00
|
|
|
|
ProgramCounterMemoryIndirectPreindexed = 0b10'101,
|
|
|
|
|
|
|
|
|
|
/// (xxx).W
|
|
|
|
|
AbsoluteShort = 0b01'000,
|
|
|
|
|
/// (xxx).L
|
|
|
|
|
AbsoluteLong = 0b01'001,
|
|
|
|
|
|
|
|
|
|
/// #
|
|
|
|
|
ImmediateData = 0b01'100,
|
2022-04-13 13:29:12 +00:00
|
|
|
|
|
2022-04-19 12:37:35 +00:00
|
|
|
|
/// .q; value is embedded in the opcode.
|
2022-04-24 14:43:06 +00:00
|
|
|
|
Quick = 0b01'110,
|
2022-04-11 19:00:55 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
|
A preinstruction is as much of an instruction as can be decoded with
|
|
|
|
|
only the first instruction word — i.e. an operation, and:
|
|
|
|
|
|
|
|
|
|
* on the 68000 and 68010, the complete addressing modes;
|
|
|
|
|
* on subsequent, a decent proportion of the addressing mode. See
|
|
|
|
|
the notes on @c AddressingMode for potential aliasing.
|
|
|
|
|
*/
|
|
|
|
|
class Preinstruction {
|
|
|
|
|
public:
|
|
|
|
|
Operation operation = Operation::Undefined;
|
|
|
|
|
|
2022-04-12 13:14:46 +00:00
|
|
|
|
// Instructions come with 0, 1 or 2 operands;
|
|
|
|
|
// the getters below act to provide a list of operands
|
|
|
|
|
// that is terminated by an AddressingMode::None.
|
|
|
|
|
//
|
|
|
|
|
// For two-operand instructions, argument 0 is a source
|
|
|
|
|
// and argument 1 is a destination.
|
|
|
|
|
//
|
|
|
|
|
// For one-operand instructions, only argument 0 will
|
|
|
|
|
// be provided, and will be a source and/or destination as
|
|
|
|
|
// per the semantics of the operation.
|
2022-05-01 17:00:20 +00:00
|
|
|
|
//
|
|
|
|
|
// The versions templated on index do a range check;
|
|
|
|
|
// if using the runtime versions then results for indices
|
|
|
|
|
// other than 0 and 1 are undefined.
|
2022-04-12 13:14:46 +00:00
|
|
|
|
|
2022-05-01 17:00:20 +00:00
|
|
|
|
AddressingMode mode(int index) const {
|
|
|
|
|
return AddressingMode(operands_[index] & 0x1f);
|
|
|
|
|
}
|
2022-04-19 18:35:40 +00:00
|
|
|
|
template <int index> AddressingMode mode() const {
|
2022-04-12 13:14:46 +00:00
|
|
|
|
if constexpr (index > 1) {
|
|
|
|
|
return AddressingMode::None;
|
|
|
|
|
}
|
2022-05-01 17:00:20 +00:00
|
|
|
|
return mode(index);
|
|
|
|
|
}
|
|
|
|
|
int reg(int index) const {
|
|
|
|
|
return operands_[index] >> 5;
|
2022-04-11 19:00:55 +00:00
|
|
|
|
}
|
2022-04-19 18:35:40 +00:00
|
|
|
|
template <int index> int reg() const {
|
2022-04-12 13:14:46 +00:00
|
|
|
|
if constexpr (index > 1) {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2022-05-01 17:00:20 +00:00
|
|
|
|
return reg(index);
|
2022-04-11 19:00:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
2022-05-01 17:00:20 +00:00
|
|
|
|
bool requires_supervisor() const {
|
2022-04-26 23:44:02 +00:00
|
|
|
|
return flags_ & 0x80;
|
|
|
|
|
}
|
2022-05-06 13:22:38 +00:00
|
|
|
|
DataSize operand_size() const {
|
2022-04-30 13:00:47 +00:00
|
|
|
|
return DataSize(flags_ & 0x03);
|
|
|
|
|
}
|
2022-05-01 17:00:20 +00:00
|
|
|
|
Condition condition() const {
|
2022-04-30 13:00:47 +00:00
|
|
|
|
return Condition((flags_ >> 2) & 0x0f);
|
2022-04-26 23:44:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
2022-04-11 19:00:55 +00:00
|
|
|
|
private:
|
2022-04-12 13:14:46 +00:00
|
|
|
|
uint8_t operands_[2] = { uint8_t(AddressingMode::None), uint8_t(AddressingMode::None)};
|
2022-04-26 23:44:02 +00:00
|
|
|
|
uint8_t flags_ = 0;
|
2022-04-11 19:00:55 +00:00
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
Preinstruction(
|
|
|
|
|
Operation operation,
|
2022-04-12 13:14:46 +00:00
|
|
|
|
AddressingMode op1_mode, int op1_reg,
|
2022-04-12 18:54:11 +00:00
|
|
|
|
AddressingMode op2_mode, int op2_reg,
|
2022-04-26 23:44:02 +00:00
|
|
|
|
bool is_supervisor,
|
2022-04-30 13:00:47 +00:00
|
|
|
|
DataSize size,
|
|
|
|
|
Condition condition) : operation(operation)
|
2022-04-12 13:14:46 +00:00
|
|
|
|
{
|
|
|
|
|
operands_[0] = uint8_t(op1_mode) | uint8_t(op1_reg << 5);
|
|
|
|
|
operands_[1] = uint8_t(op2_mode) | uint8_t(op2_reg << 5);
|
2022-04-30 13:00:47 +00:00
|
|
|
|
flags_ = uint8_t(
|
|
|
|
|
(is_supervisor ? 0x80 : 0x00) |
|
|
|
|
|
(int(condition) << 2) |
|
|
|
|
|
int(size)
|
|
|
|
|
);
|
2022-04-11 19:00:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Preinstruction() {}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif /* InstructionSets_68k_Instruction_hpp */
|