2022-04-11 15:00:55 -04:00
|
|
|
//
|
|
|
|
// Decoder.hpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 10/04/2022.
|
|
|
|
// Copyright © 2022 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
2022-04-15 09:40:37 -04:00
|
|
|
#ifndef InstructionSets_M68k_Decoder_hpp
|
|
|
|
#define InstructionSets_M68k_Decoder_hpp
|
2022-04-11 15:00:55 -04:00
|
|
|
|
|
|
|
#include "Instruction.hpp"
|
2022-04-15 09:40:37 -04:00
|
|
|
#include "Model.hpp"
|
2022-10-25 10:13:12 -04:00
|
|
|
#include "../../Numeric/Sizes.hpp"
|
2022-04-11 15:00:55 -04:00
|
|
|
|
|
|
|
namespace InstructionSet {
|
|
|
|
namespace M68k {
|
|
|
|
|
|
|
|
/*!
|
|
|
|
A stateless decoder that can map from instruction words to preinstructions
|
|
|
|
(i.e. enough to know the operation and size, and either know the addressing mode
|
|
|
|
and registers or else know how many further extension words are needed).
|
2022-04-26 19:37:07 -04:00
|
|
|
|
|
|
|
WARNING: at present this handles the original 68000 instruction set only. It
|
|
|
|
requires a model only for the sake of not baking in assumptions about MOVE SR, etc,
|
|
|
|
and supporting extended addressing modes in some cases.
|
|
|
|
|
|
|
|
But it does not yet decode any operations which were not present on the 68000.
|
2022-04-11 15:00:55 -04:00
|
|
|
*/
|
2022-04-15 09:40:37 -04:00
|
|
|
template <Model model> class Predecoder {
|
2022-04-11 15:00:55 -04:00
|
|
|
public:
|
|
|
|
Preinstruction decode(uint16_t instruction);
|
|
|
|
|
|
|
|
private:
|
|
|
|
// Page by page decoders; each gets a bit ad hoc so
|
|
|
|
// it is neater to separate them.
|
2022-04-12 08:16:29 -04:00
|
|
|
Preinstruction decode0(uint16_t instruction);
|
|
|
|
Preinstruction decode1(uint16_t instruction);
|
|
|
|
Preinstruction decode2(uint16_t instruction);
|
|
|
|
Preinstruction decode3(uint16_t instruction);
|
2022-04-12 07:49:08 -04:00
|
|
|
Preinstruction decode4(uint16_t instruction);
|
2022-04-12 08:16:29 -04:00
|
|
|
Preinstruction decode5(uint16_t instruction);
|
|
|
|
Preinstruction decode6(uint16_t instruction);
|
|
|
|
Preinstruction decode7(uint16_t instruction);
|
2022-04-11 16:32:57 -04:00
|
|
|
Preinstruction decode8(uint16_t instruction);
|
2022-04-12 08:16:29 -04:00
|
|
|
Preinstruction decode9(uint16_t instruction);
|
|
|
|
Preinstruction decodeA(uint16_t instruction);
|
2022-04-12 07:49:08 -04:00
|
|
|
Preinstruction decodeB(uint16_t instruction);
|
2022-04-11 15:00:55 -04:00
|
|
|
Preinstruction decodeC(uint16_t instruction);
|
2022-04-12 08:16:29 -04:00
|
|
|
Preinstruction decodeD(uint16_t instruction);
|
|
|
|
Preinstruction decodeE(uint16_t instruction);
|
|
|
|
Preinstruction decodeF(uint16_t instruction);
|
2022-04-11 15:00:55 -04:00
|
|
|
|
2022-10-25 10:13:12 -04:00
|
|
|
// Yuckiness here: 67 is a count of the number of things contained below in
|
|
|
|
// ExtendedOperation; this acts to ensure ExtendedOperation is the minimum
|
|
|
|
// integer size large enough to hold all actual operations plus the ephemeral
|
|
|
|
// ones used here. Intention is to support table-based decoding, which will mean
|
|
|
|
// making those integers less ephemeral, hence the desire to pick a minimum size.
|
|
|
|
using OpT = typename MinIntTypeValue<
|
|
|
|
uint64_t(OperationMax<model>::value) + 67
|
|
|
|
>::type;
|
|
|
|
static constexpr auto OpMax = OpT(OperationMax<model>::value);
|
2022-04-15 09:40:37 -04:00
|
|
|
|
2022-04-11 15:00:55 -04:00
|
|
|
// Specific instruction decoders.
|
2022-04-18 07:23:25 -04:00
|
|
|
template <OpT operation, bool validate = true> Preinstruction decode(uint16_t instruction);
|
2022-04-26 19:37:07 -04:00
|
|
|
template <OpT operation, bool validate> Preinstruction validated(
|
|
|
|
AddressingMode op1_mode = AddressingMode::None, int op1_reg = 0,
|
2022-04-30 09:00:47 -04:00
|
|
|
AddressingMode op2_mode = AddressingMode::None, int op2_reg = 0,
|
2022-10-28 13:36:40 -04:00
|
|
|
Condition condition = Condition::True,
|
|
|
|
int further_extension_words = 0
|
2022-04-26 19:37:07 -04:00
|
|
|
);
|
2022-10-25 09:50:19 -04:00
|
|
|
template <OpT operation> uint32_t invalid_operands();
|
2022-04-12 16:17:30 -04:00
|
|
|
|
2022-04-15 09:40:37 -04:00
|
|
|
// Extended operation list; collapses into a single byte enough information to
|
|
|
|
// know both the type of operation and how to decode the operands. Most of the
|
|
|
|
// time that's knowable from the Operation alone, hence the rather awkward
|
|
|
|
// extension of @c Operation.
|
2022-04-15 20:33:59 -04:00
|
|
|
enum ExtendedOperation: OpT {
|
2022-10-25 10:13:12 -04:00
|
|
|
MOVEPtoRl = OpMax + 1, MOVEPtoRw,
|
2022-04-12 16:17:30 -04:00
|
|
|
MOVEPtoMl, MOVEPtoMw,
|
|
|
|
|
2022-04-25 19:58:19 -04:00
|
|
|
MOVEQ,
|
|
|
|
|
2022-04-13 09:29:12 -04:00
|
|
|
ADDQb, ADDQw, ADDQl,
|
|
|
|
ADDQAw, ADDQAl,
|
|
|
|
SUBQb, SUBQw, SUBQl,
|
|
|
|
SUBQAw, SUBQAl,
|
|
|
|
|
2022-04-15 09:40:37 -04:00
|
|
|
ADDIb, ADDIw, ADDIl,
|
|
|
|
ORIb, ORIw, ORIl,
|
|
|
|
SUBIb, SUBIw, SUBIl,
|
|
|
|
ANDIb, ANDIw, ANDIl,
|
|
|
|
EORIb, EORIw, EORIl,
|
|
|
|
CMPIb, CMPIw, CMPIl,
|
|
|
|
|
2022-04-15 15:33:54 -04:00
|
|
|
BTSTI, BCHGI, BCLRI, BSETI,
|
2022-04-15 09:40:37 -04:00
|
|
|
|
2022-04-22 09:24:16 -04:00
|
|
|
CMPMb, CMPMw, CMPMl,
|
|
|
|
|
2022-04-22 20:37:09 -04:00
|
|
|
ADDtoMb, ADDtoMw, ADDtoMl,
|
|
|
|
ADDtoRb, ADDtoRw, ADDtoRl,
|
|
|
|
|
|
|
|
SUBtoMb, SUBtoMw, SUBtoMl,
|
|
|
|
SUBtoRb, SUBtoRw, SUBtoRl,
|
|
|
|
|
|
|
|
ANDtoMb, ANDtoMw, ANDtoMl,
|
|
|
|
ANDtoRb, ANDtoRw, ANDtoRl,
|
|
|
|
|
|
|
|
ORtoMb, ORtoMw, ORtoMl,
|
|
|
|
ORtoRb, ORtoRw, ORtoRl,
|
2022-04-25 11:43:30 -04:00
|
|
|
|
|
|
|
EXGRtoR, EXGAtoA, EXGRtoA,
|
2022-04-12 16:17:30 -04:00
|
|
|
};
|
2022-04-15 09:40:37 -04:00
|
|
|
|
2022-04-15 20:33:59 -04:00
|
|
|
static constexpr Operation operation(OpT op);
|
2022-04-11 15:00:55 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-15 09:40:37 -04:00
|
|
|
#endif /* InstructionSets_M68k_Decoder_hpp */
|