/* SPDX-FileCopyrightText: © 2022 Decompollaborate */ /* SPDX-License-Identifier: MIT */ #ifndef RABBITIZER_INSTRDESCRIPTOR_H #define RABBITIZER_INSTRDESCRIPTOR_H #pragma once #include #include "common/Utils.h" #include "RabbitizerOperandType.h" #include "RabbitizerInstrId.h" #include "RabbitizerInstrSuffix.h" #include "RabbitizerAccessType.h" #ifdef __cplusplus extern "C" { #endif //! @deprecated typedef enum RabbitizerInstrType { RABBITIZER_INSTR_TYPE_UNKNOWN, RABBITIZER_INSTR_TYPE_J, RABBITIZER_INSTR_TYPE_I, RABBITIZER_INSTR_TYPE_R, RABBITIZER_INSTR_TYPE_REGIMM, RABBITIZER_INSTR_TYPE_MAX, } RabbitizerInstrType; typedef struct RabbitizerInstrDescriptor { RabbitizerOperandType operands[4]; RabbitizerInstrType instrType; RabbitizerInstrSuffix instrSuffix; /** * Local branch with "restricted" range, usually it doesn't jump outside the current function */ bool isBranch; bool isBranchLikely; /** * The instruction can jump inside or outside its current function */ bool isJump; /** * The target address of this jump is encoded in the instruction (MIPS: J and JAL) */ bool isJumpWithAddress; /** * Triggers a trap on the processor */ bool isTrap; /** * The instruction performs float (any kind of float, including double precision) operations */ bool isFloat; /** * The instruction performs double precision float operations */ bool isDouble; /** * The instruction holds an immediate which is treated as an unsigned value, * otherwise the immediate it may hold should be treated as a Two's complement value */ bool isUnsigned; /** * The instruction modifies the state of the MIPS `rt` register */ bool modifiesRt; /** * The instruction modifies the state of the MIPS `rd` register */ bool modifiesRd; /** * The instruction reads the value which the MIPS `rs` register holds */ bool readsRs; /** * The instruction reads the value which the MIPS `rt` register holds */ bool readsRt; /** * The instruction reads the value which the MIPS `rd` register holds */ bool readsRd; bool readsHI; bool readsLO; bool modifiesHI; bool modifiesLO; /** * This instruction is not emited by a C compiler */ bool notEmitedByCompilers; /** * The instruction can hold the "hi" value of a %hi/%lo pair */ bool canBeHi; /** * The instruction can hold the "lo" value of a %hi/%lo pair */ bool canBeLo; /** * "and link" family of instructions * The instruction stores the return address link in the MIPS $ra (GPR 31) register */ bool doesLink; /** * This instruction performs a pointer dereference, either by loading from RAM or storing into RAM */ bool doesDereference; /** * Dereferences a pointer and loads data from RAM */ bool doesLoad; /** * Dereferences a pointer and stores data to RAM */ bool doesStore; /** * This instruction may be the result of expanding the `move` pseudo-instruction */ bool maybeIsMove; /** * This instruction is a pseudo-instruction */ bool isPseudo; RabbitizerAccessType accessType; bool doesUnsignedMemoryAccess; } RabbitizerInstrDescriptor; // TODO: less redundant name extern const RabbitizerInstrDescriptor RabbitizerInstrDescriptor_Descriptors[]; NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_hasSpecificOperand(const RabbitizerInstrDescriptor *self, RabbitizerOperandType operand); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_hasOperandAlias(const RabbitizerInstrDescriptor *self, RabbitizerOperandType operand); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isUnknownType(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isJType(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isIType(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isRType(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isRegimmType(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE RabbitizerInstrSuffix RabbitizerInstrDescriptor_instrSuffix(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isBranch(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isBranchLikely(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isJump(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isJumpWithAddress(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isTrap(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isFloat(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isDouble(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isUnsigned(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_modifiesRt(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_modifiesRd(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_readsRs(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_readsRt(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_readsRd(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_readsHI(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_readsLO(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_modifiesHI(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_modifiesLO(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_notEmitedByCompilers(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_canBeHi(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_canBeLo(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_doesLink(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_doesDereference(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_doesLoad(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_doesStore(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_maybeIsMove(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_isPseudo(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE RabbitizerAccessType RabbitizerInstrDescriptor_getAccessType(const RabbitizerInstrDescriptor *self); NODISCARD NON_NULL(1) PURE bool RabbitizerInstrDescriptor_doesUnsignedMemoryAccess(const RabbitizerInstrDescriptor *self); #ifdef __cplusplus } #endif #endif