2020-12-31 03:55:59 +00:00
|
|
|
//
|
2021-01-15 23:16:01 +00:00
|
|
|
// Instruction.hpp
|
2020-12-31 03:55:59 +00:00
|
|
|
// Clock Signal
|
|
|
|
//
|
2021-01-17 03:09:19 +00:00
|
|
|
// Created by Thomas Harte on 15/01/21.
|
2021-01-15 23:16:01 +00:00
|
|
|
// Copyright © 2021 Thomas Harte. All rights reserved.
|
2020-12-31 03:55:59 +00:00
|
|
|
//
|
|
|
|
|
2021-01-15 23:16:01 +00:00
|
|
|
#ifndef InstructionSets_PowerPC_Instruction_h
|
|
|
|
#define InstructionSets_PowerPC_Instruction_h
|
2020-12-31 03:55:59 +00:00
|
|
|
|
|
|
|
#include <cstdint>
|
|
|
|
|
2021-01-16 02:30:30 +00:00
|
|
|
namespace InstructionSet {
|
2020-12-31 03:55:59 +00:00
|
|
|
namespace PowerPC {
|
|
|
|
|
2022-03-21 14:17:55 +00:00
|
|
|
enum class CacheLine: uint32_t {
|
|
|
|
Instruction = 0b01100,
|
|
|
|
Data = 0b1101,
|
|
|
|
Minimum = 0b01110,
|
|
|
|
Maximum = 0b01111,
|
|
|
|
};
|
|
|
|
|
2022-03-21 14:19:30 +00:00
|
|
|
enum class BranchOptions: uint32_t {
|
|
|
|
// Naming convention:
|
|
|
|
//
|
|
|
|
// Dec_ prefix => decrement the CTR;
|
|
|
|
// condition starting NotZero or Zero => test CTR;
|
|
|
|
// condition ending Set or Clear => test for condition bit.
|
|
|
|
//
|
|
|
|
// Numerical suffixes are present because there's some redundancy
|
|
|
|
// in encodings.
|
|
|
|
//
|
|
|
|
// Note that the encodings themselves may suggest alternative means
|
|
|
|
// of interpretation than mapping via this enum.
|
|
|
|
Dec_NotZeroAndClear1 = 0b00000, Dec_NotZeroAndClear2 = 0b00001,
|
|
|
|
Dec_ZeroAndClear1 = 0b00010, Dec_ZeroAndClear2 = 0b00011,
|
|
|
|
Clear1 = 0b00100, Clear2 = 0b00101,
|
|
|
|
Dec_NotZeroAndSet1 = 0b01000, Dec_NotZeroAndSet2 = 0b01001,
|
|
|
|
Dec_ZeroAndSet1 = 0b01010, Dec_ZeroAndSet2 = 0b01011,
|
|
|
|
Set1 = 0b01100, Set2 = 0b01101,
|
|
|
|
Dec_NotZero1 = 0b10000, Dec_NotZero2 = 0b10001,
|
|
|
|
Dec_Zero1 = 0b10010, Dec_Zero2 = 0b10011,
|
|
|
|
Always = 0b10100,
|
|
|
|
};
|
|
|
|
|
2020-12-31 03:55:59 +00:00
|
|
|
enum class Operation: uint8_t {
|
|
|
|
Undefined,
|
|
|
|
|
|
|
|
// These 601-exclusive instructions; a lot of them are carry-overs
|
2022-03-21 14:17:55 +00:00
|
|
|
// from POWER. These are not part of the PowerPC architecture.
|
|
|
|
|
|
|
|
/// |rA| is placed into rD. If rA = 0x8000'0000 then 0x8000'0000 is placed into rD
|
|
|
|
/// and XER[OV] is set if oe() indicates that overflow is enabled.
|
|
|
|
absx,
|
|
|
|
|
|
|
|
/// The size of the cache line specified by rA is placed into rD. Cf. the CacheLine enum.
|
|
|
|
/// As an aside: all cache lines are 64 bytes on the MPC601.
|
|
|
|
clcs,
|
|
|
|
|
|
|
|
/// div, div., divo, div.; unsigned 64-bit divide. rA|MQ / rB is placed into rD and the
|
|
|
|
/// remainder is placed into MQ. The ermainder has the same sign as the dividend
|
|
|
|
/// such that remainder + divisor * quotient = dividend.
|
|
|
|
///
|
|
|
|
/// rc() != 0 => the LT, GT and EQ bits in CR are set as per the remainder.
|
|
|
|
/// oe() != 0 => SO and OV are set if the quotient exceeds 32 bits.
|
|
|
|
divx,
|
|
|
|
|
|
|
|
/// divs, divs., divso, divso.; signed 32-bit divide. rD = rA/rB; remainder is
|
|
|
|
/// placed into MQ. The ermainder has the same sign as the dividend
|
|
|
|
/// such that remainder + divisor * quotient = dividend.
|
|
|
|
///
|
|
|
|
/// rc() != 0 => the LT, GT and EQ bits in CR are set as per the remainder.
|
|
|
|
/// oe() != 0 => SO and OV are set if the quotient exceeds 32 bits.
|
2022-03-21 14:49:01 +00:00
|
|
|
divsx,
|
|
|
|
|
|
|
|
/// if rA > rB then rD = 0; else rD = NOT(rA) + rB + 1.
|
|
|
|
dozx,
|
|
|
|
|
|
|
|
/// if rA > simm() then rD = 0; else rD = NOT(rA) + simm() + 1.
|
|
|
|
dozi,
|
|
|
|
|
|
|
|
lscbxx, maskgx, maskirx, mulx,
|
2020-12-31 03:55:59 +00:00
|
|
|
nabsx, rlmix, rribx, slex, sleqx, sliqx, slliqx, sllqx, slqx,
|
|
|
|
sraiqx, sraqx, srex, sreax, sreqx, sriqx, srliqx, srlqx, srqx,
|
2021-01-15 23:16:01 +00:00
|
|
|
|
2020-12-31 03:55:59 +00:00
|
|
|
// 32- and 64-bit PowerPC instructions.
|
|
|
|
addx, addcx, addex, addi, addic, addic_, addis, addmex, addzex, andx,
|
2022-03-18 23:55:26 +00:00
|
|
|
andcx, andi_, andis_,
|
|
|
|
|
|
|
|
/// Branch unconditional.
|
|
|
|
///
|
|
|
|
/// Use li() to get the included immediate value.
|
|
|
|
///
|
|
|
|
/// Use aa() to determine whether it's a relative (aa() = 0) or absolute (aa() != 0) address.
|
|
|
|
/// Also check lk() to determine whether to update the link register.
|
|
|
|
///
|
|
|
|
/// Synonyms include:
|
|
|
|
/// * b (relative, no link) [though assemblers might encode as a bcx];
|
|
|
|
/// * bl (relative, link);
|
|
|
|
/// * ba (absolute, no link);
|
|
|
|
/// * bla (absolute, link).
|
|
|
|
bx,
|
|
|
|
|
|
|
|
/// Branch conditional.
|
|
|
|
///
|
|
|
|
/// lk() determines whether to update the link register.
|
|
|
|
/// bd() supplies a relative displacment.
|
|
|
|
/// bi() specifies which CR bit to use as a condition.
|
|
|
|
/// bo() provides other branch options:
|
|
|
|
/// 0000y => decrement count register, branch if new value != 0 && !condition.
|
|
|
|
/// 0001y => decrement count register, branch if new value == 0 && !condition.
|
|
|
|
/// 001zy => branch if !condition
|
|
|
|
/// 0100y => decrement count register, branch if new value != 0 && condition.
|
|
|
|
/// 0101y => decrement count register, branch if new value == 0 && condition.
|
|
|
|
/// 011zy => branch if condition.
|
|
|
|
/// 1z00y => decrement count register, branch if new value != 0.
|
|
|
|
/// 1z01y => decrement count register, branch if new value == 0.
|
|
|
|
/// 1z1zz => branch always.
|
|
|
|
/// (z should be 0 to create a valid field; y can be any value)
|
|
|
|
///
|
|
|
|
/// Synonyms incude:
|
|
|
|
/// * b (relative, no link) [though assemblers might encode as a bx].
|
|
|
|
bcx,
|
|
|
|
|
|
|
|
/// Branch conditional to count register.
|
|
|
|
///
|
|
|
|
/// bi(), bo() and lk() are as per bcx.
|
|
|
|
///
|
|
|
|
/// On the MPC601, anything that decrements the count register will use the non-decremented
|
|
|
|
/// version as the branch target. Other processors will use the decremented version.
|
|
|
|
bcctrx,
|
|
|
|
|
|
|
|
bclrx, cmp, cmpi, cmpl, cmpli,
|
2020-12-31 03:55:59 +00:00
|
|
|
cntlzwx, crand, crandc, creqv, crnand, crnor, cror, crorc, crxor, dcbf,
|
|
|
|
dcbst, dcbt, dcbtst, dcbz, divwx, divwux, eciwx, ecowx, eieio, eqvx,
|
|
|
|
extsbx, extshx, fabsx, faddx, faddsx, fcmpo, fcmpu, fctiwx, fctiwzx,
|
|
|
|
fdivx, fdivsx, fmaddx, fmaddsx, fmrx, fmsubx, fmsubsx, fmulx, fmulsx,
|
|
|
|
fnabsx, fnegx, fnmaddx, fnmaddsx, fnmsubx, fnmsubsx, frspx, fsubx, fsubsx,
|
|
|
|
icbi, isync, lbz, lbzu, lbzux, lbzx, lfd, lfdu, lfdux, lfdx, lfs, lfsu,
|
|
|
|
lfsux, lfsx, lha, lhau, lhaux, lhax, lhbrx, lhz, lhzu, lhzux, lhzx, lmw,
|
|
|
|
lswi, lswx, lwarx, lwbrx, lwz, lwzu, lwzux, lwzx, mcrf, mcrfs, mcrxr,
|
|
|
|
mfcr, mffsx, mfmsr, mfspr, mfsr, mfsrin, mtcrf, mtfsb0x, mtfsb1x, mtfsfx,
|
|
|
|
mtfsfix, mtmsr, mtspr, mtsr, mtsrin, mulhwx, mulhwux, mulli, mullwx,
|
|
|
|
nandx, negx, norx, orx, orcx, ori, oris, rfi, rlwimix, rlwinmx, rlwnmx,
|
|
|
|
sc, slwx, srawx, srawix, srwx, stb, stbu, stbux, stbx, stfd, stfdu,
|
|
|
|
stfdux, stfdx, stfs, stfsu, stfsux, stfsx, sth, sthbrx, sthu, sthux, sthx,
|
|
|
|
stmw, stswi, stswx, stw, stwbrx, stwcx_, stwu, stwux, stwx, subfx, subfcx,
|
2020-12-31 23:14:38 +00:00
|
|
|
subfex, subfic, subfmex, subfzex, sync, tw, twi, xorx, xori, xoris, mftb,
|
2021-01-01 02:12:36 +00:00
|
|
|
|
2020-12-31 03:55:59 +00:00
|
|
|
// 32-bit, supervisor level.
|
|
|
|
dcbi,
|
2021-01-01 02:12:36 +00:00
|
|
|
|
2020-12-31 23:14:38 +00:00
|
|
|
// Supervisor, optional.
|
2021-01-01 02:12:36 +00:00
|
|
|
tlbia, tlbie, tlbsync,
|
|
|
|
|
2020-12-31 03:55:59 +00:00
|
|
|
// Optional.
|
2021-01-01 16:46:26 +00:00
|
|
|
fresx, frsqrtex, fselx, fsqrtx, slbia, slbie, stfiwx,
|
2020-12-31 03:55:59 +00:00
|
|
|
|
|
|
|
// 64-bit only PowerPC instructions.
|
2021-01-01 02:12:36 +00:00
|
|
|
cntlzdx, divdx, divdux, extswx, fcfidx, fctidx, fctidzx, tdi, mulhdux,
|
|
|
|
ldx, sldx, ldux, td, mulhdx, ldarx, stdx, stdux, mulld, lwax, lwaux,
|
2021-01-01 16:46:26 +00:00
|
|
|
sradix, srdx, sradx, extsw, fsqrtsx, std, stdu, stdcx_,
|
2020-12-31 03:55:59 +00:00
|
|
|
};
|
|
|
|
|
2020-12-31 21:51:31 +00:00
|
|
|
/*!
|
|
|
|
Holds a decoded PowerPC instruction.
|
|
|
|
|
|
|
|
Implementation note: because the PowerPC encoding is particularly straightforward,
|
|
|
|
only the operation has been decoded ahead of time; all other fields are decoded on-demand.
|
2021-01-09 03:22:07 +00:00
|
|
|
|
|
|
|
It would be possible to partition the ordering of Operations into user followed by supervisor,
|
|
|
|
eliminating the storage necessary for a flag, but it wouldn't save anything due to alignment.
|
2020-12-31 21:51:31 +00:00
|
|
|
*/
|
2020-12-31 03:55:59 +00:00
|
|
|
struct Instruction {
|
2021-01-03 03:47:42 +00:00
|
|
|
Operation operation = Operation::Undefined;
|
|
|
|
bool is_supervisor = false;
|
|
|
|
uint32_t opcode = 0;
|
2020-12-31 03:55:59 +00:00
|
|
|
|
2021-01-09 03:38:56 +00:00
|
|
|
Instruction() noexcept {}
|
|
|
|
Instruction(uint32_t opcode) noexcept : opcode(opcode) {}
|
|
|
|
Instruction(Operation operation, uint32_t opcode, bool is_supervisor = false) noexcept : operation(operation), is_supervisor(is_supervisor), opcode(opcode) {}
|
2020-12-31 03:55:59 +00:00
|
|
|
|
2021-01-01 16:46:26 +00:00
|
|
|
// Instruction fields are decoded below; naming is a compromise between
|
|
|
|
// Motorola's documentation and IBM's.
|
|
|
|
//
|
|
|
|
// I've dutifully implemented various synonyms with unique entry points,
|
|
|
|
// in order to capture that information here rather than thrusting it upon
|
|
|
|
// the reader of whatever implementation may follow.
|
2020-12-31 21:51:31 +00:00
|
|
|
|
2021-01-01 21:38:40 +00:00
|
|
|
// Currently omitted: OPCD and XO, which I think are unnecessary given that
|
|
|
|
// full decoding has already occurred.
|
2021-01-01 02:12:36 +00:00
|
|
|
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Immediate field used to specify an unsigned 16-bit integer.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint16_t uimm() const { return uint16_t(opcode & 0xffff); }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Immediate field used to specify a signed 16-bit integer.
|
2021-01-05 03:36:39 +00:00
|
|
|
int16_t simm() const { return int16_t(opcode & 0xffff); }
|
2021-01-01 21:38:40 +00:00
|
|
|
/// Immediate field used to specify a signed 16-bit integer.
|
2021-01-05 03:36:39 +00:00
|
|
|
int16_t d() const { return int16_t(opcode & 0xffff); }
|
2021-01-01 21:38:40 +00:00
|
|
|
/// Immediate field used to specify a signed 14-bit integer [64-bit only].
|
2021-01-05 03:36:39 +00:00
|
|
|
int16_t ds() const { return int16_t(opcode & 0xfffc); }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Immediate field used as data to be placed into a field in the floating point status and condition register.
|
2021-01-05 03:36:39 +00:00
|
|
|
int32_t imm() const { return (opcode >> 12) & 0xf; }
|
2021-01-01 16:46:26 +00:00
|
|
|
|
|
|
|
/// Specifies the conditions on which to trap.
|
2021-01-05 03:36:39 +00:00
|
|
|
int32_t to() const { return (opcode >> 21) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
|
|
|
|
/// Register source A or destination.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t rA() const { return (opcode >> 16) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Register source B.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t rB() const { return (opcode >> 11) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Register destination.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t rD() const { return (opcode >> 21) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Register source.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t rS() const { return (opcode >> 21) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
|
|
|
|
/// Floating point register source A.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t frA() const { return (opcode >> 16) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Floating point register source B.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t frB() const { return (opcode >> 11) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Floating point register source C.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t frC() const { return (opcode >> 6) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Floating point register source.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t frS() const { return (opcode >> 21) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Floating point register destination.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t frD() const { return (opcode >> 21) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
|
2022-03-19 16:27:28 +00:00
|
|
|
/// Branch conditional options. Cf. the BranchOptions enum.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t bo() const { return (opcode >> 21) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Source condition register bit for branch conditionals.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t bi() const { return (opcode >> 16) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Branch displacement; provided as already sign extended.
|
2021-01-05 03:36:39 +00:00
|
|
|
int16_t bd() const { return int16_t(opcode & 0xfffc); }
|
2021-01-01 16:46:26 +00:00
|
|
|
|
2021-01-01 21:38:40 +00:00
|
|
|
/// Specifies the first 1 bit of a 32/64-bit mask for rotate operations.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t mb() const { return (opcode >> 6) & 0x1f; }
|
2021-01-01 21:38:40 +00:00
|
|
|
/// Specifies the first 1 bit of a 32/64-bit mask for rotate operations.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t me() const { return (opcode >> 1) & 0x1f; }
|
2021-01-01 21:38:40 +00:00
|
|
|
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Condition register source bit A.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t crbA() const { return (opcode >> 16) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Condition register source bit B.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t crbB() const { return (opcode >> 11) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Condition register (or floating point status & condition register) destination bit.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t crbD() const { return (opcode >> 21) & 0x1f; }
|
2021-01-01 16:46:26 +00:00
|
|
|
|
|
|
|
/// Condition register (or floating point status & condition register) destination field.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t crfD() const { return (opcode >> 23) & 0x07; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Condition register (or floating point status & condition register) source field.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t crfS() const { return (opcode >> 18) & 0x07; }
|
2021-01-01 16:46:26 +00:00
|
|
|
|
|
|
|
/// Mask identifying fields to be updated by mtcrf.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t crm() const { return (opcode >> 12) & 0xff; }
|
2021-01-01 16:46:26 +00:00
|
|
|
|
|
|
|
/// Mask identifying fields to be updated by mtfsf.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t fm() const { return (opcode >> 17) & 0xff; }
|
2021-01-01 16:46:26 +00:00
|
|
|
|
2021-01-01 21:38:40 +00:00
|
|
|
/// Specifies the number of bytes to move in an immediate string load or store.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t nb() const { return (opcode >> 11) & 0x1f; }
|
2021-01-01 21:38:40 +00:00
|
|
|
|
|
|
|
/// Specifies a shift amount.
|
|
|
|
/// TODO: possibly bit 30 is also used in 64-bit mode, find out.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t sh() const { return (opcode >> 11) & 0x1f; }
|
2021-01-01 21:38:40 +00:00
|
|
|
|
|
|
|
/// Specifies one of the 16 segment registers [32-bit only].
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t sr() const { return (opcode >> 16) & 0xf; }
|
2021-01-01 21:38:40 +00:00
|
|
|
|
2021-01-01 16:46:26 +00:00
|
|
|
/// A 24-bit signed number; provided as already sign extended.
|
2021-01-05 03:36:39 +00:00
|
|
|
int32_t li() const {
|
2021-01-01 16:46:26 +00:00
|
|
|
constexpr uint32_t extensions[2] = {
|
|
|
|
0x0000'0000,
|
|
|
|
0xfc00'0000
|
|
|
|
};
|
2021-01-03 16:14:43 +00:00
|
|
|
const uint32_t value = (opcode & 0x03ff'fffc) | extensions[(opcode >> 25) & 1];
|
2021-01-01 16:46:26 +00:00
|
|
|
return int32_t(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Absolute address bit; @c 0 or @c non-0.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t aa() const { return opcode & 0x02; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Link bit; @c 0 or @c non-0.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t lk() const { return opcode & 0x01; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Record bit; @c 0 or @c non-0.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t rc() const { return opcode & 0x01; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Whether to compare 32-bit or 64-bit numbers [for 64-bit implementations only]; @c 0 or @c non-0.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t l() const { return opcode & 0x200000; }
|
2021-01-01 16:46:26 +00:00
|
|
|
/// Enables setting of OV and SO in the XER; @c 0 or @c non-0.
|
2021-01-05 03:36:39 +00:00
|
|
|
uint32_t oe() const { return opcode & 0x800; }
|
2020-12-31 03:55:59 +00:00
|
|
|
};
|
|
|
|
|
2021-01-15 23:16:01 +00:00
|
|
|
// Sanity check on Instruction size.
|
2021-01-09 03:38:56 +00:00
|
|
|
static_assert(sizeof(Instruction) <= 8);
|
|
|
|
|
2020-12-31 03:55:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-15 23:16:01 +00:00
|
|
|
#endif /* InstructionSets_PowerPC_Instruction_h */
|