/* $Id: DisasmCore-armv8.cpp 107417 2025-01-06 14:14:09Z vboxsync $ */ /** @file * VBox Disassembler - Core Components. */ /* * Copyright (C) 2023-2024 Oracle and/or its affiliates. * * This file is part of VirtualBox base platform packages, as * available from https://www.virtualbox.org. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation, in version 3 of the * License. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, see . * * SPDX-License-Identifier: GPL-3.0-only */ /********************************************************************************************************************************* * Header Files * *********************************************************************************************************************************/ #define LOG_GROUP LOG_GROUP_DIS #include #include #include /* Required to get Armv8A64ConvertImmRImmS2Mask64() from armv8.h. */ #include #include #include #include #include #include #include "DisasmInternal-armv8.h" /********************************************************************************************************************************* * Structures and Typedefs * *********************************************************************************************************************************/ /** Parser callback. * @remark no DECLCALLBACK() here because it's considered to be internal and * there is no point in enforcing CDECL. */ typedef int FNDISPARSEARMV8(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit); /** Pointer to a disassembler parser function. */ typedef FNDISPARSEARMV8 *PFNDISPARSEARMV8; /** Opcode decoder callback. * @remark no DECLCALLBACK() here because it's considered to be internal and * there is no point in enforcing CDECL. */ typedef uint32_t FNDISDECODEARMV8(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8INSNCLASS pInsnClass); /** Pointer to a disassembler parser function. */ typedef FNDISDECODEARMV8 *PFNDISDECODEARMV8; /********************************************************************************************************************************* * Defined Constants And Macros * *********************************************************************************************************************************/ /********************************************************************************************************************************* * Internal Functions * *********************************************************************************************************************************/ /** @name Parsers * @{ */ static FNDISPARSEARMV8 disArmV8ParseIllegal; static FNDISPARSEARMV8 disArmV8ParseSize; static FNDISPARSEARMV8 disArmV8ParseImm; static FNDISPARSEARMV8 disArmV8ParseImmRel; static FNDISPARSEARMV8 disArmV8ParseImmAdr; static FNDISPARSEARMV8 disArmV8ParseImmZero; static FNDISPARSEARMV8 disArmV8ParseGprZr; static FNDISPARSEARMV8 disArmV8ParseGprZr32; static FNDISPARSEARMV8 disArmV8ParseGprZr64; static FNDISPARSEARMV8 disArmV8ParseGprSp; static FNDISPARSEARMV8 disArmV8ParseGprOff; static FNDISPARSEARMV8 disArmV8ParseVecReg; static FNDISPARSEARMV8 disArmV8ParseAddrGprSp; static FNDISPARSEARMV8 disArmV8ParseRegFixed31; static FNDISPARSEARMV8 disArmV8ParseGprCount; static FNDISPARSEARMV8 disArmV8ParseImmsImmrN; static FNDISPARSEARMV8 disArmV8ParseHw; static FNDISPARSEARMV8 disArmV8ParseCond; static FNDISPARSEARMV8 disArmV8ParsePState; static FNDISPARSEARMV8 disArmV8ParseCRnCRm; static FNDISPARSEARMV8 disArmV8ParseSysReg; static FNDISPARSEARMV8 disArmV8ParseSh12; static FNDISPARSEARMV8 disArmV8ParseImmTbz; static FNDISPARSEARMV8 disArmV8ParseShift; static FNDISPARSEARMV8 disArmV8ParseShiftAmount; static FNDISPARSEARMV8 disArmV8ParseImmMemOff; static FNDISPARSEARMV8 disArmV8ParseSImmMemOff; static FNDISPARSEARMV8 disArmV8ParseSImmMemOffUnscaled; static FNDISPARSEARMV8 disArmV8ParseOption; static FNDISPARSEARMV8 disArmV8ParseS; static FNDISPARSEARMV8 disArmV8ParseSetPreIndexed; static FNDISPARSEARMV8 disArmV8ParseSetPostIndexed; static FNDISPARSEARMV8 disArmV8ParseFpType; static FNDISPARSEARMV8 disArmV8ParseFpReg; static FNDISPARSEARMV8 disArmV8ParseFpScale; static FNDISPARSEARMV8 disArmV8ParseFpFixupFCvt; static FNDISPARSEARMV8 disArmV8ParseSimdRegSize; static FNDISPARSEARMV8 disArmV8ParseSimdRegSize32; static FNDISPARSEARMV8 disArmV8ParseSimdRegSize64; static FNDISPARSEARMV8 disArmV8ParseSimdRegSize128; static FNDISPARSEARMV8 disArmV8ParseSimdRegScalar; static FNDISPARSEARMV8 disArmV8ParseImmHImmB; static FNDISPARSEARMV8 disArmV8ParseSf; static FNDISPARSEARMV8 disArmV8ParseImmX16; static FNDISPARSEARMV8 disArmV8ParseSImmTags; static FNDISPARSEARMV8 disArmV8ParseLdrPacImm; static FNDISPARSEARMV8 disArmV8ParseLdrPacW; static FNDISPARSEARMV8 disArmV8ParseVecRegElemSize; static FNDISPARSEARMV8 disArmV8ParseVecRegQ; static FNDISPARSEARMV8 disArmV8ParseVecGrp; static FNDISPARSEARMV8 disArmV8ParseSimdLdStPostIndexImm; /** @} */ /** @name Decoders * @{ */ static FNDISDECODEARMV8 disArmV8DecodeIllegal; static FNDISDECODEARMV8 disArmV8DecodeLookup; static FNDISDECODEARMV8 disArmV8DecodeCollate; /** @} */ /********************************************************************************************************************************* * Global Variables * *********************************************************************************************************************************/ /** Parser opcode table for full disassembly. */ static PFNDISPARSEARMV8 const g_apfnDisasm[kDisParmParseMax] = { disArmV8ParseIllegal, disArmV8ParseSize, disArmV8ParseImm, disArmV8ParseImmRel, disArmV8ParseImmAdr, disArmV8ParseImmZero, disArmV8ParseGprZr, disArmV8ParseGprZr32, disArmV8ParseGprZr64, disArmV8ParseGprSp, disArmV8ParseGprOff, disArmV8ParseVecReg, disArmV8ParseAddrGprSp, disArmV8ParseRegFixed31, disArmV8ParseGprCount, disArmV8ParseImmsImmrN, disArmV8ParseHw, disArmV8ParseCond, disArmV8ParsePState, disArmV8ParseCRnCRm, disArmV8ParseSysReg, disArmV8ParseSh12, disArmV8ParseImmTbz, disArmV8ParseShift, disArmV8ParseShiftAmount, disArmV8ParseImmMemOff, disArmV8ParseSImmMemOff, disArmV8ParseSImmMemOffUnscaled, disArmV8ParseOption, disArmV8ParseS, disArmV8ParseSetPreIndexed, disArmV8ParseSetPostIndexed, disArmV8ParseFpType, disArmV8ParseFpReg, disArmV8ParseFpScale, disArmV8ParseFpFixupFCvt, disArmV8ParseSimdRegSize, disArmV8ParseSimdRegSize32, disArmV8ParseSimdRegSize64, disArmV8ParseSimdRegSize128, disArmV8ParseSimdRegScalar, disArmV8ParseImmHImmB, disArmV8ParseSf, disArmV8ParseImmX16, disArmV8ParseSImmTags, disArmV8ParseLdrPacImm, disArmV8ParseLdrPacW, disArmV8ParseVecRegElemSize, disArmV8ParseVecRegQ, disArmV8ParseVecGrp, disArmV8ParseSimdLdStPostIndexImm }; /** Opcode decoder table. */ static PFNDISDECODEARMV8 const g_apfnOpcDecode[kDisArmV8OpcDecodeMax] = { disArmV8DecodeIllegal, disArmV8DecodeLookup, disArmV8DecodeCollate }; DECLINLINE(uint32_t) disArmV8ExtractBitVecFromInsn(uint32_t u32Insn, uint8_t idxBitStart, uint8_t cBits) { uint32_t fMask = (uint32_t)(RT_BIT_64(idxBitStart + cBits) - 1); return (u32Insn & fMask) >> idxBitStart; } DECLINLINE(int32_t) disArmV8ExtractBitVecFromInsnSignExtend(uint32_t u32Insn, uint8_t idxBitStart, uint8_t cBits) { uint32_t const fMask = RT_BIT_32(cBits) - 1; uint32_t const fSignBit = RT_BIT_32(cBits - 1); uint32_t const u32 = (u32Insn >> idxBitStart) & fMask; return (int32_t)((u32 ^ fSignBit) - fSignBit); } static int disArmV8ParseIllegal(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, u32Insn, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); AssertFailed(); return VERR_INTERNAL_ERROR; } static int disArmV8ParseSize(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pInsnClass, pParam); Assert(pInsnParm->cBits == 2); uint32_t u32Size = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); switch (u32Size) { case 0: pDis->armv8.cbOperand = sizeof(uint8_t); break; case 1: pDis->armv8.cbOperand = sizeof(uint16_t); break; case 2: pDis->armv8.cbOperand = sizeof(uint32_t); break; case 3: pDis->armv8.cbOperand = sizeof(uint64_t); break; default: AssertReleaseFailed(); } *pf64Bit = pDis->armv8.cbOperand == sizeof(uint64_t) || (pOp->fFlags & DISARMV8INSNCLASS_F_FORCED_64BIT); return VINF_SUCCESS; } static int disArmV8ParseImm(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); AssertReturn(pInsnParm->idxBitStart + pInsnParm->cBits < 32, VERR_INTERNAL_ERROR_2); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmImm; pParam->uValue = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); if (pInsnParm->cBits <= 8) { pParam->armv8.cb = sizeof(uint8_t); pParam->fUse |= DISUSE_IMMEDIATE8; } else if (pInsnParm->cBits <= 16) { pParam->armv8.cb = sizeof(uint16_t); pParam->fUse |= DISUSE_IMMEDIATE16; } else if (pInsnParm->cBits <= 32) { pParam->armv8.cb = sizeof(uint32_t); pParam->fUse |= DISUSE_IMMEDIATE32; } else AssertReleaseFailed(); return VINF_SUCCESS; } static int disArmV8ParseImmRel(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); AssertReturn(pInsnParm->idxBitStart + pInsnParm->cBits < 32, VERR_INTERNAL_ERROR_2); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmImmRel; pParam->uValue = (int64_t)disArmV8ExtractBitVecFromInsnSignExtend(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits) * sizeof(uint32_t); if (pInsnParm->cBits <= 8) { pParam->armv8.cb = sizeof(int8_t); pParam->fUse |= DISUSE_IMMEDIATE8_REL; } else if (pInsnParm->cBits <= 16) { pParam->armv8.cb = sizeof(int16_t); pParam->fUse |= DISUSE_IMMEDIATE16_REL; } else if (pInsnParm->cBits <= 32) { pParam->armv8.cb = sizeof(int32_t); pParam->fUse |= DISUSE_IMMEDIATE32_REL; } else AssertReleaseFailed(); return VINF_SUCCESS; } static int disArmV8ParseImmAdr(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit, pInsnParm); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmImmRel; pParam->uValue = disArmV8ExtractBitVecFromInsn(u32Insn, 5, 19); pParam->uValue |= disArmV8ExtractBitVecFromInsn(u32Insn, 29, 2) << 29; pParam->fUse |= DISUSE_IMMEDIATE32; return VINF_SUCCESS; } static int disArmV8ParseImmZero(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, u32Insn, pOp, pInsnClass, pf64Bit, pInsnParm); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmImm; pParam->uValue = 0; pParam->fUse |= DISUSE_IMMEDIATE8; return VINF_SUCCESS; } static int disArmV8ParseGprZr(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmReg; pParam->armv8.Op.Reg.cRegs = 1; pParam->armv8.Op.Reg.idReg = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); if (*pf64Bit || (pParam->armv8.enmType == kDisArmv8OpParmAddrInGpr)) pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Gpr_64Bit; else pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Gpr_32Bit; return VINF_SUCCESS; } static int disArmV8ParseGprZr32(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmReg; pParam->armv8.Op.Reg.cRegs = 1; pParam->armv8.Op.Reg.idReg = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Gpr_32Bit; return VINF_SUCCESS; } static int disArmV8ParseGprZr64(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmReg; pParam->armv8.Op.Reg.cRegs = 1; pParam->armv8.Op.Reg.idReg = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Gpr_64Bit; return VINF_SUCCESS; } static int disArmV8ParseGprSp(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmReg; pParam->armv8.Op.Reg.cRegs = 1; pParam->armv8.Op.Reg.idReg = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); if (pParam->armv8.Op.Reg.idReg == 31) pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Sp; else if (*pf64Bit) pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Gpr_64Bit; else pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Gpr_32Bit; return VINF_SUCCESS; } static int disArmV8ParseGprOff(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); Assert(pParam->armv8.enmType != kDisArmv8OpParmNone); pParam->armv8.GprIndex.cRegs = 1; pParam->armv8.GprIndex.idReg = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); pParam->armv8.GprIndex.enmRegType = kDisOpParamArmV8RegType_Gpr_64Bit; /* Might get overwritten later on. */ pParam->fUse |= DISUSE_INDEX; return VINF_SUCCESS; } static int disArmV8ParseVecReg(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmReg; pParam->armv8.Op.Reg.cRegs = 1; pParam->armv8.Op.Reg.idReg = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Simd_Vector; pParam->armv8.Op.Reg.enmVecType = pDis->armv8.enmVecRegType; return VINF_SUCCESS; } static int disArmV8ParseAddrGprSp(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmAddrInGpr; pParam->armv8.Op.Reg.cRegs = 1; pParam->armv8.Op.Reg.idReg = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); if (pParam->armv8.Op.Reg.idReg == 31) pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Sp; else pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Gpr_64Bit; return VINF_SUCCESS; } static int disArmV8ParseRegFixed31(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pParam, pf64Bit); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); if (disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits) != 31) return VERR_DIS_INVALID_OPCODE; return VINF_SUCCESS; } static int disArmV8ParseGprCount(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, u32Insn, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); /* This is special as it doesn't really parse the instruction but sets the register count of the given parameter based on the number if bits. */ Assert(pInsnParm->cBits <= 2); Assert(pInsnParm->idxBitStart == 0); Assert(pParam->armv8.Op.Reg.enmRegType == kDisOpParamArmV8RegType_Gpr_64Bit); pParam->armv8.Op.Reg.cRegs = pInsnParm->cBits; return VINF_SUCCESS; } static int disArmV8ParseImmsImmrN(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass); AssertReturn(pInsnParm->cBits == 13, VERR_INTERNAL_ERROR_2); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmImm; uint32_t u32ImmRaw = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); /* N bit must be 0 if 32-bit variant is used. */ if ( ( (u32ImmRaw & RT_BIT_32(12)) && !*pf64Bit) || ( !(u32ImmRaw & RT_BIT_32(12)) && *pf64Bit)) return VERR_DIS_INVALID_OPCODE; uint32_t uImm7SizeLen = ((u32ImmRaw & RT_BIT_32(12)) >> 6) | (u32ImmRaw & 0x3f); uint32_t uImm6Rotations = (u32ImmRaw >> 6) & 0x3f; pParam->uValue = *pf64Bit ? Armv8A64ConvertImmRImmS2Mask64(uImm7SizeLen, uImm6Rotations) : Armv8A64ConvertImmRImmS2Mask32(uImm7SizeLen, uImm6Rotations); pParam->armv8.cb = pParam->uValue > UINT32_MAX ? sizeof(uint64_t) : sizeof(uint32_t); pParam->fUse |= pParam->uValue > UINT32_MAX ? DISUSE_IMMEDIATE64 : DISUSE_IMMEDIATE32; return VINF_SUCCESS; } static int disArmV8ParseHw(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pParam); Assert(pInsnParm->cBits == 2); uint32_t u32 = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); /* hw<1> must be 0 if this is the 32-bit variant. */ if ( !*pf64Bit && (u32 & RT_BIT_32(1))) return VERR_DIS_INVALID_OPCODE; Assert(pParam->armv8.enmType == kDisArmv8OpParmImm); Assert(pParam->fUse & (DISUSE_IMMEDIATE8 | DISUSE_IMMEDIATE16 | DISUSE_IMMEDIATE32)); if (u32) { pParam->armv8.enmExtend = kDisArmv8OpParmExtendLsl; pParam->armv8.u.cExtend = ((uint8_t)u32 & 0x3) << 4; } return VINF_SUCCESS; } static int disArmV8ParseCond(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pInsnClass, pOp, pParam, pf64Bit); Assert(pInsnParm->cBits <= 4); if (pParam) { /* Conditional as a parameter (CCMP/CCMN). */ Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmCond; pParam->armv8.Op.enmCond = (DISARMV8INSTRCOND)disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); } else /* Conditional for the base instruction. */ pDis->armv8.enmCond = (DISARMV8INSTRCOND)disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); return VINF_SUCCESS; } static int disArmV8ParsePState(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pInsnParm, pf64Bit); uint32_t u32Op1 = disArmV8ExtractBitVecFromInsn(u32Insn, 16, 3); uint32_t u32Op2 = disArmV8ExtractBitVecFromInsn(u32Insn, 5, 3); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); Assert(pDis->aParams[1].armv8.enmType == kDisArmv8OpParmImm); Assert(pDis->aParams[1].armv8.cb == sizeof(uint8_t)); Assert(pDis->aParams[1].uValue < 16); /* 4 bit field. */ pParam->armv8.enmType = kDisArmv8OpParmPState; uint8_t bCRm = (uint8_t)pDis->aParams[1].uValue; /* See C6.2.249 for the defined values. */ switch ((u32Op1 << 3) | u32Op2) { case 0x03: /* 000 011 */ pParam->armv8.Op.enmPState = kDisArmv8InstrPState_UAO; break; case 0x04: /* 000 100 */ pParam->armv8.Op.enmPState = kDisArmv8InstrPState_PAN; break; case 0x05: /* 000 101 */ pParam->armv8.Op.enmPState = kDisArmv8InstrPState_SPSel; break; case 0x08: /* 001 000 */ { pDis->aParams[1].uValue = bCRm & 0x1; switch (bCRm & 0xe) { case 0: /* 000x */ pParam->armv8.Op.enmPState = kDisArmv8InstrPState_ALLINT; break; case 2: /* 001x */ pParam->armv8.Op.enmPState = kDisArmv8InstrPState_PM; break; default: return VERR_DIS_INVALID_OPCODE; } break; } case 0x19: /* 011 001 */ pParam->armv8.Op.enmPState = kDisArmv8InstrPState_SSBS; break; case 0x1a: /* 011 010 */ pParam->armv8.Op.enmPState = kDisArmv8InstrPState_DIT; break; case 0x1b: /* 011 011 */ { pDis->aParams[1].uValue = bCRm & 0x1; switch (bCRm & 0xe) { case 2: /* 001x */ pParam->armv8.Op.enmPState = kDisArmv8InstrPState_SVCRSM; break; case 4: /* 010x */ pParam->armv8.Op.enmPState = kDisArmv8InstrPState_SVCRZA; break; case 6: /* 011x */ pParam->armv8.Op.enmPState = kDisArmv8InstrPState_SVCRSMZA; break; default: return VERR_DIS_INVALID_OPCODE; } break; } case 0x1c: /* 011 100 */ pParam->armv8.Op.enmPState = kDisArmv8InstrPState_TCO; break; case 0x1e: /* 011 110 */ pParam->armv8.Op.enmPState = kDisArmv8InstrPState_DAIFSet; break; case 0x1f: /* 011 111 */ pParam->armv8.Op.enmPState = kDisArmv8InstrPState_DAIFClr; break; default: return VERR_DIS_INVALID_OPCODE; } return VINF_SUCCESS; } static int disArmV8ParseCRnCRm(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, u32Insn, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); Assert(pInsnParm->cBits == 8); /** @todo Needs implementation. */ return VERR_DIS_INVALID_OPCODE; } static int disArmV8ParseSysReg(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); AssertReturn(pInsnParm->cBits == 15, VERR_INTERNAL_ERROR_2); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmSysReg; /* Assumes a op0:op1:CRn:CRm:op2 encoding in the instruction starting at the given bit position. */ uint32_t u32ImmRaw = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); pParam->armv8.Op.idSysReg = ARMV8_AARCH64_SYSREG_ID_CREATE(2 + ((u32ImmRaw >> 14) & 0x1), (u32ImmRaw >> 11) & 0x7, (u32ImmRaw >> 7) & 0xf, (u32ImmRaw >> 3) & 0xf, u32ImmRaw & 0x7); pParam->armv8.cb = 0; pParam->fUse |= DISUSE_REG_SYSTEM; return VINF_SUCCESS; } static int disArmV8ParseSh12(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); Assert(pInsnParm->cBits == 1); Assert(pParam->armv8.enmType == kDisArmv8OpParmImm); if (u32Insn & RT_BIT_32(pInsnParm->idxBitStart)) { /* Shift the immediate pointed to. */ pParam->uValue <<= 12; /* Re-evaluate the immediate data size. */ pParam->fUse &= ~(DISUSE_IMMEDIATE8 | DISUSE_IMMEDIATE16 | DISUSE_IMMEDIATE32); if (pParam->uValue <= UINT8_MAX) { pParam->armv8.cb = sizeof(uint8_t); pParam->fUse |= DISUSE_IMMEDIATE8; } else if (pParam->uValue <= UINT16_MAX) { pParam->armv8.cb = sizeof(uint16_t); pParam->fUse |= DISUSE_IMMEDIATE16; } else if (pParam->uValue <= UINT32_MAX) { pParam->armv8.cb = sizeof(uint32_t); pParam->fUse |= DISUSE_IMMEDIATE32; } else AssertReleaseFailed(); } return VINF_SUCCESS; } static int disArmV8ParseImmTbz(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); AssertReturn(!pInsnParm->idxBitStart && !pInsnParm->cBits, VERR_INTERNAL_ERROR_2); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmImm; pParam->uValue = disArmV8ExtractBitVecFromInsn(u32Insn, 19, 5); pParam->uValue |= (u32Insn & RT_BIT_32(31)) >> 26; pParam->armv8.cb = sizeof(uint8_t); pParam->fUse |= DISUSE_IMMEDIATE8; return VINF_SUCCESS; } static int disArmV8ParseShift(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); AssertReturn(pInsnParm->cBits == 2, VERR_INTERNAL_ERROR_2); uint32_t u32Shift = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); switch (u32Shift) { case 0: pParam->armv8.enmExtend = kDisArmv8OpParmExtendLsl; break; case 1: pParam->armv8.enmExtend = kDisArmv8OpParmExtendLsr; break; case 2: pParam->armv8.enmExtend = kDisArmv8OpParmExtendAsr; break; case 3: pParam->armv8.enmExtend = kDisArmv8OpParmExtendRor; break; default: AssertReleaseFailed(); } return VINF_SUCCESS; } static int disArmV8ParseShiftAmount(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); uint32_t u32Amount = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); /* For a 32-bit operand it is impossible to shift/rotate more than 31 bits. */ if ( !*pf64Bit && u32Amount > 31) return VERR_DIS_INVALID_OPCODE; Assert(pParam->armv8.enmType != kDisArmv8OpParmNone); Assert(pParam->armv8.enmExtend != kDisArmv8OpParmExtendNone); Assert(u32Amount < 64); pParam->armv8.u.cExtend = (uint8_t)u32Amount; /* Any shift operation with a 0 is essentially no shift being applied. */ if (pParam->armv8.u.cExtend == 0) pParam->armv8.enmExtend = kDisArmv8OpParmExtendNone; return VINF_SUCCESS; } static int disArmV8ParseImmMemOff(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pInsnClass, pOp, pf64Bit); AssertReturn(pInsnParm->cBits <= 12, VERR_INTERNAL_ERROR_2); AssertReturn(pDis->armv8.cbOperand != 0, VERR_INTERNAL_ERROR_2); Assert(pParam->armv8.enmType != kDisArmv8OpParmNone); pParam->armv8.u.offBase = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); switch (pDis->armv8.cbOperand) { case sizeof(uint8_t): break; case sizeof(uint16_t): pParam->armv8.u.offBase <<= 1; break; case sizeof(uint32_t): pParam->armv8.u.offBase <<= 2; break; case sizeof(uint64_t): pParam->armv8.u.offBase <<= 3; break; case 16: pParam->armv8.u.offBase <<= 4; break; default: AssertReleaseFailed(); } pParam->armv8.cb = sizeof(int16_t); return VINF_SUCCESS; } static int disArmV8ParseSImmMemOff(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pInsnClass, pf64Bit); AssertReturn(pInsnParm->cBits <= 7, VERR_INTERNAL_ERROR_2); AssertReturn( (pOp->fFlags & DISARMV8INSNCLASS_F_FORCED_32BIT) || (pOp->fFlags & DISARMV8INSNCLASS_F_FORCED_64BIT) || pDis->armv8.cbOperand != 0, VERR_INTERNAL_ERROR_2); Assert(pParam->armv8.enmType != kDisArmv8OpParmNone); pParam->armv8.cb = sizeof(int16_t); pParam->armv8.u.offBase = disArmV8ExtractBitVecFromInsnSignExtend(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); /** @todo Merge DISARMV8INSNCLASS_F_FORCED_32BIT | DISARMV8INSNCLASS_F_FORCED_64BIT into cbOperand. */ if (pDis->armv8.cbOperand) { switch (pDis->armv8.cbOperand) { case sizeof(uint8_t): break; case sizeof(uint16_t): pParam->armv8.u.offBase <<= 1; break; case sizeof(uint32_t): pParam->armv8.u.offBase <<= 2; break; case sizeof(uint64_t): pParam->armv8.u.offBase <<= 3; break; case 16: pParam->armv8.u.offBase <<= 4; break; default: AssertReleaseFailed(); } } else pParam->armv8.u.offBase <<= (pOp->fFlags & DISARMV8INSNCLASS_F_FORCED_32BIT) ? 2 : 3; return VINF_SUCCESS; } static int disArmV8ParseSImmMemOffUnscaled(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); AssertReturn(pInsnParm->cBits <= 9, VERR_INTERNAL_ERROR_2); pParam->armv8.u.offBase = disArmV8ExtractBitVecFromInsnSignExtend(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); return VINF_SUCCESS; } static int disArmV8ParseOption(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, u32Insn, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); AssertReturn(pInsnParm->cBits == 3, VERR_INTERNAL_ERROR_2); Assert(pParam->armv8.enmType != kDisArmv8OpParmNone); uint32_t u32Opt = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); Assert( pParam->armv8.enmExtend == kDisArmv8OpParmExtendNone /*&& (pParam->fUse & DISUSE_INDEX)*/); /* For add/sub extended register. */ switch (u32Opt) { case 0: pParam->armv8.enmExtend = kDisArmv8OpParmExtendUxtB; break; case 1: pParam->armv8.enmExtend = kDisArmv8OpParmExtendUxtH; break; case 2: pParam->armv8.enmExtend = kDisArmv8OpParmExtendUxtW; break; case 3: pParam->armv8.enmExtend = kDisArmv8OpParmExtendUxtX; break; case 4: pParam->armv8.enmExtend = kDisArmv8OpParmExtendSxtB; break; case 5: pParam->armv8.enmExtend = kDisArmv8OpParmExtendSxtH; break; case 6: pParam->armv8.enmExtend = kDisArmv8OpParmExtendSxtW; break; case 7: pParam->armv8.enmExtend = kDisArmv8OpParmExtendSxtX; break; default: AssertFailed(); } /* When option<1:0> is b11, the 64-bit name of the GPR is used, 32-bit otherwise. */ if (pParam->fUse & DISUSE_INDEX) pParam->armv8.GprIndex.enmRegType = (u32Opt & 0x3) == 0x3 ? kDisOpParamArmV8RegType_Gpr_64Bit : kDisOpParamArmV8RegType_Gpr_32Bit; else { Assert(pParam->armv8.enmType == kDisArmv8OpParmReg); pParam->armv8.Op.Reg.enmRegType = ((u32Opt & 0x3) == 0x3 && *pf64Bit) ? kDisOpParamArmV8RegType_Gpr_64Bit : kDisOpParamArmV8RegType_Gpr_32Bit; } return VINF_SUCCESS; } static int disArmV8ParseS(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, u32Insn, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); AssertReturn(pInsnParm->cBits == 1, VERR_INTERNAL_ERROR_2); Assert(pParam->armv8.enmType != kDisArmv8OpParmNone); bool const fS = RT_BOOL(u32Insn & RT_BIT_32(pInsnParm->idxBitStart)); Assert( pParam->armv8.enmExtend != kDisArmv8OpParmExtendNone && pDis->armv8.cbOperand > 0 && pDis->armv8.cbOperand <= 16); if (fS) { switch (pDis->armv8.cbOperand) { case sizeof(uint8_t): pParam->armv8.u.cExtend = 0; break; case sizeof(uint16_t): pParam->armv8.u.cExtend = 1; break; case sizeof(uint32_t): pParam->armv8.u.cExtend = 2; break; case sizeof(uint64_t): pParam->armv8.u.cExtend = 3; break; case 16: pParam->armv8.u.cExtend = 4; break; default: AssertReleaseFailed(); } } else if (pParam->armv8.enmExtend == kDisArmv8OpParmExtendUxtX) /* UXTX aka LSL can be ignored if S is not set. */ { pParam->armv8.u.cExtend = 0; pParam->armv8.enmExtend = kDisArmv8OpParmExtendNone; } return VINF_SUCCESS; } static int disArmV8ParseSetPreIndexed(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, u32Insn, pOp, pInsnClass, pInsnParm, pf64Bit); pParam->fUse |= DISUSE_PRE_INDEXED; return VINF_SUCCESS; } static int disArmV8ParseSetPostIndexed(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, u32Insn, pOp, pInsnClass, pInsnParm, pf64Bit); pParam->fUse |= DISUSE_POST_INDEXED; return VINF_SUCCESS; } static int disArmV8ParseFpType(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pOp, pInsnClass, pParam, pf64Bit); Assert(pDis->armv8.enmFpType == kDisArmv8InstrFpType_Invalid); uint32_t u32FpType = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); switch (u32FpType) { case 0: pDis->armv8.enmFpType = kDisArmv8InstrFpType_Single; break; case 1: pDis->armv8.enmFpType = kDisArmv8InstrFpType_Double; break; case 3: pDis->armv8.enmFpType = kDisArmv8InstrFpType_Half; break; default: return VERR_DIS_INVALID_OPCODE; } return VINF_SUCCESS; } static int disArmV8ParseFpReg(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pOp, pInsnClass, pParam, pf64Bit); Assert(pDis->armv8.enmFpType != kDisArmv8InstrFpType_Invalid); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmReg; pParam->armv8.Op.Reg.cRegs = 1; pParam->armv8.Op.Reg.idReg = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); switch (pDis->armv8.enmFpType) { case kDisArmv8InstrFpType_Single: pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_FpReg_Single; break; case kDisArmv8InstrFpType_Double: pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_FpReg_Double; break; case kDisArmv8InstrFpType_Half: pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_FpReg_Half; break; default: return VERR_DIS_INVALID_OPCODE; } return VINF_SUCCESS; } static int disArmV8ParseFpScale(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass); Assert(pDis->armv8.enmFpType != kDisArmv8InstrFpType_Invalid); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmImm; uint32_t u32Scale = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); if ( !*pf64Bit && (u32Scale & RT_BIT_32(5)) == 0) return VERR_DIS_INVALID_OPCODE; pParam->uValue = 64 - u32Scale; pParam->armv8.cb = sizeof(uint8_t); pParam->fUse |= DISUSE_IMMEDIATE8; return VINF_SUCCESS; } static int disArmV8ParseFpFixupFCvt(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pInsnClass, pParam, pInsnParm, pf64Bit); /* Nothing to do if this isn't about fcvt. */ if (pOp->Opc.uOpcode != OP_ARMV8_A64_FCVT) return VINF_SUCCESS; Assert(pDis->armv8.enmFpType != kDisArmv8InstrFpType_Invalid); Assert( pDis->aParams[0].armv8.enmType == kDisArmv8OpParmReg && pDis->aParams[1].armv8.enmType == kDisArmv8OpParmReg); /* Convert source and guest register floating point types to the correct widths. */ uint32_t u32Opc = (u32Insn & (RT_BIT_32(15) | RT_BIT_32(16))) >> 15; #ifdef VBOX_STRICT uint32_t u32FpType = disArmV8ExtractBitVecFromInsn(u32Insn, 22, 2); Assert( u32Opc != u32FpType && u32Opc != 2); #endif static const DISOPPARAMARMV8REGTYPE s_aOpc2FpWidth[] = { kDisOpParamArmV8RegType_FpReg_Single, kDisOpParamArmV8RegType_FpReg_Double, (DISOPPARAMARMV8REGTYPE)UINT8_MAX, /* Invalid encoding. */ kDisOpParamArmV8RegType_FpReg_Half }; pDis->aParams[0].armv8.Op.Reg.enmRegType = s_aOpc2FpWidth[u32Opc]; return VINF_SUCCESS; } static int disArmV8ParseSimdRegSize(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pOp, pInsnClass, pParam, pf64Bit); Assert(pInsnParm->cBits == 2); uint32_t u32Size = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); switch (u32Size) { case 0: pDis->armv8.cbOperand = sizeof(uint8_t); break; case 1: pDis->armv8.cbOperand = sizeof(uint16_t); break; case 2: pDis->armv8.cbOperand = sizeof(uint32_t); break; case 3: pDis->armv8.cbOperand = sizeof(uint64_t); break; default: AssertReleaseFailed(); } return VINF_SUCCESS; } static int disArmV8ParseSimdRegSize32(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(u32Insn, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); pDis->armv8.cbOperand = sizeof(uint32_t); return VINF_SUCCESS; } static int disArmV8ParseSimdRegSize64(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(u32Insn, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); pDis->armv8.cbOperand = sizeof(uint64_t); return VINF_SUCCESS; } static int disArmV8ParseSimdRegSize128(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(u32Insn, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); pDis->armv8.cbOperand = 16; return VINF_SUCCESS; } static int disArmV8ParseSimdRegScalar(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); Assert(pDis->armv8.cbOperand != 0); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmReg; pParam->armv8.Op.Reg.cRegs = 1; pParam->armv8.Op.Reg.idReg = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); switch (pDis->armv8.cbOperand) { case sizeof(uint8_t): pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Simd_Scalar_8Bit; break; case sizeof(uint16_t): pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Simd_Scalar_16Bit; break; case sizeof(uint32_t): pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Simd_Scalar_32Bit; break; case sizeof(uint64_t): pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Simd_Scalar_64Bit; break; case 16: pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Simd_Scalar_128Bit; break; } return VINF_SUCCESS; } static int disArmV8ParseImmHImmB(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); Assert(pInsnParm->cBits == 7); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); uint32_t u32ImmRaw = disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits); if (!(u32ImmRaw & RT_BIT_32(6))) /* immh == 0xxx is reserved for the scalar variant. */ return VERR_DIS_INVALID_OPCODE; pParam->armv8.enmType = kDisArmv8OpParmImm; pParam->uValue = 2 * 64 - u32ImmRaw; pParam->armv8.cb = sizeof(uint8_t); pParam->fUse |= DISUSE_IMMEDIATE8; return VINF_SUCCESS; } static int disArmV8ParseSf(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pParam, pInsnParm); Assert(pInsnParm->cBits == 1); Assert(pInsnParm->idxBitStart == 31 || pInsnParm->idxBitStart == 30); *pf64Bit = RT_BOOL(u32Insn & RT_BIT_32(pInsnParm->idxBitStart)); return VINF_SUCCESS; } static int disArmV8ParseImmX16(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); AssertReturn(pInsnParm->idxBitStart + pInsnParm->cBits < 32, VERR_INTERNAL_ERROR_2); Assert(pParam->armv8.enmType == kDisArmv8OpParmNone); pParam->armv8.enmType = kDisArmv8OpParmImm; pParam->uValue = (uint64_t)disArmV8ExtractBitVecFromInsn(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits) * 16; if (pParam->uValue <= UINT8_MAX) { pParam->armv8.cb = sizeof(uint8_t); pParam->fUse |= DISUSE_IMMEDIATE8; } else if (pParam->uValue <= UINT16_MAX) { pParam->armv8.cb = sizeof(uint16_t); pParam->fUse |= DISUSE_IMMEDIATE16; } else if (pParam->uValue <= UINT32_MAX) { pParam->armv8.cb = sizeof(uint32_t); pParam->fUse |= DISUSE_IMMEDIATE32; } else AssertReleaseFailed(); return VINF_SUCCESS; } static int disArmV8ParseSImmTags(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pf64Bit); AssertReturn(pInsnParm->cBits <= 9, VERR_INTERNAL_ERROR_2); Assert(pParam->armv8.enmType != kDisArmv8OpParmNone); pParam->armv8.cb = sizeof(int16_t); pParam->armv8.u.offBase = disArmV8ExtractBitVecFromInsnSignExtend(u32Insn, pInsnParm->idxBitStart, pInsnParm->cBits) << 4; return VINF_SUCCESS; } static int disArmV8ParseLdrPacImm(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); Assert(pInsnParm->cBits == 0); Assert(pInsnParm->idxBitStart == 0); Assert(pParam->armv8.enmType != kDisArmv8OpParmNone); uint32_t u32 = disArmV8ExtractBitVecFromInsn(u32Insn, 12, 9) | ((u32Insn & RT_BIT_32(22)) >> 13); pParam->armv8.cb = sizeof(int16_t); pParam->armv8.u.offBase = disArmV8ExtractBitVecFromInsnSignExtend(u32, 0, 10) << 3; return VINF_SUCCESS; } static int disArmV8ParseLdrPacW(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); Assert(pInsnParm->cBits == 1); Assert(pInsnParm->idxBitStart == 11); if (u32Insn & RT_BIT_32(11)) pParam->fUse |= DISUSE_PRE_INDEXED; return VINF_SUCCESS; } static int disArmV8ParseVecRegElemSize(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); Assert(pInsnParm->cBits == 2); Assert(pInsnParm->idxBitStart == 10); Assert(pDis->armv8.enmVecRegType == kDisOpParamArmV8VecRegType_None); uint32_t u32 = disArmV8ExtractBitVecFromInsn(u32Insn, 10, 2); switch (u32) { case 0: pDis->armv8.enmVecRegType = kDisOpParamArmV8VecRegType_8B; break; case 1: pDis->armv8.enmVecRegType = kDisOpParamArmV8VecRegType_4H; break; case 2: pDis->armv8.enmVecRegType = kDisOpParamArmV8VecRegType_2S; break; case 3: pDis->armv8.enmVecRegType = kDisOpParamArmV8VecRegType_1D; break; default: AssertFailed(); break; } return VINF_SUCCESS; } static int disArmV8ParseVecRegQ(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); Assert(pInsnParm->cBits == 1); Assert(pInsnParm->idxBitStart == 30); Assert( pDis->armv8.enmVecRegType != kDisOpParamArmV8VecRegType_None && pDis->armv8.enmVecRegType < kDisOpParamArmV8VecRegType_2D); /* This ASSUMES that the vector register type for the 64-bit and 128-bit vector register lengths are adjacent. */ if (u32Insn & RT_BIT_32(30)) pDis->armv8.enmVecRegType = (DISOPPARAMARMV8VECREGTYPE)((uint8_t)pDis->armv8.enmVecRegType + 1); return VINF_SUCCESS; } static int disArmV8ParseVecGrp(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, u32Insn, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); /* This is special as it doesn't really parse the instruction but sets the given parameter from vector to group vector and sets the register count based on the number if bits. */ Assert(pInsnParm->cBits <= 4); Assert(pInsnParm->idxBitStart == 0); Assert(pParam->armv8.Op.Reg.enmRegType == kDisOpParamArmV8RegType_Simd_Vector); pParam->armv8.Op.Reg.enmRegType = kDisOpParamArmV8RegType_Simd_Vector_Group; pParam->armv8.Op.Reg.cRegs = pInsnParm->cBits; return VINF_SUCCESS; } static int disArmV8ParseSimdLdStPostIndexImm(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass, PDISOPPARAM pParam, PCDISARMV8INSNPARAM pInsnParm, bool *pf64Bit) { RT_NOREF(pDis, u32Insn, pOp, pInsnClass, pParam, pInsnParm, pf64Bit); /* * Special decoder for Advanced SIMD load/store multiple structures (post-indexed), immediate variant. * The immediate for when Q == 0 is stored in idxBitStart, and cBits when Q == 1. */ Assert(pInsnParm->cBits == 16 || pInsnParm->cBits == 32 || pInsnParm->cBits == 48 || pInsnParm->cBits == 64); Assert(pInsnParm->idxBitStart == 8 || pInsnParm->idxBitStart == 16 || pInsnParm->idxBitStart == 24 || pInsnParm->idxBitStart == 32); Assert(pParam->armv8.Op.Reg.enmRegType == kDisOpParamArmV8RegType_Gpr_64Bit || pParam->armv8.Op.Reg.enmRegType == kDisOpParamArmV8RegType_Sp); pParam->armv8.u.offBase = RT_BOOL(u32Insn & RT_BIT_32(30)) ? pInsnParm->cBits : pInsnParm->idxBitStart; return VINF_SUCCESS; } static uint32_t disArmV8DecodeIllegal(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8INSNCLASS pInsnClass) { RT_NOREF(pDis, u32Insn, pInsnClass); AssertFailed(); return UINT32_MAX; } static uint32_t disArmV8DecodeLookup(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8INSNCLASS pInsnClass) { RT_NOREF(pDis); for (uint32_t i = 0; i < pInsnClass->Hdr.cDecode; i++) { PCDISARMV8OPCODE pOp = &pInsnClass->paOpcodes[i]; if (u32Insn == pOp->fValue) return i; } return UINT32_MAX; } static uint32_t disArmV8DecodeCollate(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8INSNCLASS pInsnClass) { RT_NOREF(pDis); /* Need to build a compact representation of the relevant bits from the mask to create an index. */ uint32_t fMask = pInsnClass->fMask >> pInsnClass->cShift; /** @todo Optimize. */ uint32_t idx = 0; uint32_t cShift = 0; while (fMask) { if (fMask & 0x1) { idx |= (u32Insn & 1) << cShift; cShift++; } u32Insn >>= 1; fMask >>= 1; } if (RT_LIKELY(idx < pInsnClass->Hdr.cDecode)) return idx; return UINT32_MAX; } /** * Looks for possible alias conversions for the given disassembler state. * * @param pDis The disassembler state to process. */ static void disArmV8A64InsnAliasesProcess(PDISSTATE pDis) { #define DIS_ARMV8_ALIAS(a_Name) s_DisArmv8Alias ## a_Name #define DIS_ARMV8_ALIAS_CREATE(a_Name, a_szOpcode, a_uOpcode, a_fOpType) static const DISOPCODE DIS_ARMV8_ALIAS(a_Name) = OP(a_szOpcode, 0, 0, 0, a_uOpcode, 0, 0, 0, a_fOpType) #define DIS_ARMV8_ALIAS_REF(a_Name) &DIS_ARMV8_ALIAS(a_Name) switch (pDis->pCurInstr->uOpcode) { case OP_ARMV8_A64_ORR: { /* Check for possible MOV conversion for the register variant when: shift is None and the first source is the zero register. */ Assert(pDis->aParams[1].armv8.enmType == kDisArmv8OpParmReg); Assert( pDis->aParams[1].armv8.Op.Reg.enmRegType == kDisOpParamArmV8RegType_Gpr_32Bit || pDis->aParams[1].armv8.Op.Reg.enmRegType == kDisOpParamArmV8RegType_Gpr_64Bit); if ( pDis->aParams[2].armv8.enmType == kDisArmv8OpParmReg && pDis->aParams[2].armv8.enmExtend == kDisArmv8OpParmExtendNone && pDis->aParams[1].armv8.Op.Reg.idReg == ARMV8_A64_REG_XZR) { DIS_ARMV8_ALIAS_CREATE(Mov, "mov", OP_ARMV8_A64_MOV, DISOPTYPE_HARMLESS); pDis->pCurInstr = DIS_ARMV8_ALIAS_REF(Mov); pDis->aParams[1] = pDis->aParams[2]; pDis->aParams[2].armv8.enmType = kDisArmv8OpParmNone; } /** @todo Immediate variant. */ break; } case OP_ARMV8_A64_SUBS: { Assert(pDis->aParams[0].armv8.enmType == kDisArmv8OpParmReg); Assert( pDis->aParams[0].armv8.Op.Reg.enmRegType == kDisOpParamArmV8RegType_Gpr_32Bit || pDis->aParams[0].armv8.Op.Reg.enmRegType == kDisOpParamArmV8RegType_Gpr_64Bit); if (pDis->aParams[0].armv8.Op.Reg.idReg == ARMV8_A64_REG_XZR) { DIS_ARMV8_ALIAS_CREATE(Cmp, "cmp", OP_ARMV8_A64_CMP, DISOPTYPE_HARMLESS); pDis->pCurInstr = DIS_ARMV8_ALIAS_REF(Cmp); pDis->aParams[0] = pDis->aParams[1]; pDis->aParams[1] = pDis->aParams[2]; pDis->aParams[2].armv8.enmType = kDisArmv8OpParmNone; } break; } default: break; /* No conversion */ } #undef DIS_ARMV8_ALIAS_REF #undef DIS_ARMV8_ALIAS_CREATE #undef DIS_ARMV8_ALIAS } static int disArmV8A64ParseInstruction(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8OPCODE pOp, PCDISARMV8INSNCLASS pInsnClass) { AssertPtr(pOp); AssertPtr(pDis); //Assert((u32Insn & pInsnClass->fFixedInsn) == pOp->fValue); if ((u32Insn & pInsnClass->fFixedInsn) != pOp->fValue) return VERR_DIS_INVALID_OPCODE; /* Should contain the parameter type on input. */ pDis->aParams[0].fUse = 0; pDis->aParams[1].fUse = 0; pDis->aParams[2].fUse = 0; pDis->aParams[3].fUse = 0; pDis->aParams[0].armv8.enmType = kDisArmv8OpParmNone; pDis->aParams[1].armv8.enmType = kDisArmv8OpParmNone; pDis->aParams[2].armv8.enmType = kDisArmv8OpParmNone; pDis->aParams[3].armv8.enmType = kDisArmv8OpParmNone; pDis->aParams[0].armv8.enmExtend = kDisArmv8OpParmExtendNone; pDis->aParams[1].armv8.enmExtend = kDisArmv8OpParmExtendNone; pDis->aParams[2].armv8.enmExtend = kDisArmv8OpParmExtendNone; pDis->aParams[3].armv8.enmExtend = kDisArmv8OpParmExtendNone; pDis->armv8.enmCond = kDisArmv8InstrCond_Al; pDis->armv8.enmFpType = kDisArmv8InstrFpType_Invalid; pDis->armv8.enmVecRegType = kDisOpParamArmV8VecRegType_None; pDis->armv8.cbOperand = 0; pDis->pCurInstr = &pOp->Opc; Assert(&pOp->Opc != &g_ArmV8A64InvalidOpcode[0]); bool f64Bit = true; if (pOp->fFlags & DISARMV8INSNCLASS_F_FORCED_32BIT) f64Bit = false; else if (pOp->fFlags & DISARMV8INSNCLASS_F_FORCED_64BIT) f64Bit = true; int rc = VINF_SUCCESS; PCDISARMV8INSNPARAM pDecode = &pInsnClass->paParms[0]; if (pOp->paDecode) pDecode = &pOp->paDecode[0]; while ( (pDecode->idxParse != kDisParmParseNop) && RT_SUCCESS(rc)) { Assert(pDecode->idxParse < kDisParmParseMax); Assert(g_apfnDisasm[pDecode->idxParse]); rc = g_apfnDisasm[pDecode->idxParse](pDis, u32Insn, pOp, pInsnClass, pDecode->idxParam != DIS_ARMV8_INSN_PARAM_UNSET ? &pDis->aParams[pDecode->idxParam] : NULL, pDecode, &f64Bit); pDecode++; } /* If parameter parsing returned an invalid opcode error the encoding is invalid. */ if (RT_SUCCESS(rc)) /** @todo Introduce flag to switch alias conversion on/off. */ disArmV8A64InsnAliasesProcess(pDis); else if (rc == VERR_DIS_INVALID_OPCODE) { pDis->pCurInstr = &g_ArmV8A64InvalidOpcode[0]; pDis->aParams[0].armv8.enmType = kDisArmv8OpParmNone; pDis->aParams[1].armv8.enmType = kDisArmv8OpParmNone; pDis->aParams[2].armv8.enmType = kDisArmv8OpParmNone; pDis->aParams[3].armv8.enmType = kDisArmv8OpParmNone; } pDis->rc = rc; return rc; } static int disArmV8A64ParseInvOpcode(PDISSTATE pDis) { pDis->pCurInstr = &g_ArmV8A64InvalidOpcode[0]; pDis->rc = VERR_DIS_INVALID_OPCODE; return VERR_DIS_INVALID_OPCODE; } static int disInstrArmV8DecodeWorker(PDISSTATE pDis, uint32_t u32Insn, PCDISARMV8DECODEHDR pHdr) { while ( pHdr && pHdr->enmDecodeType != kDisArmV8DecodeType_InsnClass) { if (pHdr->enmDecodeType == kDisArmV8DecodeType_Map) { PCDISARMV8DECODEMAP pMap = (PCDISARMV8DECODEMAP)pHdr; uint32_t idxNext = (u32Insn & pMap->fMask) >> pMap->cShift; if (RT_LIKELY(idxNext < pMap->Hdr.cDecode)) pHdr = pMap->papNext[idxNext]; else { pHdr = NULL; break; } } else { Assert(pHdr->enmDecodeType == kDisArmV8DecodeType_Table); PCDISARMV8DECODETBL pTbl = (PCDISARMV8DECODETBL)pHdr; /* Walk all entries in the table and select the best match. */ pHdr = NULL; for (uint32_t i = 0; i < pTbl->Hdr.cDecode; i++) { PCDISARMV8DECODETBLENTRY pEntry = &pTbl->paEntries[i]; if ((u32Insn & pEntry->fMask) == pEntry->fValue) { pHdr = pEntry->pHdrNext; break; } } } } if (pHdr) { Assert(pHdr->enmDecodeType == kDisArmV8DecodeType_InsnClass); PCDISARMV8INSNCLASS pInsnClass = (PCDISARMV8INSNCLASS)pHdr; /* Decode the opcode from the instruction class. */ uint32_t uOpcRaw = 0; if (pInsnClass->Hdr.cDecode > 1) { uOpcRaw = (u32Insn & pInsnClass->fMask) >> pInsnClass->cShift; if (pInsnClass->enmOpcDecode != kDisArmV8OpcDecodeNop) uOpcRaw = g_apfnOpcDecode[pInsnClass->enmOpcDecode](pDis, uOpcRaw, pInsnClass); } if (uOpcRaw < pInsnClass->Hdr.cDecode) { PCDISARMV8OPCODE pOp = &pInsnClass->paOpcodes[uOpcRaw]; return disArmV8A64ParseInstruction(pDis, u32Insn, pOp, pInsnClass); } } return disArmV8A64ParseInvOpcode(pDis); } /** * Internal worker for DISInstrEx and DISInstrWithPrefetchedBytes. * * @returns VBox status code. * @param pDis Initialized disassembler state. * @param paOneByteMap The one byte opcode map to use. * @param pcbInstr Where to store the instruction size. Can be NULL. */ DECLHIDDEN(int) disInstrWorkerArmV8(PDISSTATE pDis, PCDISOPCODE paOneByteMap, uint32_t *pcbInstr) { RT_NOREF(paOneByteMap); if (pDis->uCpuMode == DISCPUMODE_ARMV8_A64) { if (pcbInstr) *pcbInstr = sizeof(uint32_t); /* Instructions are always little endian and 4 bytes. */ uint32_t u32Insn = disReadDWord(pDis, 0 /*offInstr*/); if (RT_FAILURE(pDis->rc)) return pDis->rc; /** @todo r=bird: This is a waste of time if the host is little endian... */ pDis->Instr.u32 = RT_LE2H_U32(u32Insn); pDis->cbInstr = sizeof(u32Insn); return disInstrArmV8DecodeWorker(pDis, u32Insn, &g_aArmV8A64InsnDecodeL0.Hdr); } AssertReleaseFailed(); return VERR_NOT_IMPLEMENTED; } /** * Inlined worker that initializes the disassembler state. * * @returns The primary opcode map to use. * @param pDis The disassembler state. * @param uInstrAddr The instruction address. * @param enmCpuMode The CPU mode. * @param fFilter The instruction filter settings. * @param pfnReadBytes The byte reader, can be NULL. * @param pvUser The user data for the reader. */ DECLHIDDEN(PCDISOPCODE) disInitializeStateArmV8(PDISSTATE pDis, DISCPUMODE enmCpuMode, uint32_t fFilter) { RT_NOREF(pDis, enmCpuMode, fFilter); return NULL; }