/* armdefs.h -- ARMulator common definitions: ARM6 Instruction Emulator. Copyright (C) 1994 Advanced RISC Machines Ltd. 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; either version 2 of the License, or (at your option) any later version. 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, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #pragma once #include #include #include "common/common_types.h" #include "core/arm/skyeye_common/arm_regformat.h" #include "core/gdbstub/gdbstub.h" namespace Core { class System; } namespace Memory { class MemorySystem; } // Signal levels enum { LOW = 0, HIGH = 1, LOWHIGH = 1, HIGHLOW = 2 }; // Cache types enum { NONCACHE = 0, DATACACHE = 1, INSTCACHE = 2, }; // ARM privilege modes enum PrivilegeMode { USER32MODE = 16, FIQ32MODE = 17, IRQ32MODE = 18, SVC32MODE = 19, ABORT32MODE = 23, UNDEF32MODE = 27, SYSTEM32MODE = 31 }; // ARM privilege mode register banks enum { USERBANK = 0, FIQBANK = 1, IRQBANK = 2, SVCBANK = 3, ABORTBANK = 4, UNDEFBANK = 5, DUMMYBANK = 6, SYSTEMBANK = 7 }; // Hardware vector addresses enum { ARMResetV = 0, ARMUndefinedInstrV = 4, ARMSWIV = 8, ARMPrefetchAbortV = 12, ARMDataAbortV = 16, ARMAddrExceptnV = 20, ARMIRQV = 24, ARMFIQV = 28, ARMErrorV = 32, // This is an offset, not an address! ARMul_ResetV = ARMResetV, ARMul_UndefinedInstrV = ARMUndefinedInstrV, ARMul_SWIV = ARMSWIV, ARMul_PrefetchAbortV = ARMPrefetchAbortV, ARMul_DataAbortV = ARMDataAbortV, ARMul_AddrExceptnV = ARMAddrExceptnV, ARMul_IRQV = ARMIRQV, ARMul_FIQV = ARMFIQV }; // Coprocessor status values enum { ARMul_FIRST = 0, ARMul_TRANSFER = 1, ARMul_BUSY = 2, ARMul_DATA = 3, ARMul_INTERRUPT = 4, ARMul_DONE = 0, ARMul_CANT = 1, ARMul_INC = 3 }; // Instruction condition codes enum ConditionCode { EQ = 0, NE = 1, CS = 2, CC = 3, MI = 4, PL = 5, VS = 6, VC = 7, HI = 8, LS = 9, GE = 10, LT = 11, GT = 12, LE = 13, AL = 14, NV = 15, }; // Flags for use with the APSR. enum : u32 { NBIT = (1U << 31U), ZBIT = (1 << 30), CBIT = (1 << 29), VBIT = (1 << 28), QBIT = (1 << 27), JBIT = (1 << 24), EBIT = (1 << 9), ABIT = (1 << 8), IBIT = (1 << 7), FBIT = (1 << 6), TBIT = (1 << 5), // Masks for groups of bits in the APSR. MODEBITS = 0x1F, INTBITS = 0x1C0, }; // Values for Emulate. enum { STOP = 0, // Stop CHANGEMODE = 1, // Change mode ONCE = 2, // Execute just one iteration RUN = 3 // Continuous execution }; struct ARMul_State final { public: explicit ARMul_State(Core::System& system, Memory::MemorySystem& memory, PrivilegeMode initial_mode); void ChangePrivilegeMode(u32 new_mode); void Reset(); // Reads/writes data in big/little endian format based on the // state of the E (endian) bit in the APSR. u8 ReadMemory8(u32 address) const; u16 ReadMemory16(u32 address) const; u32 ReadMemory32(u32 address) const; u64 ReadMemory64(u32 address) const; void WriteMemory8(u32 address, u8 data); void WriteMemory16(u32 address, u16 data); void WriteMemory32(u32 address, u32 data); void WriteMemory64(u32 address, u64 data); u32 ReadCP15Register(u32 crn, u32 opcode_1, u32 crm, u32 opcode_2) const; void WriteCP15Register(u32 value, u32 crn, u32 opcode_1, u32 crm, u32 opcode_2); // Exclusive memory access functions bool IsExclusiveMemoryAccess(u32 address) const { return exclusive_state && exclusive_tag == (address & RESERVATION_GRANULE_MASK); } void SetExclusiveMemoryAddress(u32 address) { exclusive_tag = address & RESERVATION_GRANULE_MASK; exclusive_state = true; } void UnsetExclusiveMemoryAddress() { exclusive_tag = 0xFFFFFFFF; exclusive_state = false; } // Whether or not the given CPU is in big endian mode (E bit is set) bool InBigEndianMode() const { return (Cpsr & (1 << 9)) != 0; } // Whether or not the given CPU is in a mode other than user mode. bool InAPrivilegedMode() const { return (Mode != USER32MODE); } // Whether or not the current CPU mode has a Saved Program Status Register bool CurrentModeHasSPSR() const { return Mode != SYSTEM32MODE && InAPrivilegedMode(); } // Note that for the 3DS, a Thumb instruction will only ever be // two bytes in size. Thus we don't need to worry about ThumbEE // or Thumb-2 where instructions can be 4 bytes in length. u32 GetInstructionSize() const { return TFlag ? 2 : 4; } void RecordBreak(GDBStub::BreakpointAddress bkpt) { last_bkpt = bkpt; last_bkpt_hit = true; } void ServeBreak(); Core::System& system; Memory::MemorySystem& memory; std::array Reg{}; // The current register file std::array Reg_usr{}; std::array Reg_svc{}; // R13_SVC R14_SVC std::array Reg_abort{}; // R13_ABORT R14_ABORT std::array Reg_undef{}; // R13 UNDEF R14 UNDEF std::array Reg_irq{}; // R13_IRQ R14_IRQ std::array Reg_firq{}; // R8---R14 FIRQ std::array Spsr{}; // The exception psr's std::array CP15{}; // FPSID, FPSCR, and FPEXC std::array VFP{}; // VFPv2 and VFPv3-D16 has 16 doubleword registers (D0-D16 or S0-S31). // VFPv3-D32/ASIMD may have up to 32 doubleword registers (D0-D31), // and only 32 singleword registers are accessible (S0-S31). std::array ExtReg{}; u32 Emulate; // To start and stop emulation u32 Cpsr; // The current PSR u32 Spsr_copy; u32 phys_pc; u32 Mode; // The current mode u32 Bank; // The current register bank u32 NFlag, ZFlag, CFlag, VFlag, IFFlags; // Dummy flags for speed unsigned int shifter_carry_out; u32 TFlag; // Thumb state unsigned long long NumInstrs; // The number of instructions executed u64 NumInstrsToExecute; unsigned NresetSig; // Reset the processor unsigned NfiqSig; unsigned NirqSig; unsigned abortSig; unsigned NtransSig; unsigned bigendSig; unsigned syscallSig; // TODO(bunnei): Move this cache to a better place - it should be per codeset (likely per // process for our purposes), not per ARMul_State (which tracks CPU core state). std::unordered_map instruction_cache; private: void ResetMPCoreCP15Registers(); // Defines a reservation granule of 2 words, which protects the first 2 words starting at the // tag. This is the smallest granule allowed by the v7 spec, and is coincidentally just large // enough to support LDR/STREXD. static const u32 RESERVATION_GRANULE_MASK = 0xFFFFFFF8; u32 exclusive_tag; // The address for which the local monitor is in exclusive access mode bool exclusive_state; GDBStub::BreakpointAddress last_bkpt{}; bool last_bkpt_hit = false; };