mirror of
				https://github.com/RaySollium99/picodrive.git
				synced 2025-10-26 16:29:37 -04:00 
			
		
		
		
	switch Cyclone to submodule on it's own git repo
that version also has it's license clearly stated by fdave
This commit is contained in:
		
							parent
							
								
									e743be2070
								
							
						
					
					
						commit
						1b85bf1c23
					
				
					 38 changed files with 4 additions and 7740 deletions
				
			
		
							
								
								
									
										3
									
								
								.gitmodules
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitmodules
									
										
									
									
										vendored
									
									
								
							|  | @ -1,3 +1,6 @@ | |||
| [submodule "platform/libpicofe"] | ||||
| 	path = platform/libpicofe | ||||
| 	url = git://notaz.gp2x.de/~notaz/libpicofe.git | ||||
| [submodule "cpu/cyclone"] | ||||
| 	path = cpu/cyclone | ||||
| 	url = git://notaz.gp2x.de/~notaz/cyclone68000.git | ||||
|  |  | |||
|  | @ -1,97 +0,0 @@ | |||
| 
 | ||||
| // Cyclone 68000 Emulator - Header File
 | ||||
| 
 | ||||
| // (c) Copyright 2004 Dave, All rights reserved.
 | ||||
| // (c) 2005-2007 notaz
 | ||||
| // Cyclone 68000 is free for non-commercial use.
 | ||||
| 
 | ||||
| // For commercial use, separate licencing terms must be obtained.
 | ||||
| 
 | ||||
| 
 | ||||
| #ifndef __CYCLONE_H__ | ||||
| #define __CYCLONE_H__ | ||||
| 
 | ||||
| #ifdef __cplusplus | ||||
| extern "C" { | ||||
| #endif | ||||
| 
 | ||||
| extern int CycloneVer; // Version number of library
 | ||||
| 
 | ||||
| struct Cyclone | ||||
| { | ||||
|   unsigned int d[8];    // [r7,#0x00]
 | ||||
|   unsigned int a[8];    // [r7,#0x20]
 | ||||
|   unsigned int pc;      // [r7,#0x40] Memory Base (.membase) + 68k PC
 | ||||
|   unsigned char srh;    // [r7,#0x44] Status Register high (T_S__III)
 | ||||
|   unsigned char unused; // [r7,#0x45] Unused
 | ||||
|   unsigned char flags;  // [r7,#0x46] Flags (ARM order: ____NZCV) [68k order is XNZVC]
 | ||||
|   unsigned char irq;    // [r7,#0x47] IRQ level
 | ||||
|   unsigned int osp;     // [r7,#0x48] Other Stack Pointer (USP/SSP)
 | ||||
|   unsigned int xc;      // [r7,#0x4c] Extend flag (bit29: ??X? _)
 | ||||
|   unsigned int prev_pc; // [r7,#0x50] Set to start address of currently executed opcode + 2 (if enabled in config.h)
 | ||||
|   unsigned int jumptab; // [r7,#0x54] Jump table pointer
 | ||||
|   int state_flags;      // [r7,#0x58] bit: 0: stopped state, 1: trace state, 2: activity bit, 3: addr error, 4: fatal halt
 | ||||
|   int cycles;           // [r7,#0x5c] Number of cycles to execute - 1. Updates to cycles left after CycloneRun()
 | ||||
|   int membase;          // [r7,#0x60] Memory Base (ARM address minus 68000 address)
 | ||||
|   unsigned int (*checkpc)(unsigned int pc); // [r7,#0x64] called to recalc Memory Base+pc
 | ||||
|   unsigned int (*read8  )(unsigned int a);  // [r7,#0x68]
 | ||||
|   unsigned int (*read16 )(unsigned int a);  // [r7,#0x6c]
 | ||||
|   unsigned int (*read32 )(unsigned int a);  // [r7,#0x70]
 | ||||
|   void (*write8 )(unsigned int a,unsigned char  d); // [r7,#0x74]
 | ||||
|   void (*write16)(unsigned int a,unsigned short d); // [r7,#0x78]
 | ||||
|   void (*write32)(unsigned int a,unsigned int   d); // [r7,#0x7c]
 | ||||
|   unsigned int (*fetch8 )(unsigned int a);  // [r7,#0x80]
 | ||||
|   unsigned int (*fetch16)(unsigned int a);  // [r7,#0x84]
 | ||||
|   unsigned int (*fetch32)(unsigned int a);  // [r7,#0x88]
 | ||||
|   int  (*IrqCallback)(int int_level);       // [r7,#0x8c] optional irq callback function, see config.h
 | ||||
|   void (*ResetCallback)(void);              // [r7,#0x90] if enabled in config.h, calls this whenever RESET opcode is encountered.
 | ||||
|   int  (*UnrecognizedCallback)(void);       // [r7,#0x94] if enabled in config.h, calls this whenever unrecognized opcode is encountered.
 | ||||
|   unsigned int internal[6];                 // [r7,#0x98] reserved for internal use, do not change.
 | ||||
| }; | ||||
| 
 | ||||
| // Initialize. Used only if Cyclone was compiled with compressed jumptable, see config.h
 | ||||
| void CycloneInit(void); | ||||
| 
 | ||||
| // Reset
 | ||||
| void CycloneReset(struct Cyclone *pcy); | ||||
| 
 | ||||
| // Run cyclone. Cycles should be specified in context (pcy->cycles)
 | ||||
| void CycloneRun(struct Cyclone *pcy); | ||||
| 
 | ||||
| // Utility functions to get and set SR
 | ||||
| void CycloneSetSr(struct Cyclone *pcy, unsigned int sr); | ||||
| unsigned int CycloneGetSr(const struct Cyclone *pcy); | ||||
| 
 | ||||
| // Generates irq exception if needed (if pcy->irq > mask).
 | ||||
| // Returns cycles used for exception if it was generated, 0 otherwise.
 | ||||
| int CycloneFlushIrq(struct Cyclone *pcy); | ||||
| 
 | ||||
| // Functions for saving and restoring state.
 | ||||
| // CycloneUnpack() uses checkpc(), so it must be initialized.
 | ||||
| // save_buffer must point to buffer of 128 (0x80) bytes of size.
 | ||||
| void CyclonePack(const struct Cyclone *pcy, void *save_buffer); | ||||
| void CycloneUnpack(struct Cyclone *pcy, const void *save_buffer); | ||||
| 
 | ||||
| // genesis: if 1, switch to normal TAS handlers
 | ||||
| void CycloneSetRealTAS(int use_real); | ||||
| 
 | ||||
| 
 | ||||
| // These values are special return values for IrqCallback.
 | ||||
| 
 | ||||
| // Causes an interrupt autovector (0x18 + interrupt level) to be taken.
 | ||||
| // This happens in a real 68K if VPA or AVEC is asserted during an interrupt
 | ||||
| // acknowledge cycle instead of DTACK (the most common situation).
 | ||||
| #define CYCLONE_INT_ACK_AUTOVECTOR    -1 | ||||
| 
 | ||||
| // Causes the spurious interrupt vector (0x18) to be taken
 | ||||
| // This happens in a real 68K if BERR is asserted during the interrupt
 | ||||
| // acknowledge cycle (i.e. no devices responded to the acknowledge).
 | ||||
| #define CYCLONE_INT_ACK_SPURIOUS      -2 | ||||
| 
 | ||||
| 
 | ||||
| #ifdef __cplusplus | ||||
| } // End of extern "C"
 | ||||
| #endif | ||||
| 
 | ||||
| #endif // __CYCLONE_H__
 | ||||
| 
 | ||||
|  | @ -1,627 +0,0 @@ | |||
| 
 | ||||
|       _____            __                                      | ||||
|      / ___/__ __ ____ / /___   ___  ___   ___________________  | ||||
|     / /__ / // // __// // _ \ / _ \/ -_) ___________________   | ||||
|     \___/ \_, / \__//_/ \___//_//_/\__/ ___________________    | ||||
|          /___/                                                 | ||||
|          ___________________  ____ ___   ___   ___   ___       | ||||
|         ___________________  / __// _ \ / _ \ / _ \ / _ \      | ||||
|        ___________________  / _ \/ _  // // // // // // /      | ||||
|                             \___/\___/ \___/ \___/ \___/       | ||||
|                                                                | ||||
| ___________________________________________________________________________ | ||||
| 
 | ||||
|   Cyclone 68000 (c) Copyright 2004 Dave.   Free for non-commercial use | ||||
| 
 | ||||
|   Homepage: http://www.finalburn.com/ | ||||
|   Dave's e-mail: emudave(atsymbol)googlemail.com | ||||
|   Replace (atsymbol) with @ | ||||
| 
 | ||||
|   Additional coding and bugfixes done by notaz, 2005-2007 | ||||
|   Homepage: http://notaz.gp2x.de | ||||
|   e-mail: notasas(atsymbol)gmail.com | ||||
| ___________________________________________________________________________ | ||||
| 
 | ||||
| 
 | ||||
| About | ||||
| ----- | ||||
| 
 | ||||
| Cyclone 68000 is an emulator for the 68000 microprocessor, written in ARM 32-bit assembly. | ||||
| It is aimed at chips such as ARM7 and ARM9 cores, StrongARM and XScale, to interpret 68000 | ||||
| code as fast as possible. It can emulate all 68000 instructions quite accurately, instruction | ||||
| timing was synchronized with MAME's Musashi. Most 68k features are emulated (trace mode, | ||||
| address errors), but prefetch is not emulated. | ||||
| 
 | ||||
| 
 | ||||
| How to Compile | ||||
| -------------- | ||||
| 
 | ||||
| Like Starscream and A68K, Cyclone uses a 'Core Creator' program which calculates and outputs | ||||
| all possible 68000 Opcodes and a jump table into file called Cyclone.s or Cyclone.asm. | ||||
| Only Cyclone.h and the mentioned .s or .asm file will be needed for your project, other files | ||||
| are here to produce or test it. | ||||
| 
 | ||||
| First unzip "Cyclone.zip" into a "Cyclone" directory. The next thing to do is to edit config.h | ||||
| file to tune Cyclone for your project. There are lots of options in config.h, but all of them | ||||
| are documented and have defaults. You should set a define value to 1 to enable option, and | ||||
| to 0 to disable. | ||||
| 
 | ||||
| After you are done with config.h, save it and compile Cyclone. If you are using Linux, Cygwin, | ||||
| mingw or similar, you can simply cd to Cyclone/proj and type "make". If you are under Windows | ||||
| and have Visual Studio installed, you can import cyclone.dsp in the proj/ directory and compile | ||||
| it from there (this will produce cyclone.exe which you will have to run to get .s or .asm). | ||||
| You can also use Microsoft command line compile tools by entering Cyclone/proj directory and | ||||
| typing "nmake -f Makefile.win". Note that this step is done only to produce .s or .asm, and it | ||||
| is done using native tools on your PC (not using cross-compiler or similar). | ||||
| 
 | ||||
| The .s file is meant to be compiled with GNU assembler, and .asm with ARMASM.EXE | ||||
| (the Microsoft ARM assembler). Once you have the file, you can add it to your | ||||
| Makefile/project/whatever. | ||||
| 
 | ||||
| 
 | ||||
| Adding to your project | ||||
| ---------------------- | ||||
| 
 | ||||
| Compiling the .s or .asm (from previous step) for your target platform may require custom | ||||
| build rules in your Makefile/project. | ||||
| 
 | ||||
| If you use some gcc-based toolchain, you will need to add Cyclone.o to an object list in | ||||
| the Makefile. GNU make will use "as" to build Cyclone.o from Cyclone.s by default, so | ||||
| you may need to define correct cross-assembler by setting AS variable like this: | ||||
| 
 | ||||
| AS = arm-linux-as | ||||
| 
 | ||||
| This might be different in your case, basically it should be same prefix as for gcc. | ||||
| You may also need to specify floating point type in your assembler flags for Cyclone.o | ||||
| to link properly. This is done like this: | ||||
| 
 | ||||
| ASFLAGS = -mfloat-abi=soft | ||||
| 
 | ||||
| Note that Cyclone does not use floating points, this is just to make the linker happy. | ||||
| 
 | ||||
| 
 | ||||
| If you are using Visual Studio, you may need to add "custom build step", which creates | ||||
| Cyclone.obj from Cyclone.asm (asmasm.exe Cyclone.asm). Alternatively you can create | ||||
| Cyclone.obj by using armasm once and then just add it to you project. | ||||
| 
 | ||||
| Don't worry if this seem very minimal - its all you need to run as many 68000s as you want. | ||||
| It works with both C and C++. | ||||
| 
 | ||||
| 
 | ||||
| Byteswapped Memory | ||||
| ------------------ | ||||
| 
 | ||||
| If you have used Starscream, A68K or Turbo68K or similar emulators you'll be familiar with this! | ||||
| 
 | ||||
| Any memory which the 68000 can access directly must be have every two bytes swapped around. | ||||
| This is to speed up 16-bit memory accesses, because the 68000 has Big-Endian memory | ||||
| and ARM has Little-Endian memory (in most cases). | ||||
| 
 | ||||
| Now you may think you only technically have to byteswap ROM, not RAM, because | ||||
| 16-bit RAM reads go through a memory handler and you could just return (mem[a]<<8) | mem[a+1]. | ||||
| 
 | ||||
| This would work, but remember some systems can execute code from RAM as well as ROM, and | ||||
| that would fail. | ||||
| So it's best to use byteswapped ROM and RAM if the 68000 can access it directly. | ||||
| It's also faster for the memory handlers, because you can do this: | ||||
|    | ||||
|   return *(unsigned short *)(mem+a) | ||||
| 
 | ||||
| 
 | ||||
| Declaring Memory handlers | ||||
| ------------------------- | ||||
| 
 | ||||
| Before you can reset or execute 68000 opcodes you must first set up a set of memory handlers. | ||||
| There are 7 functions you have to set up per CPU, like this: | ||||
| 
 | ||||
|   static unsigned int   MyCheckPc(unsigned int pc) | ||||
|   static unsigned char  MyRead8  (unsigned int a) | ||||
|   static unsigned short MyRead16 (unsigned int a) | ||||
|   static unsigned int   MyRead32 (unsigned int a) | ||||
|   static void MyWrite8 (unsigned int a,unsigned char  d) | ||||
|   static void MyWrite16(unsigned int a,unsigned short d) | ||||
|   static void MyWrite32(unsigned int a,unsigned int   d) | ||||
| 
 | ||||
| You can think of these functions representing the 68000's memory bus. | ||||
| The Read and Write functions are called whenever the 68000 reads or writes memory. | ||||
| For example you might set MyRead8 like this: | ||||
| 
 | ||||
|   unsigned char MyRead8(unsigned int a) | ||||
|   { | ||||
|     a&=0xffffff; // Clip address to 24-bits | ||||
| 
 | ||||
|     if (a<RomLength) return RomData[a^1]; // ^1 because the memory is byteswapped | ||||
|     if (a>=0xe00000) return RamData[(a^1)&0xffff]; | ||||
|     return 0xff; // Out of range memory access | ||||
|   } | ||||
| 
 | ||||
| The other 5 read/write functions are similar. I'll describe the CheckPc function later on. | ||||
| 
 | ||||
| 
 | ||||
| Declaring a CPU Context | ||||
| ----------------------- | ||||
| 
 | ||||
| To declare a CPU simple declare a struct Cyclone in your code (don't forget to include Cyclone.h). | ||||
| For example to declare two 68000s: | ||||
| 
 | ||||
|   struct Cyclone MyCpu; | ||||
|   struct Cyclone MyCpu2; | ||||
| 
 | ||||
| It's probably a good idea to initialize the memory to zero: | ||||
| 
 | ||||
|   memset(&MyCpu, 0,sizeof(MyCpu)); | ||||
|   memset(&MyCpu2,0,sizeof(MyCpu2)); | ||||
| 
 | ||||
| Next point to your memory handlers: | ||||
| 
 | ||||
|   MyCpu.checkpc=MyCheckPc; | ||||
|   MyCpu.read8  =MyRead8; | ||||
|   MyCpu.read16 =MyRead16; | ||||
|   MyCpu.read32 =MyRead32; | ||||
|   MyCpu.write8 =MyWrite8; | ||||
|   MyCpu.write16=MyWrite16; | ||||
|   MyCpu.write32=MyWrite32; | ||||
| 
 | ||||
| You also need to point the fetch handlers - for most systems out there you can just | ||||
| point them at the read handlers: | ||||
|   MyCpu.fetch8  =MyRead8; | ||||
|   MyCpu.fetch16 =MyRead16; | ||||
|   MyCpu.fetch32 =MyRead32; | ||||
| 
 | ||||
| ( Why a different set of function pointers for fetch? | ||||
|   Well there are some systems, the main one being CPS2, which return different data | ||||
|   depending on whether the 'fetch' line on the 68000 bus is high or low. | ||||
|   If this is the case, you can set up different functions for fetch reads. | ||||
|   Generally though you don't need to. ) | ||||
| 
 | ||||
| Now you are nearly ready to reset the 68000, except a few more functions, | ||||
| one of them is: checkpc(). | ||||
| 
 | ||||
| 
 | ||||
| The checkpc() function | ||||
| ---------------------- | ||||
| 
 | ||||
| When Cyclone reads opcodes, it doesn't use a memory handler every time, this would be | ||||
| far too slow, instead it uses a direct pointer to ARM memory. | ||||
| For example if your Rom image was at 0x3000000 and the program counter was $206, | ||||
| Cyclone's program counter would be 0x3000206. | ||||
| 
 | ||||
| The difference between an ARM address and a 68000 address is also stored in a variable called | ||||
| 'membase'. In the above example it's 0x3000000. To retrieve the real 68k PC, Cyclone just | ||||
| subtracts 'membase'. | ||||
| 
 | ||||
| When a long jump happens, Cyclone calls checkpc(). If the PC is in a different bank, | ||||
| for example Ram instead of Rom, change 'membase', recalculate the new PC and return it: | ||||
| 
 | ||||
| static int MyCheckPc(unsigned int pc) | ||||
| { | ||||
|   pc-=MyCpu.membase; // Get the real program counter | ||||
| 
 | ||||
|   if (pc<RomLength) MyCpu.membase=(int)RomMem;          // Jump to Rom | ||||
|   if (pc>=0xff0000) MyCpu.membase=(int)RamMem-0xff0000; // Jump to Ram | ||||
| 
 | ||||
|   return MyCpu.membase+pc; // New program counter | ||||
| } | ||||
| 
 | ||||
| Notice that the membase is always ARM address minus 68000 address. | ||||
| 
 | ||||
| The above example doesn't consider mirrored ram, but for an example of what to do see | ||||
| PicoDrive (in Memory.c). | ||||
| 
 | ||||
| The exact cases when checkpc() is called can be configured in config.h. | ||||
| 
 | ||||
| 
 | ||||
| Initialization | ||||
| -------------- | ||||
| 
 | ||||
| Add a call to CycloneInit(). This is really only needed to be called once at startup | ||||
| if you enabled COMPRESS_JUMPTABLE in config.h, but you can add this in any case, | ||||
| it won't hurt. | ||||
| 
 | ||||
| 
 | ||||
| Almost there - Reset the 68000! | ||||
| ------------------------------- | ||||
| 
 | ||||
| Cyclone doesn't provide a reset function, so next we need to Reset the 68000 to get | ||||
| the initial Program Counter and Stack Pointer. This is obtained from addresses | ||||
| 000000 and 000004. | ||||
| 
 | ||||
| Here is code which resets the 68000 (using your memory handlers): | ||||
| 
 | ||||
|   MyCpu.state_flags=0; // Go to default state (not stopped, halted, etc.) | ||||
|   MyCpu.srh=0x27; // Set supervisor mode | ||||
|   MyCpu.a[7]=MyCpu.read32(0); // Get Stack Pointer | ||||
|   MyCpu.membase=0; // Will be set by checkpc() | ||||
|   MyCpu.pc=MyCpu.checkpc(MyCpu.read32(4)); // Get Program Counter | ||||
| 
 | ||||
| And that's ready to go. | ||||
| 
 | ||||
| 
 | ||||
| Executing the 68000 | ||||
| ------------------- | ||||
| 
 | ||||
| To execute the 68000, set the 'cycles' variable to the number of cycles you wish to execute, | ||||
| and then call CycloneRun with a pointer to the Cyclone structure. | ||||
| 
 | ||||
| e.g.: | ||||
|   // Execute 1000 cycles on the 68000: | ||||
|   MyCpu.cycles=1000; CycloneRun(&MyCpu); | ||||
| 
 | ||||
| For each opcode, the number of cycles it took is subtracted and the function returns when | ||||
| it reaches negative number. The result is stored back to MyCpu.cycles. | ||||
| 
 | ||||
| e.g. | ||||
|   // Execute one instruction on the 68000: | ||||
|   MyCpu.cycles=0; CycloneRun(&MyCpu); | ||||
|   printf("  The opcode took %d cycles\n", -MyCpu.cycles); | ||||
| 
 | ||||
| You should try to execute as many cycles as you can for maximum speed. | ||||
| The number actually executed may be slightly more than requested, i.e. cycles may come | ||||
| out with a small negative value: | ||||
| 
 | ||||
| e.g. | ||||
|   int todo=12000000/60; // 12Mhz, for one 60hz frame | ||||
|   MyCpu.cycles=todo; CycloneRun(&MyCpu); | ||||
|   printf("  Actually executed %d cycles\n", todo-MyCpu.cycles); | ||||
| 
 | ||||
| To calculate the number of cycles executed, use this formula: | ||||
|   Number of cycles requested - Cycle counter at the end | ||||
| 
 | ||||
| 
 | ||||
| Interrupts | ||||
| ---------- | ||||
| 
 | ||||
| Causing an interrupt is very simple, simply set the irq variable in the Cyclone structure | ||||
| to the IRQ number. | ||||
| To lower the IRQ line, set it to zero. | ||||
| 
 | ||||
| e.g: | ||||
|   MyCpu.irq=6; // Interrupt level 6 | ||||
|   MyCpu.cycles=20000; CycloneRun(&MyCpu); | ||||
| 
 | ||||
| Note that the interrupt is not actually processed until the next call to CycloneRun, | ||||
| and the interrupt may not be taken until the 68000 interrupt mask is changed to allow it. | ||||
| 
 | ||||
| If you need to force interrupt processing, you can use CycloneFlushIrq() function. | ||||
| It is the same as doing | ||||
| 
 | ||||
| MyCpu.cycles=0; CycloneRun(&MyCpu); | ||||
| 
 | ||||
| but is better optimized and doesn't update .cycles (returns them instead). | ||||
| This function can't be used from memory handlers and has no effect if interrupt is masked. | ||||
| 
 | ||||
| The IRQ isn't checked on exiting from a memory handler. If you need to cause interrupt | ||||
| check immediately, you should change cycle counter to 0 to cause a return from CycloneRun(), | ||||
| and then call CycloneRun() again or just call CycloneFlushIrq(). Note that you need to | ||||
| enable MEMHANDLERS_CHANGE_CYCLES in config.h for this to work. | ||||
| 
 | ||||
| If you need to do something during the interrupt acknowledge (the moment when interrupt | ||||
| is taken), you can set USE_INT_ACK_CALLBACK in config.h and specify IrqCallback function. | ||||
| This function should update the IRQ level (.irq variable in context) and return the | ||||
| interrupt vector number. But for most cases it should return special constant | ||||
| CYCLONE_INT_ACK_AUTOVECTOR so that Cyclone uses autovectors, which is what most real | ||||
| systems were doing. Another less commonly used option is to return CYCLONE_INT_ACK_SPURIOUS | ||||
| for spurious interrupt. | ||||
| 
 | ||||
| 
 | ||||
| Accessing Program Counter and registers | ||||
| --------------------------------------- | ||||
| 
 | ||||
| You can read most Cyclone's registers directly from the structure at any time. | ||||
| However, the PC value, CCR and cycle counter are cached in ARM registers and can't | ||||
| be accessed from memory handlers by default. They are written back and can be | ||||
| accessed after execution. | ||||
| 
 | ||||
| But if you need to access the mentioned registers during execution, you can set | ||||
| MEMHANDLERS_NEED_* and MEMHANDLERS_CHANGE_* options in config.h | ||||
| 
 | ||||
| The Program Counter, should you need to read or write it, is stored with membase | ||||
| added on. So use this formula to calculate the real 68000 program counter: | ||||
| 
 | ||||
|   pc = MyCpu.pc - MyCpu.membase; | ||||
| 
 | ||||
| For performance reasons Cyclone keeps the status register split into .srh | ||||
| (status register "high" supervisor byte), .xc for the X flag, and .flags for remaining | ||||
| CCR flags (in ARM order). To easily read/write the status register as normal 68k | ||||
| 16bit SR register, use CycloneGetSr() and CycloneSetSr() utility functions. | ||||
| 
 | ||||
| 
 | ||||
| Emulating more than one CPU | ||||
| --------------------------- | ||||
| 
 | ||||
| Since everything is based on the structures, emulating more than one cpu at the same time | ||||
| is just a matter of declaring more than one structures and timeslicing. You can emulate | ||||
| as many 68000s as you want. | ||||
| Just set up the memory handlers for each cpu and run each cpu for a certain number of cycles. | ||||
| 
 | ||||
| e.g. | ||||
|   // Execute 1000 cycles on 68000 #1: | ||||
|   MyCpu.cycles=1000; CycloneRun(&MyCpu); | ||||
| 
 | ||||
|   // Execute 1000 cycles on 68000 #2: | ||||
|   MyCpu2.cycles=1000; CycloneRun(&MyCpu2); | ||||
| 
 | ||||
| 
 | ||||
| Quick API reference | ||||
| ------------------- | ||||
| 
 | ||||
| void CycloneInit(void); | ||||
|   Initializes Cyclone. Must be called if the jumptable is compressed, | ||||
|   doesn't matter otherwise. | ||||
| 
 | ||||
| void CycloneRun(struct Cyclone *pcy); | ||||
|   Runs cyclone for pcy->cycles. Writes amount of cycles left back to | ||||
|   pcy->cycles (always negative). | ||||
| 
 | ||||
| unsigned int CycloneGetSr(const struct Cyclone *pcy); | ||||
|   Reads status register in internal form from pcy, converts to standard 68k SR and returns it. | ||||
| 
 | ||||
| void CycloneSetSr(struct Cyclone *pcy, unsigned int sr); | ||||
|   Takes standard 68k status register (sr), and updates Cyclone context with it. | ||||
|    | ||||
| int CycloneFlushIrq(struct Cyclone *pcy); | ||||
|   If .irq is greater than IRQ mask in SR, or it is equal to 7 (NMI), processes interrupt | ||||
|   exception and returns number of cycles used. Otherwise, does nothing and returns 0. | ||||
| 
 | ||||
| void CyclonePack(const struct Cyclone *pcy, void *save_buffer); | ||||
|   Writes Cyclone state to save_buffer. This allows to avoid all the trouble figuring what | ||||
|   actually needs to be saved from the Cyclone structure, as saving whole struct Cyclone | ||||
|   to a file will also save various pointers, which may become invalid after your program | ||||
|   is restarted, so simply reloading the structure will cause a crash. save_buffer size | ||||
|   should be 128 bytes (now it is really using less, but this allows future expansion). | ||||
| 
 | ||||
| void CycloneUnpack(struct Cyclone *pcy, const void *save_buffer); | ||||
|   Reloads Cyclone state from save_buffer, which was previously saved by CyclonePack(). | ||||
|   This function uses checkpc() callback to rebase the PC, so .checkpc must be initialized | ||||
|   before calling it. | ||||
| 
 | ||||
| Callbacks: | ||||
| 
 | ||||
| .checkpc | ||||
| unsigned int (*checkpc)(unsigned int pc); | ||||
|   This function is called when PC changes are performed in 68k code or because of exceptions. | ||||
|   It is passed ARM pointer and should return ARM pointer casted to int. It must also update | ||||
|   .membase if needed. See "The checkpc() function" section above. | ||||
| 
 | ||||
| unsigned int (*read8  )(unsigned int a); | ||||
| unsigned int (*read16 )(unsigned int a); | ||||
| unsigned int (*read32 )(unsigned int a); | ||||
|   These are the read memory handler callbacks. They are called when 68k code reads from memory. | ||||
|   The parameter is a 68k address in data space, return value is a data value read. Data value | ||||
|   doesn't have to be masked to 8 or 16 bits for read8 or read16, Cyclone will do that itself | ||||
|   if needed. | ||||
| 
 | ||||
| unsigned int (*fetch8 )(unsigned int a); | ||||
| unsigned int (*fetch16)(unsigned int a); | ||||
| unsigned int (*fetch32)(unsigned int a); | ||||
|   Same as above, but these are reads from program space (PC relative reads mostly). | ||||
|   | ||||
| void (*write8 )(unsigned int a,unsigned char  d); | ||||
| void (*write16)(unsigned int a,unsigned short d); | ||||
| void (*write32)(unsigned int a,unsigned int   d); | ||||
|   These are called when 68k code writes to data space. d is the data value. | ||||
| 
 | ||||
| int (*IrqCallback)(int int_level); | ||||
|   This function is called when Cyclone acknowledges an interrupt. The parameter is the IRQ | ||||
|   level being acknowledged, and return value is exception vector to use, or one of these special | ||||
|   values: CYCLONE_INT_ACK_AUTOVECTOR or CYCLONE_INT_ACK_SPURIOUS. Can be disabled in config.h. | ||||
|   See "Interrupts" section for more information. | ||||
| 
 | ||||
| void (*ResetCallback)(void); | ||||
|   Cyclone will call this function if it encounters RESET 68k instruction. | ||||
|   Can be disabled in config.h. | ||||
| 
 | ||||
| int (*UnrecognizedCallback)(void); | ||||
|   Cyclone will call this function if it encounters illegal instructions (including A-line and | ||||
|   F-line ones). Can be tuned / disabled in config.h. | ||||
| 
 | ||||
| 
 | ||||
| Function codes | ||||
| -------------- | ||||
| 
 | ||||
| Cyclone doesn't pass function codes to it's memory handlers, but they can be calculated: | ||||
| FC2: just use supervisor state bit from status register (eg. (MyCpu.srh & 0x20) >> 5) | ||||
| FC1: if we are in fetch* function, then 1, else 0. | ||||
| FC0: if we are in read* or write*, then 1, else 0. | ||||
| CPU state (all FC bits set) is active in IrqCallback function. | ||||
| 
 | ||||
| 
 | ||||
| References | ||||
| ---------- | ||||
| 
 | ||||
| These documents were used while writing Cyclone and should be useful for those who want to | ||||
| understand deeper how the 68000 works. | ||||
| 
 | ||||
| MOTOROLA M68000 FAMILY Programmer's Reference Manual | ||||
| common name: 68kPM.pdf | ||||
| 
 | ||||
| M68000 8-/16-/32-Bit Microprocessors User's Manual | ||||
| common name: MC68000UM.pdf | ||||
| 
 | ||||
| 68000 Undocumented Behavior Notes by Bart Trzynadlowski | ||||
| http://www.trzy.org/files/68knotes.txt | ||||
| 
 | ||||
| Instruction prefetch on the Motorola 68000 processor by Jorge Cwik | ||||
| http://pasti.fxatari.com/68kdocs/68kPrefetch.html | ||||
| 
 | ||||
| 
 | ||||
| ARM Register Usage | ||||
| ------------------ | ||||
| 
 | ||||
| See source code for up to date of register usage, however a summary is here: | ||||
| 
 | ||||
|   r0-3: Temporary registers | ||||
|   r4  : Current PC + Memory Base (i.e. pointer to next opcode) | ||||
|   r5  : Cycles remaining | ||||
|   r6  : Pointer to Opcode Jump table | ||||
|   r7  : Pointer to Cpu Context | ||||
|   r8  : Current Opcode | ||||
|   r10 : Flags (NZCV) in highest four bits | ||||
|  (r11 : Temporary register) | ||||
| 
 | ||||
| Flags are mapped onto ARM flags whenever possible, which speeds up the processing of opcode. | ||||
| r9 is not used intentionally, because AAPCS defines it as "platform register", so it's | ||||
| reserved in some systems. | ||||
| 
 | ||||
| 
 | ||||
| Thanks to... | ||||
| ------------ | ||||
| 
 | ||||
| * All the previous code-generating assembler cpu core guys! | ||||
|   Who are iirc... Neill Corlett, Neil Bradley, Mike Coates, Darren Olafson | ||||
|     Karl Stenerud and Bart Trzynadlowski | ||||
| 
 | ||||
| * Charles Macdonald, for researching just about every console ever | ||||
| * MameDev+FBA, for keeping on going and going and going | ||||
| 
 | ||||
| 
 | ||||
| What's New | ||||
| ---------- | ||||
| v0.0099 notaz | ||||
|   * Cyclone no longer uses r9, because AAPCS defines it as "platform register", | ||||
|     so it's reserved in some systems. | ||||
|   * Made SPLIT_MOVEL_PD to affect MOVEM too. | ||||
| 
 | ||||
| v0.0088 notaz | ||||
|   - Reduced amount of code in opcode handlers by ~23% by doing the following: | ||||
|     - Removed duplicate opcode handlers | ||||
|     - Optimized code to use less ARM instructions | ||||
|     - Merged some duplicate handler endings | ||||
|   + Cyclone now does better job avoiding pipeline interlocks. | ||||
|   + Replaced incorrect handler of DBT with proper one. | ||||
|   + Changed "MOVEA (An)+ An" behavior. | ||||
|   + Fixed flag behavior of ROXR, ASL, LSR and NBCD in certain situations. | ||||
|     Hopefully got them right now. | ||||
|   + Cyclone no longer sets most significant bits while pushing PC to stack. | ||||
|     Amiga Kickstart depends on this. | ||||
|   + Added optional trace mode emulation. | ||||
|   + Added optional address error emulation. | ||||
|   + Additional functionality added for MAME and other ports (see config.h). | ||||
|   + Added return value for IrqCallback to make it suitable for emulating devices which | ||||
|     pass the vector number during interrupt acknowledge cycle. For usual autovector | ||||
|     processing this function must return CYCLONE_INT_ACK_AUTOVECTOR, so those who are | ||||
|     upgrading must add "return CYCLONE_INT_ACK_AUTOVECTOR;" to their IrqCallback functions. | ||||
|   * Updated documentation. | ||||
| 
 | ||||
| v0.0086 notaz | ||||
|   + Cyclone now can be customized to better suit your project, see config.h . | ||||
|   + Added an option to compress the jumptable at compile-time. Must call CycloneInit() | ||||
|     at runtime to decompress it if enabled (see config.h). | ||||
|   + Added missing CHK opcode handler (used by SeaQuest DSV). | ||||
|   + Added missing TAS opcode handler (Gargoyles,Bubba N Stix,...). As in real genesis, | ||||
|     memory write-back phase is ignored (but can be enabled in config.h if needed). | ||||
|   + Added missing NBCD and TRAPV opcode handlers. | ||||
|   + Added missing addressing mode for CMP/EOR. | ||||
|   + Added some minor optimizations. | ||||
|   - Removed 216 handlers for 2927 opcodes which were generated for invalid addressing modes. | ||||
|   + Fixed flags for ASL, NEG, NEGX, DIVU, ADDX, SUBX, ROXR. | ||||
|   + Bugs fixed in MOVEP, LINK, ADDQ, DIVS handlers. | ||||
|   * Undocumented flags for CHK, ABCD, SBCD and NBCD are now emulated the same way as in Musashi. | ||||
|   + Added Uninitialized Interrupt emulation. | ||||
|   + Altered timing for about half of opcodes to match Musashi's. | ||||
| 
 | ||||
| v0.0082 Reesy | ||||
|   + Change cyclone to clear cycles before returning when halted | ||||
|   + Added Irq call back function.  This allows emulators to be notified | ||||
|     when cyclone has taken an interrupt allowing them to set internal flags | ||||
|     which can help fix timing problems. | ||||
| 
 | ||||
| v0.0081 notaz | ||||
|   + .asm version was broken and did not compile with armasm. Fixed. | ||||
|   + Finished implementing Stop opcode. Now it really stops the processor. | ||||
| 
 | ||||
| v0.0080 notaz | ||||
|   + Added real cmpm opcode, it was using eor handler before this. | ||||
|     Fixes Dune and Sensible Soccer. | ||||
| 
 | ||||
| v0.0078 notaz | ||||
|   note: these bugs were actually found Reesy, I reimplemented these by | ||||
|         using his changelog as a guide. | ||||
|   + Fixed a problem with divu which was using long divisor instead of word. | ||||
|     Fixes gear switching in Top Gear 2. | ||||
|   + Fixed btst opcode, The bit to test should shifted a max of 31 or 7 | ||||
|     depending on if a register or memory location is being tested. | ||||
|   + Fixed abcd,sbcd. They did bad decimal correction on invalid BCD numbers | ||||
|     Score counters in Streets of Rage level end work now. | ||||
|   + Changed flag handling of abcd,sbcd,addx,subx,asl,lsl,... | ||||
|     Some ops did not have flag handling at all. | ||||
|     Some ops must not change Z flag when result is zero, but they did. | ||||
|     Shift ops must not change X if shift count is zero, but they did. | ||||
|     There are probably still some flag problems left. | ||||
|   + Patially implemented Stop and Reset opcodes - Fixes Thunderforce IV | ||||
| 
 | ||||
| v0.0075 notaz | ||||
|   + Added missing displacement addressing mode for movem (Fantastic Dizzy) | ||||
|   + Added OSP <-> A7 swapping code in opcodes, which change privilege mode | ||||
|   + Implemented privilege violation, line emulator and divide by zero exceptions | ||||
|   + Added negx opcode (Shining Force works!) | ||||
|   + Added overflow detection for divs/divu | ||||
| 
 | ||||
| v0.0072 notaz | ||||
|   note: I could only get v0.0069 cyclone, so I had to implement these myself using Dave's | ||||
|         changelog as a guide. | ||||
|   + Fixed a problem with divs - remainder should be negative when divident is negative | ||||
|   + Added movep opcode (Sonic 3 works) | ||||
|   + Fixed a problem with DBcc incorrectly decrementing if the condition is true (Shadow of the Beast) | ||||
| 
 | ||||
| v0.0069 | ||||
|   + Added SBCD and the flags for ABCD/SBCD. Score and time now works in games such as | ||||
|     Rolling Thunder 2, Ghouls 'N Ghosts | ||||
|   + Fixed a problem with addx and subx with 8-bit and 16-bit values. | ||||
|     Ghouls 'N' Ghosts now works! | ||||
| 
 | ||||
| v0.0068 | ||||
|   + Added ABCD opcode (Streets of Rage works now!) | ||||
| 
 | ||||
| v0.0067 | ||||
|   + Added dbCC (After Burner) | ||||
|   + Added asr EA (Sonic 1 Boss/Labyrinth Zone) | ||||
|   + Added andi/ori/eori ccr (Altered Beast) | ||||
|   + Added trap (After Burner) | ||||
|   + Added special case for move.b (a7)+ and -(a7), stepping by 2 | ||||
|     After Burner is playable! Eternal Champions shows more | ||||
|   + Fixed lsr.b/w zero flag (Ghostbusters) | ||||
|     Rolling Thunder 2 now works! | ||||
|   + Fixed N flag for .b and .w arithmetic. Golden Axe works! | ||||
| 
 | ||||
| v0.0066 | ||||
|   + Fixed a stupid typo for exg (orr r10,r10, not orr r10,r8), which caused alignment | ||||
|     crashes on Strider | ||||
| 
 | ||||
| v0.0065 | ||||
|   + Fixed a problem with immediate values - they weren't being shifted up correctly for some | ||||
|     opcodes. Spiderman works, After Burner shows a bit of graphics. | ||||
|   + Fixed a problem with EA:"110nnn" extension word. 32-bit offsets were being decoded as 8-bit | ||||
|     offsets by mistake. Castlevania Bloodlines seems fine now. | ||||
|   + Added exg opcode | ||||
|   + Fixed asr opcode (Sonic jumping left is fixed) | ||||
|   + Fixed a problem with the carry bit in rol.b (Marble Madness) | ||||
| 
 | ||||
| v0.0064 | ||||
|   + Added rtr | ||||
|   + Fixed addq/subq.l (all An opcodes are 32-bit) (Road Rash) | ||||
|   + Fixed various little timings | ||||
| 
 | ||||
| v0.0063 | ||||
|   + Added link/unlk opcodes | ||||
|   + Fixed various little timings | ||||
|   + Fixed a problem with dbCC opcode being emitted at set opcodes | ||||
|   + Improved long register access, the EA fetch now does ldr r0,[r7,r0,lsl #2] whenever | ||||
|      possible, saving 1 or 2 cycles on many opcodes, which should give a nice speed up. | ||||
|   + May have fixed N flag on ext opcode? | ||||
|   + Added dasm for link opcode. | ||||
| 
 | ||||
| v0.0062 | ||||
|   * I was a bit too keen with the Arithmetic opcodes! Some of them should have been abcd, | ||||
|     exg and addx. Removed the incorrect opcodes, pending re-adding them as abcd, exg and addx. | ||||
|   + Changed unknown opcodes to act as nops. | ||||
|     Not very technical, but fun - a few more games show more graphics ;) | ||||
| 
 | ||||
| v0.0060 | ||||
|   + Fixed divu (EA intro) | ||||
|   + Added sf (set false) opcode - SOR2 | ||||
|   * Todo: pea/link/unlk opcodes | ||||
| 
 | ||||
| v0.0059: Added remainder to divide opcodes. | ||||
| 
 | ||||
| 
 | ||||
|  | @ -1,886 +0,0 @@ | |||
| 
 | ||||
| // Dave's Disa 68000 Disassembler
 | ||||
| #ifndef __GNUC__ | ||||
| #pragma warning(disable:4115) | ||||
| #endif | ||||
| #include <stdio.h> | ||||
| #include <string.h> | ||||
| #include "Disa.h" | ||||
| 
 | ||||
| unsigned int DisaPc=0; | ||||
| char *DisaText=NULL; // Text buffer to write in
 | ||||
| static char Tasm[]="bwl?"; | ||||
| static char Comment[64]=""; | ||||
| unsigned short (*DisaWord)(unsigned int a)=NULL; | ||||
| 
 | ||||
| static unsigned int DisaLong(unsigned int a) | ||||
| { | ||||
|   unsigned int d=0; | ||||
|   if (DisaWord==NULL) return d; | ||||
| 
 | ||||
|   d= DisaWord(a)<<16; | ||||
|   d|=DisaWord(a+2)&0xffff; | ||||
|   return d; | ||||
| } | ||||
| 
 | ||||
| // Get text version of the effective address
 | ||||
| int DisaGetEa(char *t,int ea,int size) | ||||
| { | ||||
|   ea&=0x3f; t[0]=0; | ||||
|   if ((ea&0x38)==0x00) { sprintf(t,"d%d",ea  ); return 0; }    // 000rrr
 | ||||
|   if ((ea&0x38)==0x08) { sprintf(t,"a%d",ea&7); return 0; }    // 001rrr
 | ||||
|   if ((ea&0x38)==0x10) { sprintf(t,"(a%d)",ea&7); return 0; }  // 010rrr
 | ||||
|   if ((ea&0x38)==0x18) { sprintf(t,"(a%d)+",ea&7); return 0; } // 011rrr
 | ||||
|   if ((ea&0x38)==0x20) { sprintf(t,"-(a%d)",ea&7); return 0; } // 100rrr
 | ||||
|   if ((ea&0x38)==0x28) { sprintf(t,"($%x,a%d)",DisaWord(DisaPc)&0xffff,ea&7); DisaPc+=2; return 0; } // 101rrr
 | ||||
| 
 | ||||
|   if ((ea&0x38)==0x30) | ||||
|   { | ||||
|     // 110nnn - An + Disp + D/An
 | ||||
|     int areg=0,ext=0,off=0,da=0,reg=0,wol=0,scale=0; | ||||
|     ext=DisaWord(DisaPc)&0xffff; | ||||
| 
 | ||||
|     areg=ea&7; | ||||
|     off=ext&0xff;    da =ext&0x8000?'a':'d'; | ||||
|     reg=(ext>>12)&7; wol=ext&0x0800?'l':'w'; | ||||
|     scale=1<<((ext>>9)&3); | ||||
| 
 | ||||
|     if (scale<2) sprintf(t,"($%x,a%d,%c%d.%c)",   off,areg,da,reg,wol); | ||||
|     else         sprintf(t,"($%x,a%d,%c%d.%c*%d)",off,areg,da,reg,wol,scale); // 68020
 | ||||
| 
 | ||||
|     DisaPc+=2; | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   if (ea==0x38) { sprintf(t,"$%x.w",DisaWord(DisaPc)&0xffff); DisaPc+=2; return 0; } // 111000 - Absolute short
 | ||||
|   if (ea==0x39) { sprintf(t,"$%x.l",DisaLong(DisaPc));        DisaPc+=4; return 0; } // 111001 - Absolute long
 | ||||
| 
 | ||||
|   if (ea==0x3a) | ||||
|   { | ||||
|     // 111010 - PC Relative
 | ||||
|     int ext=DisaWord(DisaPc)&0xffff; | ||||
|     sprintf(t,"($%x,pc)",ext); | ||||
|     sprintf(Comment,"; =%x",DisaPc+(short)ext); // Comment where pc+ext is
 | ||||
|     DisaPc+=2; | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   if (ea==0x3b) | ||||
|   { | ||||
|     // 111011 - PC Relative + D/An
 | ||||
|     int ext=0,off=0,da=0,reg=0,wol=0,scale=0; | ||||
|     ext=DisaWord(DisaPc)&0xffff; | ||||
| 
 | ||||
|     off=ext&0xff;    da =ext&0x8000?'a':'d'; | ||||
|     reg=(ext>>12)&7; wol=ext&0x0800?'l':'w'; | ||||
|     scale=1<<((ext>>9)&3); | ||||
| 
 | ||||
|     if (scale<2) sprintf(t,"($%x,pc,%c%d.%c)",   off,da,reg,wol); | ||||
|     else         sprintf(t,"($%x,pc,%c%d.%c*%d)",off,da,reg,wol,scale); // 68020
 | ||||
| 
 | ||||
|     sprintf(Comment,"; =%x",DisaPc+(char)off); // Comment where pc+ext is
 | ||||
|     DisaPc+=2; | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   if (ea==0x3c) | ||||
|   { | ||||
|     // 111100 - Immediate
 | ||||
|     switch (size) | ||||
|     { | ||||
|       case 0: sprintf(t,"#$%x",DisaWord(DisaPc)&0x00ff); DisaPc+=2; return 0; | ||||
|       case 1: sprintf(t,"#$%x",DisaWord(DisaPc)&0xffff); DisaPc+=2; return 0; | ||||
|       case 2: sprintf(t,"#$%x",DisaLong(DisaPc)       ); DisaPc+=4; return 0; | ||||
|     } | ||||
|     return 1; | ||||
|   } | ||||
| 
 | ||||
| // Unknown effective address
 | ||||
|   sprintf(t,"ea=(%d%d%d %d%d%d)", | ||||
|     (ea>>5)&1,(ea>>4)&1,(ea>>3)&1, | ||||
|     (ea>>2)&1,(ea>>1)&1, ea    &1); | ||||
|   return 1; | ||||
| } | ||||
| 
 | ||||
| static void GetOffset(char *text) | ||||
| { | ||||
|   int off=(short)DisaWord(DisaPc); DisaPc+=2; | ||||
| 
 | ||||
|   if (off<0) sprintf(text,"-$%x",-off); | ||||
|   else       sprintf(text,"$%x",  off); | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x0000+ ================
 | ||||
| static int DisaArithImm(int op) | ||||
| { | ||||
|   // Or/And/Sub/Add/Eor/Cmp Immediate 0000ttt0 xxDDDddd (tt=type, xx=size extension, DDDddd=Dest ea)
 | ||||
|   int dea=0; | ||||
|   char seat[64]="",deat[64]=""; | ||||
|   int type=0,size=0; | ||||
|   char *arith[8]={"or","and","sub","add","?","eor","cmp","?"}; | ||||
| 
 | ||||
|   type=(op>>9)&7; if (type==4 || type>=7) return 1; | ||||
|   size=(op>>6)&3; if (size>=3) return 1; | ||||
|   dea=op&0x3f; if (dea==0x3c) return 1; | ||||
| 
 | ||||
|   DisaGetEa(seat,0x3c,size); | ||||
|   DisaGetEa(deat,dea, size); | ||||
| 
 | ||||
|   sprintf(DisaText,"%si.%c %s, %s",arith[type],Tasm[size],seat,deat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x0108+ ================
 | ||||
| static int DisaMovep(int op) | ||||
| { | ||||
|   // movep.x (Aa),Dn - 0000nnn1 dx001aaa  nn
 | ||||
|   int dn=0,dir=0,size=0,an=0; | ||||
|   char offset[32]=""; | ||||
| 
 | ||||
|   dn  =(op>>9)&7; | ||||
|   dir =(op>>7)&1; | ||||
|   size=(op>>6)&1; size++; | ||||
|   an  = op    &7; | ||||
| 
 | ||||
|   GetOffset(offset); | ||||
|   if (dir) sprintf(DisaText,"movep.%c d%d, (%s,a%d)",Tasm[size],dn,offset,an); | ||||
|   else     sprintf(DisaText,"movep.%c (%s,a%d), d%d",Tasm[size],offset,an,dn); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x007c+ ================
 | ||||
| static int DisaArithSr(int op) | ||||
| { | ||||
|   // Ori/Andi/Eori $nnnn,sr 0000t0tx 0s111100
 | ||||
|   char *opcode[6]={"ori","andi","","","","eori"}; | ||||
|   char seat[64]=""; | ||||
|   int type=0,size=0; | ||||
| 
 | ||||
|   type=(op>>9)&5; | ||||
|   size=(op>>6)&1; | ||||
| 
 | ||||
|   DisaGetEa(seat,0x3c,size); | ||||
|   sprintf(DisaText,"%s.%c %s, %s", opcode[type], Tasm[size], seat, size?"sr":"ccr"); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x0100+ ================
 | ||||
| static int DisaBtstReg(int op) | ||||
| { | ||||
|   // Btst/Bchg/Bclr/Bset 0000nnn1 tteeeeee (nn=reg number, eeeeee=Dest ea)
 | ||||
|   int type=0; | ||||
|   int sea=0,dea=0; | ||||
|   char seat[64]="",deat[64]=""; | ||||
|   char *opcode[4]={"btst","bchg","bclr","bset"}; | ||||
| 
 | ||||
|   sea =(op>>9)&7; | ||||
|   type=(op>>6)&3; | ||||
|   dea=  op&0x3f; | ||||
| 
 | ||||
|   if ((dea&0x38)==0x08) return 1; // movep
 | ||||
|   DisaGetEa(seat,sea,0); | ||||
|   DisaGetEa(deat,dea,0); | ||||
| 
 | ||||
|   sprintf(DisaText,"%s %s, %s",opcode[type],seat,deat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x0800+ ================
 | ||||
| static int DisaBtstImm(int op) | ||||
| { | ||||
|   // Btst/Bchg/Bclr/Bset 00001000 tteeeeee 00 nn (eeeeee=ea, nn=bit number)
 | ||||
|   int type=0; | ||||
|   char seat[64]="",deat[64]=""; | ||||
|   char *opcode[4]={"btst","bchg","bclr","bset"}; | ||||
| 
 | ||||
|   type=(op>>6)&3; | ||||
|   DisaGetEa(seat,   0x3c,0); | ||||
|   DisaGetEa(deat,op&0x3f,0); | ||||
| 
 | ||||
|   sprintf(DisaText,"%s %s, %s",opcode[type],seat,deat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x1000+ ================
 | ||||
| static int DisaMove(int op) | ||||
| { | ||||
|   // Move 00xxdddD DDssssss (xx=size extension, ssssss=Source EA, DDDddd=Dest ea)
 | ||||
|   int sea=0,dea=0; | ||||
|   char inst[64]="",seat[64]="",deat[64]=""; | ||||
|   char *movea=""; | ||||
|   int size=0; | ||||
| 
 | ||||
|   if ((op&0x01c0)==0x0040) movea="a"; // See if it's a movea opcode
 | ||||
| 
 | ||||
|   // Find size extension
 | ||||
|   switch (op&0x3000) | ||||
|   { | ||||
|     case 0x1000: size=0; break; | ||||
|     case 0x3000: size=1; break; | ||||
|     case 0x2000: size=2; break; | ||||
|     default: return 1; | ||||
|   } | ||||
| 
 | ||||
|   sea = op&0x003f; | ||||
|   DisaGetEa(seat,sea,size); | ||||
| 
 | ||||
|   dea =(op&0x01c0)>>3; | ||||
|   dea|=(op&0x0e00)>>9; | ||||
|   DisaGetEa(deat,dea,size); | ||||
| 
 | ||||
|   sprintf(inst,"move%s.%c",movea,Tasm[size]); | ||||
|   sprintf(DisaText,"%s %s, %s",inst,seat,deat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x4000+ ================
 | ||||
| static int DisaNeg(int op) | ||||
| { | ||||
|   // 01000tt0 xxeeeeee (tt=negx/clr/neg/not, xx=size, eeeeee=EA)
 | ||||
|   char eat[64]=""; | ||||
|   int type=0,size=0; | ||||
|   char *opcode[4]={"negx","clr","neg","not"}; | ||||
| 
 | ||||
|   type=(op>>9)&3; | ||||
|   size=(op>>6)&3; if (size>=3) return 1; | ||||
|   DisaGetEa(eat,op&0x3f,size); | ||||
| 
 | ||||
|   sprintf(DisaText,"%s.%c %s",opcode[type],Tasm[size],eat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x40c0+ ================
 | ||||
| static int DisaMoveSr(int op) | ||||
| { | ||||
|   // 01000tt0 11eeeeee (tt=type, xx=size, eeeeee=EA)
 | ||||
|   int type=0,ea=0; | ||||
|   char eat[64]=""; | ||||
| 
 | ||||
|   type=(op>>9)&3; | ||||
|   ea=op&0x3f; | ||||
|   DisaGetEa(eat,ea,1); | ||||
| 
 | ||||
|   switch (type) | ||||
|   { | ||||
|     default: sprintf(DisaText,"move sr, %s", eat); break; | ||||
|     case 1:  sprintf(DisaText,"move ccr, %s",eat); break; | ||||
|     case 2:  sprintf(DisaText,"move %s, ccr",eat); break; | ||||
|     case 3:  sprintf(DisaText,"move %s, sr", eat); break; | ||||
|   } | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| static int OpChk(int op) | ||||
| { | ||||
|   int sea=0,dea=0; | ||||
|   char seat[64]="",deat[64]=""; | ||||
| 
 | ||||
|   sea=op&0x003f; | ||||
|   DisaGetEa(seat,sea,0); | ||||
| 
 | ||||
|   dea=(op>>9)&7; dea|=8; | ||||
|   DisaGetEa(deat,dea,2); | ||||
| 
 | ||||
|   sprintf(DisaText,"chk %s, %s",seat,deat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x41c0+ ================
 | ||||
| static int DisaLea(int op) | ||||
| { | ||||
|   // Lea 0100nnn1 11eeeeee (eeeeee=ea)
 | ||||
|   int sea=0,dea=0; | ||||
|   char seat[64]="",deat[64]=""; | ||||
| 
 | ||||
|   sea=op&0x003f; | ||||
|   DisaGetEa(seat,sea,0); | ||||
| 
 | ||||
|   dea=(op>>9)&7; dea|=8; | ||||
|   DisaGetEa(deat,dea,2); | ||||
| 
 | ||||
|   sprintf(DisaText,"lea %s, %s",seat,deat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| static int MakeRegList(char *list,int mask,int ea) | ||||
| { | ||||
|   int reverse=0,i=0,low=0,len=0; | ||||
| 
 | ||||
|   if ((ea&0x38)==0x20) reverse=1; // -(An), bitfield is reversed
 | ||||
| 
 | ||||
|   mask&=0xffff; list[0]=0; | ||||
| 
 | ||||
|   for (i=0;i<17;i++) | ||||
|   { | ||||
|     int bit=0; | ||||
| 
 | ||||
|     // Mask off bit i:
 | ||||
|     if (reverse) bit=0x8000>>i; else bit=1<<i; | ||||
|     bit&=mask; | ||||
| 
 | ||||
|     if (bit==0 || i==8) | ||||
|     { | ||||
|       // low to i-1 are a continuous section, add it:
 | ||||
|       char add[16]=""; | ||||
|       int ad=low&8?'a':'d'; | ||||
|       if (low==i-1) sprintf(add,"%c%d/",     ad,low&7); | ||||
|       if (low< i-1) sprintf(add,"%c%d-%c%d/",ad,low&7, ad,(i-1)&7); | ||||
|       strcat(list,add); | ||||
| 
 | ||||
|       low=i; // Next section
 | ||||
|     } | ||||
| 
 | ||||
|     if (bit==0) low=i+1; | ||||
|   } | ||||
| 
 | ||||
|   // Knock off trailing '/'
 | ||||
|   len=strlen(list); | ||||
|   if (len>0) if (list[len-1]=='/') list[len-1]=0; | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x4800+ ================
 | ||||
| static int DisaNbcd(int op) | ||||
| { | ||||
|   // Nbcd 01001000 00eeeeee  (eeeeee=ea)
 | ||||
|   int ea=0; | ||||
|   char eat[64]=""; | ||||
| 
 | ||||
|   ea=op&0x003f; | ||||
|   DisaGetEa(eat,ea,0); | ||||
| 
 | ||||
|   sprintf(DisaText,"nbcd %s",eat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x4840+ ================
 | ||||
| static int DisaSwap(int op) | ||||
| { | ||||
|   // Swap, 01001000 01000nnn swap Dn
 | ||||
|   sprintf(DisaText,"swap d%d",op&7); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x4850+ ================
 | ||||
| static int DisaPea(int op) | ||||
| { | ||||
|   // Pea 01001000 01eeeeee  (eeeeee=ea)  pea
 | ||||
|   int ea=0; | ||||
|   char eat[64]=""; | ||||
| 
 | ||||
|   ea=op&0x003f; if (ea<0x10) return 1; // swap opcode
 | ||||
|   DisaGetEa(eat,ea,2); | ||||
| 
 | ||||
|   sprintf(DisaText,"pea %s",eat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x4880+ ================
 | ||||
| static int DisaExt(int op) | ||||
| { | ||||
|   // Ext 01001000 1x000nnn (x=size, eeeeee=EA)
 | ||||
|   char eat[64]=""; | ||||
|   int size=0; | ||||
| 
 | ||||
|   size=(op>>6)&1; size++; | ||||
|   DisaGetEa(eat,op&0x3f,size); | ||||
| 
 | ||||
|   sprintf(DisaText,"ext.%c %s",Tasm[size],eat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x4890+ ================
 | ||||
| static int DisaMovem(int op) | ||||
| { | ||||
|   // Movem 01001d00 1xeeeeee regmask  d=direction, x=size, eeeeee=EA
 | ||||
|   int dir=0,size=0; | ||||
|   int ea=0,mask=0; | ||||
|   char list[64]="",eat[64]=""; | ||||
| 
 | ||||
|   dir=(op>>10)&1; | ||||
|   size=((op>>6)&1)+1; | ||||
|   ea=op&0x3f; if (ea<0x10) return 1; // ext opcode
 | ||||
| 
 | ||||
|   mask=DisaWord(DisaPc)&0xffff; DisaPc+=2; | ||||
| 
 | ||||
|   MakeRegList(list,mask,ea); // Turn register mask into text
 | ||||
|   DisaGetEa(eat,ea,size); | ||||
| 
 | ||||
|   if (dir) sprintf(DisaText,"movem.%c %s, %s",Tasm[size],eat,list); | ||||
|   else     sprintf(DisaText,"movem.%c %s, %s",Tasm[size],list,eat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x4e40+ ================
 | ||||
| static int DisaTrap(int op) | ||||
| { | ||||
|   sprintf(DisaText,"trap #%d",op&0xf); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x4e50+ ================
 | ||||
| static int DisaLink(int op) | ||||
| { | ||||
|   // Link opcode, 01001110 01010nnn dd   link An,#offset
 | ||||
|   char eat[64]=""; | ||||
|   char offset[32]=""; | ||||
| 
 | ||||
|   DisaGetEa(eat,(op&7)|8,0); | ||||
|   GetOffset(offset); | ||||
| 
 | ||||
|   sprintf(DisaText,"link %s,#%s",eat,offset); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x4e58+ ================
 | ||||
| static int DisaUnlk(int op) | ||||
| { | ||||
|   // Link opcode, 01001110 01011nnn dd   unlk An
 | ||||
|   char eat[64]=""; | ||||
| 
 | ||||
|   DisaGetEa(eat,(op&7)|8,0); | ||||
|   sprintf(DisaText,"unlk %s",eat); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x4e60+ ================
 | ||||
| static int DisaMoveUsp(int op) | ||||
| { | ||||
|   // Move USP opcode, 01001110 0110dnnn move An to/from USP (d=direction)
 | ||||
|   int ea=0,dir=0; | ||||
|   char eat[64]=""; | ||||
| 
 | ||||
|   dir=(op>>3)&1; | ||||
|   ea=(op&7)|8; | ||||
|   DisaGetEa(eat,ea,0); | ||||
| 
 | ||||
|   if (dir) sprintf(DisaText,"move usp, %s",eat); | ||||
|   else     sprintf(DisaText,"move %s, usp",eat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x4e70+ ================
 | ||||
| static int Disa4E70(int op) | ||||
| { | ||||
|   char *inst[8]={"reset","nop","stop","rte","rtd","rts","trapv","rtr"}; | ||||
|   int n=0; | ||||
| 
 | ||||
|   n=op&7; | ||||
| 
 | ||||
|   sprintf(DisaText,"%s",inst[n]); | ||||
| 
 | ||||
|   //todo - 'stop' with 16 bit data
 | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x4a00+ ================
 | ||||
| static int DisaTst(int op) | ||||
| { | ||||
|   // Tst 01001010 xxeeeeee  (eeeeee=ea)
 | ||||
|   int ea=0; | ||||
|   char eat[64]=""; | ||||
|   int size=0; | ||||
| 
 | ||||
|   ea=op&0x003f; | ||||
|   DisaGetEa(eat,ea,0); | ||||
|   size=(op>>6)&3; if (size>=3) return 1; | ||||
| 
 | ||||
|   sprintf(DisaText,"tst.%c %s",Tasm[size],eat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| static int DisaTas(int op) | ||||
| { | ||||
|   // Tas 01001010 11eeeeee  (eeeeee=ea)
 | ||||
|   int ea=0; | ||||
|   char eat[64]=""; | ||||
| 
 | ||||
|   ea=op&0x003f; | ||||
|   DisaGetEa(eat,ea,0); | ||||
| 
 | ||||
|   sprintf(DisaText,"tas %s",eat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x4e80+ ================
 | ||||
| static int DisaJsr(int op) | ||||
| { | ||||
|   // Jsr/Jmp 0100 1110 1mEE Eeee (eeeeee=ea m=1=jmp)
 | ||||
|   int sea=0; | ||||
|   char seat[64]=""; | ||||
| 
 | ||||
|   sea=op&0x003f; | ||||
|   DisaGetEa(seat,sea,0); | ||||
| 
 | ||||
|   sprintf(DisaText,"j%s %s", op&0x40?"mp":"sr", seat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x5000+ ================
 | ||||
| static int DisaAddq(int op) | ||||
| { | ||||
|   // 0101nnnt xxeeeeee (nnn=#8,1-7 t=addq/subq xx=size, eeeeee=EA)
 | ||||
|   int num=0,type=0,size=0,ea=0; | ||||
|   char eat[64]=""; | ||||
| 
 | ||||
|   num =(op>>9)&7; if (num==0) num=8; | ||||
|   type=(op>>8)&1; | ||||
|   size=(op>>6)&3; if (size>=3) return 1; | ||||
|   ea  = op&0x3f; | ||||
| 
 | ||||
|   DisaGetEa(eat,ea,size); | ||||
| 
 | ||||
|   sprintf(DisaText,"%s.%c #%d, %s",type?"subq":"addq",Tasm[size],num,eat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x50c0+ ================
 | ||||
| static int DisaSet(int op) | ||||
| { | ||||
|   // 0101cccc 11eeeeee (sxx ea)
 | ||||
|   static char *cond[16]= | ||||
|   {"t" ,"f", "hi","ls","cc","cs","ne","eq", | ||||
|    "vc","vs","pl","mi","ge","lt","gt","le"}; | ||||
|   char *cc=""; | ||||
|   int ea=0; | ||||
|   char eat[64]=""; | ||||
| 
 | ||||
|   cc=cond[(op>>8)&0xf]; // Get condition code
 | ||||
|   ea=op&0x3f; | ||||
|   if ((ea&0x38)==0x08) return 1; // dbra, not scc
 | ||||
| 
 | ||||
|   DisaGetEa(eat,ea,0); | ||||
|   sprintf(DisaText,"s%s %s",cc,eat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x50c8+ ================
 | ||||
| static int DisaDbra(int op) | ||||
| { | ||||
|   // 0101cccc 11001nnn offset  (dbra/dbxx Rn,offset)
 | ||||
|   int dea=0; char deat[64]=""; | ||||
|   int pc=0,Offset=0; | ||||
| 
 | ||||
|   static char *BraCode[16]= | ||||
|   {"bt" ,"bra","bhi","bls","bcc","bcs","bne","beq", | ||||
|    "bvc","bvs","bpl","bmi","bge","blt","bgt","ble"}; | ||||
|   char *Bra=""; | ||||
| 
 | ||||
|   dea=op&7; | ||||
|   DisaGetEa(deat,dea,2); | ||||
| 
 | ||||
|   // Get condition code
 | ||||
|   Bra=BraCode[(op>>8)&0xf]; | ||||
| 
 | ||||
|   // Get offset
 | ||||
|   pc=DisaPc; | ||||
|   Offset=(short)DisaWord(DisaPc); DisaPc+=2; | ||||
| 
 | ||||
|   sprintf(DisaText,"d%s %s, %x",Bra,deat,pc+Offset); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x6000+ ================
 | ||||
| static int DisaBranch(int op) | ||||
| { | ||||
|   // Branch 0110cccc nn  (cccc=condition)
 | ||||
|   int pc=0,Offset=0; | ||||
| 
 | ||||
|   static char *BraCode[16]= | ||||
|   {"bra","bsr","bhi","bls","bcc","bcs","bne","beq", | ||||
|    "bvc","bvs","bpl","bmi","bge","blt","bgt","ble"}; | ||||
|   char *Bra=""; | ||||
| 
 | ||||
|   // Get condition code
 | ||||
|   Bra=BraCode[(op>>8)&0x0f]; | ||||
| 
 | ||||
|   // Get offset
 | ||||
|   pc=DisaPc; | ||||
|   Offset=(char)(op&0xff); | ||||
|        if (Offset== 0) { Offset=(short)DisaWord(DisaPc); DisaPc+=2; } | ||||
|   else if (Offset==-1) { Offset=       DisaLong(DisaPc); DisaPc+=4; } | ||||
| 
 | ||||
|   sprintf(DisaText,"%s %x",Bra,pc+Offset); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x7000+ ================
 | ||||
| static int DisaMoveq(int op) | ||||
| { | ||||
|   // Moveq 0111rrr0 nn (rrr=Dest register, nn=data)
 | ||||
| 
 | ||||
|   int dea=0; char deat[64]=""; | ||||
|   char *inst="moveq"; | ||||
|   int val=0; | ||||
| 
 | ||||
|   dea=(op>>9)&7; | ||||
|   DisaGetEa(deat,dea,2); | ||||
| 
 | ||||
|   val=(char)(op&0xff); | ||||
|   sprintf(DisaText,"%s #$%x, %s",inst,val,deat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x8000+ ================
 | ||||
| static int DisaArithReg(int op) | ||||
| { | ||||
|   // 1t0tnnnd xxeeeeee (tt=type:or/sub/and/add xx=size, eeeeee=EA)
 | ||||
|   int type=0,size=0,dir=0,rea=0,ea=0; | ||||
|   char reat[64]="",eat[64]=""; | ||||
|   char *opcode[]={"or","sub","","","and","add"}; | ||||
| 
 | ||||
|   type=(op>>12)&5; | ||||
|   rea =(op>> 9)&7; | ||||
|   dir =(op>> 8)&1; | ||||
|   size=(op>> 6)&3; if (size>=3) return 1; | ||||
|   ea  = op&0x3f; | ||||
| 
 | ||||
|   if (dir && ea<0x10) return 1; // addx opcode
 | ||||
| 
 | ||||
|   DisaGetEa(reat,rea,size); | ||||
|   DisaGetEa( eat, ea,size); | ||||
| 
 | ||||
|   if (dir) sprintf(DisaText,"%s.%c %s, %s",opcode[type],Tasm[size],reat,eat); | ||||
|   else     sprintf(DisaText,"%s.%c %s, %s",opcode[type],Tasm[size],eat,reat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x8100+ ================
 | ||||
| static int DisaAbcd(int op) | ||||
| { | ||||
|   // 1t00ddd1 0000asss - sbcd/abcd Ds,Dd or -(As),-(Ad)
 | ||||
|   int type=0; | ||||
|   int dn=0,addr=0,sn=0; | ||||
|   char *opcode[]={"sbcd","abcd"}; | ||||
| 
 | ||||
|   type=(op>>14)&1; | ||||
|   dn  =(op>> 9)&7; | ||||
|   addr=(op>> 3)&1; | ||||
|   sn  = op     &7; | ||||
| 
 | ||||
|   if (addr) sprintf(DisaText,"%s -(a%d), -(a%d)",opcode[type],sn,dn); | ||||
|   else      sprintf(DisaText,"%s d%d, d%d",       opcode[type],sn,dn); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x80c0+ ================
 | ||||
| static int DisaMul(int op) | ||||
| { | ||||
|   // Div/Mul: 1m00nnns 11eeeeee (m=Mul, nnn=Register Dn, s=signed, eeeeee=EA)
 | ||||
|   int type=0,rea=0,sign=0,ea=0,size=1; | ||||
|   char reat[64]="",eat[64]=""; | ||||
|   char *opcode[2]={"div","mul"}; | ||||
| 
 | ||||
|   type=(op>>14)&1; // div/mul
 | ||||
|   rea =(op>> 9)&7; | ||||
|   sign=(op>> 8)&1; | ||||
|   ea  = op&0x3f; | ||||
| 
 | ||||
|   DisaGetEa(reat,rea,size); | ||||
|   DisaGetEa( eat, ea,size); | ||||
| 
 | ||||
|   sprintf(DisaText,"%s%c.%c %s, %s",opcode[type],sign?'s':'u',Tasm[size],eat,reat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0x90c0+ ================
 | ||||
| static int DisaAritha(int op) | ||||
| { | ||||
|   // Suba/Cmpa/Adda 1tt1nnnx 11eeeeee (tt=type, x=size, eeeeee=Source EA)
 | ||||
|   int type=0,size=0,sea=0,dea=0; | ||||
|   char seat[64]="",deat[64]=""; | ||||
|   char *aritha[4]={"suba","cmpa","adda",""}; | ||||
| 
 | ||||
|   type=(op>>13)&3; if (type>=3) return 1; | ||||
|   size=(op>>8)&1; size++; | ||||
|   dea =(op>>9)&7; dea|=8; // Dest=An
 | ||||
|   sea = op&0x003f; // Source
 | ||||
| 
 | ||||
|   DisaGetEa(seat,sea,size); | ||||
|   DisaGetEa(deat,dea,size); | ||||
| 
 | ||||
|   sprintf(DisaText,"%s.%c %s, %s",aritha[type],Tasm[size],seat,deat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0xb000+ ================
 | ||||
| static int DisaCmpEor(int op) | ||||
| { | ||||
|   // Cmp/Eor 1011rrrt xxeeeeee (rrr=Dn, t=cmp/eor, xx=size extension, eeeeee=ea)
 | ||||
|   char reat[64]="",eat[64]=""; | ||||
|   int type=0,size=0; | ||||
| 
 | ||||
|   type=(op>>8)&1; | ||||
|   size=(op>>6)&3; if (size>=3) return 1; | ||||
|   DisaGetEa(reat,(op>>9)&7,size); | ||||
|   DisaGetEa(eat,  op&0x3f, size); | ||||
| 
 | ||||
|   if (type) sprintf(DisaText,"eor.%c %s, %s",Tasm[size],reat,eat); | ||||
|   else      sprintf(DisaText,"cmp.%c %s, %s",Tasm[size],eat,reat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| static int DisaCmpm(int op) | ||||
| { | ||||
|   char seat[64]="",deat[64]=""; | ||||
|   int type=0,size=0,sea,dea; | ||||
| 
 | ||||
|   type=(op>>8)&1; | ||||
|   size=(op>>6)&3; if (size>=3) return 1; | ||||
|   sea=(op&7)|0x18; | ||||
|   dea=(op>>9)&0x3f; | ||||
|   DisaGetEa(seat,sea,size); | ||||
|   DisaGetEa(deat,dea,size); | ||||
| 
 | ||||
|   sprintf(DisaText,"cmpm.%c %s, %s",Tasm[size],seat,deat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // ================ Opcodes 0xc140+ ================
 | ||||
| // 1100ttt1 01000sss  exg ds,dt
 | ||||
| // 1100ttt1 01001sss  exg as,at
 | ||||
| // 1100ttt1 10001sss  exg as,dt
 | ||||
| static int DisaExg(int op) | ||||
| { | ||||
|   int tr=0,type=0,sr=0; | ||||
| 
 | ||||
|   tr  =(op>>9)&7; | ||||
|   type= op&0xf8; | ||||
|   sr  = op&7; | ||||
| 
 | ||||
|        if (type==0x40) sprintf(DisaText,"exg d%d, d%d",sr,tr); | ||||
|   else if (type==0x48) sprintf(DisaText,"exg a%d, a%d",sr,tr); | ||||
|   else if (type==0x88) sprintf(DisaText,"exg a%d, d%d",sr,tr); | ||||
|   else return 1; | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0xd100+ ================
 | ||||
| static int DisaAddx(int op) | ||||
| { | ||||
|   // 1t01ddd1 xx000sss addx
 | ||||
|   int type=0,size=0,dea=0,sea=0,mem; | ||||
|   char deat[64]="",seat[64]=""; | ||||
|   char *opcode[6]={"","subx","","","","addx"}; | ||||
| 
 | ||||
|   type=(op>>12)&5; | ||||
|   dea =(op>> 9)&7; | ||||
|   size=(op>> 6)&3; if (size>=3) return 1; | ||||
|   sea = op&7; | ||||
|   mem = op&8; | ||||
|   if(mem) { sea+=0x20; dea+=0x20; } | ||||
| 
 | ||||
|   DisaGetEa(deat,dea,size); | ||||
|   DisaGetEa(seat,sea,size); | ||||
| 
 | ||||
|   sprintf(DisaText,"%s.%c %s, %s",opcode[type],Tasm[size],seat,deat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ================ Opcodes 0xe000+ ================
 | ||||
| static char *AsrName[4]={"as","ls","rox","ro"}; | ||||
| static int DisaAsr(int op) | ||||
| { | ||||
|   // Asr/l/Ror/l etc - 1110cccd xxuttnnn
 | ||||
|   // (ccc=count, d=direction xx=size extension, u=use reg for count, tt=type, nnn=register Dn)
 | ||||
|   int count=0,dir=0,size=0,usereg=0,type=0,num=0; | ||||
| 
 | ||||
|   count =(op>>9)&7; | ||||
|   dir   =(op>>8)&1; | ||||
|   size  =(op>>6)&3; if (size>=3) return 1; // todo Asr EA
 | ||||
|   usereg=(op>>5)&1; | ||||
|   type  =(op>>3)&3; | ||||
|   num   = op    &7; // Register number
 | ||||
| 
 | ||||
|   if (usereg==0) count=((count-1)&7)+1; // because ccc=000 means 8
 | ||||
| 
 | ||||
|   sprintf(DisaText,"%s%c.%c %c%d, d%d", | ||||
|     AsrName[type], dir?'l':'r', Tasm[size], | ||||
|     usereg?'d':'#', count, num); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| static int DisaAsrEa(int op) | ||||
| { | ||||
|   // Asr/l/Ror/l etc EA - 11100ttd 11eeeeee
 | ||||
|   int type=0,dir=0,size=1; | ||||
|   char eat[64]=""; | ||||
| 
 | ||||
|   type=(op>>9)&3; | ||||
|   dir =(op>>8)&1; | ||||
|   DisaGetEa(eat,op&0x3f,size); | ||||
| 
 | ||||
|   sprintf(DisaText,"%s%c.w %s", AsrName[type], dir?'l':'r', eat); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // =================================================================
 | ||||
| 
 | ||||
| static int TryOp(int op) | ||||
| { | ||||
|   if ((op&0xf100)==0x0000) DisaArithImm(op); // Ori/And/Sub/Add/Eor/Cmp Immediate
 | ||||
|   if ((op&0xf5bf)==0x003c) DisaArithSr(op); // Ori/Andi/Eori $nnnn,sr
 | ||||
|   if ((op&0xf100)==0x0100) DisaBtstReg(op); | ||||
|   if ((op&0xf138)==0x0108) DisaMovep(op); | ||||
|   if ((op&0xff00)==0x0800) DisaBtstImm(op); // Btst/Bchg/Bclr/Bset
 | ||||
|   if ((op&0xc000)==0x0000) DisaMove(op); | ||||
|   if ((op&0xf900)==0x4000) DisaNeg(op); // Negx/Clr/Neg/Not
 | ||||
|   if ((op&0xf140)==0x4100) OpChk(op); | ||||
|   if ((op&0xf1c0)==0x41c0) DisaLea(op); | ||||
|   if ((op&0xf9c0)==0x40c0) DisaMoveSr(op); | ||||
|   if ((op&0xffc0)==0x4800) DisaNbcd(op); | ||||
|   if ((op&0xfff8)==0x4840) DisaSwap(op); | ||||
|   if ((op&0xffc0)==0x4840) DisaPea(op); | ||||
|   if ((op&0xffb8)==0x4880) DisaExt(op); | ||||
|   if ((op&0xfb80)==0x4880) DisaMovem(op); | ||||
|   if ((op&0xff00)==0x4a00) DisaTst(op); | ||||
|   if ((op&0xffc0)==0x4ac0) DisaTas(op); | ||||
|   if ((op&0xfff0)==0x4e40) DisaTrap(op); | ||||
|   if ((op&0xfff8)==0x4e50) DisaLink(op); | ||||
|   if ((op&0xfff8)==0x4e58) DisaUnlk(op); | ||||
|   if ((op&0xfff0)==0x4e60) DisaMoveUsp(op); | ||||
|   if ((op&0xfff8)==0x4e70) Disa4E70(op); | ||||
|   if ((op&0xff80)==0x4e80) DisaJsr(op); | ||||
|   if ((op&0xf000)==0x5000) DisaAddq(op); | ||||
|   if ((op&0xf0c0)==0x50c0) DisaSet(op); | ||||
|   if ((op&0xf0f8)==0x50c8) DisaDbra(op); | ||||
|   if ((op&0xf000)==0x6000) DisaBranch(op); | ||||
|   if ((op&0xa000)==0x8000) DisaArithReg(op); // Or/Sub/And/Add
 | ||||
|   if ((op&0xb1f0)==0x8100) DisaAbcd(op); | ||||
|   if ((op&0xb130)==0x9100) DisaAddx(op); | ||||
|   if ((op&0xb0c0)==0x80c0) DisaMul(op); | ||||
|   if ((op&0xf100)==0x7000) DisaMoveq(op); | ||||
|   if ((op&0x90c0)==0x90c0) DisaAritha(op); | ||||
|   if ((op&0xf000)==0xb000) DisaCmpEor(op); | ||||
|   if ((op&0xf138)==0xb108) DisaCmpm(op); | ||||
|   if ((op&0xf130)==0xc100) DisaExg(op); | ||||
|   if ((op&0xf000)==0xe000) DisaAsr(op); | ||||
|   if ((op&0xf8c0)==0xe0c0) DisaAsrEa(op); | ||||
| 
 | ||||
|   // Unknown opcoode
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| int DisaGet() | ||||
| { | ||||
|   int op=0; | ||||
|   if (DisaWord==NULL) return 1; | ||||
| 
 | ||||
|   Comment[0]=0; | ||||
|   DisaText[0]=0; // Assume opcode unknown
 | ||||
| 
 | ||||
|   op=DisaWord(DisaPc)&0xffff; DisaPc+=2; | ||||
|   TryOp(op); | ||||
|   strcat(DisaText,Comment); | ||||
| 
 | ||||
|   // Unknown opcoode
 | ||||
|   return 0; | ||||
| } | ||||
|  | @ -1,18 +0,0 @@ | |||
| 
 | ||||
| // Dave's Disa 68000 Disassembler
 | ||||
| 
 | ||||
| #ifdef __cplusplus | ||||
| extern "C" { | ||||
| #endif | ||||
| 
 | ||||
| extern unsigned int DisaPc; | ||||
| extern char *DisaText; // Text buffer to write in
 | ||||
| 
 | ||||
| extern unsigned short (*DisaWord)(unsigned int a); | ||||
| int DisaGetEa(char *t,int ea,int size); | ||||
| 
 | ||||
| int DisaGet(); | ||||
| 
 | ||||
| #ifdef __cplusplus | ||||
| } // End of extern "C"
 | ||||
| #endif | ||||
|  | @ -1,493 +0,0 @@ | |||
| 
 | ||||
| #include "app.h" | ||||
| 
 | ||||
| int earead_check_addrerr = 1, eawrite_check_addrerr = 0; | ||||
| 
 | ||||
| // some ops use non-standard cycle counts for EAs, so are listed here.
 | ||||
| // all constants borrowed from the MUSASHI core by Karl Stenerud.
 | ||||
| 
 | ||||
| /* Extra cycles for JMP instruction (000, 010) */ | ||||
| int g_jmp_cycle_table[8] = | ||||
| { | ||||
| 	 4, /* EA_MODE_AI   */ | ||||
| 	 6, /* EA_MODE_DI   */ | ||||
| 	10, /* EA_MODE_IX   */ | ||||
| 	 6, /* EA_MODE_AW   */ | ||||
| 	 8, /* EA_MODE_AL   */ | ||||
| 	 6, /* EA_MODE_PCDI */ | ||||
| 	10, /* EA_MODE_PCIX */ | ||||
| 	 0, /* EA_MODE_I    */ | ||||
| }; | ||||
| 
 | ||||
| /* Extra cycles for JSR instruction (000, 010) */ | ||||
| int g_jsr_cycle_table[8] = | ||||
| { | ||||
| 	 4, /* EA_MODE_AI   */ | ||||
| 	 6, /* EA_MODE_DI   */ | ||||
| 	10, /* EA_MODE_IX   */ | ||||
| 	 6, /* EA_MODE_AW   */ | ||||
| 	 8, /* EA_MODE_AL   */ | ||||
| 	 6, /* EA_MODE_PCDI */ | ||||
| 	10, /* EA_MODE_PCIX */ | ||||
| 	 0, /* EA_MODE_I    */ | ||||
| }; | ||||
| 
 | ||||
| /* Extra cycles for LEA instruction (000, 010) */ | ||||
| int g_lea_cycle_table[8] = | ||||
| { | ||||
| 	 4, /* EA_MODE_AI   */ | ||||
| 	 8, /* EA_MODE_DI   */ | ||||
| 	12, /* EA_MODE_IX   */ | ||||
| 	 8, /* EA_MODE_AW   */ | ||||
| 	12, /* EA_MODE_AL   */ | ||||
| 	 8, /* EA_MODE_PCDI */ | ||||
| 	12, /* EA_MODE_PCIX */ | ||||
| 	 0, /* EA_MODE_I    */ | ||||
| }; | ||||
| 
 | ||||
| /* Extra cycles for PEA instruction (000, 010) */ | ||||
| int g_pea_cycle_table[8] = | ||||
| { | ||||
| 	 6, /* EA_MODE_AI   */ | ||||
| 	10, /* EA_MODE_DI   */ | ||||
| 	14, /* EA_MODE_IX   */ | ||||
| 	10, /* EA_MODE_AW   */ | ||||
| 	14, /* EA_MODE_AL   */ | ||||
| 	10, /* EA_MODE_PCDI */ | ||||
| 	14, /* EA_MODE_PCIX */ | ||||
| 	 0, /* EA_MODE_I    */ | ||||
| }; | ||||
| 
 | ||||
| /* Extra cycles for MOVEM instruction (000, 010) */ | ||||
| int g_movem_cycle_table[8] = | ||||
| { | ||||
| 	 0, /* EA_MODE_AI   */ | ||||
| 	 4, /* EA_MODE_DI   */ | ||||
| 	 6, /* EA_MODE_IX   */ | ||||
| 	 4, /* EA_MODE_AW   */ | ||||
| 	 8, /* EA_MODE_AL   */ | ||||
| 	 0, /* EA_MODE_PCDI */ | ||||
| 	 0, /* EA_MODE_PCIX */ | ||||
| 	 0, /* EA_MODE_I    */ | ||||
| }; | ||||
| 
 | ||||
| // add nonstandard EA
 | ||||
| int Ea_add_ns(int *tab, int ea) | ||||
| { | ||||
|     if(ea<0x10)         return 0; | ||||
|     if((ea&0x38)==0x10) return tab[0]; // (An) (ai)
 | ||||
|     if(ea<0x28)         return 0; | ||||
|     if(ea<0x30)         return tab[1]; // ($nn,An) (di)
 | ||||
|     if(ea<0x38)         return tab[2]; // ($nn,An,Rn) (ix)
 | ||||
|     if(ea==0x38)        return tab[3]; // (aw)
 | ||||
|     if(ea==0x39)        return tab[4]; // (al)
 | ||||
|     if(ea==0x3a)        return tab[5]; // ($nn,PC) (pcdi)
 | ||||
|     if(ea==0x3b)        return tab[6]; // ($nn,pc,Rn) (pcix)
 | ||||
|     if(ea==0x3c)        return tab[7]; // #$nnnn (i)
 | ||||
|     return 0; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // ---------------------------------------------------------------------------
 | ||||
| // Gets the offset of a register for an ea, and puts it in 'r'
 | ||||
| // Shifted left by 'shift'
 | ||||
| // Doesn't trash anything
 | ||||
| static int EaCalcReg(int r,int ea,int mask,int forceor,int shift,int noshift=0) | ||||
| { | ||||
|   int i=0,low=0,needor=0; | ||||
|   int lsl=0; | ||||
| 
 | ||||
|   for (i=mask|0x8000; (i&1)==0; i>>=1) low++; // Find out how high up the EA mask is
 | ||||
|   mask&=0xf<<low; // This is the max we can do
 | ||||
| 
 | ||||
|   if (ea>=8) | ||||
|   { | ||||
|     needor=1; // Need to OR to access A0-7
 | ||||
|     if ((g_op>>low)&8) { needor=0; mask|=8<<low; } // Ah - no we don't actually need to or, since the bit is high in r8
 | ||||
|     if (forceor) needor=1; // Special case for 0x30-0x38 EAs ;)
 | ||||
|   } | ||||
| 
 | ||||
|   ot("  and r%d,r8,#0x%.4x\n",r,mask); | ||||
|   if (needor) ot("  orr r%d,r%d,#0x%x ;@ A0-7\n",r,r,8<<low); | ||||
| 
 | ||||
|   // Find out amount to shift left:
 | ||||
|   lsl=shift-low; | ||||
| 
 | ||||
|   if (lsl&&!noshift) | ||||
|   { | ||||
|     ot("  mov r%d,r%d,",r,r); | ||||
|     if (lsl>0) ot("lsl #%d\n", lsl); | ||||
|     else       ot("lsr #%d\n",-lsl); | ||||
|   } | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // EaCalc - ARM Register 'a' = Effective Address
 | ||||
| // If ea>=0x10, trashes r0,r2 and r3, else nothing
 | ||||
| // size values 0, 1, 2 ~ byte, word, long
 | ||||
| // mask shows usable bits in r8
 | ||||
| int EaCalc(int a,int mask,int ea,int size,int top,int sign_extend) | ||||
| { | ||||
|   char text[32]=""; | ||||
|   int func=0; | ||||
| 
 | ||||
|   DisaPc=2; DisaGetEa(text,ea,size); // Get text version of the effective address
 | ||||
|   func=0x68+(size<<2); // Get correct read handler
 | ||||
| 
 | ||||
|   if (ea<0x10) | ||||
|   { | ||||
|     int noshift=0; | ||||
|     if (size>=2||(size==0&&(top||!sign_extend))) noshift=1; // Saves one opcode
 | ||||
| 
 | ||||
|     ot(";@ EaCalc : Get register index into r%d:\n",a); | ||||
| 
 | ||||
|     EaCalcReg(a,ea,mask,0,2,noshift); | ||||
|     return 0; | ||||
|   } | ||||
|    | ||||
|   ot(";@ EaCalc : Get '%s' into r%d:\n",text,a); | ||||
|   // (An), (An)+, -(An)
 | ||||
|   if (ea<0x28) | ||||
|   { | ||||
|     int step=1<<size, strr=a; | ||||
|     int low=0,lsl=0,i; | ||||
| 
 | ||||
|     if ((ea&7)==7 && step<2) step=2; // move.b (a7)+ or -(a7) steps by 2 not 1
 | ||||
| 
 | ||||
|     if (ea==0x1f||ea==0x27) // A7 handlers are always separate
 | ||||
|     { | ||||
|       ot("  ldr r%d,[r7,#0x3c] ;@ A7\n",a); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|       EaCalcReg(2,ea,mask,0,0,1); | ||||
|       if(mask) | ||||
|         for (i=mask|0x8000; (i&1)==0; i>>=1) low++; // Find out how high up the EA mask is
 | ||||
|       lsl=2-low; // Having a lsl #x here saves one opcode
 | ||||
|       if      (lsl>=0) ot("  ldr r%d,[r7,r2,lsl #%i]\n",a,lsl); | ||||
|       else if (lsl<0)  ot("  ldr r%d,[r7,r2,lsr #%i]\n",a,-lsl); | ||||
|     } | ||||
| 
 | ||||
|     if ((ea&0x38)==0x18) // (An)+
 | ||||
|     { | ||||
|       ot("  add r3,r%d,#%d ;@ Post-increment An\n",a,step); | ||||
|       strr=3; | ||||
|     } | ||||
| 
 | ||||
|     if ((ea&0x38)==0x20) // -(An)
 | ||||
|       ot("  sub r%d,r%d,#%d ;@ Pre-decrement An\n",a,a,step); | ||||
| 
 | ||||
|     if ((ea&0x38)==0x18||(ea&0x38)==0x20) | ||||
|     { | ||||
|       if (ea==0x1f||ea==0x27) | ||||
|       { | ||||
|         ot("  str r%d,[r7,#0x3c] ;@ A7\n",strr); | ||||
|       } | ||||
|       else | ||||
|       { | ||||
|         if      (lsl>=0) ot("  str r%d,[r7,r2,lsl #%i]\n",strr,lsl); | ||||
|         else if (lsl<0)  ot("  str r%d,[r7,r2,lsr #%i]\n",strr,-lsl); | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     if ((ea&0x38)==0x20) Cycles+=size<2 ? 6:10; // -(An) Extra cycles
 | ||||
|     else                 Cycles+=size<2 ? 4:8;  // (An),(An)+ Extra cycles
 | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   if (ea<0x30) // ($nn,An) (di)
 | ||||
|   { | ||||
|     ot("  ldrsh r0,[r4],#2 ;@ Fetch offset\n"); pc_dirty=1; | ||||
|     EaCalcReg(2,8,mask,0,0); | ||||
|     ot("  ldr r2,[r7,r2,lsl #2]\n"); | ||||
|     ot("  add r%d,r0,r2 ;@ Add on offset\n",a); | ||||
|     Cycles+=size<2 ? 8:12; // Extra cycles
 | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   if (ea<0x38) // ($nn,An,Rn) (ix)
 | ||||
|   { | ||||
|     ot(";@ Get extension word into r3:\n"); | ||||
|     ot("  ldrh r3,[r4],#2 ;@ ($Disp,PC,Rn)\n"); pc_dirty=1; | ||||
|     ot("  mov r2,r3,lsr #10\n"); | ||||
|     ot("  tst r3,#0x0800 ;@ Is Rn Word or Long\n"); | ||||
|     ot("  and r2,r2,#0x3c ;@ r2=Index of Rn\n"); | ||||
|     ot("  ldreqsh r2,[r7,r2] ;@ r2=Rn.w\n"); | ||||
|     ot("  ldrne   r2,[r7,r2] ;@ r2=Rn.l\n"); | ||||
|     ot("  mov r0,r3,asl #24 ;@ r0=Get 8-bit signed Disp\n"); | ||||
|     ot("  add r3,r2,r0,asr #24 ;@ r3=Disp+Rn\n"); | ||||
| 
 | ||||
|     EaCalcReg(2,8,mask,1,0); | ||||
|     ot("  ldr r2,[r7,r2,lsl #2]\n"); | ||||
|     ot("  add r%d,r2,r3 ;@ r%d=Disp+An+Rn\n",a,a); | ||||
|     Cycles+=size<2 ? 10:14; // Extra cycles
 | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   if (ea==0x38) // (aw)
 | ||||
|   { | ||||
|     ot("  ldrsh r%d,[r4],#2 ;@ Fetch Absolute Short address\n",a); pc_dirty=1; | ||||
|     Cycles+=size<2 ? 8:12; // Extra cycles
 | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   if (ea==0x39) // (al)
 | ||||
|   { | ||||
|     ot("  ldrh r2,[r4],#2 ;@ Fetch Absolute Long address\n"); | ||||
|     ot("  ldrh r0,[r4],#2\n"); pc_dirty=1; | ||||
|     ot("  orr r%d,r0,r2,lsl #16\n",a); | ||||
|     Cycles+=size<2 ? 12:16; // Extra cycles
 | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   if (ea==0x3a) // ($nn,PC) (pcdi)
 | ||||
|   { | ||||
|     ot("  ldr r0,[r7,#0x60] ;@ Get Memory base\n"); | ||||
|     ot("  sub r0,r4,r0 ;@ Real PC\n"); | ||||
|     ot("  ldrsh r2,[r4],#2 ;@ Fetch extension\n"); pc_dirty=1; | ||||
|     ot("  mov r0,r0,lsl #8\n"); | ||||
|     ot("  add r%d,r2,r0,asr #8 ;@ ($nn,PC)\n",a); | ||||
|     Cycles+=size<2 ? 8:12; // Extra cycles
 | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   if (ea==0x3b) // ($nn,pc,Rn) (pcix)
 | ||||
|   { | ||||
|     ot("  ldr r0,[r7,#0x60] ;@ Get Memory base\n"); | ||||
|     ot("  ldrh r3,[r4] ;@ Get extension word\n"); | ||||
|     ot("  sub r0,r4,r0 ;@ r0=PC\n"); | ||||
|     ot("  add r4,r4,#2\n"); pc_dirty=1; | ||||
|     ot("  mov r0,r0,asl #8 ;@ use only 24bits of PC\n"); | ||||
|     ot("  mov r2,r3,lsr #10\n"); | ||||
|     ot("  tst r3,#0x0800 ;@ Is Rn Word or Long\n"); | ||||
|     ot("  and r2,r2,#0x3c ;@ r2=Index of Rn\n"); | ||||
|     ot("  ldreqsh r2,[r7,r2] ;@ r2=Rn.w\n"); | ||||
|     ot("  ldrne   r2,[r7,r2] ;@ r2=Rn.l\n"); | ||||
|     ot("  mov r3,r3,asl #24 ;@ r3=Get 8-bit signed Disp\n"); | ||||
|     ot("  add r2,r2,r3,asr #24 ;@ r2=Disp+Rn\n"); | ||||
|     ot("  add r%d,r2,r0,asr #8 ;@ r%d=Disp+PC+Rn\n",a,a); | ||||
|     Cycles+=size<2 ? 10:14; // Extra cycles
 | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   if (ea==0x3c) // #$nnnn (i)
 | ||||
|   { | ||||
|     if (size<2) | ||||
|     { | ||||
|       ot("  ldr%s r%d,[r4],#2 ;@ Fetch immediate value\n",Sarm[size&3],a); pc_dirty=1; | ||||
|       Cycles+=4; // Extra cycles
 | ||||
|       return 0; | ||||
|     } | ||||
| 
 | ||||
|     ot("  ldrh r2,[r4],#2 ;@ Fetch immediate value\n"); | ||||
|     ot("  ldrh r3,[r4],#2\n"); pc_dirty=1; | ||||
|     ot("  orr r%d,r3,r2,lsl #16\n",a); | ||||
|     Cycles+=8; // Extra cycles
 | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   return 1; | ||||
| } | ||||
| 
 | ||||
| // ---------------------------------------------------------------------------
 | ||||
| // Read effective address in (ARM Register 'a') to ARM register 'v'
 | ||||
| // 'a' and 'v' can be anything but 0 is generally best (for both)
 | ||||
| // If (ea<0x10) nothing is trashed, else r0-r3 is trashed
 | ||||
| // If 'top' is given, the ARM register v shifted to the top, e.g. 0xc000 -> 0xc0000000
 | ||||
| // If top is 0 and sign_extend is not, then ARM register v is sign extended,
 | ||||
| // e.g. 0xc000 -> 0xffffc000 (else it may or may not be sign extended)
 | ||||
| 
 | ||||
| int EaRead(int a,int v,int ea,int size,int mask,int top,int sign_extend) | ||||
| { | ||||
|   char text[32]=""; | ||||
|   int shift=0; | ||||
|   | ||||
|   shift=32-(8<<size); | ||||
| 
 | ||||
|   DisaPc=2; DisaGetEa(text,ea,size); // Get text version of the effective address
 | ||||
| 
 | ||||
|   if (ea<0x10) | ||||
|   { | ||||
|     int lsl=0,low=0,nsarm=size&3,i; | ||||
|     if (size>=2||(size==0&&(top||!sign_extend))) { | ||||
|       if(mask) | ||||
|         for (i=mask|0x8000; (i&1)==0; i>>=1) low++; // Find out how high up the EA mask is
 | ||||
|       lsl=2-low; // Having a lsl #2 here saves one opcode
 | ||||
|     } | ||||
| 
 | ||||
|     if (top||!sign_extend) nsarm=3; | ||||
| 
 | ||||
|     ot(";@ EaRead : Read register[r%d] into r%d:\n",a,v); | ||||
| 
 | ||||
|     if      (lsl>0) ot("  ldr%s r%d,[r7,r%d,lsl #%i]\n",Narm[nsarm],v,a,lsl); | ||||
|     else if (lsl<0) ot("  ldr%s r%d,[r7,r%d,lsr #%i]\n",Narm[nsarm],v,a,-lsl); | ||||
|     else            ot("  ldr%s r%d,[r7,r%d]\n",Sarm[nsarm],v,a); | ||||
| 
 | ||||
|     if (top&&shift) ot("  mov r%d,r%d,asl #%d\n",v,v,shift); | ||||
| 
 | ||||
|     ot("\n"); return 0; | ||||
|   } | ||||
| 
 | ||||
|   ot(";@ EaRead : Read '%s' (address in r%d) into r%d:\n",text,a,v); | ||||
| 
 | ||||
|   if (ea==0x3c) | ||||
|   { | ||||
|     int asl=0; | ||||
| 
 | ||||
|     if (top) asl=shift; | ||||
| 
 | ||||
|     if (asl) ot("  mov r%d,r%d,asl #%d\n",v,a,asl); | ||||
|     else if (v!=a) ot("  mov r%d,r%d\n",v,a); | ||||
|     ot("\n"); return 0; | ||||
|   } | ||||
| 
 | ||||
|   if (ea>=0x3a && ea<=0x3b) MemHandler(2,size,a,earead_check_addrerr); // Fetch
 | ||||
|   else                      MemHandler(0,size,a,earead_check_addrerr); // Read
 | ||||
| 
 | ||||
|   // defaults to 1, as most things begins with a read
 | ||||
|   earead_check_addrerr=1; | ||||
| 
 | ||||
|   if (sign_extend) | ||||
|   { | ||||
|     int d_reg=0; | ||||
|     if (shift) { | ||||
|       ot("  mov r%d,r%d,asl #%d\n",v,d_reg,shift); | ||||
|       d_reg=v; | ||||
|     } | ||||
|     if (!top && shift) { | ||||
|       ot("  mov r%d,r%d,asr #%d\n",v,d_reg,shift); | ||||
|       d_reg=v; | ||||
|     } | ||||
|     if (d_reg != v) | ||||
|       ot("  mov r%d,r%d\n",v,d_reg); | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     if (top && shift) | ||||
|       ot("  mov r%d,r0,asl #%d\n",v,shift); | ||||
|     else if (v!=0) | ||||
|       ot("  mov r%d,r0\n",v); | ||||
|   } | ||||
| 
 | ||||
|   ot("\n"); return 0; | ||||
| } | ||||
| 
 | ||||
| // calculate EA and  read
 | ||||
| // if (ea  < 0x10) nothing is trashed
 | ||||
| // if (ea == 0x3c) r2 and r3 are trashed
 | ||||
| // else r0-r3 are trashed
 | ||||
| // size values 0, 1, 2 ~ byte, word, long
 | ||||
| // r_ea is reg to store ea in (-1 means ea is not needed), r is dst reg
 | ||||
| // if sign_extend is 0, non-32bit values will have MS bits undefined
 | ||||
| int EaCalcRead(int r_ea,int r,int ea,int size,int mask,int sign_extend) | ||||
| { | ||||
|   if (ea<0x10) | ||||
|   { | ||||
|     if (r_ea==-1) | ||||
|     { | ||||
|       r_ea=r; | ||||
|       if (!sign_extend) size=2; | ||||
|     } | ||||
|   } | ||||
|   else if (ea==0x3c) // #imm
 | ||||
|   { | ||||
|     r_ea=r; | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     if (r_ea==-1) r_ea=0; | ||||
|   } | ||||
| 
 | ||||
|   EaCalc (r_ea,mask,ea,size,0,sign_extend); | ||||
|   EaRead (r_ea,   r,ea,size,mask,0,sign_extend); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| int EaCalcReadNoSE(int r_ea,int r,int ea,int size,int mask) | ||||
| { | ||||
|   return EaCalcRead(r_ea,r,ea,size,mask,0); | ||||
| } | ||||
| 
 | ||||
| // Return 1 if we can read this ea
 | ||||
| int EaCanRead(int ea,int size) | ||||
| { | ||||
|   if (size<0) | ||||
|   { | ||||
|     // LEA:
 | ||||
|     // These don't make sense?:
 | ||||
|     if (ea< 0x10) return 0; // Register
 | ||||
|     if (ea==0x3c) return 0; // Immediate
 | ||||
|     if (ea>=0x18 && ea<0x28) return 0; // Pre/Post inc/dec An
 | ||||
|   } | ||||
| 
 | ||||
|   if (ea<=0x3c) return 1; | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ---------------------------------------------------------------------------
 | ||||
| // Write effective address (ARM Register 'a') with ARM register 'v'
 | ||||
| // Trashes r0-r3,r12,lr; 'a' can be 0 or 2+, 'v' can be 1 or higher
 | ||||
| // If a==0 and v==1 it's faster though.
 | ||||
| int EaWrite(int a,int v,int ea,int size,int mask,int top,int sign_extend_ea) | ||||
| { | ||||
|   char text[32]=""; | ||||
|   int shift=0; | ||||
| 
 | ||||
|   if(a == 1) { printf("Error! EaWrite a==1 !\n"); return 1; } | ||||
| 
 | ||||
|   if (top) shift=32-(8<<size); | ||||
| 
 | ||||
|   DisaPc=2; DisaGetEa(text,ea,size); // Get text version of the effective address
 | ||||
| 
 | ||||
|   if (ea<0x10) | ||||
|   { | ||||
|     int lsl=0,low=0,i; | ||||
|     if (size>=2||(size==0&&(top||!sign_extend_ea))) { | ||||
|       if(mask) | ||||
|         for (i=mask|0x8000; (i&1)==0; i>>=1) low++; // Find out how high up the EA mask is
 | ||||
|       lsl=2-low; // Having a lsl #x here saves one opcode
 | ||||
|     } | ||||
| 
 | ||||
|     ot(";@ EaWrite: r%d into register[r%d]:\n",v,a); | ||||
|     if (shift)  ot("  mov r%d,r%d,asr #%d\n",v,v,shift); | ||||
| 
 | ||||
|     if      (lsl>0) ot("  str%s r%d,[r7,r%d,lsl #%i]\n",Narm[size&3],v,a,lsl); | ||||
|     else if (lsl<0) ot("  str%s r%d,[r7,r%d,lsr #%i]\n",Narm[size&3],v,a,-lsl); | ||||
|     else            ot("  str%s r%d,[r7,r%d]\n",Narm[size&3],v,a); | ||||
| 
 | ||||
|     ot("\n"); return 0; | ||||
|   } | ||||
| 
 | ||||
|   ot(";@ EaWrite: Write r%d into '%s' (address in r%d):\n",v,text,a); | ||||
| 
 | ||||
|   if (ea==0x3c) { ot("Error! Write EA=0x%x\n\n",ea); return 1; } | ||||
| 
 | ||||
|   if (shift)     ot("  mov r1,r%d,asr #%d\n",v,shift); | ||||
|   else if (v!=1) ot("  mov r1,r%d\n",v); | ||||
| 
 | ||||
|   MemHandler(1,size,a,eawrite_check_addrerr); // Call write handler
 | ||||
| 
 | ||||
|   // not check by default, because most cases are rmw and
 | ||||
|   // address was already checked before reading
 | ||||
|   eawrite_check_addrerr = 0; | ||||
| 
 | ||||
|   ot("\n"); return 0; | ||||
| } | ||||
| 
 | ||||
| // Return 1 if we can write this ea
 | ||||
| int EaCanWrite(int ea) | ||||
| { | ||||
|   if (ea<=0x39) return 1; // 3b?
 | ||||
|   return 0; | ||||
| } | ||||
| // ---------------------------------------------------------------------------
 | ||||
| 
 | ||||
| // Return 1 if EA is An reg
 | ||||
| int EaAn(int ea) | ||||
| { | ||||
|   if((ea&0x38)==8) return 1; | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
							
								
								
									
										1298
									
								
								cpu/Cyclone/Main.cpp
									
										
									
									
									
								
							
							
						
						
									
										1298
									
								
								cpu/Cyclone/Main.cpp
									
										
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							|  | @ -1,207 +0,0 @@ | |||
| 
 | ||||
| #include "app.h" | ||||
| 
 | ||||
| int opend_op_changes_cycles, opend_check_interrupt, opend_check_trace; | ||||
| 
 | ||||
| static unsigned char OpData[16]={0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}; | ||||
| 
 | ||||
| static unsigned short OpRead16(unsigned int a) | ||||
| { | ||||
|   return (unsigned short)( (OpData[a&15]<<8) | OpData[(a+1)&15] ); | ||||
| } | ||||
| 
 | ||||
| // For opcode 'op' use handler 'use'
 | ||||
| void OpUse(int op,int use) | ||||
| { | ||||
|   char text[64]=""; | ||||
|   CyJump[op]=use; | ||||
| 
 | ||||
|   if (op!=use) return; | ||||
| 
 | ||||
|   // Disassemble opcode
 | ||||
|   DisaPc=0; | ||||
|   DisaText=text; | ||||
|   DisaWord=OpRead16; | ||||
| 
 | ||||
|   DisaGet(); | ||||
|   ot(";@ ---------- [%.4x] %s uses Op%.4x ----------\n",op,text,use); | ||||
| } | ||||
| 
 | ||||
| void OpStart(int op, int sea, int tea, int op_changes_cycles, int supervisor_check) | ||||
| { | ||||
|   int last_op_count=arm_op_count; | ||||
| 
 | ||||
|   Cycles=0; | ||||
|   OpUse(op,op); // This opcode obviously uses this handler
 | ||||
|   ot("Op%.4x%s\n", op, ms?"":":"); | ||||
| 
 | ||||
|   if (supervisor_check) | ||||
|   { | ||||
|     // checks for supervisor bit, if not set, jumps to SuperEnd()
 | ||||
|     // also sets r11 to SR high value, SuperChange() uses this
 | ||||
|     ot("  ldr r11,[r7,#0x44] ;@ Get SR high\n"); | ||||
|   } | ||||
|   if ((sea >= 0x10 && sea != 0x3c) || (tea >= 0x10 && tea != 0x3c)) | ||||
|   { | ||||
| #if MEMHANDLERS_NEED_PREV_PC | ||||
|     ot("  str r4,[r7,#0x50] ;@ Save prev PC + 2\n"); | ||||
| #endif | ||||
| #if MEMHANDLERS_NEED_CYCLES | ||||
|     ot("  str r5,[r7,#0x5c] ;@ Save Cycles\n"); | ||||
| #endif | ||||
|   } | ||||
|   if (supervisor_check) | ||||
|   { | ||||
|     ot("  tst r11,#0x20 ;@ Check we are in supervisor mode\n"); | ||||
|     ot("  beq WrongPrivilegeMode ;@ No\n"); | ||||
|   } | ||||
|   if ((sea >= 0x10 && sea != 0x3c) || (tea >= 0x10 && tea != 0x3c)) { | ||||
| #if MEMHANDLERS_CHANGE_CYCLES | ||||
|     if (op_changes_cycles) | ||||
|       ot("  mov r5,#0\n"); | ||||
| #endif | ||||
|   } | ||||
|   if (last_op_count!=arm_op_count) | ||||
|     ot("\n"); | ||||
|   pc_dirty = 1; | ||||
|   opend_op_changes_cycles = opend_check_interrupt = opend_check_trace = 0; | ||||
| } | ||||
| 
 | ||||
| void OpEnd(int sea, int tea) | ||||
| { | ||||
|   int did_fetch=0; | ||||
|   opend_check_trace = opend_check_trace && EMULATE_TRACE; | ||||
| #if MEMHANDLERS_CHANGE_CYCLES | ||||
|   if ((sea >= 0x10 && sea != 0x3c) || (tea >= 0x10 && tea != 0x3c)) | ||||
|   { | ||||
|     if (opend_op_changes_cycles) | ||||
|     { | ||||
|       ot("  ldr r0,[r7,#0x5c] ;@ Load Cycles\n"); | ||||
|       ot("  ldrh r8,[r4],#2 ;@ Fetch next opcode\n"); | ||||
|       ot("  add r5,r0,r5\n"); | ||||
|       did_fetch=1; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|       ot("  ldr r5,[r7,#0x5c] ;@ Load Cycles\n"); | ||||
|     } | ||||
|   } | ||||
| #endif | ||||
|   if (!did_fetch) | ||||
|     ot("  ldrh r8,[r4],#2 ;@ Fetch next opcode\n"); | ||||
|   if (opend_check_trace) | ||||
|     ot("  ldr r1,[r7,#0x44]\n"); | ||||
|   ot("  subs r5,r5,#%d ;@ Subtract cycles\n",Cycles); | ||||
|   if (opend_check_trace) | ||||
|   { | ||||
|     ot(";@ CheckTrace:\n"); | ||||
|     ot("  tst r1,#0x80\n"); | ||||
|     ot("  bne CycloneDoTraceWithChecks\n"); | ||||
|     ot("  cmp r5,#0\n"); | ||||
|   } | ||||
|   if (opend_check_interrupt) | ||||
|   { | ||||
|     ot("  blt CycloneEnd\n"); | ||||
|     ot(";@ CheckInterrupt:\n"); | ||||
|     if (!opend_check_trace) | ||||
|       ot("  ldr r1,[r7,#0x44]\n"); | ||||
|     ot("  movs r0,r1,lsr #24 ;@ Get IRQ level\n"); // same as  ldrb r0,[r7,#0x47]
 | ||||
|     ot("  ldreq pc,[r6,r8,asl #2] ;@ Jump to next opcode handler\n"); | ||||
|     ot("  cmp r0,#6 ;@ irq>6 ?\n"); | ||||
|     ot("  andle r1,r1,#7 ;@ Get interrupt mask\n"); | ||||
|     ot("  cmple r0,r1 ;@ irq<=6: Is irq<=mask ?\n"); | ||||
|     ot("  ldrle pc,[r6,r8,asl #2] ;@ Jump to next opcode handler\n"); | ||||
|     ot("  b CycloneDoInterruptGoBack\n"); | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     ot("  ldrge pc,[r6,r8,asl #2] ;@ Jump to opcode handler\n"); | ||||
|     ot("  b CycloneEnd\n"); | ||||
|   } | ||||
|   ot("\n"); | ||||
| } | ||||
| 
 | ||||
| int OpBase(int op,int size,int sepa) | ||||
| { | ||||
|   int ea=op&0x3f; // Get Effective Address
 | ||||
|   if (ea<0x10) return sepa?(op&~0x7):(op&~0xf); // Use 1 handler for d0-d7 and a0-a7
 | ||||
|   if (size==0&&(ea==0x1f || ea==0x27)) return op; // Specific handler for (a7)+ and -(a7)
 | ||||
|   if (ea<0x38) return op&~7;   // Use 1 handler for (a0)-(a7), etc...
 | ||||
|   return op; | ||||
| } | ||||
| 
 | ||||
| // Get flags, trashes r2
 | ||||
| int OpGetFlags(int subtract,int xbit,int specialz) | ||||
| { | ||||
|   if (specialz) ot("  orr r2,r10,#0xb0000000 ;@ for old Z\n"); | ||||
| 
 | ||||
|   ot("  mrs r10,cpsr ;@ r10=flags\n"); | ||||
| 
 | ||||
|   if (specialz) ot("  andeq r10,r10,r2 ;@ fix Z\n"); | ||||
| 
 | ||||
|   if (subtract) ot("  eor r10,r10,#0x20000000 ;@ Invert carry\n"); | ||||
| 
 | ||||
|   if (xbit) | ||||
|   { | ||||
|     ot("  str r10,[r7,#0x4c] ;@ Save X bit\n"); | ||||
|   } | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // -----------------------------------------------------------------
 | ||||
| 
 | ||||
| int g_op; | ||||
| 
 | ||||
| void OpAny(int op) | ||||
| { | ||||
|   memset(OpData,0x33,sizeof(OpData)); | ||||
|   OpData[0]=(unsigned char)(op>>8); | ||||
|   OpData[1]=(unsigned char)op; | ||||
|   g_op=op; | ||||
| 
 | ||||
|   if ((op&0xf100)==0x0000) OpArith(op);    // +
 | ||||
|   if ((op&0xc000)==0x0000) OpMove(op);     // +
 | ||||
|   if ((op&0xf5bf)==0x003c) OpArithSr(op);  // + Ori/Andi/Eori $nnnn,sr
 | ||||
|   if ((op&0xf100)==0x0100) OpBtstReg(op);  // +
 | ||||
|   if ((op&0xf138)==0x0108) OpMovep(op);    // +
 | ||||
|   if ((op&0xff00)==0x0800) OpBtstImm(op);  // +
 | ||||
|   if ((op&0xf900)==0x4000) OpNeg(op);      // +
 | ||||
|   if ((op&0xf140)==0x4100) OpChk(op);      // +
 | ||||
|   if ((op&0xf1c0)==0x41c0) OpLea(op);      // +
 | ||||
|   if ((op&0xf9c0)==0x40c0) OpMoveSr(op);   // +
 | ||||
|   if ((op&0xffc0)==0x4800) OpNbcd(op);     // +
 | ||||
|   if ((op&0xfff8)==0x4840) OpSwap(op);     // +
 | ||||
|   if ((op&0xffc0)==0x4840) OpPea(op);      // +
 | ||||
|   if ((op&0xffb8)==0x4880) OpExt(op);      // +
 | ||||
|   if ((op&0xfb80)==0x4880) OpMovem(op);    // +
 | ||||
|   if ((op&0xff00)==0x4a00) OpTst(op);      // +
 | ||||
|   if ((op&0xffc0)==0x4ac0) OpTas(op);      // +
 | ||||
|   if ((op&0xfff0)==0x4e40) OpTrap(op);     // +
 | ||||
|   if ((op&0xfff8)==0x4e50) OpLink(op);     // +
 | ||||
|   if ((op&0xfff8)==0x4e58) OpUnlk(op);     // +
 | ||||
|   if ((op&0xfff0)==0x4e60) OpMoveUsp(op);  // +
 | ||||
|   if ((op&0xfff8)==0x4e70) Op4E70(op);     // + Reset/Rts etc
 | ||||
|   if ((op&0xfffd)==0x4e70) OpStopReset(op);// +
 | ||||
|   if ((op&0xff80)==0x4e80) OpJsr(op);      // +
 | ||||
|   if ((op&0xf000)==0x5000) OpAddq(op);     // +
 | ||||
|   if ((op&0xf0c0)==0x50c0) OpSet(op);      // +
 | ||||
|   if ((op&0xf0f8)==0x50c8) OpDbra(op);     // +
 | ||||
|   if ((op&0xf000)==0x6000) OpBranch(op);   // +
 | ||||
|   if ((op&0xf100)==0x7000) OpMoveq(op);    // +
 | ||||
|   if ((op&0xa000)==0x8000) OpArithReg(op); // + Or/Sub/And/Add
 | ||||
|   if ((op&0xb1f0)==0x8100) OpAbcd(op);     // +
 | ||||
|   if ((op&0xb0c0)==0x80c0) OpMul(op);      // +
 | ||||
|   if ((op&0x90c0)==0x90c0) OpAritha(op);   // +
 | ||||
|   if ((op&0xb130)==0x9100) OpAddx(op);     // +
 | ||||
|   if ((op&0xf000)==0xb000) OpCmpEor(op);   // +
 | ||||
|   if ((op&0xf138)==0xb108) OpCmpm(op);     // +
 | ||||
|   if ((op&0xf130)==0xc100) OpExg(op);      // +
 | ||||
|   if ((op&0xf000)==0xe000) OpAsr(op);      // + Asr/l/Ror/l etc
 | ||||
|   if ((op&0xf8c0)==0xe0c0) OpAsrEa(op);    // +
 | ||||
| 
 | ||||
|   if (op==0xffff) | ||||
|   { | ||||
|     SuperEnd(); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
|  | @ -1,819 +0,0 @@ | |||
| 
 | ||||
| #include "app.h" | ||||
| 
 | ||||
| // --------------------- Opcodes 0x0000+ ---------------------
 | ||||
| // Emit an Ori/And/Sub/Add/Eor/Cmp Immediate opcode, 0000ttt0 ssaaaaaa
 | ||||
| int OpArith(int op) | ||||
| { | ||||
|   int type=0,size=0; | ||||
|   int sea=0,tea=0; | ||||
|   int use=0; | ||||
|   const char *shiftstr=""; | ||||
| 
 | ||||
|   // Get source and target EA
 | ||||
|   type=(op>>9)&7; if (type==4 || type>=7) return 1; | ||||
|   size=(op>>6)&3; if (size>=3) return 1; | ||||
|   sea=   0x003c; | ||||
|   tea=op&0x003f; | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanRead(tea,size)==0) return 1; | ||||
|   if (EaCanWrite(tea)==0 || EaAn(tea)) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op, sea, tea); Cycles=4; | ||||
| 
 | ||||
|   // imm must be read first
 | ||||
|   EaCalcReadNoSE(-1,10,sea,size,0); | ||||
|   EaCalcReadNoSE((type!=6)?11:-1,0,tea,size,0x003f); | ||||
| 
 | ||||
|   if (size<2) shiftstr=(char *)(size?",asl #16":",asl #24"); | ||||
|   if (size<2) ot("  mov r10,r10,asl #%i\n",size?16:24); | ||||
| 
 | ||||
|   ot(";@ Do arithmetic:\n"); | ||||
| 
 | ||||
|   if (type==0) ot("  orr r1,r10,r0%s\n",shiftstr); | ||||
|   if (type==1) ot("  and r1,r10,r0%s\n",shiftstr); | ||||
|   if (type==2||type==6) | ||||
|                ot("  rsbs r1,r10,r0%s ;@ Defines NZCV\n",shiftstr); | ||||
|   if (type==3) ot("  adds r1,r10,r0%s ;@ Defines NZCV\n",shiftstr); | ||||
|   if (type==5) ot("  eor r1,r10,r0%s\n",shiftstr); | ||||
| 
 | ||||
|   if (type<2 || type==5) ot("  adds r1,r1,#0 ;@ Defines NZ, clears CV\n"); // 0,1,5
 | ||||
| 
 | ||||
|   if (type< 2) OpGetFlags(0,0); // Ori/And
 | ||||
|   if (type==2) OpGetFlags(1,1); // Sub: Subtract/X-bit
 | ||||
|   if (type==3) OpGetFlags(0,1); // Add: X-bit
 | ||||
|   if (type==5) OpGetFlags(0,0); // Eor
 | ||||
|   if (type==6) OpGetFlags(1,0); // Cmp: Subtract
 | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   if (type!=6) | ||||
|   { | ||||
|     EaWrite(11, 1, tea,size,0x003f,1); | ||||
|   } | ||||
| 
 | ||||
|   // Correct cycles:
 | ||||
|   if (type==6) | ||||
|   { | ||||
|     if (size>=2 && tea<0x10) Cycles+=2; | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     if (size>=2) Cycles+=4; | ||||
|     if (tea>=8)  Cycles+=4; | ||||
|     if (type==1 && size>=2 && tea<8) Cycles-=2; | ||||
|   } | ||||
| 
 | ||||
|   OpEnd(sea,tea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x5000+ ---------------------
 | ||||
| int OpAddq(int op) | ||||
| { | ||||
|   // 0101nnnt xxeeeeee (nnn=#8,1-7 t=addq/subq xx=size, eeeeee=EA)
 | ||||
|   int num=0,type=0,size=0,ea=0; | ||||
|   int use=0; | ||||
|   char count[16]=""; | ||||
|   int shift=0; | ||||
| 
 | ||||
|   num =(op>>9)&7; if (num==0) num=8; | ||||
|   type=(op>>8)&1; | ||||
|   size=(op>>6)&3; if (size>=3) return 1; | ||||
|   ea  = op&0x3f; | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanRead (ea,size)==0) return 1; | ||||
|   if (EaCanWrite(ea)     ==0) return 1; | ||||
|   if (size == 0 && EaAn(ea) ) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,size,1); | ||||
| 
 | ||||
|   if (num!=8) use|=0x0e00; // If num is not 8, use same handler
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,ea); | ||||
|   Cycles=ea<8?4:8; | ||||
|   if(type==0&&size==1) Cycles=ea<0x10?4:8; | ||||
|   if(size>=2) Cycles=ea<0x10?8:12; | ||||
| 
 | ||||
|   if (size>0 && (ea&0x38)==0x08) size=2; // addq.w #n,An is also 32-bit
 | ||||
| 
 | ||||
|   EaCalcReadNoSE(11,0,ea,size,0x003f); | ||||
| 
 | ||||
|   shift=32-(8<<size); | ||||
| 
 | ||||
|   if (num!=8) | ||||
|   { | ||||
|     int lsr=9-shift; | ||||
| 
 | ||||
|     ot("  and r2,r8,#0x0e00 ;@ Get quick value\n"); | ||||
| 
 | ||||
|     if (lsr>=0) sprintf(count,"r2,lsr #%d",  lsr); | ||||
|     else        sprintf(count,"r2,lsl #%d", -lsr); | ||||
| 
 | ||||
|     ot("\n"); | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     sprintf(count,"#0x%.4x",8<<shift); | ||||
|   } | ||||
| 
 | ||||
|   if (size<2)  ot("  mov r0,r0,asl #%d\n\n",size?16:24); | ||||
| 
 | ||||
|   if (type==0) ot("  adds r1,r0,%s\n",count); | ||||
|   if (type==1) ot("  subs r1,r0,%s\n",count); | ||||
| 
 | ||||
|   if ((ea&0x38)!=0x08) OpGetFlags(type,1); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   EaWrite(11,     1, ea,size,0x003f,1); | ||||
| 
 | ||||
|   OpEnd(ea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x8000+ ---------------------
 | ||||
| // 1t0tnnnd xxeeeeee (tt=type:or/sub/and/add xx=size, eeeeee=EA)
 | ||||
| int OpArithReg(int op) | ||||
| { | ||||
|   int use=0; | ||||
|   int type=0,size=0,dir=0,rea=0,ea=0; | ||||
|   const char *asl=""; | ||||
|   const char *strop=0; | ||||
| 
 | ||||
|   type=(op>>12)&5; | ||||
|   rea =(op>> 9)&7; | ||||
|   dir =(op>> 8)&1; // er,re
 | ||||
|   size=(op>> 6)&3; if (size>=3) return 1; | ||||
|   ea  = op&0x3f; | ||||
| 
 | ||||
|   if (dir && ea<0x10) return 1; // addx/subx opcode
 | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (dir==0 && EaCanRead (ea,size)==0) return 1; | ||||
|   if (dir    && EaCanWrite(ea)==0)      return 1; | ||||
|   if ((size==0||!(type&1))&&EaAn(ea))   return 1; | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   use&=~0x0e00; // Use same opcode for Dn
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,ea); Cycles=4; | ||||
| 
 | ||||
|   EaCalcReadNoSE(dir?11:-1,0,ea,size,0x003f); | ||||
| 
 | ||||
|   EaCalcReadNoSE(dir?-1:11,1,rea,size,0x0e00); | ||||
| 
 | ||||
|   ot(";@ Do arithmetic:\n"); | ||||
|   if (type==0) strop = "orr"; | ||||
|   if (type==1) strop = (char *) (dir ? "subs" : "rsbs"); | ||||
|   if (type==4) strop = "and"; | ||||
|   if (type==5) strop = "adds"; | ||||
| 
 | ||||
|   if (size==0) asl=",asl #24"; | ||||
|   if (size==1) asl=",asl #16"; | ||||
| 
 | ||||
|   if (size<2) ot("  mov r0,r0%s\n",asl); | ||||
|   ot("  %s r1,r0,r1%s\n",strop,asl); | ||||
| 
 | ||||
|   if ((type&1)==0) ot("  adds r1,r1,#0 ;@ Defines NZ, clears CV\n"); | ||||
| 
 | ||||
|   OpGetFlags(type==1,type&1); // 1==subtract
 | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   ot(";@ Save result:\n"); | ||||
|   if (size<2) ot("  mov r1,r1,asr #%d\n",size?16:24); | ||||
|   if (dir) EaWrite(11, 1, ea,size,0x003f,0,0); | ||||
|   else     EaWrite(11, 1,rea,size,0x0e00,0,0); | ||||
| 
 | ||||
|   if(rea==ea) { | ||||
|     if(ea<8) Cycles=(size>=2)?8:4; else Cycles+=(size>=2)?26:14; | ||||
|   } else if(dir) { | ||||
|     Cycles+=4; | ||||
|     if(size>=2) Cycles+=4; | ||||
|   } else { | ||||
|     if(size>=2) { | ||||
|       Cycles+=2; | ||||
|       if(ea<0x10||ea==0x3c) Cycles+=2; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   OpEnd(ea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x80c0+ ---------------------
 | ||||
| int OpMul(int op) | ||||
| { | ||||
|   // Div/Mul: 1m00nnns 11eeeeee (m=Mul, nnn=Register Dn, s=signed, eeeeee=EA)
 | ||||
|   int type=0,rea=0,sign=0,ea=0; | ||||
|   int use=0; | ||||
| 
 | ||||
|   type=(op>>14)&1; // div/mul
 | ||||
|   rea =(op>> 9)&7; | ||||
|   sign=(op>> 8)&1; | ||||
|   ea  = op&0x3f; | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanRead(ea,1)==0||EaAn(ea)) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,1); | ||||
|   use&=~0x0e00; // Use same for all registers
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,ea); | ||||
|   if(type) Cycles=54; | ||||
|   else     Cycles=sign?158:140; | ||||
| 
 | ||||
|   EaCalcReadNoSE(-1,0,ea,1,0x003f); | ||||
| 
 | ||||
|   EaCalc(11,0x0e00,rea, 2); | ||||
|   EaRead(11,     2,rea, 2,0x0e00); | ||||
| 
 | ||||
|   ot("  movs r1,r0,asl #16\n"); | ||||
| 
 | ||||
|   if (type==0) // div
 | ||||
|   { | ||||
|     // the manual says C is always cleared, but neither Musashi nor FAME do that
 | ||||
|     //ot("  bic r10,r10,#0x20000000 ;@ always clear C\n");
 | ||||
|     ot("  beq divzero%.4x ;@ division by zero\n",op); | ||||
|     ot("\n"); | ||||
|      | ||||
|     if (sign) | ||||
|     { | ||||
|       ot("  mov r12,#0 ;@ r12 = 1 or 2 if the result is negative\n"); | ||||
|       ot("  tst r2,r2\n"); | ||||
|       ot("  orrmi r12,r12,#2\n"); | ||||
|       ot("  rsbmi r2,r2,#0 ;@ Make r2 positive\n"); | ||||
|       ot("\n"); | ||||
|       ot("  movs r0,r1,asr #16\n"); | ||||
|       ot("  orrmi r12,r12,#1\n"); | ||||
|       ot("  rsbmi r0,r0,#0 ;@ Make r0 positive\n"); | ||||
|       ot("\n"); | ||||
|       ot(";@ detect the nasty 0x80000000 / -1 situation\n"); | ||||
|       ot("  mov r3,r2,asr #31\n"); | ||||
|       ot("  eors r3,r3,r1,asr #16\n"); | ||||
|       ot("  beq wrendofop%.4x\n",op); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|       ot("  mov r0,r1,lsr #16 ;@ use only 16 bits of divisor\n"); | ||||
|     } | ||||
| 
 | ||||
|     ot("\n"); | ||||
|     ot(";@ Divide r2 by r0\n"); | ||||
|     ot("  mov r3,#0\n"); | ||||
|     ot("  mov r1,r0\n"); | ||||
|     ot("\n"); | ||||
|     ot(";@ Shift up divisor till it's just less than numerator\n"); | ||||
|     ot("Shift%.4x%s\n",op,ms?"":":"); | ||||
|     ot("  cmp r1,r2,lsr #1\n"); | ||||
|     ot("  movls r1,r1,lsl #1\n"); | ||||
|     ot("  bcc Shift%.4x\n",op); | ||||
|     ot("\n"); | ||||
| 
 | ||||
|     ot("Divide%.4x%s\n",op,ms?"":":"); | ||||
|     ot("  cmp r2,r1\n"); | ||||
|     ot("  adc r3,r3,r3 ;@ Double r3 and add 1 if carry set\n"); | ||||
|     ot("  subcs r2,r2,r1\n"); | ||||
|     ot("  teq r1,r0\n"); | ||||
|     ot("  movne r1,r1,lsr #1\n"); | ||||
|     ot("  bne Divide%.4x\n",op); | ||||
|     ot("\n"); | ||||
|     ot(";@r3==quotient,r2==remainder\n"); | ||||
| 
 | ||||
|     if (sign) | ||||
|     { | ||||
|       // sign correction
 | ||||
|       ot("  and r1,r12,#1\n"); | ||||
|       ot("  teq r1,r12,lsr #1\n"); | ||||
|       ot("  rsbne r3,r3,#0 ;@ negate if quotient is negative\n"); | ||||
|       ot("  tst r12,#2\n"); | ||||
|       ot("  rsbne r2,r2,#0 ;@ negate the remainder if divident was negative\n"); | ||||
|       ot("\n"); | ||||
| 
 | ||||
|       // signed overflow check
 | ||||
|       ot("  mov r1,r3,asl #16\n"); | ||||
|       ot("  cmp r3,r1,asr #16 ;@ signed overflow?\n"); | ||||
|       ot("  orrne r10,r10,#0x10000000 ;@ set overflow flag\n"); | ||||
|       ot("  bne endofop%.4x ;@ overflow!\n",op); | ||||
|       ot("\n"); | ||||
|       ot("wrendofop%.4x%s\n",op,ms?"":":"); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|       // overflow check
 | ||||
|       ot("  movs r1,r3,lsr #16 ;@ check for overflow condition\n"); | ||||
|       ot("  orrne r10,r10,#0x10000000 ;@ set overflow flag\n"); | ||||
|       ot("  bne endofop%.4x ;@ overflow!\n",op); | ||||
|       ot("\n"); | ||||
|     } | ||||
| 
 | ||||
|     ot("  mov r1,r3,lsl #16 ;@ Clip to 16-bits\n"); | ||||
|     ot("  adds r1,r1,#0 ;@ Defines NZ, clears CV\n"); | ||||
|     OpGetFlags(0,0); | ||||
| 
 | ||||
|     ot("  mov r1,r1,lsr #16\n"); | ||||
|     ot("  orr r1,r1,r2,lsl #16 ;@ Insert remainder\n"); | ||||
|   } | ||||
| 
 | ||||
|   if (type==1) | ||||
|   { | ||||
|     ot(";@ Get 16-bit signs right:\n"); | ||||
|     ot("  mov r0,r1,%s #16\n",sign?"asr":"lsr"); | ||||
|     ot("  mov r2,r2,lsl #16\n"); | ||||
|     ot("  mov r2,r2,%s #16\n",sign?"asr":"lsr"); | ||||
|     ot("\n"); | ||||
| 
 | ||||
|     ot("  mul r1,r2,r0\n"); | ||||
|     ot("  adds r1,r1,#0 ;@ Defines NZ, clears CV\n"); | ||||
|     OpGetFlags(0,0); | ||||
|   } | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   EaWrite(11,    1,rea, 2,0x0e00,1); | ||||
| 
 | ||||
|   if (type==0) ot("endofop%.4x%s\n",op,ms?"":":"); | ||||
|   OpEnd(ea); | ||||
| 
 | ||||
|   if (type==0) // div
 | ||||
|   { | ||||
|     ot("divzero%.4x%s\n",op,ms?"":":"); | ||||
|     ot("  mov r0,#5 ;@ Divide by zero\n"); | ||||
|     ot("  bl Exception\n"); | ||||
|     Cycles+=38; | ||||
|     OpEnd(ea); | ||||
|     ot("\n"); | ||||
|   } | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // Get X Bit into carry - trashes r2
 | ||||
| int GetXBit(int subtract) | ||||
| { | ||||
|   ot(";@ Get X bit:\n"); | ||||
|   ot("  ldr r2,[r7,#0x4c]\n"); | ||||
|   if (subtract) ot("  mvn r2,r2 ;@ Invert it\n"); | ||||
|   ot("  msr cpsr_flg,r2 ;@ Get into Carry\n"); | ||||
|   ot("\n"); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x8100+ ---------------------
 | ||||
| // 1t00ddd1 0000asss - sbcd/abcd Ds,Dd or -(As),-(Ad)
 | ||||
| int OpAbcd(int op) | ||||
| { | ||||
|   int use=0; | ||||
|   int type=0,sea=0,mem=0,dea=0; | ||||
|    | ||||
|   type=(op>>14)&1; // sbcd/abcd
 | ||||
|   dea =(op>> 9)&7; | ||||
|   mem =(op>> 3)&1; | ||||
|   sea = op     &7; | ||||
| 
 | ||||
|   if (mem) { sea|=0x20; dea|=0x20; } | ||||
| 
 | ||||
|   use=op&~0x0e07; // Use same opcode for all registers..
 | ||||
|   if (sea==0x27) use|=0x0007; // ___x.b -(a7)
 | ||||
|   if (dea==0x27) use|=0x0e00; // ___x.b -(a7)
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,sea,dea); Cycles=6; | ||||
| 
 | ||||
|   if (mem) | ||||
|   { | ||||
|     ot(";@ Get src/dest EA vals\n"); | ||||
|     EaCalc (0,0x000f, sea,0,1); | ||||
|     EaRead (0,     6, sea,0,0x000f,1); | ||||
|     EaCalcReadNoSE(11,0,dea,0,0x0e00); | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     ot(";@ Get src/dest reg vals\n"); | ||||
|     EaCalcReadNoSE(-1,6,sea,0,0x0007); | ||||
|     EaCalcReadNoSE(11,0,dea,0,0x0e00); | ||||
|     ot("  mov r6,r6,asl #24\n"); | ||||
|   } | ||||
|   ot("  mov r1,r0,asl #24\n\n"); | ||||
| 
 | ||||
|   ot("  bic r10,r10,#0xb1000000 ;@ clear all flags except old Z\n"); | ||||
| 
 | ||||
|   if (type) | ||||
|   { | ||||
|     ot("  ldr r0,[r7,#0x4c] ;@ Get X bit\n"); | ||||
|     ot("  mov r3,#0x00f00000\n"); | ||||
|     ot("  and r2,r3,r1,lsr #4\n"); | ||||
|     ot("  tst r0,#0x20000000\n"); | ||||
|     ot("  and r0,r3,r6,lsr #4\n"); | ||||
|     ot("  add r0,r0,r2\n"); | ||||
|     ot("  addne r0,r0,#0x00100000\n"); | ||||
| //    ot("  tst r0,#0x00800000\n");
 | ||||
| //    ot("  orreq r10,r10,#0x01000000 ;@ Undefined V behavior\n");
 | ||||
|     ot("  cmp r0,#0x00900000\n"); | ||||
|     ot("  addhi r0,r0,#0x00600000 ;@ Decimal adjust units\n"); | ||||
| 
 | ||||
|     ot("  mov r2,r1,lsr #28\n"); | ||||
|     ot("  add r0,r0,r2,lsl #24\n"); | ||||
|     ot("  mov r2,r6,lsr #28\n"); | ||||
|     ot("  add r0,r0,r2,lsl #24\n"); | ||||
|     ot("  cmp r0,#0x09900000\n"); | ||||
|     ot("  orrhi r10,r10,#0x20000000 ;@ C\n"); | ||||
|     ot("  subhi r0,r0,#0x0a000000\n"); | ||||
| //    ot("  and r3,r10,r0,lsr #3 ;@ Undefined V behavior part II\n");
 | ||||
| //    ot("  orr r10,r10,r3,lsl #4 ;@ V\n");
 | ||||
|     ot("  movs r0,r0,lsl #4\n"); | ||||
|     ot("  orrmi r10,r10,#0x90000000 ;@ Undefined N+V behavior\n"); // this is what Musashi really does
 | ||||
|     ot("  bicne r10,r10,#0x40000000 ;@ Z flag\n"); | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     ot("  ldr r0,[r7,#0x4c] ;@ Get X bit\n"); | ||||
|     ot("  mov r3,#0x00f00000\n"); | ||||
|     ot("  and r2,r3,r6,lsr #4\n"); | ||||
|     ot("  tst r0,#0x20000000\n"); | ||||
|     ot("  and r0,r3,r1,lsr #4\n"); | ||||
|     ot("  sub r0,r0,r2\n"); | ||||
|     ot("  subne r0,r0,#0x00100000\n"); | ||||
| //    ot("  tst r0,#0x00800000\n");
 | ||||
| //    ot("  orreq r10,r10,#0x01000000 ;@ Undefined V behavior\n");
 | ||||
|     ot("  cmp r0,#0x00900000\n"); | ||||
|     ot("  subhi r0,r0,#0x00600000 ;@ Decimal adjust units\n"); | ||||
| 
 | ||||
|     ot("  mov r2,r1,lsr #28\n"); | ||||
|     ot("  add r0,r0,r2,lsl #24\n"); | ||||
|     ot("  mov r2,r6,lsr #28\n"); | ||||
|     ot("  sub r0,r0,r2,lsl #24\n"); | ||||
|     ot("  cmp r0,#0x09900000\n"); | ||||
|     ot("  orrhi r10,r10,#0xa0000000 ;@ N and C\n"); | ||||
|     ot("  addhi r0,r0,#0x0a000000\n"); | ||||
| //    ot("  and r3,r10,r0,lsr #3 ;@ Undefined V behavior part II\n");
 | ||||
| //    ot("  orr r10,r10,r3,lsl #4 ;@ V\n");
 | ||||
|     ot("  movs r0,r0,lsl #4\n"); | ||||
| //    ot("  orrmi r10,r10,#0x80000000 ;@ Undefined N behavior\n");
 | ||||
|     ot("  bicne r10,r10,#0x40000000 ;@ Z flag\n"); | ||||
|   } | ||||
| 
 | ||||
|   ot("  str r10,[r7,#0x4c] ;@ Save X bit\n"); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   EaWrite(11,     0, dea,0,0x0e00,1); | ||||
| 
 | ||||
|   ot("  ldr r6,[r7,#0x54]\n"); | ||||
|   OpEnd(sea,dea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // 01001000 00eeeeee - nbcd <ea>
 | ||||
| int OpNbcd(int op) | ||||
| { | ||||
|   int use=0; | ||||
|   int ea=0; | ||||
|    | ||||
|   ea=op&0x3f; | ||||
| 
 | ||||
|   if(EaCanWrite(ea)==0||EaAn(ea)) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,0); | ||||
|   if(op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,ea); Cycles=6; | ||||
|   if(ea >= 8)  Cycles+=2; | ||||
| 
 | ||||
|   EaCalcReadNoSE(6,0,ea,0,0x003f); | ||||
| 
 | ||||
|   // this is rewrite of Musashi's code
 | ||||
|   ot("  ldr r2,[r7,#0x4c]\n"); | ||||
|   ot("  bic r10,r10,#0xb0000000 ;@ clear all flags, except Z\n"); | ||||
|   ot("  mov r0,r0,asl #24\n"); | ||||
|   ot("  and r2,r2,#0x20000000\n"); | ||||
|   ot("  add r2,r0,r2,lsr #5 ;@ add X\n"); | ||||
|   ot("  rsb r11,r2,#0x9a000000 ;@ do arithmetic\n"); | ||||
| 
 | ||||
|   ot("  cmp r11,#0x9a000000\n"); | ||||
|   ot("  beq finish%.4x\n",op); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   ot("  mvn r3,r11,lsr #31 ;@ Undefined V behavior\n",op); | ||||
|   ot("  and r2,r11,#0x0f000000\n"); | ||||
|   ot("  cmp r2,#0x0a000000\n"); | ||||
|   ot("  andeq r11,r11,#0xf0000000\n"); | ||||
|   ot("  addeq r11,r11,#0x10000000\n"); | ||||
|   ot("  and r3,r3,r11,lsr #31 ;@ Undefined V behavior part II\n",op); | ||||
|   ot("  movs r1,r11,asr #24\n"); | ||||
|   ot("  bicne r10,r10,#0x40000000 ;@ Z\n"); | ||||
|   ot("  orr r10,r10,r3,lsl #28 ;@ save V\n",op); | ||||
|   ot("  orr r10,r10,#0x20000000 ;@ C\n"); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   EaWrite(6, 1, ea,0,0x3f,0,0); | ||||
| 
 | ||||
|   ot("finish%.4x%s\n",op,ms?"":":"); | ||||
|   ot("  tst r11,r11\n"); | ||||
|   ot("  orrmi r10,r10,#0x80000000 ;@ N\n"); | ||||
|   ot("  str r10,[r7,#0x4c] ;@ Save X\n"); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   ot("  ldr r6,[r7,#0x54]\n"); | ||||
|   OpEnd(ea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x90c0+ ---------------------
 | ||||
| // Suba/Cmpa/Adda 1tt1nnnx 11eeeeee (tt=type, x=size, eeeeee=Source EA)
 | ||||
| int OpAritha(int op) | ||||
| { | ||||
|   int use=0; | ||||
|   int type=0,size=0,sea=0,dea=0; | ||||
|   const char *asr=""; | ||||
| 
 | ||||
|   // Suba/Cmpa/Adda/(invalid):
 | ||||
|   type=(op>>13)&3; if (type>=3) return 1; | ||||
| 
 | ||||
|   size=(op>>8)&1; size++; | ||||
|   dea=(op>>9)&7; dea|=8; // Dest=An
 | ||||
|   sea=op&0x003f; // Source
 | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanRead(sea,size)==0) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   use&=~0x0e00; // Use same opcode for An
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,sea); Cycles=(size==2)?6:8; | ||||
|   if(size==2&&(sea<0x10||sea==0x3c)) Cycles+=2; | ||||
|   if(type==1) Cycles=6; | ||||
| 
 | ||||
|   // EA calculation order defines how situations like  suba.w (A0)+, A0 get handled.
 | ||||
|   // different emus act differently in this situation, I couldn't fugure which is right behaviour.
 | ||||
|   //if (type == 1)
 | ||||
|   { | ||||
|     EaCalcReadNoSE(-1,0,sea,size,0x003f); | ||||
|     EaCalcReadNoSE(type!=1?11:-1,1,dea,2,0x0e00); | ||||
|   } | ||||
| #if 0 | ||||
|   else | ||||
|   { | ||||
|     EaCalcReadNoSE(type!=1?11:-1,1,dea,2,0x0e00); | ||||
|     EaCalcReadNoSE(-1,0,sea,size,0x003f); | ||||
|   } | ||||
| #endif | ||||
| 
 | ||||
|   if (size<2) ot("  mov r0,r0,asl #%d\n\n",size?16:24); | ||||
|   if (size<2) asr=(char *)(size?",asr #16":",asr #24"); | ||||
| 
 | ||||
|   if (type==0) ot("  sub r1,r1,r0%s\n",asr); | ||||
|   if (type==1) ot("  cmp r1,r0%s ;@ Defines NZCV\n",asr); | ||||
|   if (type==1) OpGetFlags(1,0); // Get Cmp flags
 | ||||
|   if (type==2) ot("  add r1,r1,r0%s\n",asr); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   if (type!=1) EaWrite(11, 1, dea,2,0x0e00); | ||||
| 
 | ||||
|   OpEnd(sea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x9100+ ---------------------
 | ||||
| // Emit a Subx/Addx opcode, 1t01ddd1 zz00rsss addx.z Ds,Dd
 | ||||
| int OpAddx(int op) | ||||
| { | ||||
|   int use=0; | ||||
|   int type=0,size=0,dea=0,sea=0,mem=0; | ||||
|   const char *asl=""; | ||||
| 
 | ||||
|   type=(op>>14)&1; | ||||
|   dea =(op>> 9)&7; | ||||
|   size=(op>> 6)&3; if (size>=3) return 1; | ||||
|   sea = op&7; | ||||
|   mem =(op>> 3)&1; | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanRead(sea,size)==0) return 1; | ||||
|   if (EaCanWrite(dea)==0) return 1; | ||||
| 
 | ||||
|   if (mem) { sea+=0x20; dea+=0x20; } | ||||
| 
 | ||||
|   use=op&~0x0e07; // Use same opcode for Dn
 | ||||
|   if (size==0&&sea==0x27) use|=0x0007; // ___x.b -(a7)
 | ||||
|   if (size==0&&dea==0x27) use|=0x0e00; // ___x.b -(a7)
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,sea,dea); Cycles=4; | ||||
|   if(size>=2)   Cycles+=4; | ||||
|   if(sea>=0x10) Cycles+=2; | ||||
| 
 | ||||
|   if (mem) | ||||
|   { | ||||
|     ot(";@ Get src/dest EA vals\n"); | ||||
|     EaCalc (0,0x000f, sea,size,1); | ||||
|     EaRead (0,     6, sea,size,0x000f,1); | ||||
|     EaCalcReadNoSE(11,0,dea,size,0x0e00); | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     ot(";@ Get src/dest reg vals\n"); | ||||
|     EaCalcReadNoSE(-1,6,sea,size,0x0007); | ||||
|     EaCalcReadNoSE(11,0,dea,size,0x0e00); | ||||
|     if (size<2) ot("  mov r6,r6,asl #%d\n\n",size?16:24); | ||||
|   } | ||||
| 
 | ||||
|   if (size<2) asl=(char *)(size?",asl #16":",asl #24"); | ||||
| 
 | ||||
|   ot(";@ Do arithmetic:\n"); | ||||
|   GetXBit(type==0); | ||||
| 
 | ||||
|   if (type==1 && size<2) | ||||
|   { | ||||
|     ot(";@ Make sure the carry bit will tip the balance:\n"); | ||||
|     ot("  mvn r2,#0\n"); | ||||
|     ot("  orr r6,r6,r2,lsr #%i\n",(size==0)?8:16); | ||||
|     ot("\n"); | ||||
|   } | ||||
| 
 | ||||
|   if (type==0) ot("  rscs r1,r6,r0%s\n",asl); | ||||
|   if (type==1) ot("  adcs r1,r6,r0%s\n",asl); | ||||
|   ot("  orr r3,r10,#0xb0000000 ;@ for old Z\n"); | ||||
|   OpGetFlags(type==0,1,0); // subtract
 | ||||
|   if (size<2) { | ||||
|     ot("  movs r2,r1,lsr #%i\n", size?16:24); | ||||
|     ot("  orreq r10,r10,#0x40000000 ;@ add potentially missed Z\n"); | ||||
|   } | ||||
|   ot("  andeq r10,r10,r3 ;@ fix Z\n"); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   ot(";@ Save result:\n"); | ||||
|   EaWrite(11, 1, dea,size,0x0e00,1); | ||||
| 
 | ||||
|   ot("  ldr r6,[r7,#0x54]\n"); | ||||
|   OpEnd(sea,dea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0xb000+ ---------------------
 | ||||
| // Emit a Cmp/Eor opcode, 1011rrrt xxeeeeee (rrr=Dn, t=cmp/eor, xx=size extension, eeeeee=ea)
 | ||||
| int OpCmpEor(int op) | ||||
| { | ||||
|   int rea=0,eor=0; | ||||
|   int size=0,ea=0,use=0; | ||||
|   const char *asl=""; | ||||
| 
 | ||||
|   // Get EA and register EA
 | ||||
|   rea=(op>>9)&7; | ||||
|   eor=(op>>8)&1; | ||||
|   size=(op>>6)&3; if (size>=3) return 1; | ||||
|   ea=op&0x3f; | ||||
| 
 | ||||
|   if (eor && (ea>>3) == 1) return 1; // not a valid mode for eor
 | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanRead(ea,size)==0) return 1; | ||||
|   if (eor && EaCanWrite(ea)==0) return 1; | ||||
|   if (EaAn(ea)&&(eor||size==0)) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   use&=~0x0e00; // Use 1 handler for register d0-7
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,ea); Cycles=4; | ||||
|   if(eor) { | ||||
|     if(ea>8)     Cycles+=4; | ||||
|     if(size>=2)  Cycles+=4; | ||||
|   } else { | ||||
|     if(size>=2)  Cycles+=2; | ||||
|   } | ||||
| 
 | ||||
|   ot(";@ Get EA into r11 and value into r0:\n"); | ||||
|   EaCalcReadNoSE(eor?11:-1,0,ea,size,0x003f); | ||||
| 
 | ||||
|   ot(";@ Get register operand into r1:\n"); | ||||
|   EaCalcReadNoSE(-1,1,rea,size,0x0e00); | ||||
| 
 | ||||
|   if (size<2) ot("  mov r0,r0,asl #%d\n\n",size?16:24); | ||||
|   if (size<2) asl=(char *)(size?",asl #16":",asl #24"); | ||||
| 
 | ||||
|   ot(";@ Do arithmetic:\n"); | ||||
|   if (eor==0) ot("  rsbs r1,r0,r1%s\n",asl); | ||||
|   if (eor) | ||||
|   { | ||||
|     ot("  eor r1,r0,r1%s\n",asl); | ||||
|     ot("  adds r1,r1,#0 ;@ Defines NZ, clears CV\n"); | ||||
|   } | ||||
| 
 | ||||
|   OpGetFlags(eor==0,0); // Cmp like subtract
 | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   if (eor) EaWrite(11, 1,ea,size,0x003f,1); | ||||
| 
 | ||||
|   OpEnd(ea); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // Emit a Cmpm opcode, 1011ddd1 xx001sss (rrr=Adst, xx=size extension, sss=Asrc)
 | ||||
| int OpCmpm(int op) | ||||
| { | ||||
|   int size=0,sea=0,dea=0,use=0; | ||||
|   const char *asl=""; | ||||
| 
 | ||||
|   // get size, get EAs
 | ||||
|   size=(op>>6)&3; if (size>=3) return 1; | ||||
|   sea=(op&7)|0x18; | ||||
|   dea=(op>>9)&0x3f; | ||||
| 
 | ||||
|   use=op&~0x0e07; // Use 1 handler for all registers..
 | ||||
|   if (size==0&&sea==0x1f) use|=0x0007; // ..except (a7)+
 | ||||
|   if (size==0&&dea==0x1f) use|=0x0e00; | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,sea); Cycles=4; | ||||
| 
 | ||||
|   ot(";@ Get src operand into r11:\n"); | ||||
|   EaCalc (0,0x0007, sea,size,1); | ||||
|   EaRead (0,    11, sea,size,0x0007,1); | ||||
| 
 | ||||
|   ot(";@ Get dst operand into r0:\n"); | ||||
|   EaCalcReadNoSE(-1,0,dea,size,0x0e00); | ||||
| 
 | ||||
|   if (size<2) asl=(char *)(size?",asl #16":",asl #24"); | ||||
| 
 | ||||
|   ot("  rsbs r0,r11,r0%s\n",asl); | ||||
|   OpGetFlags(1,0); // Cmp like subtract
 | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   OpEnd(sea); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // Emit a Chk opcode, 0100ddd1 x0eeeeee (rrr=Dn, x=size extension, eeeeee=ea)
 | ||||
| int OpChk(int op) | ||||
| { | ||||
|   int rea=0; | ||||
|   int size=0,ea=0,use=0; | ||||
| 
 | ||||
|   // Get EA and register EA
 | ||||
|   rea=(op>>9)&7; | ||||
|   if((op>>7)&1) | ||||
|        size=1; // word operation
 | ||||
|   else size=2; // long
 | ||||
|   ea=op&0x3f; | ||||
| 
 | ||||
|   if (EaAn(ea)) return 1; // not a valid mode
 | ||||
|   if (size!=1)  return 1; // 000 variant only supports word
 | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanRead(ea,size)==0) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   use&=~0x0e00; // Use 1 handler for register d0-7
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,ea); Cycles=10; | ||||
| 
 | ||||
|   ot(";@ Get value into r0:\n"); | ||||
|   EaCalcReadNoSE(-1,0,ea,size,0x003f); | ||||
| 
 | ||||
|   ot(";@ Get register operand into r1:\n"); | ||||
|   EaCalcReadNoSE(-1,1,rea,size,0x0e00); | ||||
| 
 | ||||
|   if (size<2) ot("  mov r0,r0,asl #%d\n",size?16:24); | ||||
|   if (size<2) ot("  mov r1,r1,asl #%d\n\n",size?16:24); | ||||
| 
 | ||||
|   ot(";@ get flags, including undocumented ones\n"); | ||||
|   ot("  and r3,r10,#0x80000000\n"); | ||||
|   ot("  adds r1,r1,#0 ;@ Defines NZ, clears CV\n"); | ||||
|   OpGetFlags(0,0); | ||||
| 
 | ||||
|   ot(";@ is reg negative?\n"); | ||||
|   ot("  bmi chktrap%.4x\n",op); | ||||
| 
 | ||||
|   ot(";@ Do arithmetic:\n"); | ||||
|   ot("  bic r10,r10,#0x80000000 ;@ N\n"); | ||||
|   ot("  cmp r1,r0\n"); | ||||
|   ot("  bgt chktrap%.4x\n",op); | ||||
| 
 | ||||
|   ot(";@ old N remains\n"); | ||||
|   ot("  orr r10,r10,r3\n"); | ||||
|   OpEnd(ea); | ||||
| 
 | ||||
|   ot("chktrap%.4x%s ;@ CHK exception:\n",op,ms?"":":"); | ||||
|   ot("  mov r0,#6\n"); | ||||
|   ot("  bl Exception\n"); | ||||
|   Cycles+=40; | ||||
|   OpEnd(ea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
|  | @ -1,517 +0,0 @@ | |||
| 
 | ||||
| #include "app.h" | ||||
| 
 | ||||
| // in/out address in r0, trashes all temp regs
 | ||||
| static void CheckPc(void) | ||||
| { | ||||
| #if USE_CHECKPC_CALLBACK | ||||
|  #ifdef MEMHANDLERS_DIRECT_PREFIX | ||||
|   ot("  bl %scheckpc ;@ Call checkpc()\n", MEMHANDLERS_DIRECT_PREFIX); | ||||
|  #else | ||||
|   ot(";@ Check Memory Base+pc\n"); | ||||
|   ot("  mov lr,pc\n"); | ||||
|   ot("  ldr pc,[r7,#0x64] ;@ Call checkpc()\n"); | ||||
|   ot("\n"); | ||||
|  #endif | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| // Push 32-bit value in r1 - trashes r0-r3,r12,lr
 | ||||
| void OpPush32() | ||||
| { | ||||
|   ot(";@ Push r1 onto stack\n"); | ||||
|   ot("  ldr r0,[r7,#0x3c]\n"); | ||||
|   ot("  sub r0,r0,#4 ;@ Predecrement A7\n"); | ||||
|   ot("  str r0,[r7,#0x3c] ;@ Save A7\n"); | ||||
|   MemHandler(1,2); | ||||
|   ot("\n"); | ||||
| } | ||||
| 
 | ||||
| // Push SR - trashes r0-r3,r12,lr
 | ||||
| void OpPushSr(int high) | ||||
| { | ||||
|   ot(";@ Push SR:\n"); | ||||
|   OpFlagsToReg(high); | ||||
|   ot("  ldr r0,[r7,#0x3c]\n"); | ||||
|   ot("  sub r0,r0,#2 ;@ Predecrement A7\n"); | ||||
|   ot("  str r0,[r7,#0x3c] ;@ Save A7\n"); | ||||
|   MemHandler(1,1); | ||||
|   ot("\n"); | ||||
| } | ||||
| 
 | ||||
| // Pop SR - trashes r0-r3
 | ||||
| static void PopSr(int high) | ||||
| { | ||||
|   ot(";@ Pop SR:\n"); | ||||
|   ot("  ldr r0,[r7,#0x3c]\n"); | ||||
|   ot("  add r1,r0,#2 ;@ Postincrement A7\n"); | ||||
|   ot("  str r1,[r7,#0x3c] ;@ Save A7\n"); | ||||
|   MemHandler(0,1); | ||||
|   ot("\n"); | ||||
|   OpRegToFlags(high); | ||||
| } | ||||
| 
 | ||||
| // Pop PC - trashes r0-r3
 | ||||
| static void PopPc() | ||||
| { | ||||
|   ot(";@ Pop PC:\n"); | ||||
|   ot("  ldr r0,[r7,#0x3c]\n"); | ||||
|   ot("  add r1,r0,#4 ;@ Postincrement A7\n"); | ||||
|   ot("  str r1,[r7,#0x3c] ;@ Save A7\n"); | ||||
|   MemHandler(0,2); | ||||
|   ot("  ldr r1,[r7,#0x60] ;@ Get Memory base\n"); | ||||
|   ot("  add r0,r0,r1 ;@ Memory Base+PC\n"); | ||||
|   ot("\n"); | ||||
|   CheckPc(); | ||||
| #if EMULATE_ADDRESS_ERRORS_JUMP | ||||
|   ot("  mov r4,r0\n"); | ||||
| #else | ||||
|   ot("  bic r4,r0,#1\n"); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| int OpTrap(int op) | ||||
| { | ||||
|   int use=0; | ||||
| 
 | ||||
|   use=op&~0xf; | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,0x10); | ||||
|   ot("  and r0,r8,#0xf ;@ Get trap number\n"); | ||||
|   ot("  orr r0,r0,#0x20 ;@ 32+n\n"); | ||||
|   ot("  bl Exception\n"); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   Cycles=38; OpEnd(0x10); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x4e50+ ---------------------
 | ||||
| int OpLink(int op) | ||||
| { | ||||
|   int use=0,reg; | ||||
| 
 | ||||
|   use=op&~7; | ||||
|   reg=op&7; | ||||
|   if (reg==7) use=op; | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,0x10); | ||||
| 
 | ||||
|   if(reg!=7) { | ||||
|     ot(";@ Get An\n"); | ||||
|     EaCalc(11, 7, 8, 2, 1); | ||||
|     EaRead(11, 1, 8, 2, 7, 1); | ||||
|   } | ||||
| 
 | ||||
|   ot("  ldr r0,[r7,#0x3c] ;@ Get A7\n"); | ||||
|   ot("  sub r0,r0,#4 ;@ A7-=4\n"); | ||||
|   ot("  mov r8,r0 ;@ abuse r8\n"); | ||||
|   if(reg==7) ot("  mov r1,r0\n"); | ||||
|   ot("\n"); | ||||
|    | ||||
|   ot(";@ Write An to Stack\n"); | ||||
|   MemHandler(1,2); | ||||
| 
 | ||||
|   ot(";@ Save to An\n"); | ||||
|   if(reg!=7) | ||||
|     EaWrite(11,8, 8, 2, 7, 1); | ||||
| 
 | ||||
|   ot(";@ Get offset:\n"); | ||||
|   EaCalc(0,0,0x3c,1);    // abused r8 is ok because of imm EA
 | ||||
|   EaRead(0,0,0x3c,1,0); | ||||
| 
 | ||||
|   ot("  add r8,r8,r0 ;@ Add offset to A7\n"); | ||||
|   ot("  str r8,[r7,#0x3c]\n"); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   Cycles=16; | ||||
|   OpEnd(0x10); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x4e58+ ---------------------
 | ||||
| int OpUnlk(int op) | ||||
| { | ||||
|   int use=0; | ||||
| 
 | ||||
|   use=op&~7; | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,0x10); | ||||
| 
 | ||||
|   ot(";@ Get An\n"); | ||||
|   EaCalc(11, 0xf, 8, 2,   1); | ||||
|   EaRead(11,   0, 8, 2, 0xf, 1); | ||||
| 
 | ||||
|   ot("  add r8,r0,#4 ;@ A7+=4, abuse r8\n"); | ||||
|   ot("\n"); | ||||
|   ot(";@ Pop An from stack:\n"); | ||||
|   MemHandler(0,2); | ||||
|   ot("\n"); | ||||
|   ot("  str r8,[r7,#0x3c] ;@ Save A7\n"); | ||||
|   ot("\n"); | ||||
|   ot(";@ An = value from stack:\n"); | ||||
|   EaWrite(11, 0, 8, 2, 7, 1); | ||||
| 
 | ||||
|   Cycles=12; | ||||
|   OpEnd(0x10); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x4e70+ ---------------------
 | ||||
| // 01001110 01110ttt
 | ||||
| int Op4E70(int op) | ||||
| { | ||||
|   int type=0; | ||||
| 
 | ||||
|   type=op&7; // reset/nop/stop/rte/rtd/rts/trapv/rtr
 | ||||
| 
 | ||||
|   switch (type) | ||||
|   { | ||||
|     case 1:  // nop
 | ||||
|     OpStart(op); | ||||
|     Cycles=4; | ||||
|     OpEnd(); | ||||
|     return 0; | ||||
| 
 | ||||
|     case 3: // rte
 | ||||
|     OpStart(op,0x10,0,0,1); Cycles=20; | ||||
|     PopSr(1); | ||||
|     PopPc(); | ||||
|     ot("  ldr r1,[r7,#0x44] ;@ reload SR high\n"); | ||||
|     SuperChange(op,1); | ||||
| #if EMULATE_ADDRESS_ERRORS_JUMP || EMULATE_ADDRESS_ERRORS_IO || EMULATE_HALT | ||||
|     ot("  ldr r1,[r7,#0x58]\n"); | ||||
|     ot("  bic r1,r1,#0x0c ;@ clear 'not processing instruction' and 'doing addr error' bits\n"); | ||||
|     ot("  str r1,[r7,#0x58]\n"); | ||||
| #endif | ||||
| #if EMULATE_ADDRESS_ERRORS_JUMP | ||||
|     ot("  tst r4,#1 ;@ address error?\n"); | ||||
|     ot("  bne ExceptionAddressError_r_prg_r4\n"); | ||||
| #endif | ||||
|     opend_check_interrupt = 1; | ||||
|     opend_check_trace = 1; | ||||
|     OpEnd(0x10,0); | ||||
|     return 0; | ||||
| 
 | ||||
|     case 5: // rts
 | ||||
|     OpStart(op,0x10); Cycles=16; | ||||
|     PopPc(); | ||||
| #if EMULATE_ADDRESS_ERRORS_JUMP | ||||
|     ot("  tst r4,#1 ;@ address error?\n"); | ||||
|     ot("  bne ExceptionAddressError_r_prg_r4\n"); | ||||
| #endif | ||||
|     OpEnd(0x10); | ||||
|     return 0; | ||||
| 
 | ||||
|     case 6: // trapv
 | ||||
|     OpStart(op,0x10,0,1); Cycles=4; | ||||
|     ot("  tst r10,#0x10000000\n"); | ||||
|     ot("  subne r5,r5,#%i\n",34); | ||||
|     ot("  movne r0,#7 ;@ TRAPV exception\n"); | ||||
|     ot("  blne Exception\n"); | ||||
|     opend_op_changes_cycles = 1; | ||||
|     OpEnd(0x10,0); | ||||
|     return 0; | ||||
| 
 | ||||
|     case 7: // rtr
 | ||||
|     OpStart(op,0x10); Cycles=20; | ||||
|     PopSr(0); | ||||
|     PopPc(); | ||||
| #if EMULATE_ADDRESS_ERRORS_JUMP | ||||
|     ot("  tst r4,#1 ;@ address error?\n"); | ||||
|     ot("  bne ExceptionAddressError_r_prg_r4\n"); | ||||
| #endif | ||||
|     OpEnd(0x10); | ||||
|     return 0; | ||||
| 
 | ||||
|     default: | ||||
|     return 1; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x4e80+ ---------------------
 | ||||
| // Emit a Jsr/Jmp opcode, 01001110 1meeeeee
 | ||||
| int OpJsr(int op) | ||||
| { | ||||
|   int use=0; | ||||
|   int sea=0; | ||||
| 
 | ||||
|   sea=op&0x003f; | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanRead(sea,-1)==0) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,0); | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,(op&0x40)?0:0x10); | ||||
| 
 | ||||
|   ot("  ldr r11,[r7,#0x60] ;@ Get Memory base\n"); | ||||
|   ot("\n"); | ||||
|   EaCalc(12,0x003f,sea,0); | ||||
| 
 | ||||
|   ot(";@ Jump - Get new PC from r12\n"); | ||||
|   ot("  add r0,r12,r11 ;@ Memory Base + New PC\n"); | ||||
|   ot("\n"); | ||||
|   CheckPc(); | ||||
|   if (!(op&0x40)) | ||||
|   { | ||||
|     ot("  ldr r2,[r7,#0x3c]\n"); | ||||
|     ot("  sub r1,r4,r11 ;@ r1 = Old PC\n"); | ||||
|   } | ||||
| #if EMULATE_ADDRESS_ERRORS_JUMP | ||||
|   // jsr prefetches next instruction before pushing old PC,
 | ||||
|   // according to http://pasti.fxatari.com/68kdocs/68kPrefetch.html
 | ||||
|   ot("  mov r4,r0\n"); | ||||
|   ot("  tst r4,#1 ;@ address error?\n"); | ||||
|   ot("  bne ExceptionAddressError_r_prg_r4\n"); | ||||
| #else | ||||
|   ot("  bic r4,r0,#1\n"); | ||||
| #endif | ||||
| 
 | ||||
|   if (!(op&0x40)) | ||||
|   { | ||||
|     ot(";@ Push old PC onto stack\n"); | ||||
|     ot("  sub r0,r2,#4 ;@ Predecrement A7\n"); | ||||
|     ot("  str r0,[r7,#0x3c] ;@ Save A7\n"); | ||||
|     MemHandler(1,2); | ||||
|   } | ||||
| 
 | ||||
|   Cycles=(op&0x40) ? 4 : 12; | ||||
|   Cycles+=Ea_add_ns((op&0x40) ? g_jmp_cycle_table : g_jsr_cycle_table, sea); | ||||
| 
 | ||||
|   OpEnd((op&0x40)?0:0x10); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x50c8+ ---------------------
 | ||||
| 
 | ||||
| // ARM version of 68000 condition codes:
 | ||||
| static const char * const Cond[16]= | ||||
| { | ||||
|   "",  "",  "hi","ls","cc","cs","ne","eq", | ||||
|   "vc","vs","pl","mi","ge","lt","gt","le" | ||||
| }; | ||||
| 
 | ||||
| // Emit a Dbra opcode, 0101cccc 11001nnn vv
 | ||||
| int OpDbra(int op) | ||||
| { | ||||
|   int use=0; | ||||
|   int cc=0; | ||||
| 
 | ||||
|   use=op&~7; // Use same handler
 | ||||
|   cc=(op>>8)&15; | ||||
|    | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
|   OpStart(op); | ||||
| 
 | ||||
|   switch (cc) | ||||
|   { | ||||
|     case 0: // T
 | ||||
|     case 1: // F
 | ||||
|       break; | ||||
|     case 2: // hi
 | ||||
|       ot("  tst r10,#0x60000000 ;@ hi: !C && !Z\n"); | ||||
|       ot("  beq DbraTrue\n\n"); | ||||
|       break; | ||||
|     case 3: // ls
 | ||||
|       ot("  tst r10,#0x60000000 ;@ ls: C || Z\n"); | ||||
|       ot("  bne DbraTrue\n\n"); | ||||
|       break; | ||||
|     default: | ||||
|       ot(";@ Is the condition true?\n"); | ||||
|       ot("  msr cpsr_flg,r10 ;@ ARM flags = 68000 flags\n"); | ||||
|       ot(";@ If so, don't dbra\n"); | ||||
|       ot("  b%s DbraTrue\n\n",Cond[cc]); | ||||
|       break; | ||||
|   } | ||||
| 
 | ||||
|   if (cc!=0) | ||||
|   { | ||||
|     ot(";@ Decrement Dn.w\n"); | ||||
|     ot("  and r1,r8,#0x0007\n"); | ||||
|     ot("  mov r1,r1,lsl #2\n"); | ||||
|     ot("  ldrsh r0,[r7,r1]\n"); | ||||
|     ot("  sub r0,r0,#1\n"); | ||||
|     ot("  strh r0,[r7,r1]\n"); | ||||
|     ot("\n"); | ||||
| 
 | ||||
|     ot(";@ Check if Dn.w is -1\n"); | ||||
|     ot("  cmn r0,#1\n"); | ||||
| 
 | ||||
| #if (USE_CHECKPC_CALLBACK && USE_CHECKPC_DBRA) || EMULATE_ADDRESS_ERRORS_JUMP | ||||
|     ot("  beq DbraMin1\n"); | ||||
|     ot("\n"); | ||||
| 
 | ||||
|     ot(";@ Get Branch offset:\n"); | ||||
|     ot("  ldrsh r0,[r4]\n"); | ||||
|     ot("  add r0,r4,r0 ;@ r0 = New PC\n"); | ||||
|     CheckPc(); | ||||
| #if EMULATE_ADDRESS_ERRORS_JUMP | ||||
|     ot("  mov r4,r0\n"); | ||||
|     ot("  tst r4,#1 ;@ address error?\n"); | ||||
|     ot("  bne ExceptionAddressError_r_prg_r4\n"); | ||||
| #else | ||||
|     ot("  bic r4,r0,#1\n"); | ||||
| #endif | ||||
| #else | ||||
|     ot("\n"); | ||||
|     ot(";@ Get Branch offset:\n"); | ||||
|     ot("  ldrnesh r0,[r4]\n"); | ||||
|     ot("  addeq r4,r4,#2 ;@ Skip branch offset\n"); | ||||
|     ot("  subeq r5,r5,#4 ;@ additional cycles\n"); | ||||
|     ot("  addne r4,r4,r0 ;@ r4 = New PC\n"); | ||||
|     ot("  bic r4,r4,#1\n"); // we do not emulate address errors
 | ||||
|     ot("\n"); | ||||
| #endif | ||||
|     Cycles=12-2; | ||||
|     OpEnd(); | ||||
|   } | ||||
|    | ||||
|   //if (cc==0||cc>=2)
 | ||||
|   if (op==0x50c8) | ||||
|   { | ||||
|     ot(";@ condition true:\n"); | ||||
|     ot("DbraTrue%s\n", ms?"":":"); | ||||
|     ot("  add r4,r4,#2 ;@ Skip branch offset\n"); | ||||
|     ot("\n"); | ||||
|     Cycles=12; | ||||
|     OpEnd(); | ||||
|   } | ||||
| 
 | ||||
| #if (USE_CHECKPC_CALLBACK && USE_CHECKPC_DBRA) || EMULATE_ADDRESS_ERRORS_JUMP | ||||
|   if (op==0x51c8) | ||||
|   { | ||||
|     ot(";@ Dn.w is -1:\n"); | ||||
|     ot("DbraMin1%s\n", ms?"":":"); | ||||
|     ot("  add r4,r4,#2 ;@ Skip branch offset\n"); | ||||
|     ot("\n"); | ||||
|     Cycles=12+2; | ||||
|     OpEnd(); | ||||
|   } | ||||
| #endif | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x6000+ ---------------------
 | ||||
| // Emit a Branch opcode 0110cccc nn  (cccc=condition)
 | ||||
| int OpBranch(int op) | ||||
| { | ||||
|   int size=0,use=0,checkpc=0; | ||||
|   int offset=0; | ||||
|   int cc=0; | ||||
|   const char *asr_r11=""; | ||||
| 
 | ||||
|   offset=(char)(op&0xff); | ||||
|   cc=(op>>8)&15; | ||||
| 
 | ||||
|   // Special offsets:
 | ||||
|   if (offset==0)  size=1; | ||||
|   if (offset==-1) size=2; | ||||
| 
 | ||||
|   if (size==2) size=0; // 000 model does not support long displacement
 | ||||
|   if (size) use=op; // 16-bit or 32-bit
 | ||||
|   else use=(op&0xff00)+1; // Use same opcode for all 8-bit branches
 | ||||
| 
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
|   OpStart(op,size?0x10:0); | ||||
|   Cycles=10; // Assume branch taken
 | ||||
| 
 | ||||
|   switch (cc) | ||||
|   { | ||||
|     case 0: // T
 | ||||
|     case 1: // F
 | ||||
|       break; | ||||
|     case 2: // hi
 | ||||
|       ot("  tst r10,#0x60000000 ;@ hi: !C && !Z\n"); | ||||
|       ot("  bne BccDontBranch%i\n\n",8<<size); | ||||
|       break; | ||||
|     case 3: // ls
 | ||||
|       ot("  tst r10,#0x60000000 ;@ ls: C || Z\n"); | ||||
|       ot("  beq BccDontBranch%i\n\n",8<<size); | ||||
|       break; | ||||
|     default: | ||||
|       ot(";@ Is the condition true?\n"); | ||||
|       ot("  msr cpsr_flg,r10 ;@ ARM flags = 68000 flags\n"); | ||||
|       ot("  b%s BccDontBranch%i\n\n",Cond[cc^1],8<<size); | ||||
|       break; | ||||
|   } | ||||
| 
 | ||||
|   if (size)  | ||||
|   { | ||||
|     if (size<2) | ||||
|     { | ||||
|       ot("  ldrsh r11,[r4] ;@ Fetch Branch offset\n"); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|       ot("  ldrh r2,[r4] ;@ Fetch Branch offset\n"); | ||||
|       ot("  ldrh r11,[r4,#2]\n"); | ||||
|       ot("  orr r11,r11,r2,lsl #16\n"); | ||||
|     } | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     ot("  mov r11,r8,asl #24 ;@ Shift 8-bit signed offset up...\n\n"); | ||||
|     asr_r11=",asr #24"; | ||||
|   } | ||||
| 
 | ||||
|   ot(";@ Branch taken - Add on r0 to PC\n"); | ||||
| 
 | ||||
|   if (cc==1) | ||||
|   { | ||||
|     ot(";@ Bsr - remember old PC\n"); | ||||
|     ot("  ldr r12,[r7,#0x60] ;@ Get Memory base\n"); | ||||
|     ot("  ldr r2,[r7,#0x3c]\n"); | ||||
|     ot("  sub r1,r4,r12 ;@ r1 = Old PC\n"); | ||||
|     if (size) ot("  add r1,r1,#%d\n",1<<size); | ||||
|     ot("\n"); | ||||
|     ot(";@ Push r1 onto stack\n"); | ||||
|     ot("  sub r0,r2,#4 ;@ Predecrement A7\n"); | ||||
|     ot("  str r0,[r7,#0x3c] ;@ Save A7\n"); | ||||
|     MemHandler(1,2); | ||||
|     ot("\n"); | ||||
|     Cycles=18; // always 18
 | ||||
|   } | ||||
| 
 | ||||
|   ot("  add r0,r4,r11%s ;@ r4 = New PC\n",asr_r11); | ||||
| 
 | ||||
| #if USE_CHECKPC_CALLBACK && USE_CHECKPC_OFFSETBITS_8 | ||||
|   if (offset!=0 && offset!=-1) checkpc=1; | ||||
| #endif | ||||
| #if USE_CHECKPC_CALLBACK && USE_CHECKPC_OFFSETBITS_16 | ||||
|   if (offset==0)  checkpc=1; | ||||
| #endif | ||||
| #if USE_CHECKPC_CALLBACK | ||||
|   if (offset==-1) checkpc=1; | ||||
| #endif | ||||
|   if (checkpc) CheckPc(); | ||||
| #if EMULATE_ADDRESS_ERRORS_JUMP | ||||
|   ot("  mov r4,r0\n"); | ||||
|   ot("  tst r4,#1 ;@ address error?\n"); | ||||
|   ot("  bne ExceptionAddressError_r_prg_r4\n"); | ||||
| #else | ||||
|   ot("  bic r4,r0,#1\n"); | ||||
| #endif | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   OpEnd(size?0x10:0); | ||||
| 
 | ||||
|   // since all "DontBranch" code is same for every size, output only once
 | ||||
|   if (cc>=2&&(op&0xff00)==0x6700) | ||||
|   { | ||||
|     ot("BccDontBranch%i%s\n", 8<<size, ms?"":":"); | ||||
|     if (size) ot("  add r4,r4,#%d\n",1<<size); | ||||
|     Cycles+=(size==1) ? 2 : -2; // Branch not taken
 | ||||
|     OpEnd(0); | ||||
|   } | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
|  | @ -1,702 +0,0 @@ | |||
| #include "app.h" | ||||
| 
 | ||||
| // --------------------- Opcodes 0x0100+ ---------------------
 | ||||
| // Emit a Btst (Register) opcode 0000nnn1 ttaaaaaa
 | ||||
| int OpBtstReg(int op) | ||||
| { | ||||
|   int use=0; | ||||
|   int type=0,sea=0,tea=0; | ||||
|   int size=0; | ||||
| 
 | ||||
|   type=(op>>6)&3; // Btst/Bchg/Bclr/Bset
 | ||||
|   // Get source and target EA
 | ||||
|   sea=(op>>9)&7; | ||||
|   tea=op&0x003f; | ||||
|   if (tea<0x10) size=2; // For registers, 32-bits
 | ||||
| 
 | ||||
|   if ((tea&0x38)==0x08) return 1; // movep
 | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanRead(tea,0)==0) return 1; | ||||
|   if (type>0) | ||||
|   { | ||||
|     if (EaCanWrite(tea)==0) return 1; | ||||
|   } | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   use&=~0x0e00; // Use same handler for all registers
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,tea); | ||||
| 
 | ||||
|   if(type==1||type==3) { | ||||
|     Cycles=8; | ||||
|   } else { | ||||
|     Cycles=type?8:4; | ||||
|     if(size>=2) Cycles+=2; | ||||
|   } | ||||
| 
 | ||||
|   EaCalcReadNoSE(-1,11,sea,0,0x0e00); | ||||
| 
 | ||||
|   EaCalcReadNoSE((type>0)?8:-1,0,tea,size,0x003f); | ||||
| 
 | ||||
|   if (tea>=0x10) | ||||
|        ot("  and r11,r11,#7  ;@ mem - do mod 8\n");  // size always 0
 | ||||
|   else ot("  and r11,r11,#31 ;@ reg - do mod 32\n"); // size always 2
 | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   ot("  mov r1,#1\n"); | ||||
|   ot("  tst r0,r1,lsl r11 ;@ Do arithmetic\n"); | ||||
|   ot("  bicne r10,r10,#0x40000000\n"); | ||||
|   ot("  orreq r10,r10,#0x40000000 ;@ Get Z flag\n"); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   if (type>0) | ||||
|   { | ||||
|     if (type==1) ot("  eor r1,r0,r1,lsl r11 ;@ Toggle bit\n"); | ||||
|     if (type==2) ot("  bic r1,r0,r1,lsl r11 ;@ Clear bit\n"); | ||||
|     if (type==3) ot("  orr r1,r0,r1,lsl r11 ;@ Set bit\n"); | ||||
|     ot("\n"); | ||||
|     EaWrite(8,1,tea,size,0x003f,0,0); | ||||
|   } | ||||
|   OpEnd(tea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x0800+ ---------------------
 | ||||
| // Emit a Btst/Bchg/Bclr/Bset (Immediate) opcode 00001000 ttaaaaaa nn
 | ||||
| int OpBtstImm(int op) | ||||
| { | ||||
|   int type=0,sea=0,tea=0; | ||||
|   int use=0; | ||||
|   int size=0; | ||||
| 
 | ||||
|   type=(op>>6)&3; | ||||
|   // Get source and target EA
 | ||||
|   sea=   0x003c; | ||||
|   tea=op&0x003f; | ||||
|   if (tea<0x10) size=2; // For registers, 32-bits
 | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanRead(tea,0)==0||EaAn(tea)||tea==0x3c) return 1; | ||||
|   if (type>0) | ||||
|   { | ||||
|     if (EaCanWrite(tea)==0) return 1; | ||||
|   } | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,sea,tea); | ||||
| 
 | ||||
|   ot("\n"); | ||||
|   EaCalcReadNoSE(-1,0,sea,0,0); | ||||
|   ot("  mov r11,#1\n"); | ||||
|   ot("  bic r10,r10,#0x40000000 ;@ Blank Z flag\n"); | ||||
|   if (tea>=0x10) | ||||
|        ot("  and r0,r0,#7    ;@ mem - do mod 8\n");  // size always 0
 | ||||
|   else ot("  and r0,r0,#0x1F ;@ reg - do mod 32\n"); // size always 2
 | ||||
|   ot("  mov r11,r11,lsl r0 ;@ Make bit mask\n"); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   if(type==1||type==3) { | ||||
|     Cycles=12; | ||||
|   } else { | ||||
|     Cycles=type?12:8; | ||||
|     if(size>=2) Cycles+=2; | ||||
|   } | ||||
| 
 | ||||
|   EaCalcReadNoSE((type>0)?8:-1,0,tea,size,0x003f); | ||||
|   ot("  tst r0,r11 ;@ Do arithmetic\n"); | ||||
|   ot("  orreq r10,r10,#0x40000000 ;@ Get Z flag\n"); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   if (type>0) | ||||
|   { | ||||
|     if (type==1) ot("  eor r1,r0,r11 ;@ Toggle bit\n"); | ||||
|     if (type==2) ot("  bic r1,r0,r11 ;@ Clear bit\n"); | ||||
|     if (type==3) ot("  orr r1,r0,r11 ;@ Set bit\n"); | ||||
|     ot("\n"); | ||||
|     EaWrite(8,   1,tea,size,0x003f,0,0); | ||||
| #if CYCLONE_FOR_GENESIS && !MEMHANDLERS_CHANGE_CYCLES | ||||
|     // this is a bit hacky (device handlers might modify cycles)
 | ||||
|     if (tea==0x38||tea==0x39) | ||||
|       ot("  ldr r5,[r7,#0x5c] ;@ Load Cycles\n"); | ||||
| #endif | ||||
|   } | ||||
| 
 | ||||
|   OpEnd(sea,tea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x4000+ ---------------------
 | ||||
| int OpNeg(int op) | ||||
| { | ||||
|   // 01000tt0 xxeeeeee (tt=negx/clr/neg/not, xx=size, eeeeee=EA)
 | ||||
|   int type=0,size=0,ea=0,use=0; | ||||
| 
 | ||||
|   type=(op>>9)&3; | ||||
|   ea  =op&0x003f; | ||||
|   size=(op>>6)&3; if (size>=3) return 1; | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanRead (ea,size)==0||EaAn(ea)) return 1; | ||||
|   if (EaCanWrite(ea     )==0) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,ea); Cycles=size<2?4:6; | ||||
|   if(ea >= 0x10)  Cycles*=2; | ||||
| 
 | ||||
|   EaCalc (11,0x003f,ea,size,0,0); | ||||
| 
 | ||||
|   if (type!=1) EaRead (11,0,ea,size,0x003f,0,0); // Don't need to read for 'clr' (or do we, for a dummy read?)
 | ||||
|   if (type==1) ot("\n"); | ||||
| 
 | ||||
|   if (type==0) | ||||
|   { | ||||
|     ot(";@ Negx:\n"); | ||||
|     GetXBit(1); | ||||
|     if(size!=2) ot("  mov r0,r0,asl #%i\n",size?16:24); | ||||
|     ot("  rscs r1,r0,#0 ;@ do arithmetic\n"); | ||||
|     ot("  orr r3,r10,#0xb0000000 ;@ for old Z\n"); | ||||
|     OpGetFlags(1,1,0); | ||||
|     if(size!=2) { | ||||
|       ot("  movs r1,r1,asr #%i\n",size?16:24); | ||||
|       ot("  orreq r10,r10,#0x40000000 ;@ possily missed Z\n"); | ||||
|     } | ||||
|     ot("  andeq r10,r10,r3 ;@ fix Z\n"); | ||||
|     ot("\n"); | ||||
|   } | ||||
| 
 | ||||
|   if (type==1) | ||||
|   { | ||||
|     ot(";@ Clear:\n"); | ||||
|     ot("  mov r1,#0\n"); | ||||
|     ot("  mov r10,#0x40000000 ;@ NZCV=0100\n"); | ||||
|     ot("\n"); | ||||
|   } | ||||
| 
 | ||||
|   if (type==2) | ||||
|   { | ||||
|     ot(";@ Neg:\n"); | ||||
|     if(size!=2) ot("  mov r0,r0,asl #%i\n",size?16:24); | ||||
|     ot("  rsbs r1,r0,#0\n"); | ||||
|     OpGetFlags(1,1); | ||||
|     if(size!=2) ot("  mov r1,r1,asr #%i\n",size?16:24); | ||||
|     ot("\n"); | ||||
|   } | ||||
| 
 | ||||
|   if (type==3) | ||||
|   { | ||||
|     ot(";@ Not:\n"); | ||||
|     if(size!=2) { | ||||
|       ot("  mov r0,r0,asl #%i\n",size?16:24); | ||||
|       ot("  mvn r1,r0,asr #%i\n",size?16:24); | ||||
|     } | ||||
|     else | ||||
|       ot("  mvn r1,r0\n"); | ||||
|     ot("  adds r1,r1,#0 ;@ Defines NZ, clears CV\n"); | ||||
|     OpGetFlags(0,0); | ||||
|     ot("\n"); | ||||
|   } | ||||
| 
 | ||||
|   if (type==1) eawrite_check_addrerr=1; | ||||
|   EaWrite(11,     1,ea,size,0x003f,0,0); | ||||
| 
 | ||||
|   OpEnd(ea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x4840+ ---------------------
 | ||||
| // Swap, 01001000 01000nnn swap Dn
 | ||||
| int OpSwap(int op) | ||||
| { | ||||
|   int ea=0,use=0; | ||||
| 
 | ||||
|   ea=op&7; | ||||
|   use=op&~0x0007; // Use same opcode for all An
 | ||||
| 
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op); Cycles=4; | ||||
| 
 | ||||
|   EaCalc (11,0x0007,ea,2,1); | ||||
|   EaRead (11,     0,ea,2,0x0007,1); | ||||
| 
 | ||||
|   ot("  mov r1,r0,ror #16\n"); | ||||
|   ot("  adds r1,r1,#0 ;@ Defines NZ, clears CV\n"); | ||||
|   OpGetFlags(0,0); | ||||
| 
 | ||||
|   EaWrite(11,     1,8,2,0x0007,1); | ||||
| 
 | ||||
|   OpEnd(); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x4a00+ ---------------------
 | ||||
| // Emit a Tst opcode, 01001010 xxeeeeee
 | ||||
| int OpTst(int op) | ||||
| { | ||||
|   int sea=0; | ||||
|   int size=0,use=0; | ||||
| 
 | ||||
|   sea=op&0x003f; | ||||
|   size=(op>>6)&3; if (size>=3) return 1; | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanWrite(sea)==0||EaAn(sea)) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,sea); Cycles=4; | ||||
| 
 | ||||
|   EaCalc ( 0,0x003f,sea,size,1); | ||||
|   EaRead ( 0,     0,sea,size,0x003f,1); | ||||
| 
 | ||||
|   ot("  adds r0,r0,#0 ;@ Defines NZ, clears CV\n"); | ||||
|   ot("  mrs r10,cpsr ;@ r10=flags\n"); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   OpEnd(sea); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x4880+ ---------------------
 | ||||
| // Emit an Ext opcode, 01001000 1x000nnn
 | ||||
| int OpExt(int op) | ||||
| { | ||||
|   int ea=0; | ||||
|   int size=0,use=0; | ||||
|   int shift=0; | ||||
| 
 | ||||
|   ea=op&0x0007; | ||||
|   size=(op>>6)&1; | ||||
|   shift=32-(8<<size); | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op); Cycles=4; | ||||
| 
 | ||||
|   EaCalc (11,0x0007,ea,size+1,0,0); | ||||
|   EaRead (11,     0,ea,size+1,0x0007,0,0); | ||||
| 
 | ||||
|   ot("  mov r0,r0,asl #%d\n",shift); | ||||
|   ot("  adds r0,r0,#0 ;@ Defines NZ, clears CV\n"); | ||||
|   ot("  mrs r10,cpsr ;@ r10=flags\n"); | ||||
|   ot("  mov r1,r0,asr #%d\n",shift); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   EaWrite(11,     1,ea,size+1,0x0007,0,0); | ||||
| 
 | ||||
|   OpEnd(); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x50c0+ ---------------------
 | ||||
| // Emit a Set cc opcode, 0101cccc 11eeeeee
 | ||||
| int OpSet(int op) | ||||
| { | ||||
|   int cc=0,ea=0; | ||||
|   int size=0,use=0,changed_cycles=0; | ||||
|   static const char * const cond[16]= | ||||
|   { | ||||
|     "al","", "hi","ls","cc","cs","ne","eq", | ||||
|     "vc","vs","pl","mi","ge","lt","gt","le" | ||||
|   }; | ||||
| 
 | ||||
|   cc=(op>>8)&15; | ||||
|   ea=op&0x003f; | ||||
| 
 | ||||
|   if ((ea&0x38)==0x08) return 1; // dbra, not scc
 | ||||
|    | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanWrite(ea)==0) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   changed_cycles=ea<8 && cc>=2; | ||||
|   OpStart(op,ea,0,changed_cycles); Cycles=8; | ||||
|   if (ea<8) Cycles=4; | ||||
| 
 | ||||
|   if (cc) | ||||
|     ot("  mov r1,#0\n"); | ||||
| 
 | ||||
|   switch (cc) | ||||
|   { | ||||
|     case 0: // T
 | ||||
|       ot("  mvn r1,#0\n"); | ||||
|       if (ea<8) Cycles+=2; | ||||
|       break; | ||||
|     case 1: // F
 | ||||
|       break; | ||||
|     case 2: // hi
 | ||||
|       ot("  tst r10,#0x60000000 ;@ hi: !C && !Z\n"); | ||||
|       ot("  mvneq r1,r1\n"); | ||||
|       if (ea<8) ot("  subeq r5,r5,#2 ;@ Extra cycles\n"); | ||||
|       break; | ||||
|     case 3: // ls
 | ||||
|       ot("  tst r10,#0x60000000 ;@ ls: C || Z\n"); | ||||
|       ot("  mvnne r1,r1\n"); | ||||
|       if (ea<8) ot("  subne r5,r5,#2 ;@ Extra cycles\n"); | ||||
|       break; | ||||
|     default: | ||||
|       ot(";@ Is the condition true?\n"); | ||||
|       ot("  msr cpsr_flg,r10 ;@ ARM flags = 68000 flags\n"); | ||||
|       ot("  mvn%s r1,r1\n",cond[cc]); | ||||
|       if (ea<8) ot("  sub%s r5,r5,#2 ;@ Extra cycles\n",cond[cc]); | ||||
|       break; | ||||
|   } | ||||
| 
 | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   eawrite_check_addrerr=1; | ||||
|   EaCalc (0,0x003f, ea,size,0,0); | ||||
|   EaWrite(0,     1, ea,size,0x003f,0,0); | ||||
| 
 | ||||
|   opend_op_changes_cycles=changed_cycles; | ||||
|   OpEnd(ea,0); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // Emit a Asr/Lsr/Roxr/Ror opcode
 | ||||
| static int EmitAsr(int op,int type,int dir,int count,int size,int usereg) | ||||
| { | ||||
|   char pct[8]=""; // count
 | ||||
|   int shift=32-(8<<size); | ||||
| 
 | ||||
|   if (count>=1) sprintf(pct,"#%d",count); // Fixed count
 | ||||
| 
 | ||||
|   if (usereg) | ||||
|   { | ||||
|     ot(";@ Use Dn for count:\n"); | ||||
|     ot("  and r2,r8,#0x0e00\n"); | ||||
|     ot("  ldr r2,[r7,r2,lsr #7]\n"); | ||||
|     ot("  and r2,r2,#63\n"); | ||||
|     ot("\n"); | ||||
|     strcpy(pct,"r2"); | ||||
|   } | ||||
|   else if (count<0) | ||||
|   { | ||||
|     ot("  mov r2,r8,lsr #9 ;@ Get 'n'\n"); | ||||
|     ot("  and r2,r2,#7\n\n"); strcpy(pct,"r2"); | ||||
|   } | ||||
| 
 | ||||
|   // Take 2*n cycles:
 | ||||
|   if (count<0) ot("  sub r5,r5,r2,asl #1 ;@ Take 2*n cycles\n\n"); | ||||
|   else Cycles+=count<<1; | ||||
| 
 | ||||
|   if (type<2) | ||||
|   { | ||||
|     // Asr/Lsr
 | ||||
|     if (dir==0 && size<2) | ||||
|     { | ||||
|       ot(";@ For shift right, use loworder bits for the operation:\n"); | ||||
|       ot("  mov r0,r0,%s #%d\n",type?"lsr":"asr",32-(8<<size)); | ||||
|       ot("\n"); | ||||
|     } | ||||
| 
 | ||||
|     if (type==0 && dir) ot("  adds r3,r0,#0 ;@ save old value for V flag calculation, also clear V\n"); | ||||
| 
 | ||||
|     ot(";@ Shift register:\n"); | ||||
|     if (type==0) ot("  movs r0,r0,%s %s\n",dir?"asl":"asr",pct); | ||||
|     if (type==1) ot("  movs r0,r0,%s %s\n",dir?"lsl":"lsr",pct); | ||||
| 
 | ||||
|     OpGetFlags(0,0); | ||||
|     if (usereg) { // store X only if count is not 0
 | ||||
|       ot("  cmp %s,#0 ;@ shifting by 0?\n",pct); | ||||
|       ot("  biceq r10,r10,#0x20000000 ;@ if so, clear carry\n"); | ||||
|       ot("  strne r10,[r7,#0x4c] ;@ else Save X bit\n"); | ||||
|     } else { | ||||
|       // count will never be 0 if we use immediate
 | ||||
|       ot("  str r10,[r7,#0x4c] ;@ Save X bit\n"); | ||||
|     } | ||||
|     ot("\n"); | ||||
| 
 | ||||
|     if (dir==0 && size<2) | ||||
|     { | ||||
|       ot(";@ restore after right shift:\n"); | ||||
|       ot("  movs r0,r0,lsl #%d\n",32-(8<<size)); | ||||
|       if (type) | ||||
|         ot("  orrmi r10,r10,#0x80000000 ;@ Potentially missed N flag\n"); | ||||
|       ot("\n"); | ||||
|     } | ||||
| 
 | ||||
|     if (type==0 && dir) { | ||||
|       ot(";@ calculate V flag (set if sign bit changes at anytime):\n"); | ||||
|       ot("  mov r1,#0x80000000\n"); | ||||
|       ot("  ands r3,r3,r1,asr %s\n", pct); | ||||
|       ot("  cmpne r3,r1,asr %s\n", pct); | ||||
|       ot("  eoreq r1,r0,r3\n"); // above check doesn't catch (-1)<<(32+), so we need this
 | ||||
|       ot("  tsteq r1,#0x80000000\n"); | ||||
|       ot("  orrne r10,r10,#0x10000000\n"); | ||||
|       ot("\n"); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   // --------------------------------------
 | ||||
|   if (type==2) | ||||
|   { | ||||
|     int wide=8<<size; | ||||
| 
 | ||||
|     // Roxr
 | ||||
|     if(count == 1) | ||||
|     { | ||||
|       if(dir==0) { | ||||
|         if(size!=2) { | ||||
|           ot("  orr r0,r0,r0,lsr #%i\n", size?16:24); | ||||
|           ot("  bic r0,r0,#0x%x\n", 1<<(32-wide)); | ||||
|         } | ||||
|         GetXBit(0); | ||||
|         ot("  movs r0,r0,rrx\n"); | ||||
|         OpGetFlags(0,1); | ||||
|       } else { | ||||
|         ot("  ldr r3,[r7,#0x4c]\n"); | ||||
|         ot("  movs r0,r0,lsl #1\n"); | ||||
|         OpGetFlags(0,1); | ||||
|         ot("  tst r3,#0x20000000\n"); | ||||
|         ot("  orrne r0,r0,#0x%x\n", 1<<(32-wide)); | ||||
|         ot("  bicne r10,r10,#0x40000000 ;@ clear Z in case it got there\n"); | ||||
|       } | ||||
|       ot("  bic r10,r10,#0x10000000 ;@ make suve V is clear\n"); | ||||
|       return 0; | ||||
|     } | ||||
| 
 | ||||
|     if (usereg) | ||||
|     { | ||||
|       if (size==2) | ||||
|       { | ||||
|         ot("  subs r2,r2,#33\n"); | ||||
|         ot("  addmis r2,r2,#33 ;@ Now r2=0-%d\n",wide); | ||||
|       } | ||||
|       else | ||||
|       { | ||||
|         ot(";@ Reduce r2 until <0:\n"); | ||||
|         ot("Reduce_%.4x%s\n",op,ms?"":":"); | ||||
|         ot("  subs r2,r2,#%d\n",wide+1); | ||||
|         ot("  bpl Reduce_%.4x\n",op); | ||||
|         ot("  adds r2,r2,#%d ;@ Now r2=0-%d\n",wide+1,wide); | ||||
|       } | ||||
|       ot("  beq norotx_%.4x\n",op); | ||||
|       ot("\n"); | ||||
|     } | ||||
| 
 | ||||
|     if (usereg||count < 0) | ||||
|     { | ||||
|       if (dir) ot("  rsb r2,r2,#%d ;@ Reverse direction\n",wide+1); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|       if (dir) ot("  mov r2,#%d ;@ Reversed\n",wide+1-count); | ||||
|       else     ot("  mov r2,#%d\n",count); | ||||
|     } | ||||
| 
 | ||||
|     if (shift) ot("  mov r0,r0,lsr #%d ;@ Shift down\n",shift); | ||||
| 
 | ||||
|     ot("\n"); | ||||
|     ot(";@ First get X bit (middle):\n"); | ||||
|     ot("  ldr r3,[r7,#0x4c]\n"); | ||||
|     ot("  rsb r1,r2,#%d\n",wide); | ||||
|     ot("  and r3,r3,#0x20000000\n"); | ||||
|     ot("  mov r3,r3,lsr #29\n"); | ||||
|     ot("  mov r3,r3,lsl r1\n"); | ||||
| 
 | ||||
|     ot(";@ Rotate bits:\n"); | ||||
|     ot("  orr r3,r3,r0,lsr r2 ;@ Orr right part\n"); | ||||
|     ot("  rsbs r2,r2,#%d ;@ should also clear ARM V\n",wide+1); | ||||
|     ot("  orrs r0,r3,r0,lsl r2 ;@ Orr left part, set flags\n"); | ||||
|     ot("\n"); | ||||
| 
 | ||||
|     if (shift) ot("  movs r0,r0,lsl #%d ;@ Shift up and get correct NC flags\n",shift); | ||||
|     OpGetFlags(0,!usereg); | ||||
|     if (usereg) { // store X only if count is not 0
 | ||||
|       ot("  str r10,[r7,#0x4c] ;@ if not 0, Save X bit\n"); | ||||
|       ot("  b nozerox%.4x\n",op); | ||||
|       ot("norotx_%.4x%s\n",op,ms?"":":"); | ||||
|       ot("  ldr r2,[r7,#0x4c]\n"); | ||||
|       ot("  adds r0,r0,#0 ;@ Defines NZ, clears CV\n"); | ||||
|       OpGetFlags(0,0); | ||||
|       ot("  and r2,r2,#0x20000000\n"); | ||||
|       ot("  orr r10,r10,r2 ;@ C = old_X\n"); | ||||
|       ot("nozerox%.4x%s\n",op,ms?"":":"); | ||||
|     } | ||||
| 
 | ||||
|     ot("\n"); | ||||
|   } | ||||
| 
 | ||||
|   // --------------------------------------
 | ||||
|   if (type==3) | ||||
|   { | ||||
|     // Ror
 | ||||
|     if (size<2) | ||||
|     { | ||||
|       ot(";@ Mirror value in whole 32 bits:\n"); | ||||
|       if (size<=0) ot("  orr r0,r0,r0,lsr #8\n"); | ||||
|       if (size<=1) ot("  orr r0,r0,r0,lsr #16\n"); | ||||
|       ot("\n"); | ||||
|     } | ||||
| 
 | ||||
|     ot(";@ Rotate register:\n"); | ||||
|     if (!dir) ot("  adds r0,r0,#0 ;@ first clear V and C\n"); // ARM does not clear C if rot count is 0
 | ||||
|     if (count<0) | ||||
|     { | ||||
|       if (dir) ot("  rsb %s,%s,#32\n",pct,pct); | ||||
|       ot("  movs r0,r0,ror %s\n",pct); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|       int ror=count; | ||||
|       if (dir) ror=32-ror; | ||||
|       if (ror&31) ot("  movs r0,r0,ror #%d\n",ror); | ||||
|     } | ||||
| 
 | ||||
|     OpGetFlags(0,0); | ||||
|     if (dir) | ||||
|     { | ||||
|       ot("  bic r10,r10,#0x30000000 ;@ clear CV\n"); | ||||
|       ot(";@ Get carry bit from bit 0:\n"); | ||||
|       if (usereg) | ||||
|       { | ||||
|         ot("  cmp %s,#32 ;@ rotating by 0?\n",pct); | ||||
|         ot("  tstne r0,#1 ;@ no, check bit 0\n"); | ||||
|       } | ||||
|       else | ||||
|         ot("  tst r0,#1\n"); | ||||
|       ot("  orrne r10,r10,#0x20000000\n"); | ||||
|     } | ||||
|     ot("\n"); | ||||
| 
 | ||||
|   } | ||||
|   // --------------------------------------
 | ||||
|    | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // Emit a Asr/Lsr/Roxr/Ror opcode - 1110cccd xxuttnnn
 | ||||
| // (ccc=count, d=direction(r,l) xx=size extension, u=use reg for count, tt=type, nnn=register Dn)
 | ||||
| int OpAsr(int op) | ||||
| { | ||||
|   int ea=0,use=0; | ||||
|   int count=0,dir=0; | ||||
|   int size=0,usereg=0,type=0; | ||||
| 
 | ||||
|   count =(op>>9)&7; | ||||
|   dir   =(op>>8)&1; | ||||
|   size  =(op>>6)&3; | ||||
|   if (size>=3) return 1; // use OpAsrEa()
 | ||||
|   usereg=(op>>5)&1; | ||||
|   type  =(op>>3)&3; | ||||
| 
 | ||||
|   if (usereg==0) count=((count-1)&7)+1; // because ccc=000 means 8
 | ||||
| 
 | ||||
|   // Use the same opcode for target registers:
 | ||||
|   use=op&~0x0007; | ||||
| 
 | ||||
|   // As long as count is not 8, use the same opcode for all shift counts:
 | ||||
|   if (usereg==0 && count!=8 && !(count==1&&type==2)) { use|=0x0e00; count=-1; } | ||||
|   if (usereg) { use&=~0x0e00; count=-1; } // Use same opcode for all Dn
 | ||||
| 
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,ea,0,count<0); Cycles=size<2?6:8; | ||||
| 
 | ||||
|   EaCalc(11,0x0007, ea,size,1); | ||||
|   EaRead(11,     0, ea,size,0x0007,1); | ||||
| 
 | ||||
|   EmitAsr(op,type,dir,count, size,usereg); | ||||
| 
 | ||||
|   EaWrite(11,    0, ea,size,0x0007,1); | ||||
| 
 | ||||
|   opend_op_changes_cycles = (count<0); | ||||
|   OpEnd(ea,0); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // Asr/Lsr/Roxr/Ror etc EA - 11100ttd 11eeeeee 
 | ||||
| int OpAsrEa(int op) | ||||
| { | ||||
|   int use=0,type=0,dir=0,ea=0,size=1; | ||||
| 
 | ||||
|   type=(op>>9)&3; | ||||
|   dir =(op>>8)&1; | ||||
|   ea  = op&0x3f; | ||||
| 
 | ||||
|   if (ea<0x10) return 1; | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanRead(ea,0)==0) return 1; | ||||
|   if (EaCanWrite(ea)==0) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,ea); Cycles=6; // EmitAsr() will add 2
 | ||||
| 
 | ||||
|   EaCalc (11,0x003f,ea,size,1); | ||||
|   EaRead (11,     0,ea,size,0x003f,1); | ||||
| 
 | ||||
|   EmitAsr(op,type,dir,1,size,0); | ||||
| 
 | ||||
|   EaWrite(11,     0,ea,size,0x003f,1); | ||||
| 
 | ||||
|   OpEnd(ea); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| int OpTas(int op, int gen_special) | ||||
| { | ||||
|   int ea=0; | ||||
|   int use=0; | ||||
| 
 | ||||
|   ea=op&0x003f; | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanWrite(ea)==0 || EaAn(ea)) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,0); | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   if (!gen_special) OpStart(op,ea); | ||||
|   else | ||||
|     ot("Op%.4x_%s\n", op, ms?"":":"); | ||||
| 
 | ||||
|   Cycles=4; | ||||
|   if(ea>=8) Cycles+=10; | ||||
| 
 | ||||
|   EaCalc (11,0x003f,ea,0,1); | ||||
|   EaRead (11,     1,ea,0,0x003f,1); | ||||
| 
 | ||||
|   ot("  adds r1,r1,#0 ;@ Defines NZ, clears CV\n"); | ||||
|   OpGetFlags(0,0); | ||||
|   ot("\n"); | ||||
| 
 | ||||
| #if CYCLONE_FOR_GENESIS | ||||
|   // the original Sega hardware ignores write-back phase (to memory only)
 | ||||
|   if (ea < 0x10 || gen_special) { | ||||
| #endif | ||||
|     ot("  orr r1,r1,#0x80000000 ;@ set bit7\n"); | ||||
| 
 | ||||
|     EaWrite(11,     1,ea,0,0x003f,1); | ||||
| #if CYCLONE_FOR_GENESIS | ||||
|   } | ||||
| #endif | ||||
| 
 | ||||
|   OpEnd(ea); | ||||
| 
 | ||||
| #if (CYCLONE_FOR_GENESIS == 2) | ||||
|   if (!gen_special && ea >= 0x10) { | ||||
|     OpTas(op, 1); | ||||
|   } | ||||
| #endif | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
|  | @ -1,685 +0,0 @@ | |||
| 
 | ||||
| #include "app.h" | ||||
| 
 | ||||
| 
 | ||||
| // Pack our flags into r1, in SR/CCR register format
 | ||||
| // trashes r0,r2
 | ||||
| void OpFlagsToReg(int high) | ||||
| { | ||||
|   ot("  ldr r0,[r7,#0x4c]   ;@ X bit\n"); | ||||
|   ot("  mov r1,r10,lsr #28  ;@ ____NZCV\n"); | ||||
|   ot("  eor r2,r1,r1,ror #1 ;@ Bit 0=C^V\n"); | ||||
|   ot("  tst r2,#1           ;@ 1 if C!=V\n"); | ||||
|   ot("  eorne r1,r1,#3      ;@ ____NZVC\n"); | ||||
|   ot("\n"); | ||||
|   if (high) ot("  ldrb r2,[r7,#0x44]  ;@ Include SR high\n"); | ||||
|   ot("  and r0,r0,#0x20000000\n"); | ||||
|   ot("  orr r1,r1,r0,lsr #25 ;@ ___XNZVC\n"); | ||||
|   if (high) ot("  orr r1,r1,r2,lsl #8\n"); | ||||
|   ot("\n"); | ||||
| } | ||||
| 
 | ||||
| // Convert SR/CRR register in r0 to our flags
 | ||||
| // trashes r0,r1
 | ||||
| void OpRegToFlags(int high, int srh_reg) | ||||
| { | ||||
|   ot("  eor r1,r0,r0,ror #1 ;@ Bit 0=C^V\n"); | ||||
|   ot("  mov r2,r0,lsl #25\n"); | ||||
|   ot("  tst r1,#1           ;@ 1 if C!=V\n"); | ||||
|   ot("  eorne r0,r0,#3      ;@ ___XNZCV\n"); | ||||
|   ot("  str r2,[r7,#0x4c]   ;@ Store X bit\n"); | ||||
|   ot("  mov r10,r0,lsl #28  ;@ r10=NZCV...\n"); | ||||
| 
 | ||||
|   if (high) | ||||
|   { | ||||
|     int mask=EMULATE_TRACE?0xa7:0x27; | ||||
|     ot("  mov r%i,r0,ror #8\n",srh_reg); | ||||
|     ot("  and r%i,r%i,#0x%02x ;@ only take defined bits\n",srh_reg,srh_reg,mask); | ||||
|     ot("  strb r%i,[r7,#0x44] ;@ Store SR high\n",srh_reg); | ||||
|   } | ||||
|   ot("\n"); | ||||
| } | ||||
| 
 | ||||
| void SuperEnd(void) | ||||
| { | ||||
|   ot(";@ ----------\n"); | ||||
|   ot(";@ tried execute privileged instruction in user mode\n"); | ||||
|   ot("WrongPrivilegeMode%s\n",ms?"":":"); | ||||
| #if EMULATE_ADDRESS_ERRORS_JUMP || EMULATE_ADDRESS_ERRORS_IO | ||||
|   ot("  ldr r1,[r7,#0x58]\n"); | ||||
|   ot("  sub r4,r4,#2 ;@ last opcode wasn't executed - go back\n"); | ||||
|   ot("  orr r1,r1,#4 ;@ set activity bit: 'not processing instruction'\n"); | ||||
|   ot("  str r1,[r7,#0x58]\n"); | ||||
| #else | ||||
|   ot("  sub r4,r4,#2 ;@ last opcode wasn't executed - go back\n"); | ||||
| #endif | ||||
|   ot("  mov r0,#8 ;@ privilege violation\n"); | ||||
|   ot("  bl Exception\n"); | ||||
|   Cycles=34; | ||||
|   OpEnd(0); | ||||
| } | ||||
| 
 | ||||
| // does OSP and A7 swapping if needed
 | ||||
| // new or old SR (not the one already in [r7,#0x44]) should be passed in r11
 | ||||
| // uses srh from srh_reg (loads if < 0), trashes r0,r11
 | ||||
| void SuperChange(int op,int srh_reg) | ||||
| { | ||||
|   ot(";@ A7 <-> OSP?\n"); | ||||
|   if (srh_reg < 0) { | ||||
|     ot("  ldr r0,[r7,#0x44] ;@ Get other SR high\n"); | ||||
|     srh_reg=0; | ||||
|   } | ||||
|   ot("  eor r0,r%i,r11\n",srh_reg); | ||||
|   ot("  tst r0,#0x20\n"); | ||||
|   ot("  beq no_sp_swap%.4x\n",op); | ||||
|   ot(" ;@ swap OSP and A7:\n"); | ||||
|   ot("  ldr r11,[r7,#0x3C] ;@ Get A7\n"); | ||||
|   ot("  ldr r0, [r7,#0x48] ;@ Get OSP\n"); | ||||
|   ot("  str r11,[r7,#0x48]\n"); | ||||
|   ot("  str r0, [r7,#0x3C]\n"); | ||||
|   ot("no_sp_swap%.4x%s\n", op, ms?"":":"); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| // --------------------- Opcodes 0x1000+ ---------------------
 | ||||
| // Emit a Move opcode, 00xxdddd ddssssss
 | ||||
| int OpMove(int op) | ||||
| { | ||||
|   int sea=0,tea=0; | ||||
|   int size=0,use=0; | ||||
|   int movea=0; | ||||
| 
 | ||||
|   // Get source and target EA
 | ||||
|   sea = op&0x003f; | ||||
|   tea =(op&0x01c0)>>3; | ||||
|   tea|=(op&0x0e00)>>9; | ||||
| 
 | ||||
|   if (tea>=8 && tea<0x10) movea=1; | ||||
| 
 | ||||
|   // Find size extension
 | ||||
|   switch (op&0x3000) | ||||
|   { | ||||
|     default: return 1; | ||||
|     case 0x1000: size=0; break; | ||||
|     case 0x3000: size=1; break; | ||||
|     case 0x2000: size=2; break; | ||||
|   } | ||||
| 
 | ||||
|   if (size<1 && (movea || EaAn(sea))) return 1; // move.b An,* and movea.b * are invalid
 | ||||
| 
 | ||||
|   // See if we can do this opcode:
 | ||||
|   if (EaCanRead (sea,size)==0) return 1; | ||||
|   if (EaCanWrite(tea     )==0) return 1; | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   if (tea<0x38) use&=~0x0e00; // Use same handler for register ?0-7
 | ||||
|    | ||||
|   if (tea==0x1f || tea==0x27) use|=0x0e00; // Specific handler for (a7)+ and -(a7)
 | ||||
| 
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,sea,tea); Cycles=4; | ||||
| 
 | ||||
|   if (movea==0) | ||||
|   { | ||||
|     EaCalcRead(-1,0,sea,size,0x003f); | ||||
|     ot("  adds r1,r0,#0 ;@ Defines NZ, clears CV\n"); | ||||
|     ot("  mrs r10,cpsr ;@ r10=NZCV flags\n"); | ||||
|     ot("\n"); | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     EaCalcRead(-1,1,sea,size,0x003f); | ||||
|     size=2; // movea always expands to 32-bits
 | ||||
|   } | ||||
| 
 | ||||
|   eawrite_check_addrerr=1; | ||||
| #if SPLIT_MOVEL_PD | ||||
|   if ((tea&0x38)==0x20 && size==2) { // -(An)
 | ||||
|     EaCalc (8,0x0e00,tea,size,0,0); | ||||
|     ot("  mov r11,r1\n"); | ||||
|     ot("  add r0,r8,#2\n"); | ||||
|     EaWrite(0,     1,tea,1,0x0e00,0,0); | ||||
|     EaWrite(8,    11,tea,1,0x0e00,1); | ||||
|   } | ||||
|   else | ||||
| #endif | ||||
|   { | ||||
|     EaCalc (0,0x0e00,tea,size,0,0); | ||||
|     EaWrite(0,     1,tea,size,0x0e00,0,0); | ||||
|   } | ||||
| 
 | ||||
| #if CYCLONE_FOR_GENESIS && !MEMHANDLERS_CHANGE_CYCLES | ||||
|   // this is a bit hacky (device handlers might modify cycles)
 | ||||
|   if (tea==0x39||((0x10<=tea&&tea<0x30)&&size>=1)) | ||||
|     ot("  ldr r5,[r7,#0x5c] ;@ Load Cycles\n"); | ||||
| #endif | ||||
| 
 | ||||
|   if((tea&0x38)==0x20) Cycles-=2; // less cycles when dest is -(An)
 | ||||
| 
 | ||||
|   OpEnd(sea,tea); | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x41c0+ ---------------------
 | ||||
| // Emit an Lea opcode, 0100nnn1 11aaaaaa
 | ||||
| int OpLea(int op) | ||||
| { | ||||
|   int use=0; | ||||
|   int sea=0,tea=0; | ||||
| 
 | ||||
|   sea= op&0x003f; | ||||
|   tea=(op&0x0e00)>>9; tea|=8; | ||||
| 
 | ||||
|   if (EaCanRead(sea,-1)==0) return 1; // See if we can do this opcode
 | ||||
| 
 | ||||
|   use=OpBase(op,0); | ||||
|   use&=~0x0e00; // Also use 1 handler for target ?0-7
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,sea,tea); | ||||
| 
 | ||||
|   eawrite_check_addrerr=1; | ||||
|   EaCalc (1,0x003f,sea,0); // Lea
 | ||||
|   EaCalc (0,0x0e00,tea,2); | ||||
|   EaWrite(0,     1,tea,2,0x0e00); | ||||
| 
 | ||||
|   Cycles=Ea_add_ns(g_lea_cycle_table,sea); | ||||
| 
 | ||||
|   OpEnd(sea,tea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x40c0+ ---------------------
 | ||||
| // Move SR opcode, 01000tt0 11aaaaaa move SR
 | ||||
| int OpMoveSr(int op) | ||||
| { | ||||
|   int type=0,ea=0; | ||||
|   int use=0,size=1; | ||||
| 
 | ||||
|   type=(op>>9)&3; // from SR, from CCR, to CCR, to SR
 | ||||
|   ea=op&0x3f; | ||||
| 
 | ||||
|   if(EaAn(ea)) return 1; // can't use An regs
 | ||||
| 
 | ||||
|   switch(type) | ||||
|   { | ||||
|     case 0: | ||||
|       if (EaCanWrite(ea)==0) return 1; // See if we can do this opcode:
 | ||||
|       break; | ||||
| 
 | ||||
|     case 1: | ||||
|       return 1; // no such op in 68000
 | ||||
| 
 | ||||
|     case 2: case 3: | ||||
|       if (EaCanRead(ea,size)==0) return 1; // See if we can do this opcode:
 | ||||
|       break; | ||||
|   } | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   // 68000 model allows reading whole SR in user mode (but newer models don't)
 | ||||
|   OpStart(op,ea,0,0,type==3); | ||||
|   Cycles=12; | ||||
|   if (type==0) Cycles=(ea>=8)?8:6; | ||||
| 
 | ||||
|   if (type==0 || type==1) | ||||
|   { | ||||
|     eawrite_check_addrerr=1; | ||||
|     OpFlagsToReg(type==0); | ||||
|     EaCalc (0,0x003f,ea,size,0,0); | ||||
|     EaWrite(0,     1,ea,size,0x003f,0,0); | ||||
|   } | ||||
| 
 | ||||
|   if (type==2 || type==3) | ||||
|   { | ||||
|     EaCalcReadNoSE(-1,0,ea,size,0x003f); | ||||
|     OpRegToFlags(type==3,1); | ||||
|     if (type==3) { | ||||
|       SuperChange(op,1); | ||||
|       opend_check_interrupt = 1; | ||||
|       opend_check_trace = 1; | ||||
|       OpEnd(ea); | ||||
|       return 0; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   OpEnd(ea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // Ori/Andi/Eori $nnnn,sr 0000t0t0 01111100
 | ||||
| int OpArithSr(int op) | ||||
| { | ||||
|   int type=0,ea=0; | ||||
|   int use=0,size=0; | ||||
|   int sr_mask=EMULATE_TRACE?0xa7:0x27; | ||||
| 
 | ||||
|   type=(op>>9)&5; if (type==4) return 1; | ||||
|   size=(op>>6)&1; // ccr or sr?
 | ||||
|   ea=0x3c; | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,ea,0,0,size!=0); Cycles=16; | ||||
| 
 | ||||
|   EaCalcRead(-1,0,ea,size,0x003f); | ||||
| 
 | ||||
|   ot("  eor r1,r0,r0,ror #1 ;@ Bit 0=C^V\n"); | ||||
|   ot("  tst r1,#1           ;@ 1 if C!=V\n"); | ||||
|   ot("  eorne r0,r0,#3      ;@ ___XNZCV\n"); | ||||
|   ot("  ldr r2,[r7,#0x4c]   ;@ Load old X bit\n"); | ||||
| 
 | ||||
|   // note: old srh is already in r11 (done by OpStart)
 | ||||
|   if (type==0) { | ||||
|     ot("  orr r10,r10,r0,lsl #28\n"); | ||||
|     ot("  orr r2,r2,r0,lsl #25 ;@ X bit\n"); | ||||
|     if (size!=0) { | ||||
|       ot("  orr r1,r11,r0,lsr #8\n"); | ||||
|       ot("  and r1,r1,#0x%02x ;@ mask-out unused bits\n",sr_mask); | ||||
|     } | ||||
|   } | ||||
|   if (type==1) { | ||||
|     ot("  and r10,r10,r0,lsl #28\n"); | ||||
|     ot("  and r2,r2,r0,lsl #25 ;@ X bit\n"); | ||||
|     if (size!=0) | ||||
|       ot("  and r1,r11,r0,lsr #8\n"); | ||||
|   } | ||||
|   if (type==5) { | ||||
|     ot("  eor r10,r10,r0,lsl #28\n"); | ||||
|     ot("  eor r2,r2,r0,lsl #25 ;@ X bit\n"); | ||||
|     if (size!=0) { | ||||
|       ot("  eor r1,r11,r0,lsr #8\n"); | ||||
|       ot("  and r1,r1,#0x%02x ;@ mask-out unused bits\n",sr_mask); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   ot("  str r2,[r7,#0x4c]   ;@ Save X bit\n"); | ||||
|   if (size!=0) | ||||
|     ot("  strb r1,[r7,#0x44]\n"); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   // we can't enter supervisor mode, nor unmask irqs just by using OR
 | ||||
|   if (size!=0 && type!=0) { | ||||
|     SuperChange(op,1); | ||||
|     ot("\n"); | ||||
|     opend_check_interrupt = 1; | ||||
|   } | ||||
|   // also can't set trace bit with AND
 | ||||
|   if (size!=0 && type!=1) | ||||
|     opend_check_trace = 1; | ||||
| 
 | ||||
|   OpEnd(ea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x4850+ ---------------------
 | ||||
| // Emit an Pea opcode, 01001000 01aaaaaa
 | ||||
| int OpPea(int op) | ||||
| { | ||||
|   int use=0; | ||||
|   int ea=0; | ||||
| 
 | ||||
|   ea=op&0x003f; if (ea<0x10) return 1; // Swap opcode
 | ||||
|   if (EaCanRead(ea,-1)==0) return 1; // See if we can do this opcode:
 | ||||
| 
 | ||||
|   use=OpBase(op,0); | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,ea); | ||||
| 
 | ||||
|   ot("  ldr r11,[r7,#0x3c]\n"); | ||||
|   EaCalc (1,0x003f, ea,0); | ||||
|   ot("\n"); | ||||
|   ot("  sub r0,r11,#4 ;@ Predecrement A7\n"); | ||||
|   ot("  str r0,[r7,#0x3c] ;@ Save A7\n"); | ||||
|   ot("\n"); | ||||
|   MemHandler(1,2); // Write 32-bit
 | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   Cycles=6+Ea_add_ns(g_pea_cycle_table,ea); | ||||
| 
 | ||||
|   OpEnd(ea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x4880+ ---------------------
 | ||||
| // Emit a Movem opcode, 01001d00 1xeeeeee regmask
 | ||||
| int OpMovem(int op) | ||||
| { | ||||
|   int size=0,ea=0,cea=0,dir=0; | ||||
|   int use=0,decr=0,change=0; | ||||
| 
 | ||||
|   size=((op>>6)&1)+1; // word, long
 | ||||
|   ea=op&0x003f; | ||||
|   dir=(op>>10)&1; // Direction (1==ea2reg)
 | ||||
| 
 | ||||
|   if (dir) { | ||||
|     if (ea<0x10 || ea>0x3b || (ea&0x38)==0x20) return 1; // Invalid EA
 | ||||
|   } else { | ||||
|     if (ea<0x10 || ea>0x39 || (ea&0x38)==0x18) return 1; | ||||
|   } | ||||
| 
 | ||||
|   if ((ea&0x38)==0x18 || (ea&0x38)==0x20) change=1; | ||||
|   if ((ea&0x38)==0x20) decr=1; // -(An), bitfield is decr
 | ||||
| 
 | ||||
|   cea=ea; if (change) cea=0x10; | ||||
| 
 | ||||
|   use=OpBase(op,size); | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,ea,0,1); | ||||
| 
 | ||||
|   ot("  ldrh r11,[r4],#2 ;@ r11=register mask\n"); | ||||
|   ot("\n"); | ||||
|   ot(";@ Get the address into r6:\n"); | ||||
|   EaCalc(6,0x003f,cea,size); | ||||
| 
 | ||||
| #if !MEMHANDLERS_NEED_PREV_PC | ||||
|   // must save PC, need a spare register
 | ||||
|   ot("  str r4,[r7,#0x40] ;@ Save PC\n"); | ||||
| #endif | ||||
| 
 | ||||
|   ot(";@ r4=Register Index*4:\n"); | ||||
|   if (decr) ot("  mov r4,#0x40 ;@ order reversed for -(An)\n"); | ||||
|   else      ot("  mov r4,#-4\n"); | ||||
|    | ||||
|   ot("\n"); | ||||
|   ot("  tst r11,r11\n");        // sanity check
 | ||||
|   ot("  beq NoRegs%.4x\n",op); | ||||
| 
 | ||||
| #if EMULATE_ADDRESS_ERRORS_IO | ||||
|   ot("\n"); | ||||
|   ot("  tst r6,#1 ;@ address error?\n"); | ||||
|   ot("  movne r0,r6\n"); | ||||
|   ot("  bne ExceptionAddressError_%c_data\n",dir?'r':'w'); | ||||
| #endif | ||||
| 
 | ||||
|   ot("\n"); | ||||
|   ot("Movemloop%.4x%s\n",op, ms?"":":"); | ||||
|   ot("  add r4,r4,#%d ;@ r4=Next Register\n",decr?-4:4); | ||||
|   ot("  movs r11,r11,lsr #1\n"); | ||||
|   ot("  bcc Movemloop%.4x\n",op); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   if (decr) ot("  sub r6,r6,#%d ;@ Pre-decrement address\n",1<<size); | ||||
| 
 | ||||
|   if (dir) | ||||
|   { | ||||
|     ot("  ;@ Copy memory to register:\n",1<<size); | ||||
|     earead_check_addrerr=0; // already checked
 | ||||
|     EaRead (6,0,ea,size,0x003f); | ||||
|     ot("  str r0,[r7,r4] ;@ Save value into Dn/An\n"); | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     ot("  ;@ Copy register to memory:\n",1<<size); | ||||
|     ot("  ldr r1,[r7,r4] ;@ Load value from Dn/An\n"); | ||||
| #if SPLIT_MOVEL_PD | ||||
|     if (decr && size==2) { // -(An)
 | ||||
|       ot("  add r0,r6,#2\n"); | ||||
|       EaWrite(0,1,ea,1,0x003f,0,0); | ||||
|       ot("  ldr r1,[r7,r4] ;@ Load value from Dn/An\n"); | ||||
|       ot("  mov r0,r6\n"); | ||||
|       EaWrite(0,1,ea,1,0x003f,1); | ||||
|     } | ||||
|     else | ||||
| #endif | ||||
|     { | ||||
|       EaWrite(6,1,ea,size,0x003f); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   if (decr==0) ot("  add r6,r6,#%d ;@ Post-increment address\n",1<<size); | ||||
| 
 | ||||
|   ot("  sub r5,r5,#%d ;@ Take some cycles\n",2<<size); | ||||
|   ot("  tst r11,r11\n"); | ||||
|   ot("  bne Movemloop%.4x\n",op); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   if (change) | ||||
|   { | ||||
|     ot(";@ Write back address:\n"); | ||||
|     EaCalc (0,0x0007,8|(ea&7),2); | ||||
|     EaWrite(0,     6,8|(ea&7),2,0x0007); | ||||
|   } | ||||
| 
 | ||||
|   ot("NoRegs%.4x%s\n",op, ms?"":":"); | ||||
|   ot("  ldr r4,[r7,#0x40]\n"); | ||||
|   ot("  ldr r6,[r7,#0x54] ;@ restore Opcode Jump table\n"); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   if(dir) { // er
 | ||||
|          if (ea==0x3a) Cycles=16; // ($nn,PC)
 | ||||
|     else if (ea==0x3b) Cycles=18; // ($nn,pc,Rn)
 | ||||
|     else Cycles=12; | ||||
|   } else { | ||||
|     Cycles=8; | ||||
|   } | ||||
| 
 | ||||
|   Cycles+=Ea_add_ns(g_movem_cycle_table,ea); | ||||
| 
 | ||||
|   opend_op_changes_cycles = 1; | ||||
|   OpEnd(ea); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x4e60+ ---------------------
 | ||||
| // Emit a Move USP opcode, 01001110 0110dnnn move An to/from USP
 | ||||
| int OpMoveUsp(int op) | ||||
| { | ||||
|   int use=0,dir=0; | ||||
| 
 | ||||
|   dir=(op>>3)&1; // Direction
 | ||||
|   use=op&~0x0007; // Use same opcode for all An
 | ||||
| 
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op,0,0,0,1); Cycles=4; | ||||
| 
 | ||||
|   if (dir) | ||||
|   { | ||||
|     eawrite_check_addrerr=1; | ||||
|     ot("  ldr r1,[r7,#0x48] ;@ Get from USP\n\n"); | ||||
|     EaCalc (0,0x000f,8,2,1); | ||||
|     EaWrite(0,     1,8,2,0x000f,1); | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     EaCalc (0,0x000f,8,2,1); | ||||
|     EaRead (0,     0,8,2,0x000f,1); | ||||
|     ot("  str r0,[r7,#0x48] ;@ Put in USP\n\n"); | ||||
|   } | ||||
|      | ||||
|   OpEnd(); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0x7000+ ---------------------
 | ||||
| // Emit a Move Quick opcode, 0111nnn0 dddddddd  moveq #dd,Dn
 | ||||
| int OpMoveq(int op) | ||||
| { | ||||
|   int use=0; | ||||
| 
 | ||||
|   use=op&0xf100; // Use same opcode for all values
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op); Cycles=4; | ||||
| 
 | ||||
|   ot("  movs r0,r8,asl #24\n"); | ||||
|   ot("  and r1,r8,#0x0e00\n"); | ||||
|   ot("  mov r0,r0,asr #24 ;@ Sign extended Quick value\n"); | ||||
|   ot("  mrs r10,cpsr ;@ r10=NZ flags\n"); | ||||
|   ot("  str r0,[r7,r1,lsr #7] ;@ Store into Dn\n"); | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   OpEnd(); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------- Opcodes 0xc140+ ---------------------
 | ||||
| // Emit a Exchange opcode:
 | ||||
| // 1100ttt1 01000sss  exg ds,dt
 | ||||
| // 1100ttt1 01001sss  exg as,at
 | ||||
| // 1100ttt1 10001sss  exg as,dt
 | ||||
| int OpExg(int op) | ||||
| { | ||||
|   int use=0,type=0; | ||||
| 
 | ||||
|   type=op&0xf8; | ||||
| 
 | ||||
|   if (type!=0x40 && type!=0x48 && type!=0x88) return 1; // Not an exg opcode
 | ||||
| 
 | ||||
|   use=op&0xf1f8; // Use same opcode for all values
 | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler
 | ||||
| 
 | ||||
|   OpStart(op); Cycles=6; | ||||
| 
 | ||||
|   ot("  and r2,r8,#0x0e00 ;@ Find T register\n"); | ||||
|   ot("  and r3,r8,#0x000f ;@ Find S register\n"); | ||||
|   if (type==0x48) ot("  orr r2,r2,#0x1000 ;@ T is an address register\n"); | ||||
|   ot("\n"); | ||||
|   ot("  ldr r0,[r7,r2,lsr #7] ;@ Get T\n"); | ||||
|   ot("  ldr r1,[r7,r3,lsl #2] ;@ Get S\n"); | ||||
|   ot("\n"); | ||||
|   ot("  str r0,[r7,r3,lsl #2] ;@ T->S\n"); | ||||
|   ot("  str r1,[r7,r2,lsr #7] ;@ S->T\n");   | ||||
|   ot("\n"); | ||||
| 
 | ||||
|   OpEnd(); | ||||
|    | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // ------------------------- movep -------------------------------
 | ||||
| // 0000ddd1 0z001sss
 | ||||
| // 0000sss1 1z001ddd (to mem)
 | ||||
| int OpMovep(int op) | ||||
| { | ||||
|   int ea=0,rea=0; | ||||
|   int size=1,use=0,dir,aadd=0; | ||||
| 
 | ||||
|   use=op&0xf1f8; | ||||
|   if (op!=use) { OpUse(op,use); return 0; } // Use existing handler (for all dests, srcs)
 | ||||
| 
 | ||||
|   // Get EA
 | ||||
|   ea = (op&0x0007)|0x28; | ||||
|   rea= (op&0x0e00)>>9; | ||||
|   dir = (op>>7)&1; | ||||
| 
 | ||||
|   // Find size extension
 | ||||
|   if(op&0x0040) size=2; | ||||
| 
 | ||||
|   OpStart(op,ea); | ||||
|    | ||||
|   if(dir) // reg to mem
 | ||||
|   { | ||||
|     EaCalcReadNoSE(-1,11,rea,size,0x0e00); | ||||
| 
 | ||||
|     EaCalc(8,0x000f,ea,size); | ||||
|     if(size==2) { // if operand is long
 | ||||
|       ot("  mov r1,r11,lsr #24 ;@ first byte\n"); | ||||
|       EaWrite(8,1,ea,0,0x000f); // store first byte
 | ||||
|       ot("  add r0,r8,#%i\n",(aadd+=2)); | ||||
|       ot("  mov r1,r11,lsr #16 ;@ second byte\n"); | ||||
|       EaWrite(0,1,ea,0,0x000f); // store second byte
 | ||||
|       ot("  add r0,r8,#%i\n",(aadd+=2)); | ||||
|     } else { | ||||
|       ot("  mov r0,r8\n"); | ||||
|     } | ||||
|     ot("  mov r1,r11,lsr #8 ;@ first or third byte\n"); | ||||
|     EaWrite(0,1,ea,0,0x000f); | ||||
|     ot("  add r0,r8,#%i\n",(aadd+=2)); | ||||
|     ot("  and r1,r11,#0xff\n"); | ||||
|     EaWrite(0,1,ea,0,0x000f); | ||||
|   } | ||||
|   else // mem to reg
 | ||||
|   { | ||||
|     EaCalc(6,0x000f,ea,size,1); | ||||
|     EaRead(6,11,ea,0,0x000f,1); // read first byte
 | ||||
|     ot("  add r0,r6,#2\n"); | ||||
|     EaRead(0,1,ea,0,0x000f,1); // read second byte
 | ||||
|     if(size==2) { // if operand is long
 | ||||
|       ot("  orr r11,r11,r1,lsr #8 ;@ second byte\n"); | ||||
|       ot("  add r0,r6,#4\n"); | ||||
|       EaRead(0,1,ea,0,0x000f,1); | ||||
|       ot("  orr r11,r11,r1,lsr #16 ;@ third byte\n"); | ||||
|       ot("  add r0,r6,#6\n"); | ||||
|       EaRead(0,1,ea,0,0x000f,1); | ||||
|       ot("  orr r1,r11,r1,lsr #24 ;@ fourth byte\n"); | ||||
|     } else { | ||||
|       ot("  orr r1,r11,r1,lsr #8 ;@ second byte\n"); | ||||
|     } | ||||
|     // store the result
 | ||||
|     EaCalc(0,0x0e00,rea,size,1); | ||||
|     EaWrite(0,1,rea,size,0x0e00,1); | ||||
|     ot("  ldr r6,[r7,#0x54]\n"); | ||||
|   } | ||||
| 
 | ||||
|   Cycles=(size==2)?24:16; | ||||
|   OpEnd(ea); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| // Emit a Stop/Reset opcodes, 01001110 011100t0 imm
 | ||||
| int OpStopReset(int op) | ||||
| { | ||||
|   int type=(op>>1)&1; // stop/reset
 | ||||
| 
 | ||||
|   OpStart(op,0,0,0,1); | ||||
| 
 | ||||
|   if(type) { | ||||
|     // copy immediate to SR, stop the CPU and eat all remaining cycles.
 | ||||
|     ot("  ldrh r0,[r4],#2 ;@ Fetch the immediate\n"); | ||||
|     OpRegToFlags(1); | ||||
|     SuperChange(op,0); | ||||
| 
 | ||||
|     ot("\n"); | ||||
| 
 | ||||
|     ot("  ldr r0,[r7,#0x58]\n"); | ||||
|     ot("  mov r5,#0 ;@ eat cycles\n"); | ||||
|     ot("  orr r0,r0,#1 ;@ stopped\n"); | ||||
|     ot("  str r0,[r7,#0x58]\n"); | ||||
|     ot("\n"); | ||||
| 
 | ||||
|     Cycles = 4; | ||||
|     ot("\n"); | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     Cycles = 132; | ||||
| #if USE_RESET_CALLBACK | ||||
|     ot("  str r4,[r7,#0x40] ;@ Save PC\n"); | ||||
|     ot("  mov r1,r10,lsr #28\n"); | ||||
|     ot("  strb r1,[r7,#0x46] ;@ Save Flags (NZCV)\n"); | ||||
|     ot("  str r5,[r7,#0x5c] ;@ Save Cycles\n"); | ||||
|     ot("  ldr r11,[r7,#0x90] ;@ ResetCallback\n"); | ||||
|     ot("  tst r11,r11\n"); | ||||
|     ot("  movne lr,pc\n"); | ||||
|     ot("  bxne r11 ;@ call ResetCallback if it is defined\n"); | ||||
|     ot("  ldrb r10,[r7,#0x46] ;@ r10 = Load Flags (NZCV)\n"); | ||||
|     ot("  ldr r5,[r7,#0x5c] ;@ Load Cycles\n"); | ||||
|     ot("  ldr r4,[r7,#0x40] ;@ Load PC\n"); | ||||
|     ot("  mov r10,r10,lsl #28\n"); | ||||
|     ot("\n"); | ||||
| #endif | ||||
|   } | ||||
| 
 | ||||
|   OpEnd(); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
|  | @ -1,111 +0,0 @@ | |||
| 
 | ||||
| #include <stdio.h> | ||||
| #include <stdarg.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| 
 | ||||
| #ifndef CONFIG_FILE | ||||
| #define CONFIG_FILE "config.h" | ||||
| #endif | ||||
| #include CONFIG_FILE | ||||
| 
 | ||||
| // Disa.c
 | ||||
| #include "Disa/Disa.h" | ||||
| 
 | ||||
| // Ea.cpp
 | ||||
| extern int earead_check_addrerr; | ||||
| extern int eawrite_check_addrerr; | ||||
| extern int g_jmp_cycle_table[]; | ||||
| extern int g_jsr_cycle_table[]; | ||||
| extern int g_lea_cycle_table[]; | ||||
| extern int g_pea_cycle_table[]; | ||||
| extern int g_movem_cycle_table[]; | ||||
| int Ea_add_ns(int *tab, int ea); // add nonstandard EA cycles
 | ||||
| int EaCalc(int a,int mask,int ea,int size,int top=0,int sign_extend=1); // 6
 | ||||
| int EaRead(int a,int v,int ea,int size,int mask,int top=0,int sign_extend=1); // 7
 | ||||
| int EaCalcRead(int r_ea,int r,int ea,int size,int mask,int sign_extend=1); // 6
 | ||||
| int EaCalcReadNoSE(int r_ea,int r,int ea,int size,int mask); | ||||
| int EaCanRead(int ea,int size); | ||||
| int EaWrite(int a,int v,int ea,int size,int mask,int top=0,int sign_extend_ea=1); | ||||
| int EaCanWrite(int ea); | ||||
| int EaAn(int ea); | ||||
| 
 | ||||
| // Main.cpp
 | ||||
| extern int *CyJump;   // Jump table
 | ||||
| extern int  ms;       // If non-zero, output in Microsoft ARMASM format
 | ||||
| extern const char * const Narm[4]; // Normal ARM Extensions for operand sizes 0,1,2
 | ||||
| extern const char * const Sarm[4]; // Sign-extend ARM Extensions for operand sizes 0,1,2
 | ||||
| extern int  Cycles;   // Current cycles for opcode
 | ||||
| extern int  pc_dirty; // something changed PC during processing
 | ||||
| extern int  arm_op_count; // for stats
 | ||||
| void ot(const char *format, ...); | ||||
| void ltorg(); | ||||
| int MemHandler(int type,int size,int addrreg=0,int need_addrerr_check=1); | ||||
| void FlushPC(void); | ||||
| 
 | ||||
| // OpAny.cpp
 | ||||
| extern int g_op; | ||||
| extern int opend_op_changes_cycles, opend_check_interrupt, opend_check_trace; | ||||
| int OpGetFlags(int subtract,int xbit,int sprecialz=0); | ||||
| void OpUse(int op,int use); | ||||
| void OpStart(int op,int sea=0,int tea=0,int op_changes_cycles=0,int supervisor_check=0); | ||||
| void OpEnd(int sea=0,int tea=0); | ||||
| int OpBase(int op,int size,int sepa=0); | ||||
| void OpAny(int op); | ||||
| 
 | ||||
| //----------------------
 | ||||
| // OpArith.cpp
 | ||||
| int OpArith(int op); | ||||
| int OpLea(int op); | ||||
| int OpAddq(int op); | ||||
| int OpArithReg(int op); | ||||
| int OpMul(int op); | ||||
| int OpAbcd(int op); | ||||
| int OpNbcd(int op); | ||||
| int OpAritha(int op); | ||||
| int OpAddx(int op); | ||||
| int OpCmpEor(int op); | ||||
| int OpCmpm(int op); | ||||
| int OpChk(int op); | ||||
| int GetXBit(int subtract); | ||||
| 
 | ||||
| // OpBranch.cpp
 | ||||
| void OpPush32(); | ||||
| void OpPushSr(int high); | ||||
| int OpTrap(int op); | ||||
| int OpLink(int op); | ||||
| int OpUnlk(int op); | ||||
| int Op4E70(int op); | ||||
| int OpJsr(int op); | ||||
| int OpBranch(int op); | ||||
| int OpDbra(int op); | ||||
| 
 | ||||
| // OpLogic.cpp
 | ||||
| int OpBtstReg(int op); | ||||
| int OpBtstImm(int op); | ||||
| int OpNeg(int op); | ||||
| int OpSwap(int op); | ||||
| int OpTst(int op); | ||||
| int OpExt(int op); | ||||
| int OpSet(int op); | ||||
| int OpAsr(int op); | ||||
| int OpAsrEa(int op); | ||||
| int OpTas(int op, int gen_special=0); | ||||
| 
 | ||||
| // OpMove.cpp
 | ||||
| int OpMove(int op); | ||||
| int OpLea(int op); | ||||
| void OpFlagsToReg(int high); | ||||
| void OpRegToFlags(int high,int srh_reg=0); | ||||
| int OpMoveSr(int op); | ||||
| int OpArithSr(int op); | ||||
| int OpPea(int op); | ||||
| int OpMovem(int op); | ||||
| int OpMoveq(int op); | ||||
| int OpMoveUsp(int op); | ||||
| int OpExg(int op); | ||||
| int OpMovep(int op); | ||||
| int OpStopReset(int op); | ||||
| void SuperEnd(void); | ||||
| void SuperChange(int op,int srh_reg=-1); | ||||
| 
 | ||||
|  | @ -1,182 +0,0 @@ | |||
| 
 | ||||
| 
 | ||||
| /**
 | ||||
|  * Cyclone 68000 configuration file | ||||
| **/ | ||||
| 
 | ||||
| 
 | ||||
| /*
 | ||||
|  * If this option is enabled, Microsoft ARMASM compatible output is generated | ||||
|  * (output file -  Cyclone.asm). Otherwise GNU as syntax is used (Cyclone.s). | ||||
|  */ | ||||
| #define USE_MS_SYNTAX               0 | ||||
| 
 | ||||
| /*
 | ||||
|  * Enable this option if you are going to use Cyclone to emulate Genesis / | ||||
|  * Mega Drive system. As VDP chip in these systems had control of the bus, | ||||
|  * several instructions were acting differently, for example TAS did'n have | ||||
|  * the write-back phase. That will be emulated, if this option is enabled. | ||||
|  */ | ||||
| #define CYCLONE_FOR_GENESIS         0 | ||||
| 
 | ||||
| /*
 | ||||
|  * This option compresses Cyclone's jumptable. Because of this the executable | ||||
|  * will be smaller and load slightly faster and less relocations will be needed. | ||||
|  * This also fixes the crash problem with 0xfffe and 0xffff opcodes. | ||||
|  * Warning: if you enable this, you MUST call CycloneInit() before calling | ||||
|  * CycloneRun(), or else it will crash. | ||||
|  */ | ||||
| #define COMPRESS_JUMPTABLE          1 | ||||
| 
 | ||||
| /*
 | ||||
|  * Address mask for memory hadlers. The bits set will be masked out of address | ||||
|  * parameter, which is passed to r/w memory handlers. | ||||
|  * Using 0xff000000 means that only 24 least significant bits should be used. | ||||
|  * Set to 0 if you want to mask unused address bits in the memory handlers yourself. | ||||
|  */ | ||||
| #define MEMHANDLERS_ADDR_MASK       0 | ||||
| 
 | ||||
| /*
 | ||||
|  * Cyclone keeps the 4 least significant bits of SR, PC+membase and it's cycle | ||||
|  * counter in ARM registers instead of the context for performance reasons. If you for | ||||
|  * any reason need to access them in your memory handlers, enable the options below, | ||||
|  * otherwise disable them to improve performance. | ||||
|  * | ||||
|  * MEMHANDLERS_NEED_PC updates .pc context field with PC value effective at the time | ||||
|  * when memhandler was called (opcode address + 2-10 bytes). | ||||
|  * MEMHANDLERS_NEED_PREV_PC updates .prev_pc context field to currently executed | ||||
|  * opcode address + 2. | ||||
|  * Note that .pc and .prev_pc values are always real pointers to memory, so you must | ||||
|  * subtract .membase to get M68k PC value. | ||||
|  * | ||||
|  * Warning: updating PC in memhandlers is dangerous, as Cyclone may internally | ||||
|  * increment the PC before fetching the next instruction and continue executing | ||||
|  * at wrong location. It's better to wait until Cyclone CycloneRun() finishes. | ||||
|  * | ||||
|  * Warning: if you enable MEMHANDLERS_CHANGE_CYCLES, you must also enable | ||||
|  * MEMHANDLERS_NEED_CYCLES, or else Cyclone will keep reloading the same cycle | ||||
|  * count and this will screw timing (if not cause a deadlock). | ||||
|  */ | ||||
| #define MEMHANDLERS_NEED_PC         0 | ||||
| #define MEMHANDLERS_NEED_PREV_PC    0 | ||||
| #define MEMHANDLERS_NEED_FLAGS      0 | ||||
| #define MEMHANDLERS_NEED_CYCLES     0 | ||||
| #define MEMHANDLERS_CHANGE_PC       0 | ||||
| #define MEMHANDLERS_CHANGE_FLAGS    0 | ||||
| #define MEMHANDLERS_CHANGE_CYCLES   0 | ||||
| 
 | ||||
| /*
 | ||||
|  * If the following macro is defined, Cyclone no longer calls read*, write*, | ||||
|  * fetch* and checkpc from it's context, it calls these functions directly | ||||
|  * instead, prefixed with prefix selected below. For example, if | ||||
|  * MEMHANDLERS_DIRECT_PREFIX is set to cyclone_, it will call cyclone_read8 | ||||
|  * on byte reads. | ||||
|  * This is to avoid indirect jumps, which are slower. It also saves one ARM | ||||
|  * instruction. | ||||
|  */ | ||||
| /* MEMHANDLERS_DIRECT_PREFIX "cyclone_" */ | ||||
| 
 | ||||
| /*
 | ||||
|  * If enabled, Cyclone will call .IrqCallback routine from it's context whenever it | ||||
|  * acknowledges an IRQ. IRQ level (.irq) is not cleared automatically, do this in your | ||||
|  * handler if needed. | ||||
|  * This function must either return vector number to use for interrupt exception, | ||||
|  * CYCLONE_INT_ACK_AUTOVECTOR to use autovector (this is the most common case), or | ||||
|  * CYCLONE_INT_ACK_SPURIOUS (least common case). | ||||
|  * If disabled, it simply uses appropriate autovector, clears the IRQ level and | ||||
|  * continues execution. | ||||
|  */ | ||||
| #define USE_INT_ACK_CALLBACK        0 | ||||
| 
 | ||||
| /*
 | ||||
|  * Enable this if you need old PC, flags or cycles; | ||||
|  * or you change cycles in your IrqCallback function. | ||||
|  */ | ||||
| #define INT_ACK_NEEDS_STUFF         0 | ||||
| #define INT_ACK_CHANGES_CYCLES      0 | ||||
| 
 | ||||
| /*
 | ||||
|  * If enabled, .ResetCallback is called from the context, whenever RESET opcode is | ||||
|  * encountered. All context members are valid and can be changed. | ||||
|  * If disabled, RESET opcode acts as an NOP. | ||||
|  */ | ||||
| #define USE_RESET_CALLBACK          0 | ||||
| 
 | ||||
| /*
 | ||||
|  * If enabled, UnrecognizedCallback is called if an invalid opcode is | ||||
|  * encountered. All context members are valid and can be changed. The handler | ||||
|  * should return zero if you want Cyclone to gererate "Illegal Instruction" | ||||
|  * exception after this, or nonzero if not. In the later case you should change | ||||
|  * the PC by yourself, or else Cyclone will keep executing that opcode all over | ||||
|  * again. | ||||
|  * If disabled, "Illegal Instruction" exception is generated and execution is | ||||
|  * continued. | ||||
|  */ | ||||
| #define USE_UNRECOGNIZED_CALLBACK   0 | ||||
| 
 | ||||
| /*
 | ||||
|  * This option will also call UnrecognizedCallback for a-line and f-line | ||||
|  * (0xa*** and 0xf***) opcodes the same way as described above, only appropriate | ||||
|  * exceptions will be generated. | ||||
|  */ | ||||
| #define USE_AFLINE_CALLBACK         0 | ||||
| 
 | ||||
| /*
 | ||||
|  * This makes Cyclone to call checkpc from it's context whenever it changes the PC | ||||
|  * by a large value. It takes and should return the PC value in PC+membase form. | ||||
|  * The flags and cycle counter are not valid in this function. | ||||
|  */ | ||||
| #define USE_CHECKPC_CALLBACK        1 | ||||
| 
 | ||||
| /*
 | ||||
|  * This determines if checkpc() should be called after jumps when 8 and 16 bit | ||||
|  * displacement values were used. | ||||
|  */ | ||||
| #define USE_CHECKPC_OFFSETBITS_16   1 | ||||
| #define USE_CHECKPC_OFFSETBITS_8    0 | ||||
| 
 | ||||
| /*
 | ||||
|  * Call checkpc() after DBcc jumps (which use 16bit displacement). Cyclone prior to | ||||
|  * 0.0087 never did that. | ||||
|  */ | ||||
| #define USE_CHECKPC_DBRA            0 | ||||
| 
 | ||||
| /*
 | ||||
|  * When this option is enabled Cyclone will do two word writes instead of one | ||||
|  * long write when handling MOVE.L or MOVEM.L with pre-decrementing destination, | ||||
|  * as described in Bart Trzynadlowski's doc (http://www.trzy.org/files/68knotes.txt).
 | ||||
|  * Enable this if you are emulating a 16 bit system. | ||||
|  */ | ||||
| #define SPLIT_MOVEL_PD              1 | ||||
| 
 | ||||
| /*
 | ||||
|  * Enable emulation of trace mode. Shouldn't cause any performance decrease, so it | ||||
|  * should be safe to keep this ON. | ||||
|  */ | ||||
| #define EMULATE_TRACE               1 | ||||
| 
 | ||||
| /*
 | ||||
|  * If enabled, address error exception will be generated if 68k code jumps to an | ||||
|  * odd address. Causes very small performance hit (2 ARM instructions for every | ||||
|  * emulated jump/return/exception in normal case). | ||||
|  * Note: checkpc() must not clear least significant bit of rebased address | ||||
|  * for this to work, as checks are performed after calling checkpc(). | ||||
|  */ | ||||
| #define EMULATE_ADDRESS_ERRORS_JUMP 1 | ||||
| 
 | ||||
| /*
 | ||||
|  * If enabled, address error exception will be generated if 68k code tries to | ||||
|  * access a word or longword at an odd address. The performance cost is also 2 ARM | ||||
|  * instructions per access (for address error checks). | ||||
|  */ | ||||
| #define EMULATE_ADDRESS_ERRORS_IO   0 | ||||
| 
 | ||||
| /*
 | ||||
|  * If an address error happens during another address error processing, | ||||
|  * the processor halts until it is reset (catastrophic system failure, as the manual | ||||
|  * states). This option enables halt emulation. | ||||
|  * Note that this might be not desired if it is known that emulated system should | ||||
|  * never reach this state. | ||||
|  */ | ||||
| #define EMULATE_HALT                0 | ||||
| 
 | ||||
|  | @ -1,44 +0,0 @@ | |||
| 
 | ||||
| 
 | ||||
| /**
 | ||||
|  * Cyclone 68000 configuration file | ||||
|  * | ||||
|  * Used for mamegp2x Cyclone build. | ||||
|  * See config.h in Cyclone directory for option descriptions. | ||||
| **/ | ||||
| 
 | ||||
| 
 | ||||
| #define USE_MS_SYNTAX               0 | ||||
| #define CYCLONE_FOR_GENESIS         0 | ||||
| #define COMPRESS_JUMPTABLE          1 | ||||
| #define MEMHANDLERS_ADDR_MASK       0xff000000 | ||||
| 
 | ||||
| #define MEMHANDLERS_NEED_PC         1 | ||||
| #define MEMHANDLERS_NEED_PREV_PC    1 | ||||
| #define MEMHANDLERS_NEED_FLAGS      0 | ||||
| #define MEMHANDLERS_NEED_CYCLES     1 | ||||
| #define MEMHANDLERS_CHANGE_PC       0 | ||||
| #define MEMHANDLERS_CHANGE_FLAGS    0 | ||||
| #define MEMHANDLERS_CHANGE_CYCLES   1 | ||||
| 
 | ||||
| #define USE_INT_ACK_CALLBACK        1 | ||||
| 
 | ||||
| #define INT_ACK_NEEDS_STUFF         0 | ||||
| #define INT_ACK_CHANGES_CYCLES      0 | ||||
| 
 | ||||
| #define USE_RESET_CALLBACK          0 | ||||
| #define USE_UNRECOGNIZED_CALLBACK   0 | ||||
| #define USE_AFLINE_CALLBACK         0 | ||||
| 
 | ||||
| #define USE_CHECKPC_CALLBACK        1 | ||||
| #define USE_CHECKPC_OFFSETBITS_16   1 | ||||
| #define USE_CHECKPC_OFFSETBITS_8    0 | ||||
| #define USE_CHECKPC_DBRA            0 | ||||
| 
 | ||||
| #define SPLIT_MOVEL_PD              1 | ||||
| 
 | ||||
| #define EMULATE_TRACE               1 | ||||
| #define EMULATE_ADDRESS_ERRORS_JUMP 1 | ||||
| #define EMULATE_ADDRESS_ERRORS_IO   0 | ||||
| #define EMULATE_HALT                0 | ||||
| 
 | ||||
|  | @ -1,44 +0,0 @@ | |||
| 
 | ||||
| 
 | ||||
| /**
 | ||||
|  * Cyclone 68000 configuration file | ||||
|  * | ||||
|  * Used for UAE4ALL Cyclone build. | ||||
|  * See config.h in Cyclone directory for option descriptions. | ||||
| **/ | ||||
| 
 | ||||
| 
 | ||||
| #define USE_MS_SYNTAX               0 | ||||
| #define CYCLONE_FOR_GENESIS         0 | ||||
| #define COMPRESS_JUMPTABLE          1 | ||||
| #define MEMHANDLERS_ADDR_MASK       0 | ||||
| 
 | ||||
| #define MEMHANDLERS_NEED_PC         1 | ||||
| #define MEMHANDLERS_NEED_PREV_PC    0 | ||||
| #define MEMHANDLERS_NEED_FLAGS      0 | ||||
| #define MEMHANDLERS_NEED_CYCLES     1 | ||||
| #define MEMHANDLERS_CHANGE_PC       0 | ||||
| #define MEMHANDLERS_CHANGE_FLAGS    0 | ||||
| #define MEMHANDLERS_CHANGE_CYCLES   1 | ||||
| 
 | ||||
| #define USE_INT_ACK_CALLBACK        1 | ||||
| 
 | ||||
| #define INT_ACK_NEEDS_STUFF         0 | ||||
| #define INT_ACK_CHANGES_CYCLES      0 | ||||
| 
 | ||||
| #define USE_RESET_CALLBACK          0 | ||||
| #define USE_UNRECOGNIZED_CALLBACK   1 | ||||
| #define USE_AFLINE_CALLBACK         1 | ||||
| 
 | ||||
| #define USE_CHECKPC_CALLBACK        1 | ||||
| #define USE_CHECKPC_OFFSETBITS_16   1 | ||||
| #define USE_CHECKPC_OFFSETBITS_8    0 | ||||
| #define USE_CHECKPC_DBRA            0 | ||||
| 
 | ||||
| #define SPLIT_MOVEL_PD              1 | ||||
| 
 | ||||
| #define EMULATE_TRACE               1 | ||||
| #define EMULATE_ADDRESS_ERRORS_JUMP 1 | ||||
| #define EMULATE_ADDRESS_ERRORS_IO   1 | ||||
| #define EMULATE_HALT                0 | ||||
| 
 | ||||
										
											Binary file not shown.
										
									
								
							|  | @ -1,150 +0,0 @@ | |||
| #include <stdio.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| 
 | ||||
| 
 | ||||
| typedef struct { | ||||
| 	unsigned long _dontcare1[4]; | ||||
| 	char signature[4];						// 'EPOC'
 | ||||
| 	unsigned long iCpu;						// 0x1000 = X86, 0x2000 = ARM, 0x4000 = M*Core
 | ||||
| 	unsigned long iCheckSumCode;			// sum of all 32 bit words in .text
 | ||||
| 	unsigned long _dontcare3[5]; | ||||
| 	unsigned long iCodeSize;				// size of code, import address table, constant data and export dir |+30
 | ||||
| 	unsigned long _dontcare4[12]; | ||||
| 	unsigned long iCodeOffset;				// file offset to code section    |+64
 | ||||
| 	unsigned long _dontcare5[2]; | ||||
| 	unsigned long iCodeRelocOffset;			// relocations for code and const |+70
 | ||||
| 	unsigned long iDataRelocOffset;			// relocations for data
 | ||||
| 	unsigned long iPriority;		// priority of this process (EPriorityHigh=450)
 | ||||
| } E32ImageHeader; | ||||
| 
 | ||||
| 
 | ||||
| typedef struct { | ||||
| 	unsigned long iSize;				// size of this relocation section
 | ||||
| 	unsigned long iNumberOfRelocs;		// number of relocations in this section
 | ||||
| } E32RelocSection; | ||||
| 
 | ||||
| 
 | ||||
| typedef struct { | ||||
| 	unsigned long base; | ||||
| 	unsigned long size; | ||||
| } reloc_page_header; | ||||
| 
 | ||||
| 
 | ||||
| // E32Image relocation section consists of a number of pages
 | ||||
| // every page has 8 byte header and a number or 16bit relocation entries
 | ||||
| // entry format:
 | ||||
| // 0x3000 | <12bit_reloc_offset>
 | ||||
| //
 | ||||
| // if we have page_header.base == 0x1000 and a reloc entry 0x3110,
 | ||||
| // it means that 32bit value at offset 0x1110 of .text section
 | ||||
| // is relocatable
 | ||||
| 
 | ||||
| int main(int argc, char *argv[]) | ||||
| { | ||||
| 	FILE *f = 0; | ||||
| 	unsigned char pattern[8] = { 0x12, 0x34, 0x56, 0x78, 0x90, 0x12, 0x34, 0x56 }; | ||||
| 	unsigned char *buff, *p; | ||||
| 	unsigned long patt_offset; // pattern offset in .text section
 | ||||
| 	unsigned long size = 0, i, symbols, insert_pos, *handler; | ||||
| 	unsigned short reloc_entry; | ||||
| 	E32ImageHeader  *header; | ||||
| 	E32RelocSection *reloc_section; | ||||
| 	reloc_page_header *reloc_page; | ||||
| 
 | ||||
| 	if(argc != 3) { | ||||
| 		printf("usage: %s <e32_exe> <nsymbols>\n\n", argv[0]); | ||||
| 		printf("note: this was written to fix a problem caused by as v.2.9-psion-98r2 and shouldn't be used for anything else.\n", argv[0]); | ||||
| 		return 1; | ||||
| 	} | ||||
| 	 | ||||
| 	f = fopen(argv[1], "rb+"); | ||||
| 	if(!f) { | ||||
| 		printf("%s: couldn't open %s\n", argv[0], argv[1]); | ||||
| 		return 2; | ||||
| 	} | ||||
| 
 | ||||
| 	symbols = atoi(argv[2]); | ||||
| 
 | ||||
| 	// read the file
 | ||||
| 	fseek(f,0,SEEK_END); size=ftell(f); fseek(f,0,SEEK_SET); | ||||
| 	buff = (unsigned char *) malloc(size); | ||||
| 	fread(buff,1,size,f); | ||||
| 
 | ||||
| 	header = (E32ImageHeader *) buff; | ||||
| 
 | ||||
| 	if(strncmp(header->signature, "EPOC", 4) || header->iCpu != 0x2000) { | ||||
| 		printf("%s: not a E32 executable image for ARM target.\n", argv[0]); | ||||
| 		fclose(f); | ||||
| 		free(buff); | ||||
| 		return 2; | ||||
| 	} | ||||
| 
 | ||||
| 	// find the pattern
 | ||||
| 	for(i = 0; i < size-8; i++) | ||||
| 		if(memcmp(buff+i, pattern, 8) == 0) break; | ||||
| 	if(i == size-8 || i < 4) { | ||||
| 		printf("%s: failed to find the pattern.\n", argv[0]); | ||||
| 		fclose(f); | ||||
| 		free(buff); | ||||
| 		return 3; | ||||
| 	} | ||||
| 	patt_offset = i - header->iCodeOffset; | ||||
| 
 | ||||
| 	// find suitable reloc section
 | ||||
| 	reloc_section = (E32RelocSection *) (buff + header->iCodeRelocOffset); | ||||
| 	for(i = 0, p = buff+header->iCodeRelocOffset+8; i < reloc_section->iSize; ) { | ||||
| 		reloc_page = (reloc_page_header *) p; | ||||
| 		if(patt_offset - reloc_page->base >= 0 && patt_offset - reloc_page->base < 0x1000 - symbols*4) break; | ||||
| 		i += reloc_page->size; | ||||
| 		p += reloc_page->size; | ||||
| 	} | ||||
| 
 | ||||
| 	if(i >= reloc_section->iSize) { | ||||
| 		printf("%s: suitable reloc section not found.\n", argv[0]); | ||||
| 		fclose(f); | ||||
| 		free(buff); | ||||
| 		return 4; | ||||
| 	} | ||||
| 
 | ||||
| 	// now find the insert pos and update everything
 | ||||
| 	insert_pos = p + reloc_page->size - buff; | ||||
| 	reloc_page->size     += symbols*2; | ||||
| 	reloc_section->iSize += symbols*2; | ||||
| 	reloc_section->iNumberOfRelocs += symbols; | ||||
| 	header->iDataRelocOffset += symbols*2; // data reloc section is now also pushed a little
 | ||||
| 	header->iPriority = 450; // let's boost our priority :)
 | ||||
| 
 | ||||
| 	// replace the placeholders themselves
 | ||||
| 	handler = (unsigned long *) (buff + patt_offset + header->iCodeOffset - 4); | ||||
| 	for(i = 1; i <= symbols; i++) | ||||
| 		*(handler+i) = *handler; | ||||
| 	 | ||||
| 	// recalculate checksum
 | ||||
| 	header->iCheckSumCode = 0; | ||||
| 	for(i = 0, p = buff+header->iCodeOffset; i < header->iCodeSize; i+=4, p+=4) | ||||
| 		header->iCheckSumCode += *(unsigned long *) p; | ||||
| 
 | ||||
| 	// check for possible padding
 | ||||
| 	if(!*(buff+insert_pos-1)) insert_pos -= 2; | ||||
| 
 | ||||
| 	// write all this joy
 | ||||
| 	fseek(f,0,SEEK_SET); | ||||
| 	fwrite(buff, 1, insert_pos, f); | ||||
| 
 | ||||
| 	// write new reloc entries
 | ||||
| 	for(i = 0; i < symbols; i++) { | ||||
| 		handler++; | ||||
| 		reloc_entry = ((unsigned char *) handler - buff - reloc_page->base - header->iCodeOffset) | 0x3000; | ||||
| 		fwrite(&reloc_entry, 1, 2, f); | ||||
| 	} | ||||
| 
 | ||||
| 	// write the remaining data
 | ||||
| 	fwrite(buff+insert_pos, 1, size-insert_pos, f); | ||||
| 
 | ||||
| 	// done at last!
 | ||||
| 	fclose(f); | ||||
| 	free(buff); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
|  | @ -1,133 +0,0 @@ | |||
| #include <stdio.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| 
 | ||||
| #include <windows.h> | ||||
| 
 | ||||
| #define symbols 2 | ||||
| 
 | ||||
| int main(int argc, char *argv[]) | ||||
| { | ||||
| 	FILE *f = 0; | ||||
| 	unsigned char pattern[8] = { 0x12, 0x34, 0x56, 0x78, 0x90, 0x12, 0x34, 0x56 }; | ||||
| 	unsigned char *buff, *p; | ||||
| 	unsigned long patt_offset; // pattern offset in .text section
 | ||||
| 	unsigned long size = 0, i, insert_pos, *handler;//, symbols;
 | ||||
| 	unsigned short reloc_entry; | ||||
| 	IMAGE_BASE_RELOCATION *reloc_page; | ||||
| 	IMAGE_DOS_HEADER  *dos_header; | ||||
| 	IMAGE_FILE_HEADER *file_header; | ||||
| 	IMAGE_SECTION_HEADER *sect_header, *relocsect_header = 0, *codesect_header = 0; | ||||
| 
 | ||||
| 	if(argc != 2) { | ||||
| 		printf("usage: %s <pe_exe_or_app_before_petran>\n\n", argv[0]); | ||||
| 		printf("note: this was written to fix a problem related to Cyclone and as v.2.9-psion-98r2 and shouldn't be used for anything else. See Readme.\n", argv[0]); | ||||
| 		return 1; | ||||
| 	} | ||||
| 	 | ||||
| 	f = fopen(argv[1], "rb+"); | ||||
| 	if(!f) { | ||||
| 		printf("%s: couldn't open %s\n", argv[0], argv[1]); | ||||
| 		return 2; | ||||
| 	} | ||||
| 
 | ||||
| 	//symbols = atoi(argv[2]);
 | ||||
| 
 | ||||
| 	// read the file
 | ||||
| 	fseek(f,0,SEEK_END); size=ftell(f); fseek(f,0,SEEK_SET); | ||||
| 	buff = (unsigned char *) malloc(size); | ||||
| 	fread(buff,1,size,f); | ||||
| 
 | ||||
| 	dos_header = (IMAGE_DOS_HEADER *)      buff; | ||||
| 	file_header= (IMAGE_FILE_HEADER *)    (buff+dos_header->e_lfanew+4); | ||||
| 	sect_header= (IMAGE_SECTION_HEADER *) (buff+dos_header->e_lfanew+4+sizeof(IMAGE_FILE_HEADER)+sizeof(IMAGE_OPTIONAL_HEADER32)); | ||||
| 
 | ||||
| 	if(size < 0x500 || dos_header->e_magic != IMAGE_DOS_SIGNATURE || | ||||
| 	   *(DWORD *)(buff+dos_header->e_lfanew) != IMAGE_NT_SIGNATURE || file_header->Machine != 0x0A00) { | ||||
| 		printf("%s: not a PE executable image for ARM target.\n", argv[0]); | ||||
| 		fclose(f); | ||||
| 		free(buff); | ||||
| 		return 2; | ||||
| 	} | ||||
| 
 | ||||
| 	// scan all sections for data and reloc sections
 | ||||
| 	for(i = 0; i < file_header->NumberOfSections; i++, sect_header++) { | ||||
| 		     if(strncmp(sect_header->Name, ".text",  5) == 0) codesect_header  = sect_header; | ||||
| 		else if(strncmp(sect_header->Name, ".reloc", 6) == 0) relocsect_header = sect_header; | ||||
| 	} | ||||
| 	 | ||||
| 	if(!codesect_header || !relocsect_header) { | ||||
| 		printf("%s: failed to find reloc and/or data section.\n", argv[0]); | ||||
| 		fclose(f); | ||||
| 		free(buff); | ||||
| 		return 3; | ||||
| 	} | ||||
| 
 | ||||
| 	if(relocsect_header != sect_header-1) { | ||||
| 		printf("%s: bug: reloc section is not last, this is unexpected and not supported.\n", argv[0]); | ||||
| 		fclose(f); | ||||
| 		free(buff); | ||||
| 		return 4; | ||||
| 	} | ||||
| 
 | ||||
| 	// find the pattern
 | ||||
| 	for(i = codesect_header->PointerToRawData; i < size-8; i+=2) | ||||
| 		if(memcmp(buff+i, pattern, 8) == 0) break; | ||||
| 	if(i == size-8 || i < 4) { | ||||
| 		printf("%s: failed to find the pattern.\n", argv[0]); | ||||
| 		fclose(f); | ||||
| 		free(buff); | ||||
| 		return 5; | ||||
| 	} | ||||
| 
 | ||||
| 	// calculate pattern offset in RVA (relative virtual address)
 | ||||
| 	patt_offset = i - codesect_header->PointerToRawData + codesect_header->VirtualAddress; | ||||
| 
 | ||||
| 	// replace the placeholders themselves
 | ||||
| 	handler = (unsigned long *) (buff + i - 4); | ||||
| 	for(i = 1; i <= symbols; i++) | ||||
| 		*(handler+i) = *handler; | ||||
| 
 | ||||
| 	// find suitable reloc section
 | ||||
| 	for(i = 0, p = buff+relocsect_header->PointerToRawData; i < relocsect_header->SizeOfRawData; ) { | ||||
| 		reloc_page = (IMAGE_BASE_RELOCATION *) p; | ||||
| 		if(patt_offset - reloc_page->VirtualAddress >= 0 && patt_offset - reloc_page->VirtualAddress < 0x1000 - symbols*2) break; | ||||
| 		i += reloc_page->SizeOfBlock; | ||||
| 		p += reloc_page->SizeOfBlock; | ||||
| 	} | ||||
| 
 | ||||
| 	if(i >= relocsect_header->SizeOfRawData) { | ||||
| 		printf("%s: suitable reloc section not found.\n", argv[0]); | ||||
| 		fclose(f); | ||||
| 		free(buff); | ||||
| 		return 6; | ||||
| 	} | ||||
| 
 | ||||
| 	// now find the insert pos and update everything
 | ||||
| 	insert_pos = p + reloc_page->SizeOfBlock - buff; | ||||
| 	reloc_page->SizeOfBlock += symbols*2; | ||||
| 	relocsect_header->SizeOfRawData += symbols*2; | ||||
| 
 | ||||
| 	// check for possible padding
 | ||||
| 	if(!*(buff+insert_pos-1)) insert_pos -= 2; | ||||
| 
 | ||||
| 	// write all this joy
 | ||||
| 	fseek(f,0,SEEK_SET); | ||||
| 	fwrite(buff, 1, insert_pos, f); | ||||
| 
 | ||||
| 	// write new reloc entries
 | ||||
| 	for(i = 0; i < symbols; i++) { | ||||
| 		handler++; | ||||
| 		reloc_entry = (unsigned short)(((unsigned char *) handler - buff) - reloc_page->VirtualAddress - codesect_header->PointerToRawData + codesect_header->VirtualAddress) | 0x3000; // quite nasty
 | ||||
| 		fwrite(&reloc_entry, 1, 2, f); | ||||
| 	} | ||||
| 
 | ||||
| 	// write the remaining data
 | ||||
| 	fwrite(buff+insert_pos, 1, size-insert_pos, f); | ||||
| 
 | ||||
| 	// done at last!
 | ||||
| 	fclose(f); | ||||
| 	free(buff); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
|  | @ -1,42 +0,0 @@ | |||
| *update* | ||||
| Use the "compress jumtable" Cyclone config.h option to fix this issue | ||||
| (no patcher will be needed then). | ||||
| 
 | ||||
| 
 | ||||
| There is a problem with Cyclone on symbian platform: | ||||
| GNU as generates COFF object files, which allow max of 0xFFFF (65535) relocation | ||||
| entries. Cyclone uses a jumptable of 0x10000 (65536, 1 for every opcode-word) | ||||
| antries. When the executable is loaded, all jumptable entries must be relocated | ||||
| to point to right code location. Because of this limitation, Cyclone's jumptable is | ||||
| incomplete (misses 2 entries), and if M68k opcodes 0xFFFE or 0xFFFF are ever | ||||
| encoundered when emulating, your emulator will crash. | ||||
| 
 | ||||
| I have written a little patcher to fix that. It writes those two missing entries and | ||||
| marks them as relocatable. Placeholders must not be deleted just after the jumttable | ||||
| in the Cyclone source code. | ||||
| 
 | ||||
| This version works with intermediate PE executable, which is used both for APPs and EXEs, | ||||
| and is produced by gcc toolkit just before running petran. So it's best to insert | ||||
| this in your makefile, in the rule which builds your APP/EXE, just after last 'ld' | ||||
| statement, for example: | ||||
| 
 | ||||
| $(EPOCTRGUREL)\PICODRIVEN.APP : $(EPOCBLDUREL)\PICODRIVEN.in $(EPOCSTATLINKUREL)\EDLL.LIB $(LIBSUREL) | ||||
| 	... | ||||
| 	ld  -s -e _E32Dll -u _E32Dll --dll \ | ||||
| 		"$(EPOCBLDUREL)\PICODRIVEN.exp" \ | ||||
| 		-Map "$(EPOCTRGUREL)\PICODRIVEN.APP.map" -o "$(EPOCBLDUREL)\PICODRIVEN.APP" \ | ||||
| 		"$(EPOCSTATLINKUREL)\EDLL.LIB" --whole-archive "$(EPOCBLDUREL)\PICODRIVEN.in" \ | ||||
| 		--no-whole-archive $(LIBSUREL) $(USERLDFLAGS) | ||||
| 	-$(ERASE) "$(EPOCBLDUREL)\PICODRIVEN.exp" | ||||
| 	 | ||||
| 	patchtable_symb2 "$(EPOCBLDUREL)\PICODRIVEN.APP" | ||||
| 
 | ||||
| 	petran  "$(EPOCBLDUREL)\PICODRIVEN.APP" "$@" \ | ||||
| 		 -nocall -uid1 0x10000079 -uid2 0x100039ce -uid3 0x1000c193 | ||||
| 	-$(ERASE) "$(EPOCBLDUREL)\PICODRIVEN.APP" | ||||
| 	perl -S ecopyfile.pl "$@" "PICODRIVEN.APP" | ||||
| 
 | ||||
| 
 | ||||
| This is also compatible with ECompXL. | ||||
| 
 | ||||
| To test if this thing worked, you can load crash_cyclone.bin in your emulator. | ||||
|  | @ -1,42 +0,0 @@ | |||
| CFLAGS = -Wall | ||||
| ifdef CONFIG_FILE | ||||
| CFLAGS += -DCONFIG_FILE=\"$(CONFIG_FILE)\" | ||||
| endif | ||||
| 
 | ||||
| all : cyclone.s | ||||
| 
 | ||||
| cyclone.s : Cyclone.exe | ||||
| 	./Cyclone.exe | ||||
| 
 | ||||
| Cyclone.exe : Main.o Ea.o OpAny.o OpArith.o OpBranch.o OpLogic.o Disa.o OpMove.o | ||||
| 	$(CC) $^ -o $@ -lstdc++ | ||||
| 
 | ||||
| Main.o : ../Main.cpp ../app.h | ||||
| 	$(CC) $(CFLAGS) ../Main.cpp -c -o $@ | ||||
| 
 | ||||
| Ea.o : ../Ea.cpp ../app.h | ||||
| 	$(CC) $(CFLAGS) ../Ea.cpp -c -o $@ | ||||
| 
 | ||||
| OpAny.o : ../OpAny.cpp ../app.h | ||||
| 	$(CC) $(CFLAGS) ../OpAny.cpp -c -o $@ | ||||
| 
 | ||||
| OpArith.o : ../OpArith.cpp ../app.h | ||||
| 	$(CC) $(CFLAGS) ../OpArith.cpp -c -o $@ | ||||
| 
 | ||||
| OpBranch.o : ../OpBranch.cpp ../app.h | ||||
| 	$(CC) $(CFLAGS) ../OpBranch.cpp -c -o $@ | ||||
| 
 | ||||
| OpLogic.o : ../OpLogic.cpp ../app.h | ||||
| 	$(CC) $(CFLAGS) ../OpLogic.cpp -c -o $@ | ||||
| 
 | ||||
| OpMove.o : ../OpMove.cpp ../app.h | ||||
| 	$(CC) $(CFLAGS) ../OpMove.cpp -c -o $@ | ||||
| 
 | ||||
| Disa.o : ../Disa/Disa.c ../Disa/Disa.h | ||||
| 	$(CC) $(CFLAGS) ../Disa/Disa.c -c -o $@ | ||||
| 
 | ||||
| ../app.h : ../config.h | ||||
| 
 | ||||
| clean : | ||||
| 	$(RM) *.o Cyclone.exe Cyclone.s | ||||
| 
 | ||||
|  | @ -1,59 +0,0 @@ | |||
| # Makefile for MS Visual C | ||||
| 
 | ||||
| CPP=cl.exe | ||||
| CPP_PROJ=/nologo /W4 /O2 /D "_CRT_SECURE_NO_WARNINGS" /c  | ||||
| 
 | ||||
| LINK32=link.exe | ||||
| LINK32_FLAGS=user32.lib /nologo /subsystem:console /machine:I386 /out:Cyclone.exe | ||||
| 
 | ||||
| 
 | ||||
| ALL : cyclone.s | ||||
| 
 | ||||
| cyclone.s : Cyclone.exe | ||||
| 	Cyclone.exe | ||||
| 
 | ||||
| Cyclone.exe : Main.obj Ea.obj OpAny.obj OpArith.obj OpBranch.obj OpLogic.obj Disa.obj OpMove.obj | ||||
| 	$(LINK32) Main.obj Ea.obj OpAny.obj OpArith.obj OpBranch.obj OpLogic.obj Disa.obj OpMove.obj $(LINK32_FLAGS) | ||||
| 
 | ||||
| Main.obj : ..\Main.cpp ..\app.h | ||||
| 	$(CPP) $(CPP_PROJ) ..\Main.cpp | ||||
| 
 | ||||
| Ea.obj : ..\Ea.cpp ..\app.h | ||||
| 	$(CPP) $(CPP_PROJ) ..\Ea.cpp | ||||
| 
 | ||||
| OpAny.obj : ..\OpAny.cpp ..\app.h | ||||
| 	$(CPP) $(CPP_PROJ) ..\OpAny.cpp | ||||
| 
 | ||||
| OpArith.obj : ..\OpArith.cpp ..\app.h | ||||
| 	$(CPP) $(CPP_PROJ) ..\OpArith.cpp | ||||
| 
 | ||||
| OpBranch.obj : ..\OpBranch.cpp ..\app.h | ||||
| 	$(CPP) $(CPP_PROJ) ..\OpBranch.cpp | ||||
| 
 | ||||
| OpLogic.obj : ..\OpLogic.cpp ..\app.h | ||||
| 	$(CPP) $(CPP_PROJ) ..\OpLogic.cpp | ||||
| 
 | ||||
| OpMove.obj : ..\OpMove.cpp ..\app.h | ||||
| 	$(CPP) $(CPP_PROJ) ..\OpMove.cpp | ||||
| 
 | ||||
| Disa.obj : ..\disa\Disa.c ..\disa\Disa.h | ||||
| 	$(CPP) $(CPP_PROJ) ..\disa\Disa.c | ||||
| 
 | ||||
| ..\app.h : ..\config.h | ||||
| 
 | ||||
| 
 | ||||
| CLEAN : | ||||
| 	-@erase "Disa.obj" | ||||
| 	-@erase "Ea.obj" | ||||
| 	-@erase "Main.obj" | ||||
| 	-@erase "OpAny.obj" | ||||
| 	-@erase "OpArith.obj" | ||||
| 	-@erase "OpBranch.obj" | ||||
| 	-@erase "OpLogic.obj" | ||||
| 	-@erase "OpMove.obj" | ||||
| 	-@erase "vc60.idb" | ||||
| 	-@erase "vc60.pch" | ||||
| 	-@erase "Cyclone.exe" | ||||
| 	-@erase "Cyclone.asm" | ||||
| 	-@erase "Cyclone.s" | ||||
| 
 | ||||
|  | @ -1,146 +0,0 @@ | |||
| # Microsoft Developer Studio Project File - Name="cyclone" - Package Owner=<4> | ||||
| # Microsoft Developer Studio Generated Build File, Format Version 6.00 | ||||
| # ** DO NOT EDIT ** | ||||
| 
 | ||||
| # TARGTYPE "Win32 (x86) Console Application" 0x0103 | ||||
| 
 | ||||
| CFG=cyclone - Win32 Debug | ||||
| !MESSAGE This is not a valid makefile. To build this project using NMAKE, | ||||
| !MESSAGE use the Export Makefile command and run | ||||
| !MESSAGE  | ||||
| !MESSAGE NMAKE /f "cyclone.mak". | ||||
| !MESSAGE  | ||||
| !MESSAGE You can specify a configuration when running NMAKE | ||||
| !MESSAGE by defining the macro CFG on the command line. For example: | ||||
| !MESSAGE  | ||||
| !MESSAGE NMAKE /f "cyclone.mak" CFG="cyclone - Win32 Debug" | ||||
| !MESSAGE  | ||||
| !MESSAGE Possible choices for configuration are: | ||||
| !MESSAGE  | ||||
| !MESSAGE "cyclone - Win32 Release" (based on "Win32 (x86) Console Application") | ||||
| !MESSAGE "cyclone - Win32 Debug" (based on "Win32 (x86) Console Application") | ||||
| !MESSAGE  | ||||
| 
 | ||||
| # Begin Project | ||||
| # PROP AllowPerConfigDependencies 0 | ||||
| # PROP Scc_ProjName "" | ||||
| # PROP Scc_LocalPath "" | ||||
| CPP=cl.exe | ||||
| RSC=rc.exe | ||||
| 
 | ||||
| !IF  "$(CFG)" == "cyclone - Win32 Release" | ||||
| 
 | ||||
| # PROP BASE Use_MFC 0 | ||||
| # PROP BASE Use_Debug_Libraries 0 | ||||
| # PROP BASE Output_Dir "Release" | ||||
| # PROP BASE Intermediate_Dir "Release" | ||||
| # PROP BASE Target_Dir "" | ||||
| # PROP Use_MFC 0 | ||||
| # PROP Use_Debug_Libraries 0 | ||||
| # PROP Output_Dir "Release" | ||||
| # PROP Intermediate_Dir "Release" | ||||
| # PROP Ignore_Export_Lib 0 | ||||
| # PROP Target_Dir "" | ||||
| # ADD BASE CPP /nologo /W4 /O2 /D "_CRT_SECURE_NO_WARNINGS" /c | ||||
| # ADD CPP /nologo /W4 /O2 /D "_CRT_SECURE_NO_WARNINGS" /c | ||||
| # ADD BASE RSC /l 0x427 /d "NDEBUG" | ||||
| # ADD RSC /l 0x427 /d "NDEBUG" | ||||
| BSC32=bscmake.exe | ||||
| # ADD BASE BSC32 /nologo | ||||
| # ADD BSC32 /nologo | ||||
| LINK32=link.exe | ||||
| # ADD BASE LINK32 kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /subsystem:console /machine:I386 | ||||
| # ADD LINK32 kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /subsystem:console /machine:I386 /out:"cyclone.exe" | ||||
| 
 | ||||
| !ELSEIF  "$(CFG)" == "cyclone - Win32 Debug" | ||||
| 
 | ||||
| # PROP BASE Use_MFC 0 | ||||
| # PROP BASE Use_Debug_Libraries 1 | ||||
| # PROP BASE Output_Dir "Debug" | ||||
| # PROP BASE Intermediate_Dir "Debug" | ||||
| # PROP BASE Target_Dir "" | ||||
| # PROP Use_MFC 0 | ||||
| # PROP Use_Debug_Libraries 1 | ||||
| # PROP Output_Dir "Debug" | ||||
| # PROP Intermediate_Dir "Debug" | ||||
| # PROP Ignore_Export_Lib 0 | ||||
| # PROP Target_Dir "" | ||||
| # ADD BASE CPP /nologo /W4 /O2 /D "_CRT_SECURE_NO_WARNINGS" /c | ||||
| # ADD CPP /nologo /W4 /O2 /D "_CRT_SECURE_NO_WARNINGS" /c | ||||
| # ADD BASE RSC /l 0x427 /d "_DEBUG" | ||||
| # ADD RSC /l 0x427 /d "_DEBUG" | ||||
| BSC32=bscmake.exe | ||||
| # ADD BASE BSC32 /nologo | ||||
| # ADD BSC32 /nologo | ||||
| LINK32=link.exe | ||||
| # ADD BASE LINK32 kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /subsystem:console /debug /machine:I386 /pdbtype:sept | ||||
| # ADD LINK32 kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /subsystem:console /debug /machine:I386 /out:"cyclone.exe" /pdbtype:sept | ||||
| 
 | ||||
| !ENDIF  | ||||
| 
 | ||||
| # Begin Target | ||||
| 
 | ||||
| # Name "cyclone - Win32 Release" | ||||
| # Name "cyclone - Win32 Debug" | ||||
| # Begin Group "Source Files" | ||||
| 
 | ||||
| # PROP Default_Filter "cpp;c;cxx;rc;def;r;odl;idl;hpj;bat" | ||||
| # Begin Source File | ||||
| 
 | ||||
| SOURCE=..\disa\Disa.c | ||||
| # End Source File | ||||
| # Begin Source File | ||||
| 
 | ||||
| SOURCE=..\Ea.cpp | ||||
| # End Source File | ||||
| # Begin Source File | ||||
| 
 | ||||
| SOURCE=..\Main.cpp | ||||
| # End Source File | ||||
| # Begin Source File | ||||
| 
 | ||||
| SOURCE=..\OpAny.cpp | ||||
| # End Source File | ||||
| # Begin Source File | ||||
| 
 | ||||
| SOURCE=..\OpArith.cpp | ||||
| # End Source File | ||||
| # Begin Source File | ||||
| 
 | ||||
| SOURCE=..\OpBranch.cpp | ||||
| # End Source File | ||||
| # Begin Source File | ||||
| 
 | ||||
| SOURCE=..\OpLogic.cpp | ||||
| # End Source File | ||||
| # Begin Source File | ||||
| 
 | ||||
| SOURCE=..\OpMove.cpp | ||||
| # End Source File | ||||
| # End Group | ||||
| # Begin Group "Header Files" | ||||
| 
 | ||||
| # PROP Default_Filter "h;hpp;hxx;hm;inl" | ||||
| # Begin Source File | ||||
| 
 | ||||
| SOURCE=..\app.h | ||||
| # End Source File | ||||
| # Begin Source File | ||||
| 
 | ||||
| SOURCE=..\config.h | ||||
| # End Source File | ||||
| # Begin Source File | ||||
| 
 | ||||
| SOURCE=..\Cyclone.h | ||||
| # End Source File | ||||
| # Begin Source File | ||||
| 
 | ||||
| SOURCE=..\disa\Disa.h | ||||
| # End Source File | ||||
| # End Group | ||||
| # Begin Group "Resource Files" | ||||
| 
 | ||||
| # PROP Default_Filter "ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe" | ||||
| # End Group | ||||
| # End Target | ||||
| # End Project | ||||
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							|  | @ -1,119 +0,0 @@ | |||
| #include <stdio.h> | ||||
| #include <stdlib.h> | ||||
| #include <time.h> | ||||
| 
 | ||||
| 
 | ||||
| static FILE *f; | ||||
| 
 | ||||
| #define bswap16(x) (x=(unsigned short)((x<<8)|(x>>8))) | ||||
| #define bswap32(x) (x=((x<<24)|((x<<8)&0xff0000)|((x>>8)&0x00ff00)|((unsigned)x>>24))) | ||||
| 
 | ||||
| static void write_op(unsigned short op, unsigned short word0, unsigned short word1, unsigned short word2) | ||||
| { | ||||
| 	bswap16(op); | ||||
| 	bswap16(word0); | ||||
| 	bswap16(word1); | ||||
| 	bswap16(word2); | ||||
| 
 | ||||
| 	fwrite(&op,    1, sizeof(op), f); | ||||
| 	fwrite(&word0, 1, sizeof(word0), f); | ||||
| 	fwrite(&word1, 1, sizeof(word1), f); | ||||
| 	fwrite(&word2, 1, sizeof(word2), f); | ||||
| } | ||||
| 
 | ||||
| static void write32(unsigned int a) | ||||
| { | ||||
| 	bswap32(a); | ||||
| 	fwrite(&a,     1, sizeof(a),  f); | ||||
| } | ||||
| 
 | ||||
| static int op_check(unsigned short op) | ||||
| { | ||||
| 	if ((op&0xf000) == 0x6000) return 0; // Bxx
 | ||||
| 	if ((op&0xf0f8) == 0x50c8) return 0; // DBxx
 | ||||
| 	if ((op&0xff80) == 0x4e80) return 0; // Jsr
 | ||||
| 	if ((op&0xf000) == 0xa000) return 0; // a-line
 | ||||
| 	if ((op&0xf000) == 0xf000) return 0; // f-line
 | ||||
| 	if ((op&0xfff8)==0x4e70&&op!=0x4e71&&op!=0x4e76) return 0; // reset, rte, rts
 | ||||
| 
 | ||||
| 	if ((op&0x3f) >= 0x28) op = (op&~0x3f) | (rand() % 0x28); | ||||
| 	return 1; | ||||
| } | ||||
| 
 | ||||
| static unsigned short safe_rand(void) | ||||
| { | ||||
| 	unsigned short op; | ||||
| 
 | ||||
| 	/* avoid branch opcodes */ | ||||
| 	do | ||||
| 	{ | ||||
| 		op = rand(); | ||||
| 	} | ||||
| 	while (!op_check(op)); | ||||
| 
 | ||||
| 	return op; | ||||
| } | ||||
| 
 | ||||
| int main() | ||||
| { | ||||
| 	int i, op; | ||||
| 
 | ||||
| 	srand(time(0)); | ||||
| 
 | ||||
| 	f = fopen("test_misc2.bin", "wb"); | ||||
| 	if (!f) return 1; | ||||
| 
 | ||||
| 	write32(0x00ff8000); // stack
 | ||||
| 	write32(0x300); // IP
 | ||||
| 
 | ||||
| 	for (i=0x100/4-2; i; i--) | ||||
| 	{ | ||||
| 		write32(0x200+i*4); // exception vectors
 | ||||
| 	} | ||||
| 
 | ||||
| 	for (i=0x100/4; i; i--) | ||||
| 	{ | ||||
| 		write32(0); // pad
 | ||||
| 	} | ||||
| 
 | ||||
| 	for (i=0x100/4; i; i--) | ||||
| 	{ | ||||
| 		write32(0x4e734e73); // fill with rte instructions
 | ||||
| 	} | ||||
| 
 | ||||
| 	for (op = 0; op < 0x10000; op++) | ||||
| 	{ | ||||
| 		if ((op&0xf000) == 0x6000) // Bxx
 | ||||
| 		{ | ||||
| 			if ((op&0x00ff) == 0) | ||||
| 				write_op(op, 6, 0, 0); | ||||
| 		} | ||||
| 		else if ((op&0xf0f8)==0x50c8) // DBxx
 | ||||
| 		{ | ||||
| 			write_op(op, 6, 0, 0); | ||||
| 		} | ||||
| 		else if ((op&0xff80)==0x4e80) // Jsr
 | ||||
| 		{ | ||||
| 			int addr = 0x300 + op*8 + 8; | ||||
| 			if ((op&0x3f) == 0x39) | ||||
| 				write_op(op, addr >> 16, addr & 0xffff, 0); | ||||
| 		} | ||||
| 		else if ((op&0xf000)==0xa000 || (op&0xf000)==0xf000) // a-line, f-line
 | ||||
| 		{ | ||||
| 			if (op != 0xa000 && op != 0xf000) continue; | ||||
| 		} | ||||
| 		else if ((op&0xfff8)==0x4e70&&op!=0x4e71&&op!=0x4e76); // rte, rts, stop, reset
 | ||||
| 		else | ||||
| 		{ | ||||
| 			write_op(op, safe_rand(), safe_rand(), safe_rand()); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	// jump to the beginning
 | ||||
| 	write_op(0x4ef8, 0x300, 0x4ef8, 0x300); | ||||
| 	write_op(0x4ef8, 0x300, 0x4ef8, 0x300); | ||||
| 
 | ||||
| 	fclose(f); | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							|  | @ -1,140 +0,0 @@ | |||
| | Processor:        68K | ||||
| | Target Assembler: 680x0 Assembler by GNU project | ||||
| 
 | ||||
| | ___________________________________________________________________________ | ||||
| 
 | ||||
| | Segment type: Pure code | ||||
| | segment "ROM" | ||||
| dword_0:        .long 0                 | DATA XREF: ROM:00007244r | ||||
|                                         | sub_764E+3Eo ... | ||||
|                                         | initial interrupt stack pointer | ||||
| dword_4:        .long _start            | DATA XREF: ROM:00007248r | ||||
|                                         | ROM:000142C2w | ||||
|                                         | reset initial PC | ||||
| dword_8:        .long 0x4DE             | DATA XREF: sub_20050+B54w | ||||
|                 .long 0x490
 | ||||
|                 .long 0x4AA             | illegal instruction | ||||
|                 .long 0x4C4
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long _trace            | trace | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x548             | Level 1 Interrupt Autovector | ||||
|                 .long 0x548             | 2 = ext interrupt | ||||
|                 .long 0x548
 | ||||
|                 .long 0x592             | 4 = horizontal interrupt? | ||||
|                 .long 0x548
 | ||||
|                 .long 0x594             | 6 = verticai interrupt? | ||||
|                 .long 0x552
 | ||||
| dword_80:       .long 0x45C             | DATA XREF: ROM:00152F29o | ||||
|                                         | trap vector table? trap 0? | ||||
|                 .long 0x1738
 | ||||
|                 .long 0x171C
 | ||||
|                 .long 0x1754
 | ||||
|                 .long 0x1700
 | ||||
|                 .long 0x556
 | ||||
|                 .long 0x57A
 | ||||
|                 .long 0x548
 | ||||
|                 .long 0x548
 | ||||
|                 .long 0x7CE             | 9 | ||||
|                 .long 0x548
 | ||||
|                 .long 0x548
 | ||||
|                 .long 0x548
 | ||||
|                 .long 0x548
 | ||||
|                 .long 0x548
 | ||||
|                 .long 0x548
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
|                 .long 0x4DE
 | ||||
| aSegaGenesis:   .ascii "SEGA GENESIS    " | DATA XREF: ROM:00045C6Ao | ||||
| aCSega1994_jul: .ascii "(C)SEGA 1994.JUL" | ||||
| aDumpedByTsd:   .ascii "Dumped By TSD                                   " | ||||
| aShiningForce2: .ascii "SHINING FORCE 2                                 " | ||||
| aGmMk131500:    .ascii "GM MK-1315 -00" | ||||
|                 .word 0x8921            | checksum | ||||
| aJ:             .ascii "J               " | IO_Support | ||||
|                 .long 0                 | Rom_Start_Adress | ||||
| dword_1A4:      .long 0x1FFFFF          | DATA XREF: sub_28008+F66o | ||||
|                                         | Rom_End_Adress | ||||
|                 .long 0xFF0000          | Ram_Start_Adress | ||||
|                 .long 0xFFFFFF          | Ram_End_Adress | ||||
| aRaa:           .ascii "RA° "<0>" "<0><1><0>" ?"<0xFF> | Modem_Infos | ||||
|                 .ascii "                                        " | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
| aU:             .ascii "U  "            | Countries | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
|                 .byte 0x20 |   | ||||
| _trace: | ||||
|   nop | ||||
|   nop | ||||
|   rte | ||||
| 
 | ||||
| .globl _start
 | ||||
| _start: | ||||
|   move.l   #0xFFFFFFFF, %d0 | ||||
|   move.l   #0xFFFFFFFF, %d1 | ||||
|   move.w   #0xa711, %sr | ||||
|   move.l   #0x1, %d2 | ||||
|   move.l   #0x8000, %d3 | ||||
|   negx.l   %d0 | ||||
|   negx.l   %d1 | ||||
|   move.w   #0x270f, %sr | ||||
|   negx.b   %d2 | ||||
|   negx.w   %d3 | ||||
| _loop: | ||||
|   bra      _loop | ||||
| 
 | ||||
|   nop | ||||
|   nop | ||||
|   nop | ||||
|   nop | ||||
|  | @ -1,3 +0,0 @@ | |||
| 
 | ||||
| void CycloneInitIdle(void); | ||||
| void CycloneFinishIdle(void); | ||||
|  | @ -1,176 +0,0 @@ | |||
| @ vim:filetype=armasm
 | ||||
| 
 | ||||
| @ ranges/opcodes (idle, normal):
 | ||||
| @ 71xx, 73xx  - bne.s (8bit offset)
 | ||||
| @ 75xx, 77xx  - beq.s (8bit offset)
 | ||||
| @ 7dxx, 7fxx  - bra.s (8bit offset)
 | ||||
| 
 | ||||
| .data | ||||
| .align 2
 | ||||
| 
 | ||||
| have_patches: | ||||
|   .word 0
 | ||||
| 
 | ||||
| .equ patch_desc_table_size, 10 | ||||
| 
 | ||||
| patch_desc_table: | ||||
|   .word (0x71fa<<16) | 0x66fa, idle_detector_bcc8, idle_bne, Op6601  @ bne.s
 | ||||
|   .word (0x71f8<<16) | 0x66f8, idle_detector_bcc8, idle_bne, Op6601  @ bne.s
 | ||||
|   .word (0x71f6<<16) | 0x66f6, idle_detector_bcc8, idle_bne, Op6601  @ bne.s
 | ||||
|   .word (0x71f2<<16) | 0x66f2, idle_detector_bcc8, idle_bne, Op6601  @ bne.s
 | ||||
|   .word (0x75fa<<16) | 0x67fa, idle_detector_bcc8, idle_beq, Op6701  @ beq.s
 | ||||
|   .word (0x75f8<<16) | 0x67f8, idle_detector_bcc8, idle_beq, Op6701  @ beq.s
 | ||||
|   .word (0x75f6<<16) | 0x67f6, idle_detector_bcc8, idle_beq, Op6701  @ beq.s
 | ||||
|   .word (0x75f2<<16) | 0x67f2, idle_detector_bcc8, idle_beq, Op6701  @ beq.s
 | ||||
|   .word (0x7dfe<<16) | 0x60fe, idle_detector_bcc8, idle_bra, Op6001  @ bra.s
 | ||||
|   .word (0x7dfc<<16) | 0x60fc, idle_detector_bcc8, idle_bra, Op6001  @ bra.s
 | ||||
| 
 | ||||
| 
 | ||||
| .text | ||||
| .align 2
 | ||||
| 
 | ||||
| 
 | ||||
| .global CycloneInitIdle
 | ||||
| 
 | ||||
| CycloneInitIdle: | ||||
|     ldr     r3, =CycloneJumpTab | ||||
|     ldr     r2, =patch_desc_table | ||||
|     mov     r12,#patch_desc_table_size | ||||
| 
 | ||||
| cii_loop: | ||||
|     ldrh    r0, [r2] | ||||
|     ldr     r1, [r2, #4]           @ detector
 | ||||
|     str     r1, [r3, r0, lsl #2] | ||||
|     ldrh    r0, [r2, #2] | ||||
|     ldr     r1, [r2, #8]           @ idle
 | ||||
|     add     r0, r3, r0, lsl #2 | ||||
|     str     r1, [r0] | ||||
|     ldr     r1, [r2, #12]          @ normal
 | ||||
|     str     r1, [r0, #0x800] | ||||
|     add     r2, r2, #16 | ||||
|     subs    r12,r12,#1 | ||||
|     bgt     cii_loop | ||||
| 
 | ||||
|     ldr     r0, =have_patches | ||||
|     mov     r1, #1 | ||||
|     str     r1, [r0] | ||||
|     bx      lr | ||||
| 
 | ||||
| 
 | ||||
| .global CycloneFinishIdle
 | ||||
| 
 | ||||
| CycloneFinishIdle: | ||||
|     ldr     r0, =have_patches | ||||
|     ldr     r0, [r0] | ||||
|     tst     r0, r0 | ||||
|     bxeq    lr | ||||
| 
 | ||||
|     ldr     r3, =CycloneJumpTab | ||||
|     ldr     r2, =patch_desc_table | ||||
|     mov     r12,#patch_desc_table_size | ||||
| 
 | ||||
| cfi_loop: | ||||
|     ldrh    r0, [r2] | ||||
|     ldr     r1, [r2, #12]         @ normal
 | ||||
|     str     r1, [r3, r0, lsl #2] | ||||
|     ldrh    r0, [r2, #2] | ||||
|     ldr     r1, =Op____ | ||||
|     add     r0, r3, r0, lsl #2 | ||||
|     str     r1, [r0] | ||||
|     str     r1, [r0, #0x800] | ||||
|     add     r2, r2, #16 | ||||
|     subs    r12,r12,#1 | ||||
|     bgt     cfi_loop | ||||
| 
 | ||||
|     ldr     r0, =have_patches | ||||
|     mov     r1, #0 | ||||
|     str     r1, [r0] | ||||
|     bx      lr | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| .macro inc_counter cond | ||||
| @    ldr\cond r0, [r7, #0x60]
 | ||||
| @    mov     r11,lr
 | ||||
| @    sub     r0, r4, r0
 | ||||
| @    sub     r0, r0, #2
 | ||||
| @    bl\cond SekRegisterIdleHit
 | ||||
| @    mov     lr, r11
 | ||||
| .endm | ||||
| 
 | ||||
| idle_bra: | ||||
|     mov     r5, #2 | ||||
|     inc_counter | ||||
|     b       Op6001 | ||||
| 
 | ||||
| idle_bne: | ||||
|     msr     cpsr_flg, r10 | ||||
|     movne   r5, #2                @ 2 is intentional due to strange timing issues
 | ||||
|     inc_counter ne | ||||
|     b       Op6601 | ||||
| 
 | ||||
| idle_beq: | ||||
|     msr     cpsr_flg, r10 ;@ ARM flags = 68000 flags
 | ||||
|     moveq   r5, #2 | ||||
|     inc_counter eq | ||||
|     b       Op6701 | ||||
| 
 | ||||
| 
 | ||||
| @ @@@ @
 | ||||
| 
 | ||||
| idle_detector_bcc8: | ||||
|     ldr     r0, =(Pico+0x22208)   @ Pico.m
 | ||||
|     ldr     r1, =idledet_start_frame | ||||
|     ldr     r0, [r0, #0x1c]       @ ..frame_count
 | ||||
|     ldr     r1, [r1] | ||||
|     cmp     r0, r1 | ||||
|     blt     exit_detector         @ not yet
 | ||||
| 
 | ||||
|     mov     r0, r8, asl #24       @ Shift 8-bit signed offset up...
 | ||||
|     add     r0, r4, r0, asr #24   @ jump dest
 | ||||
|     bic     r0, r0, #1 | ||||
| 
 | ||||
|     mov     r1, #0 | ||||
|     sub     r1, r1, r8, lsl #24 | ||||
|     mov     r1, r1, lsr #24 | ||||
|     sub     r1, r1, #2 | ||||
|     bic     r1, r1, #1 | ||||
| 
 | ||||
|     bl      SekIsIdleCode | ||||
|     tst     r0, r0 | ||||
|     and     r2, r8, #0x00ff | ||||
|     orr     r2, r2, #0x7100 | ||||
|     orreq   r2, r2, #0x0200 | ||||
|     mov     r0, r8, lsr #8 | ||||
|     cmp     r0, #0x66 | ||||
|     orrgt   r2, r2, #0x0400       @ 67xx (beq)
 | ||||
|     orrlt   r2, r2, #0x0c00       @ 60xx (bra)
 | ||||
| 
 | ||||
|     @ r2 = patch_opcode
 | ||||
|     sub     r0, r4, #2 | ||||
|     ldrh    r1, [r0] | ||||
|     mov     r11,r2 | ||||
|     mov     r3, r7 | ||||
|     bl      SekRegisterIdlePatch | ||||
|     cmp     r0, #1                @ 0 - ok to patch, 1 - no patch, 2 - remove detector
 | ||||
|     strlth  r11,[r4, #-2] | ||||
|     ble     exit_detector | ||||
| 
 | ||||
|     @ remove detector from Cyclone
 | ||||
|     mov     r0, r8, lsr #8 | ||||
|     cmp     r0, #0x66 | ||||
|     ldrlt   r1, =Op6001 | ||||
|     ldreq   r1, =Op6601 | ||||
|     ldrgt   r1, =Op6701 | ||||
| 
 | ||||
|     ldr     r3, =CycloneJumpTab | ||||
|     str     r1, [r3, r8, lsl #2] | ||||
|     bx      r1 | ||||
| 
 | ||||
| exit_detector: | ||||
|     mov     r0, r8, lsr #8 | ||||
|     cmp     r0, #0x66 | ||||
|     blt     Op6001 | ||||
|     beq     Op6601 | ||||
|     b       Op6701 | ||||
| 
 | ||||
							
								
								
									
										1
									
								
								cpu/cyclone
									
										
									
									
									
										Submodule
									
								
							
							
						
						
									
										1
									
								
								cpu/cyclone
									
										
									
									
									
										Submodule
									
								
							|  | @ -0,0 +1 @@ | |||
| Subproject commit a6905b4de17f4d772c7742065f2863b77ddf0b31 | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 notaz
						notaz