| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  | /*  armdefs.h -- ARMulator common definitions:  ARM6 Instruction Emulator.
 | 
					
						
							|  |  |  |     Copyright (C) 1994 Advanced RISC Machines Ltd. | 
					
						
							| 
									
										
										
										
											2015-05-25 20:34:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  |     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. | 
					
						
							| 
									
										
										
										
											2015-05-25 20:34:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  |     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. | 
					
						
							| 
									
										
										
										
											2015-05-25 20:34:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  |     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. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-30 13:24:19 -05:00
										 |  |  | #pragma once
 | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-26 11:39:57 -04:00
										 |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2015-04-27 22:44:05 -04:00
										 |  |  | #include <unordered_map>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-13 20:23:32 -05:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2015-02-13 08:08:21 -05:00
										 |  |  | #include "core/arm/skyeye_common/arm_regformat.h"
 | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-11 10:14:20 -05:00
										 |  |  | // Signal levels
 | 
					
						
							|  |  |  | enum { | 
					
						
							|  |  |  |     LOW     = 0, | 
					
						
							|  |  |  |     HIGH    = 1, | 
					
						
							|  |  |  |     LOWHIGH = 1, | 
					
						
							|  |  |  |     HIGHLOW = 2 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Cache types
 | 
					
						
							|  |  |  | enum { | 
					
						
							|  |  |  |     NONCACHE  = 0, | 
					
						
							|  |  |  |     DATACACHE = 1, | 
					
						
							|  |  |  |     INSTCACHE = 2, | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-26 05:39:54 -04:00
										 |  |  | // ARM privilege modes
 | 
					
						
							|  |  |  | enum PrivilegeMode { | 
					
						
							|  |  |  |     USER32MODE   = 16, | 
					
						
							|  |  |  |     FIQ32MODE    = 17, | 
					
						
							|  |  |  |     IRQ32MODE    = 18, | 
					
						
							|  |  |  |     SVC32MODE    = 19, | 
					
						
							|  |  |  |     ABORT32MODE  = 23, | 
					
						
							|  |  |  |     UNDEF32MODE  = 27, | 
					
						
							|  |  |  |     SYSTEM32MODE = 31 | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-26 05:39:54 -04:00
										 |  |  | // ARM privilege mode register banks
 | 
					
						
							|  |  |  | enum { | 
					
						
							|  |  |  |     USERBANK   = 0, | 
					
						
							|  |  |  |     FIQBANK    = 1, | 
					
						
							|  |  |  |     IRQBANK    = 2, | 
					
						
							|  |  |  |     SVCBANK    = 3, | 
					
						
							|  |  |  |     ABORTBANK  = 4, | 
					
						
							|  |  |  |     UNDEFBANK  = 5, | 
					
						
							|  |  |  |     DUMMYBANK  = 6, | 
					
						
							|  |  |  |     SYSTEMBANK = 7 | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-26 05:39:54 -04:00
										 |  |  | // Hardware vector addresses
 | 
					
						
							| 
									
										
										
										
											2015-01-30 12:43:58 -05:00
										 |  |  | 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 | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-26 05:39:54 -04:00
										 |  |  | // Coprocessor status values
 | 
					
						
							| 
									
										
										
										
											2015-01-30 12:43:58 -05:00
										 |  |  | 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 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-26 05:39:54 -04:00
										 |  |  | // Instruction condition codes
 | 
					
						
							| 
									
										
										
										
											2014-10-23 01:20:01 -02:00
										 |  |  | 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, | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-23 20:51:02 -04:00
										 |  |  | // Flags for use with the APSR.
 | 
					
						
							|  |  |  | enum : u32 { | 
					
						
							| 
									
										
										
										
											2015-07-25 21:10:41 -04:00
										 |  |  |     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, | 
					
						
							| 
									
										
										
										
											2015-05-23 20:51:02 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Values for Emulate.
 | 
					
						
							|  |  |  | enum { | 
					
						
							| 
									
										
										
										
											2015-07-25 21:10:41 -04:00
										 |  |  |     STOP       = 0, // Stop
 | 
					
						
							|  |  |  |     CHANGEMODE = 1, // Change mode
 | 
					
						
							|  |  |  |     ONCE       = 2, // Execute just one iteration
 | 
					
						
							|  |  |  |     RUN        = 3  // Continuous execution
 | 
					
						
							| 
									
										
										
										
											2015-05-23 20:51:02 -04:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2015-07-26 05:39:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-26 11:39:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-26 05:39:54 -04:00
										 |  |  | struct ARMul_State final | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit ARMul_State(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.
 | 
					
						
							|  |  |  |     u16 ReadMemory16(u32 address) const; | 
					
						
							|  |  |  |     u32 ReadMemory32(u32 address) const; | 
					
						
							|  |  |  |     u64 ReadMemory64(u32 address) const; | 
					
						
							|  |  |  |     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); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 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); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // 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; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-26 11:39:57 -04:00
										 |  |  |     std::array<u32, 16> Reg;      // The current register file
 | 
					
						
							|  |  |  |     std::array<u32, 2> Reg_usr; | 
					
						
							|  |  |  |     std::array<u32, 2> Reg_svc;   // R13_SVC R14_SVC
 | 
					
						
							|  |  |  |     std::array<u32, 2> Reg_abort; // R13_ABORT R14_ABORT
 | 
					
						
							|  |  |  |     std::array<u32, 2> Reg_undef; // R13 UNDEF R14 UNDEF
 | 
					
						
							|  |  |  |     std::array<u32, 2> Reg_irq;   // R13_IRQ R14_IRQ
 | 
					
						
							|  |  |  |     std::array<u32, 7> Reg_firq;  // R8---R14 FIRQ
 | 
					
						
							|  |  |  |     std::array<u32, 7> Spsr;      // The exception psr's
 | 
					
						
							|  |  |  |     std::array<u32, CP15_REGISTER_COUNT> CP15; | 
					
						
							| 
									
										
										
										
											2015-07-26 05:39:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // FPSID, FPSCR, and FPEXC
 | 
					
						
							| 
									
										
										
										
											2015-07-26 11:39:57 -04:00
										 |  |  |     std::array<u32, VFP_SYSTEM_REGISTER_COUNT> VFP; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-26 05:39:54 -04:00
										 |  |  |     // 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).
 | 
					
						
							| 
									
										
										
										
											2015-07-26 11:39:57 -04:00
										 |  |  |     std::array<u32, 64> 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 exclusive_tag; // The address for which the local monitor is in exclusive access mode
 | 
					
						
							|  |  |  |     u32 exclusive_state; | 
					
						
							|  |  |  |     u32 exclusive_result; | 
					
						
							| 
									
										
										
										
											2015-07-26 05:39:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     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
 | 
					
						
							|  |  |  |     unsigned 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<u32, int> instruction_cache; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     void ResetMPCoreCP15Registers(); | 
					
						
							|  |  |  | }; |