mirror of
				https://github.com/RaySollium99/picodrive.git
				synced 2025-10-26 08:19:38 -04:00 
			
		
		
		
	32x: memhandler improvements
- use consistent read tables (with write) - use sh2 ptr instead of id - place data_array/peri_regs in sh2 struct
This commit is contained in:
		
							parent
							
								
									c1931173ab
								
							
						
					
					
						commit
						f81107f590
					
				
					 11 changed files with 224 additions and 223 deletions
				
			
		|  | @ -739,7 +739,7 @@ static int emith_xbranch(int cond, void *target, int is_call) | ||||||
| #define emith_sh2_wcall(a, tab) { \ | #define emith_sh2_wcall(a, tab) { \ | ||||||
| 	emith_lsr(12, a, SH2_WRITE_SHIFT); \ | 	emith_lsr(12, a, SH2_WRITE_SHIFT); \ | ||||||
| 	EOP_LDR_REG_LSL(A_COND_AL,12,tab,12,2); \ | 	EOP_LDR_REG_LSL(A_COND_AL,12,tab,12,2); \ | ||||||
| 	emith_ctx_read(2, offsetof(SH2, is_slave)); \ | 	emith_move_r_r(2, CONTEXT_REG); \ | ||||||
| 	emith_jump_reg(12); \ | 	emith_jump_reg(12); \ | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -635,7 +635,7 @@ enum { xAX = 0, xCX, xDX, xBX, xSP, xBP, xSI, xDI }; | ||||||
| 	emith_lsr(xBX, a, SH2_WRITE_SHIFT); \ | 	emith_lsr(xBX, a, SH2_WRITE_SHIFT); \ | ||||||
| 	EMIT_OP_MODRM(0x8b, 0, xBX, 4); \ | 	EMIT_OP_MODRM(0x8b, 0, xBX, 4); \ | ||||||
| 	EMIT_SIB(2, xBX, tab); /* mov ebx, [tab + ebx * 4] */ \ | 	EMIT_SIB(2, xBX, tab); /* mov ebx, [tab + ebx * 4] */ \ | ||||||
| 	emith_ctx_read(arg2_, offsetof(SH2, is_slave)); \ | 	emith_move_r_r(arg2_, CONTEXT_REG); \ | ||||||
| 	emith_jump_reg(xBX); \ | 	emith_jump_reg(xBX); \ | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -349,7 +349,7 @@ static u32  REGPARM(2) (*sh2_drc_read16)(u32 a, SH2 *sh2); | ||||||
| static u32  REGPARM(2) (*sh2_drc_read32)(u32 a, SH2 *sh2); | static u32  REGPARM(2) (*sh2_drc_read32)(u32 a, SH2 *sh2); | ||||||
| static void REGPARM(2) (*sh2_drc_write8)(u32 a, u32 d); | static void REGPARM(2) (*sh2_drc_write8)(u32 a, u32 d); | ||||||
| static void REGPARM(2) (*sh2_drc_write16)(u32 a, u32 d); | static void REGPARM(2) (*sh2_drc_write16)(u32 a, u32 d); | ||||||
| static int  REGPARM(3) (*sh2_drc_write32)(u32 a, u32 d, SH2 *sh2); | static void REGPARM(3) (*sh2_drc_write32)(u32 a, u32 d, SH2 *sh2); | ||||||
| 
 | 
 | ||||||
| // address space stuff
 | // address space stuff
 | ||||||
| static int dr_ctx_get_mem_ptr(u32 a, u32 *mask) | static int dr_ctx_get_mem_ptr(u32 a, u32 *mask) | ||||||
|  | @ -363,6 +363,7 @@ static int dr_ctx_get_mem_ptr(u32 a, u32 *mask) | ||||||
|   } |   } | ||||||
|   else if ((a & 0xfffff000) == 0xc0000000) { |   else if ((a & 0xfffff000) == 0xc0000000) { | ||||||
|     // data array
 |     // data array
 | ||||||
|  |     // FIXME: access sh2->data_array instead
 | ||||||
|     poffs = offsetof(SH2, p_da); |     poffs = offsetof(SH2, p_da); | ||||||
|     *mask = 0xfff; |     *mask = 0xfff; | ||||||
|   } |   } | ||||||
|  | @ -3159,7 +3160,7 @@ void sh2_drc_mem_setup(SH2 *sh2) | ||||||
| { | { | ||||||
|   // fill the convenience pointers
 |   // fill the convenience pointers
 | ||||||
|   sh2->p_bios = sh2->is_slave ? Pico32xMem->sh2_rom_s : Pico32xMem->sh2_rom_m; |   sh2->p_bios = sh2->is_slave ? Pico32xMem->sh2_rom_s : Pico32xMem->sh2_rom_m; | ||||||
|   sh2->p_da = Pico32xMem->data_array[sh2->is_slave]; |   sh2->p_da = sh2->data_array; | ||||||
|   sh2->p_sdram = Pico32xMem->sdram; |   sh2->p_sdram = Pico32xMem->sdram; | ||||||
|   sh2->p_rom = Pico.rom; |   sh2->p_rom = Pico.rom; | ||||||
| } | } | ||||||
|  | @ -3277,7 +3278,7 @@ static void *dr_get_pc_base(u32 pc, int is_slave) | ||||||
|   } |   } | ||||||
|   else if ((pc & 0xfffff000) == 0xc0000000) { |   else if ((pc & 0xfffff000) == 0xc0000000) { | ||||||
|     // data array
 |     // data array
 | ||||||
|     ret = Pico32xMem->data_array[is_slave]; |     ret = sh2s[is_slave].data_array; | ||||||
|     mask = 0xfff; |     mask = 0xfff; | ||||||
|   } |   } | ||||||
|   else if ((pc & 0xc6000000) == 0x06000000) { |   else if ((pc & 0xc6000000) == 0x06000000) { | ||||||
|  |  | ||||||
|  | @ -14,12 +14,13 @@ | ||||||
| 
 | 
 | ||||||
| #define I 0xf0 | #define I 0xf0 | ||||||
| 
 | 
 | ||||||
| int sh2_init(SH2 *sh2, int is_slave) | int sh2_init(SH2 *sh2, int is_slave, SH2 *other_sh2) | ||||||
| { | { | ||||||
| 	int ret = 0; | 	int ret = 0; | ||||||
| 
 | 
 | ||||||
| 	memset(sh2, 0, offsetof(SH2, mult_m68k_to_sh2)); | 	memset(sh2, 0, offsetof(SH2, mult_m68k_to_sh2)); | ||||||
| 	sh2->is_slave = is_slave; | 	sh2->is_slave = is_slave; | ||||||
|  | 	sh2->other_sh2 = other_sh2; | ||||||
| 	pdb_register_cpu(sh2, PDBCT_SH2, is_slave ? "ssh2" : "msh2"); | 	pdb_register_cpu(sh2, PDBCT_SH2, is_slave ? "ssh2" : "msh2"); | ||||||
| #ifdef DRC_SH2 | #ifdef DRC_SH2 | ||||||
| 	ret = sh2_drc_init(sh2); | 	ret = sh2_drc_init(sh2); | ||||||
|  | @ -145,16 +146,13 @@ enum ctl_byte { | ||||||
| 	CTL_CYCLES = 0x85, | 	CTL_CYCLES = 0x85, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| #define SH2MAP_ADDR2OFFS_R(a) \ |  | ||||||
|   ((((a) >> 25) & 3) | (((a) >> 27) & 0x1c)) |  | ||||||
| 
 |  | ||||||
| static unsigned int local_read32(SH2 *sh2, u32 a) | static unsigned int local_read32(SH2 *sh2, u32 a) | ||||||
| { | { | ||||||
| 	const sh2_memmap *sh2_map = sh2->read16_map; | 	const sh2_memmap *sh2_map = sh2->read16_map; | ||||||
| 	u16 *pd; | 	u16 *pd; | ||||||
| 	uptr p; | 	uptr p; | ||||||
| 
 | 
 | ||||||
| 	sh2_map += SH2MAP_ADDR2OFFS_R(a); | 	sh2_map += (a >> 25); | ||||||
| 	p = sh2_map->addr; | 	p = sh2_map->addr; | ||||||
| 	if (!map_flag_set(p)) { | 	if (!map_flag_set(p)) { | ||||||
| 		pd = (u16 *)((p << 1) + ((a & sh2_map->mask) & ~1)); | 		pd = (u16 *)((p << 1) + ((a & sh2_map->mask) & ~1)); | ||||||
|  | @ -163,8 +161,7 @@ static unsigned int local_read32(SH2 *sh2, u32 a) | ||||||
| 
 | 
 | ||||||
| 	if ((a & 0xfffff000) == 0xc0000000) { | 	if ((a & 0xfffff000) == 0xc0000000) { | ||||||
| 		// data array
 | 		// data array
 | ||||||
| 		pd = (u16 *)Pico32xMem->data_array[sh2->is_slave] | 		pd = (u16 *)sh2->data_array + (a & 0xfff) / 2; | ||||||
| 			+ (a & 0xfff) / 2; |  | ||||||
| 		return (pd[0] << 16) | pd[1]; | 		return (pd[0] << 16) | pd[1]; | ||||||
| 	} | 	} | ||||||
| 	if ((a & 0xdfffffc0) == 0x4000) { | 	if ((a & 0xdfffffc0) == 0x4000) { | ||||||
|  |  | ||||||
|  | @ -63,10 +63,15 @@ typedef struct SH2_ | ||||||
| 
 | 
 | ||||||
| 	unsigned int	cycles_timeslice; | 	unsigned int	cycles_timeslice; | ||||||
| 
 | 
 | ||||||
|  | 	struct SH2_	*other_sh2; | ||||||
|  | 
 | ||||||
| 	// we use 68k reference cycles for easier sync
 | 	// we use 68k reference cycles for easier sync
 | ||||||
| 	unsigned int	m68krcycles_done; | 	unsigned int	m68krcycles_done; | ||||||
| 	unsigned int	mult_m68k_to_sh2; | 	unsigned int	mult_m68k_to_sh2; | ||||||
| 	unsigned int	mult_sh2_to_m68k; | 	unsigned int	mult_sh2_to_m68k; | ||||||
|  | 
 | ||||||
|  | 	unsigned char	data_array[0x1000]; // cache (can be used as RAM)
 | ||||||
|  | 	unsigned int	peri_regs[0x200/4]; // periphereal regs
 | ||||||
| } SH2; | } SH2; | ||||||
| 
 | 
 | ||||||
| #define CYCLE_MULT_SHIFT 10 | #define CYCLE_MULT_SHIFT 10 | ||||||
|  | @ -75,7 +80,7 @@ typedef struct SH2_ | ||||||
| #define C_SH2_TO_M68K(xsh2, c) \ | #define C_SH2_TO_M68K(xsh2, c) \ | ||||||
| 	((int)((c + 3) * (xsh2).mult_sh2_to_m68k) >> CYCLE_MULT_SHIFT) | 	((int)((c + 3) * (xsh2).mult_sh2_to_m68k) >> CYCLE_MULT_SHIFT) | ||||||
| 
 | 
 | ||||||
| int  sh2_init(SH2 *sh2, int is_slave); | int  sh2_init(SH2 *sh2, int is_slave, SH2 *other_sh2); | ||||||
| void sh2_finish(SH2 *sh2); | void sh2_finish(SH2 *sh2); | ||||||
| void sh2_reset(SH2 *sh2); | void sh2_reset(SH2 *sh2); | ||||||
| int  sh2_irl_irq(SH2 *sh2, int level, int nested_call); | int  sh2_irl_irq(SH2 *sh2, int level, int nested_call); | ||||||
|  | @ -94,9 +99,9 @@ int  sh2_execute(SH2 *sh2, int cycles); | ||||||
| unsigned int REGPARM(2) p32x_sh2_read8(unsigned int a, SH2 *sh2); | unsigned int REGPARM(2) p32x_sh2_read8(unsigned int a, SH2 *sh2); | ||||||
| unsigned int REGPARM(2) p32x_sh2_read16(unsigned int a, SH2 *sh2); | unsigned int REGPARM(2) p32x_sh2_read16(unsigned int a, SH2 *sh2); | ||||||
| unsigned int REGPARM(2) p32x_sh2_read32(unsigned int a, SH2 *sh2); | unsigned int REGPARM(2) p32x_sh2_read32(unsigned int a, SH2 *sh2); | ||||||
| int REGPARM(3) p32x_sh2_write8 (unsigned int a, unsigned int d, SH2 *sh2); | void REGPARM(3) p32x_sh2_write8 (unsigned int a, unsigned int d, SH2 *sh2); | ||||||
| int REGPARM(3) p32x_sh2_write16(unsigned int a, unsigned int d, SH2 *sh2); | void REGPARM(3) p32x_sh2_write16(unsigned int a, unsigned int d, SH2 *sh2); | ||||||
| int REGPARM(3) p32x_sh2_write32(unsigned int a, unsigned int d, SH2 *sh2); | void REGPARM(3) p32x_sh2_write32(unsigned int a, unsigned int d, SH2 *sh2); | ||||||
| 
 | 
 | ||||||
| // debug
 | // debug
 | ||||||
| #ifdef DRC_CMP | #ifdef DRC_CMP | ||||||
|  |  | ||||||
|  | @ -73,9 +73,9 @@ void Pico32xStartup(void) | ||||||
| 
 | 
 | ||||||
|   // TODO: OOM handling
 |   // TODO: OOM handling
 | ||||||
|   PicoAHW |= PAHW_32X; |   PicoAHW |= PAHW_32X; | ||||||
|   sh2_init(&msh2, 0); |   sh2_init(&msh2, 0, &ssh2); | ||||||
|   msh2.irq_callback = sh2_irq_cb; |   msh2.irq_callback = sh2_irq_cb; | ||||||
|   sh2_init(&ssh2, 1); |   sh2_init(&ssh2, 1, &msh2); | ||||||
|   ssh2.irq_callback = sh2_irq_cb; |   ssh2.irq_callback = sh2_irq_cb; | ||||||
| 
 | 
 | ||||||
|   PicoMemSetup32x(); |   PicoMemSetup32x(); | ||||||
|  | @ -85,8 +85,8 @@ void Pico32xStartup(void) | ||||||
|   if (!Pico.m.pal) |   if (!Pico.m.pal) | ||||||
|     Pico32x.vdp_regs[0] |= P32XV_nPAL; |     Pico32x.vdp_regs[0] |= P32XV_nPAL; | ||||||
| 
 | 
 | ||||||
|   PREG8(Pico32xMem->sh2_peri_regs[0], 4) = |   PREG8(msh2.peri_regs, 4) = | ||||||
|   PREG8(Pico32xMem->sh2_peri_regs[1], 4) = 0x84; // SCI SSR
 |   PREG8(ssh2.peri_regs, 4) = 0x84; // SCI SSR
 | ||||||
| 
 | 
 | ||||||
|   rendstatus_old = -1; |   rendstatus_old = -1; | ||||||
| 
 | 
 | ||||||
|  | @ -339,7 +339,7 @@ static inline void run_sh2(SH2 *sh2, int m68k_cycles) | ||||||
| // note: recursive call
 | // note: recursive call
 | ||||||
| void p32x_sync_other_sh2(SH2 *sh2, unsigned int m68k_target) | void p32x_sync_other_sh2(SH2 *sh2, unsigned int m68k_target) | ||||||
| { | { | ||||||
|   SH2 *osh2 = &sh2s[sh2->is_slave ^ 1]; |   SH2 *osh2 = sh2->other_sh2; | ||||||
|   int left_to_event; |   int left_to_event; | ||||||
|   int m68k_cycles; |   int m68k_cycles; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -432,17 +432,18 @@ static void p32x_vdp_write16(u32 a, u32 d, SH2 *sh2) | ||||||
| // ------------------------------------------------------------------
 | // ------------------------------------------------------------------
 | ||||||
| // SH2 regs
 | // SH2 regs
 | ||||||
| 
 | 
 | ||||||
| static u32 p32x_sh2reg_read16(u32 a, int cpuid) | static u32 p32x_sh2reg_read16(u32 a, SH2 *sh2) | ||||||
| { | { | ||||||
|   u16 *r = Pico32x.regs; |   u16 *r = Pico32x.regs; | ||||||
|   a &= 0xfe; // ?
 |   a &= 0xfe; // ?
 | ||||||
| 
 | 
 | ||||||
|   switch (a) { |   switch (a) { | ||||||
|     case 0x00: // adapter/irq ctl
 |     case 0x00: // adapter/irq ctl
 | ||||||
|       return (r[0] & P32XS_FM) | Pico32x.sh2_regs[0] | Pico32x.sh2irq_mask[cpuid]; |       return (r[0] & P32XS_FM) | Pico32x.sh2_regs[0] | ||||||
|  |         | Pico32x.sh2irq_mask[sh2->is_slave]; | ||||||
|     case 0x04: // H count (often as comm too)
 |     case 0x04: // H count (often as comm too)
 | ||||||
|       sh2_poll_detect(&sh2s[cpuid], a, SH2_STATE_CPOLL); |       sh2_poll_detect(sh2, a, SH2_STATE_CPOLL); | ||||||
|       sh2s_sync_on_read(&sh2s[cpuid]); |       sh2s_sync_on_read(sh2); | ||||||
|       return Pico32x.sh2_regs[4 / 2]; |       return Pico32x.sh2_regs[4 / 2]; | ||||||
|     case 0x10: // DREQ len
 |     case 0x10: // DREQ len
 | ||||||
|       return r[a / 2]; |       return r[a / 2]; | ||||||
|  | @ -457,22 +458,22 @@ static u32 p32x_sh2reg_read16(u32 a, int cpuid) | ||||||
|     if (Pico32x.comm_dirty_68k & comreg) |     if (Pico32x.comm_dirty_68k & comreg) | ||||||
|       Pico32x.comm_dirty_68k &= ~comreg; |       Pico32x.comm_dirty_68k &= ~comreg; | ||||||
|     else |     else | ||||||
|       sh2_poll_detect(&sh2s[cpuid], a, SH2_STATE_CPOLL); |       sh2_poll_detect(sh2, a, SH2_STATE_CPOLL); | ||||||
|     sh2s_sync_on_read(&sh2s[cpuid]); |     sh2s_sync_on_read(sh2); | ||||||
|     return r[a / 2]; |     return r[a / 2]; | ||||||
|   } |   } | ||||||
|   if ((a & 0x30) == 0x30) { |   if ((a & 0x30) == 0x30) { | ||||||
|     return p32x_pwm_read16(a, &sh2s[cpuid], sh2_cycles_done_m68k(&sh2s[cpuid])); |     return p32x_pwm_read16(a, sh2, sh2_cycles_done_m68k(sh2)); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   return 0; |   return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void p32x_sh2reg_write8(u32 a, u32 d, int cpuid) | static void p32x_sh2reg_write8(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   a &= 0xff; |   a &= 0xff; | ||||||
| 
 | 
 | ||||||
|   sh2s[cpuid].poll_addr = 0; |   sh2->poll_addr = 0; | ||||||
| 
 | 
 | ||||||
|   switch (a) { |   switch (a) { | ||||||
|     case 0: // FM
 |     case 0: // FM
 | ||||||
|  | @ -482,20 +483,20 @@ static void p32x_sh2reg_write8(u32 a, u32 d, int cpuid) | ||||||
|     case 1: // HEN/irq masks
 |     case 1: // HEN/irq masks
 | ||||||
|       if ((d ^ Pico32x.sh2_regs[0]) & 0x80) |       if ((d ^ Pico32x.sh2_regs[0]) & 0x80) | ||||||
|         elprintf(EL_ANOMALY|EL_32X, "HEN"); |         elprintf(EL_ANOMALY|EL_32X, "HEN"); | ||||||
|       Pico32x.sh2irq_mask[cpuid] = d & 0x8f; |       Pico32x.sh2irq_mask[sh2->is_slave] = d & 0x8f; | ||||||
|       Pico32x.sh2_regs[0] &= ~0x80; |       Pico32x.sh2_regs[0] &= ~0x80; | ||||||
|       Pico32x.sh2_regs[0] |= d & 0x80; |       Pico32x.sh2_regs[0] |= d & 0x80; | ||||||
|       if (d & 1) |       if (d & 1) | ||||||
|         p32x_pwm_schedule_sh2(&sh2s[cpuid]); |         p32x_pwm_schedule_sh2(sh2); | ||||||
|       p32x_update_irls(&sh2s[cpuid], 0); |       p32x_update_irls(sh2, 0); | ||||||
|       return; |       return; | ||||||
|     case 5: // H count
 |     case 5: // H count
 | ||||||
|       d &= 0xff; |       d &= 0xff; | ||||||
|       if (Pico32x.sh2_regs[4 / 2] != d) { |       if (Pico32x.sh2_regs[4 / 2] != d) { | ||||||
|         Pico32x.sh2_regs[4 / 2] = d; |         Pico32x.sh2_regs[4 / 2] = d; | ||||||
|         p32x_sh2_poll_event(&sh2s[cpuid ^ 1], SH2_STATE_CPOLL, |         p32x_sh2_poll_event(sh2->other_sh2, SH2_STATE_CPOLL, | ||||||
|           sh2_cycles_done_m68k(&sh2s[cpuid])); |           sh2_cycles_done_m68k(sh2)); | ||||||
|         sh2_end_run(&sh2s[cpuid], 4); |         sh2_end_run(sh2, 4); | ||||||
|       } |       } | ||||||
|       return; |       return; | ||||||
|   } |   } | ||||||
|  | @ -508,19 +509,19 @@ static void p32x_sh2reg_write8(u32 a, u32 d, int cpuid) | ||||||
| 
 | 
 | ||||||
|     r8[a ^ 1] = d; |     r8[a ^ 1] = d; | ||||||
|     p32x_m68k_poll_event(P32XF_68KCPOLL); |     p32x_m68k_poll_event(P32XF_68KCPOLL); | ||||||
|     p32x_sh2_poll_event(&sh2s[cpuid ^ 1], SH2_STATE_CPOLL, |     p32x_sh2_poll_event(sh2->other_sh2, SH2_STATE_CPOLL, | ||||||
|       sh2_cycles_done_m68k(&sh2s[cpuid])); |       sh2_cycles_done_m68k(sh2)); | ||||||
|     comreg = 1 << (a & 0x0f) / 2; |     comreg = 1 << (a & 0x0f) / 2; | ||||||
|     Pico32x.comm_dirty_sh2 |= comreg; |     Pico32x.comm_dirty_sh2 |= comreg; | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void p32x_sh2reg_write16(u32 a, u32 d, int cpuid) | static void p32x_sh2reg_write16(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   a &= 0xfe; |   a &= 0xfe; | ||||||
| 
 | 
 | ||||||
|   sh2s[cpuid].poll_addr = 0; |   sh2->poll_addr = 0; | ||||||
| 
 | 
 | ||||||
|   // comm
 |   // comm
 | ||||||
|   if ((a & 0x30) == 0x20) { |   if ((a & 0x30) == 0x20) { | ||||||
|  | @ -530,15 +531,15 @@ static void p32x_sh2reg_write16(u32 a, u32 d, int cpuid) | ||||||
| 
 | 
 | ||||||
|     Pico32x.regs[a / 2] = d; |     Pico32x.regs[a / 2] = d; | ||||||
|     p32x_m68k_poll_event(P32XF_68KCPOLL); |     p32x_m68k_poll_event(P32XF_68KCPOLL); | ||||||
|     p32x_sh2_poll_event(&sh2s[cpuid ^ 1], SH2_STATE_CPOLL, |     p32x_sh2_poll_event(sh2->other_sh2, SH2_STATE_CPOLL, | ||||||
|       sh2_cycles_done_m68k(&sh2s[cpuid])); |       sh2_cycles_done_m68k(sh2)); | ||||||
|     comreg = 1 << (a & 0x0f) / 2; |     comreg = 1 << (a & 0x0f) / 2; | ||||||
|     Pico32x.comm_dirty_sh2 |= comreg; |     Pico32x.comm_dirty_sh2 |= comreg; | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|   // PWM
 |   // PWM
 | ||||||
|   else if ((a & 0x30) == 0x30) { |   else if ((a & 0x30) == 0x30) { | ||||||
|     p32x_pwm_write16(a, d, &sh2s[cpuid], sh2_cycles_done_m68k(&sh2s[cpuid])); |     p32x_pwm_write16(a, d, sh2, sh2_cycles_done_m68k(sh2)); | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  | @ -550,18 +551,18 @@ static void p32x_sh2reg_write16(u32 a, u32 d, int cpuid) | ||||||
|     case 0x14: Pico32x.sh2irqs &= ~P32XI_VRES; goto irls; |     case 0x14: Pico32x.sh2irqs &= ~P32XI_VRES; goto irls; | ||||||
|     case 0x16: Pico32x.sh2irqs &= ~P32XI_VINT; goto irls; |     case 0x16: Pico32x.sh2irqs &= ~P32XI_VINT; goto irls; | ||||||
|     case 0x18: Pico32x.sh2irqs &= ~P32XI_HINT; goto irls; |     case 0x18: Pico32x.sh2irqs &= ~P32XI_HINT; goto irls; | ||||||
|     case 0x1a: Pico32x.sh2irqi[cpuid] &= ~P32XI_CMD; goto irls; |     case 0x1a: Pico32x.sh2irqi[sh2->is_slave] &= ~P32XI_CMD; goto irls; | ||||||
|     case 0x1c: |     case 0x1c: | ||||||
|       Pico32x.sh2irqs &= ~P32XI_PWM; |       Pico32x.sh2irqs &= ~P32XI_PWM; | ||||||
|       p32x_pwm_schedule_sh2(&sh2s[cpuid]); |       p32x_pwm_schedule_sh2(sh2); | ||||||
|       goto irls; |       goto irls; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   p32x_sh2reg_write8(a | 1, d, cpuid); |   p32x_sh2reg_write8(a | 1, d, sh2); | ||||||
|   return; |   return; | ||||||
| 
 | 
 | ||||||
| irls: | irls: | ||||||
|   p32x_update_irls(&sh2s[cpuid], 0); |   p32x_update_irls(sh2, 0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // ------------------------------------------------------------------
 | // ------------------------------------------------------------------
 | ||||||
|  | @ -821,26 +822,25 @@ static void m68k_write8_dram1_ow(u32 a, u32 d) | ||||||
|   sh2_write8_dramN(1); |   sh2_write8_dramN(1); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #define sh2_write16_dramN(n, ret) \ | #define sh2_write16_dramN(n) \ | ||||||
|   u16 *pd = &Pico32xMem->dram[n][(a & 0x1ffff) / 2]; \ |   u16 *pd = &Pico32xMem->dram[n][(a & 0x1ffff) / 2]; \ | ||||||
|   if (!(a & 0x20000)) { \ |   if (!(a & 0x20000)) { \ | ||||||
|     *pd = d; \ |     *pd = d; \ | ||||||
|     return ret; \ |     return; \ | ||||||
|   } \ |   } \ | ||||||
|   /* overwrite */ \ |   /* overwrite */ \ | ||||||
|   if (!(d & 0xff00)) d |= *pd & 0xff00; \ |   if (!(d & 0xff00)) d |= *pd & 0xff00; \ | ||||||
|   if (!(d & 0x00ff)) d |= *pd & 0x00ff; \ |   if (!(d & 0x00ff)) d |= *pd & 0x00ff; \ | ||||||
|   *pd = d; \ |   *pd = d; | ||||||
|   return ret |  | ||||||
| 
 | 
 | ||||||
| static void m68k_write16_dram0_ow(u32 a, u32 d) | static void m68k_write16_dram0_ow(u32 a, u32 d) | ||||||
| { | { | ||||||
|   sh2_write16_dramN(0,); |   sh2_write16_dramN(0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void m68k_write16_dram1_ow(u32 a, u32 d) | static void m68k_write16_dram1_ow(u32 a, u32 d) | ||||||
| { | { | ||||||
|   sh2_write16_dramN(1,); |   sh2_write16_dramN(1); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // -----------------------------------------------------------------
 | // -----------------------------------------------------------------
 | ||||||
|  | @ -898,33 +898,33 @@ static void bank_switch(int b) | ||||||
| // -----------------------------------------------------------------
 | // -----------------------------------------------------------------
 | ||||||
| 
 | 
 | ||||||
| // read8
 | // read8
 | ||||||
| static u32 sh2_read8_unmapped(u32 a, int id) | static u32 sh2_read8_unmapped(u32 a, SH2 *sh2) | ||||||
| { | { | ||||||
|   elprintf(EL_UIO, "%csh2 unmapped r8  [%08x]       %02x @%06x", |   elprintf(EL_UIO, "%csh2 unmapped r8  [%08x]       %02x @%06x", | ||||||
|     id ? 's' : 'm', a, 0, sh2_pc(id)); |     sh2->is_slave ? 's' : 'm', a, 0, sh2_pc(sh2)); | ||||||
|   return 0; |   return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static u32 sh2_read8_cs0(u32 a, int id) | static u32 sh2_read8_cs0(u32 a, SH2 *sh2) | ||||||
| { | { | ||||||
|   u32 d = 0; |   u32 d = 0; | ||||||
| 
 | 
 | ||||||
|   // 0x3ff00 is veridied
 |   // 0x3ff00 is veridied
 | ||||||
|   if ((a & 0x3ff00) == 0x4000) { |   if ((a & 0x3ff00) == 0x4000) { | ||||||
|     d = p32x_sh2reg_read16(a, id); |     d = p32x_sh2reg_read16(a, sh2); | ||||||
|     goto out_16to8; |     goto out_16to8; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   if ((a & 0x3ff00) == 0x4100) { |   if ((a & 0x3ff00) == 0x4100) { | ||||||
|     d = p32x_vdp_read16(a); |     d = p32x_vdp_read16(a); | ||||||
|     sh2_poll_detect(&sh2s[id], a, SH2_STATE_VPOLL); |     sh2_poll_detect(sh2, a, SH2_STATE_VPOLL); | ||||||
|     goto out_16to8; |     goto out_16to8; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   // TODO: mirroring?
 |   // TODO: mirroring?
 | ||||||
|   if (id == 0 && a < sizeof(Pico32xMem->sh2_rom_m)) |   if (!sh2->is_slave && a < sizeof(Pico32xMem->sh2_rom_m)) | ||||||
|     return Pico32xMem->sh2_rom_m[a ^ 1]; |     return Pico32xMem->sh2_rom_m[a ^ 1]; | ||||||
|   if (id == 1 && a < sizeof(Pico32xMem->sh2_rom_s)) |   if (sh2->is_slave  && a < sizeof(Pico32xMem->sh2_rom_s)) | ||||||
|     return Pico32xMem->sh2_rom_s[a ^ 1]; |     return Pico32xMem->sh2_rom_s[a ^ 1]; | ||||||
| 
 | 
 | ||||||
|   if ((a & 0x3fe00) == 0x4200) { |   if ((a & 0x3fe00) == 0x4200) { | ||||||
|  | @ -932,7 +932,7 @@ static u32 sh2_read8_cs0(u32 a, int id) | ||||||
|     goto out_16to8; |     goto out_16to8; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   return sh2_read8_unmapped(a, id); |   return sh2_read8_unmapped(a, sh2); | ||||||
| 
 | 
 | ||||||
| out_16to8: | out_16to8: | ||||||
|   if (a & 1) |   if (a & 1) | ||||||
|  | @ -941,29 +941,29 @@ out_16to8: | ||||||
|     d >>= 8; |     d >>= 8; | ||||||
| 
 | 
 | ||||||
|   elprintf(EL_32X, "%csh2 r8  [%08x]       %02x @%06x", |   elprintf(EL_32X, "%csh2 r8  [%08x]       %02x @%06x", | ||||||
|     id ? 's' : 'm', a, d, sh2_pc(id)); |     sh2->is_slave ? 's' : 'm', a, d, sh2_pc(sh2)); | ||||||
|   return d; |   return d; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static u32 sh2_read8_da(u32 a, int id) | static u32 sh2_read8_da(u32 a, SH2 *sh2) | ||||||
| { | { | ||||||
|   return Pico32xMem->data_array[id][(a & 0xfff) ^ 1]; |   return sh2->data_array[(a & 0xfff) ^ 1]; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // read16
 | // read16
 | ||||||
| static u32 sh2_read16_unmapped(u32 a, int id) | static u32 sh2_read16_unmapped(u32 a, SH2 *sh2) | ||||||
| { | { | ||||||
|   elprintf(EL_UIO, "%csh2 unmapped r16 [%08x]     %04x @%06x", |   elprintf(EL_UIO, "%csh2 unmapped r16 [%08x]     %04x @%06x", | ||||||
|     id ? 's' : 'm', a, 0, sh2_pc(id)); |     sh2->is_slave ? 's' : 'm', a, 0, sh2_pc(sh2)); | ||||||
|   return 0; |   return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static u32 sh2_read16_cs0(u32 a, int id) | static u32 sh2_read16_cs0(u32 a, SH2 *sh2) | ||||||
| { | { | ||||||
|   u32 d = 0; |   u32 d = 0; | ||||||
| 
 | 
 | ||||||
|   if ((a & 0x3ff00) == 0x4000) { |   if ((a & 0x3ff00) == 0x4000) { | ||||||
|     d = p32x_sh2reg_read16(a, id); |     d = p32x_sh2reg_read16(a, sh2); | ||||||
|     if (!(EL_LOGMASK & EL_PWM) && (a & 0x30) == 0x30) // hide PWM
 |     if (!(EL_LOGMASK & EL_PWM) && (a & 0x30) == 0x30) // hide PWM
 | ||||||
|       return d; |       return d; | ||||||
|     goto out; |     goto out; | ||||||
|  | @ -971,13 +971,13 @@ static u32 sh2_read16_cs0(u32 a, int id) | ||||||
| 
 | 
 | ||||||
|   if ((a & 0x3ff00) == 0x4100) { |   if ((a & 0x3ff00) == 0x4100) { | ||||||
|     d = p32x_vdp_read16(a); |     d = p32x_vdp_read16(a); | ||||||
|     sh2_poll_detect(&sh2s[id], a, SH2_STATE_VPOLL); |     sh2_poll_detect(sh2, a, SH2_STATE_VPOLL); | ||||||
|     goto out; |     goto out; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   if (id == 0 && a < sizeof(Pico32xMem->sh2_rom_m)) |   if (!sh2->is_slave && a < sizeof(Pico32xMem->sh2_rom_m)) | ||||||
|     return *(u16 *)(Pico32xMem->sh2_rom_m + a); |     return *(u16 *)(Pico32xMem->sh2_rom_m + a); | ||||||
|   if (id == 1 && a < sizeof(Pico32xMem->sh2_rom_s)) |   if (sh2->is_slave  && a < sizeof(Pico32xMem->sh2_rom_s)) | ||||||
|     return *(u16 *)(Pico32xMem->sh2_rom_s + a); |     return *(u16 *)(Pico32xMem->sh2_rom_s + a); | ||||||
| 
 | 
 | ||||||
|   if ((a & 0x3fe00) == 0x4200) { |   if ((a & 0x3fe00) == 0x4200) { | ||||||
|  | @ -985,165 +985,159 @@ static u32 sh2_read16_cs0(u32 a, int id) | ||||||
|     goto out; |     goto out; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   return sh2_read16_unmapped(a, id); |   return sh2_read16_unmapped(a, sh2); | ||||||
| 
 | 
 | ||||||
| out: | out: | ||||||
|   elprintf(EL_32X, "%csh2 r16 [%08x]     %04x @%06x", |   elprintf(EL_32X, "%csh2 r16 [%08x]     %04x @%06x", | ||||||
|     id ? 's' : 'm', a, d, sh2_pc(id)); |     sh2->is_slave ? 's' : 'm', a, d, sh2_pc(sh2)); | ||||||
|   return d; |   return d; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static u32 sh2_read16_da(u32 a, int id) | static u32 sh2_read16_da(u32 a, SH2 *sh2) | ||||||
| { | { | ||||||
|   return ((u16 *)Pico32xMem->data_array[id])[(a & 0xfff) / 2]; |   return ((u16 *)sh2->data_array)[(a & 0xfff) / 2]; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int REGPARM(3) sh2_write_ignore(u32 a, u32 d, int id) | // writes
 | ||||||
|  | static void REGPARM(3) sh2_write_ignore(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   return 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // write8
 | // write8
 | ||||||
| static int REGPARM(3) sh2_write8_unmapped(u32 a, u32 d, int id) | static void REGPARM(3) sh2_write8_unmapped(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   elprintf(EL_UIO, "%csh2 unmapped w8  [%08x]       %02x @%06x", |   elprintf(EL_UIO, "%csh2 unmapped w8  [%08x]       %02x @%06x", | ||||||
|     id ? 's' : 'm', a, d & 0xff, sh2_pc(id)); |     sh2->is_slave ? 's' : 'm', a, d & 0xff, sh2_pc(sh2)); | ||||||
|   return 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int REGPARM(3) sh2_write8_cs0(u32 a, u32 d, int id) | static void REGPARM(3) sh2_write8_cs0(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   elprintf(EL_32X, "%csh2 w8  [%08x]       %02x @%06x", |   elprintf(EL_32X, "%csh2 w8  [%08x]       %02x @%06x", | ||||||
|     id ? 's' : 'm', a, d & 0xff, sh2_pc(id)); |     sh2->is_slave ? 's' : 'm', a, d & 0xff, sh2_pc(sh2)); | ||||||
| 
 | 
 | ||||||
|   if (Pico32x.regs[0] & P32XS_FM) { |   if (Pico32x.regs[0] & P32XS_FM) { | ||||||
|     if ((a & 0x3ff00) == 0x4100) { |     if ((a & 0x3ff00) == 0x4100) { | ||||||
|       sh2s[id].poll_addr = 0; |       sh2->poll_addr = 0; | ||||||
|       p32x_vdp_write8(a, d); |       p32x_vdp_write8(a, d); | ||||||
|       return 0; |       return; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   if ((a & 0x3ff00) == 0x4000) { |   if ((a & 0x3ff00) == 0x4000) { | ||||||
|     p32x_sh2reg_write8(a, d, id); |     p32x_sh2reg_write8(a, d, sh2); | ||||||
|     return 1; |     return; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   return sh2_write8_unmapped(a, d, id); |   sh2_write8_unmapped(a, d, sh2); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int REGPARM(3) sh2_write8_dram0(u32 a, u32 d, int id) | static void REGPARM(3) sh2_write8_dram0(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   sh2_write8_dramN(0); |   sh2_write8_dramN(0); | ||||||
|   return 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int REGPARM(3) sh2_write8_dram1(u32 a, u32 d, int id) | static void REGPARM(3) sh2_write8_dram1(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   sh2_write8_dramN(1); |   sh2_write8_dramN(1); | ||||||
|   return 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int REGPARM(3) sh2_write8_sdram(u32 a, u32 d, int id) | static void REGPARM(3) sh2_write8_sdram(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   u32 a1 = a & 0x3ffff; |   u32 a1 = a & 0x3ffff; | ||||||
| #ifdef DRC_SH2 | #ifdef DRC_SH2 | ||||||
|   int t = Pico32xMem->drcblk_ram[a1 >> SH2_DRCBLK_RAM_SHIFT]; |   int t = Pico32xMem->drcblk_ram[a1 >> SH2_DRCBLK_RAM_SHIFT]; | ||||||
|   if (t) |   if (t) | ||||||
|     sh2_drc_wcheck_ram(a, t, id); |     sh2_drc_wcheck_ram(a, t, sh2->is_slave); | ||||||
| #endif | #endif | ||||||
|   Pico32xMem->sdram[a1 ^ 1] = d; |   Pico32xMem->sdram[a1 ^ 1] = d; | ||||||
|   return 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int REGPARM(3) sh2_write8_da(u32 a, u32 d, int id) | static void REGPARM(3) sh2_write8_da(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   u32 a1 = a & 0xfff; |   u32 a1 = a & 0xfff; | ||||||
| #ifdef DRC_SH2 | #ifdef DRC_SH2 | ||||||
|  |   int id = sh2->is_slave; | ||||||
|   int t = Pico32xMem->drcblk_da[id][a1 >> SH2_DRCBLK_DA_SHIFT]; |   int t = Pico32xMem->drcblk_da[id][a1 >> SH2_DRCBLK_DA_SHIFT]; | ||||||
|   if (t) |   if (t) | ||||||
|     sh2_drc_wcheck_da(a, t, id); |     sh2_drc_wcheck_da(a, t, id); | ||||||
| #endif | #endif | ||||||
|   Pico32xMem->data_array[id][a1 ^ 1] = d; |   sh2->data_array[a1 ^ 1] = d; | ||||||
|   return 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // write16
 | // write16
 | ||||||
| static int REGPARM(3) sh2_write16_unmapped(u32 a, u32 d, int id) | static void REGPARM(3) sh2_write16_unmapped(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   elprintf(EL_UIO, "%csh2 unmapped w16 [%08x]     %04x @%06x", |   elprintf(EL_UIO, "%csh2 unmapped w16 [%08x]     %04x @%06x", | ||||||
|     id ? 's' : 'm', a, d & 0xffff, sh2_pc(id)); |     sh2->is_slave ? 's' : 'm', a, d & 0xffff, sh2_pc(sh2)); | ||||||
|   return 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int REGPARM(3) sh2_write16_cs0(u32 a, u32 d, int id) | static void REGPARM(3) sh2_write16_cs0(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   if (((EL_LOGMASK & EL_PWM) || (a & 0x30) != 0x30)) // hide PWM
 |   if (((EL_LOGMASK & EL_PWM) || (a & 0x30) != 0x30)) // hide PWM
 | ||||||
|     elprintf(EL_32X, "%csh2 w16 [%08x]     %04x @%06x", |     elprintf(EL_32X, "%csh2 w16 [%08x]     %04x @%06x", | ||||||
|       id ? 's' : 'm', a, d & 0xffff, sh2_pc(id)); |       sh2->is_slave ? 's' : 'm', a, d & 0xffff, sh2_pc(sh2)); | ||||||
| 
 | 
 | ||||||
|   if (Pico32x.regs[0] & P32XS_FM) { |   if (Pico32x.regs[0] & P32XS_FM) { | ||||||
|     if ((a & 0x3ff00) == 0x4100) { |     if ((a & 0x3ff00) == 0x4100) { | ||||||
|       sh2s[id].poll_addr = 0; |       sh2->poll_addr = 0; | ||||||
|       p32x_vdp_write16(a, d, &sh2s[id]); |       p32x_vdp_write16(a, d, sh2); | ||||||
|       return 0; |       return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if ((a & 0x3fe00) == 0x4200) { |     if ((a & 0x3fe00) == 0x4200) { | ||||||
|       Pico32xMem->pal[(a & 0x1ff) / 2] = d; |       Pico32xMem->pal[(a & 0x1ff) / 2] = d; | ||||||
|       Pico32x.dirty_pal = 1; |       Pico32x.dirty_pal = 1; | ||||||
|       return 0; |       return; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   if ((a & 0x3ff00) == 0x4000) { |   if ((a & 0x3ff00) == 0x4000) { | ||||||
|     p32x_sh2reg_write16(a, d, id); |     p32x_sh2reg_write16(a, d, sh2); | ||||||
|     return 1; |     return; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   return sh2_write16_unmapped(a, d, id); |   sh2_write16_unmapped(a, d, sh2); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int REGPARM(3) sh2_write16_dram0(u32 a, u32 d, int id) | static void REGPARM(3) sh2_write16_dram0(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   sh2_write16_dramN(0, 0); |   sh2_write16_dramN(0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int REGPARM(3) sh2_write16_dram1(u32 a, u32 d, int id) | static void REGPARM(3) sh2_write16_dram1(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   sh2_write16_dramN(1, 0); |   sh2_write16_dramN(1); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int REGPARM(3) sh2_write16_sdram(u32 a, u32 d, int id) | static void REGPARM(3) sh2_write16_sdram(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   u32 a1 = a & 0x3ffff; |   u32 a1 = a & 0x3ffff; | ||||||
| #ifdef DRC_SH2 | #ifdef DRC_SH2 | ||||||
|   int t = Pico32xMem->drcblk_ram[a1 >> SH2_DRCBLK_RAM_SHIFT]; |   int t = Pico32xMem->drcblk_ram[a1 >> SH2_DRCBLK_RAM_SHIFT]; | ||||||
|   if (t) |   if (t) | ||||||
|     sh2_drc_wcheck_ram(a, t, id); |     sh2_drc_wcheck_ram(a, t, sh2->is_slave); | ||||||
| #endif | #endif | ||||||
|   ((u16 *)Pico32xMem->sdram)[a1 / 2] = d; |   ((u16 *)Pico32xMem->sdram)[a1 / 2] = d; | ||||||
|   return 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int REGPARM(3) sh2_write16_da(u32 a, u32 d, int id) | static void REGPARM(3) sh2_write16_da(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   u32 a1 = a & 0xfff; |   u32 a1 = a & 0xfff; | ||||||
| #ifdef DRC_SH2 | #ifdef DRC_SH2 | ||||||
|  |   int id = sh2->is_slave; | ||||||
|   int t = Pico32xMem->drcblk_da[id][a1 >> SH2_DRCBLK_DA_SHIFT]; |   int t = Pico32xMem->drcblk_da[id][a1 >> SH2_DRCBLK_DA_SHIFT]; | ||||||
|   if (t) |   if (t) | ||||||
|     sh2_drc_wcheck_da(a, t, id); |     sh2_drc_wcheck_da(a, t, id); | ||||||
| #endif | #endif | ||||||
|   ((u16 *)Pico32xMem->data_array[id])[a1 / 2] = d; |   ((u16 *)sh2->data_array)[a1 / 2] = d; | ||||||
|   return 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| typedef u32 (sh2_read_handler)(u32 a, int id); | typedef u32 (sh2_read_handler)(u32 a, SH2 *sh2); | ||||||
| typedef int REGPARM(3) (sh2_write_handler)(u32 a, u32 d, int id); | typedef void REGPARM(3) (sh2_write_handler)(u32 a, u32 d, SH2 *sh2); | ||||||
| 
 | 
 | ||||||
| #define SH2MAP_ADDR2OFFS_R(a) \ | #define SH2MAP_ADDR2OFFS_R(a) \ | ||||||
|   ((((a) >> 25) & 3) | (((a) >> 27) & 0x1c)) |   ((u32)(a) >> SH2_READ_SHIFT) | ||||||
| 
 | 
 | ||||||
| #define SH2MAP_ADDR2OFFS_W(a) \ | #define SH2MAP_ADDR2OFFS_W(a) \ | ||||||
|   ((u32)(a) >> SH2_WRITE_SHIFT) |   ((u32)(a) >> SH2_WRITE_SHIFT) | ||||||
|  | @ -1156,7 +1150,7 @@ u32 REGPARM(2) p32x_sh2_read8(u32 a, SH2 *sh2) | ||||||
|   sh2_map += SH2MAP_ADDR2OFFS_R(a); |   sh2_map += SH2MAP_ADDR2OFFS_R(a); | ||||||
|   p = sh2_map->addr; |   p = sh2_map->addr; | ||||||
|   if (map_flag_set(p)) |   if (map_flag_set(p)) | ||||||
|     return ((sh2_read_handler *)(p << 1))(a, sh2->is_slave); |     return ((sh2_read_handler *)(p << 1))(a, sh2); | ||||||
|   else |   else | ||||||
|     return *(u8 *)((p << 1) + ((a & sh2_map->mask) ^ 1)); |     return *(u8 *)((p << 1) + ((a & sh2_map->mask) ^ 1)); | ||||||
| } | } | ||||||
|  | @ -1169,7 +1163,7 @@ u32 REGPARM(2) p32x_sh2_read16(u32 a, SH2 *sh2) | ||||||
|   sh2_map += SH2MAP_ADDR2OFFS_R(a); |   sh2_map += SH2MAP_ADDR2OFFS_R(a); | ||||||
|   p = sh2_map->addr; |   p = sh2_map->addr; | ||||||
|   if (map_flag_set(p)) |   if (map_flag_set(p)) | ||||||
|     return ((sh2_read_handler *)(p << 1))(a, sh2->is_slave); |     return ((sh2_read_handler *)(p << 1))(a, sh2); | ||||||
|   else |   else | ||||||
|     return *(u16 *)((p << 1) + ((a & sh2_map->mask) & ~1)); |     return *(u16 *)((p << 1) + ((a & sh2_map->mask) & ~1)); | ||||||
| } | } | ||||||
|  | @ -1191,48 +1185,46 @@ u32 REGPARM(2) p32x_sh2_read32(u32 a, SH2 *sh2) | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   if (offs == 0x1f) |   if (offs == 0x1f) | ||||||
|     return sh2_peripheral_read32(a, sh2->is_slave); |     return sh2_peripheral_read32(a, sh2); | ||||||
| 
 | 
 | ||||||
|   handler = (sh2_read_handler *)(p << 1); |   handler = (sh2_read_handler *)(p << 1); | ||||||
|   return (handler(a, sh2->is_slave) << 16) | handler(a + 2, sh2->is_slave); |   return (handler(a, sh2) << 16) | handler(a + 2, sh2); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // return nonzero if write potentially causes an interrupt (used by drc)
 | void REGPARM(3) p32x_sh2_write8(u32 a, u32 d, SH2 *sh2) | ||||||
| int REGPARM(3) p32x_sh2_write8(u32 a, u32 d, SH2 *sh2) |  | ||||||
| { | { | ||||||
|   const void **sh2_wmap = sh2->write8_tab; |   const void **sh2_wmap = sh2->write8_tab; | ||||||
|   sh2_write_handler *wh; |   sh2_write_handler *wh; | ||||||
| 
 | 
 | ||||||
|   wh = sh2_wmap[SH2MAP_ADDR2OFFS_W(a)]; |   wh = sh2_wmap[SH2MAP_ADDR2OFFS_W(a)]; | ||||||
|   return wh(a, d, sh2->is_slave); |   wh(a, d, sh2); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int REGPARM(3) p32x_sh2_write16(u32 a, u32 d, SH2 *sh2) | void REGPARM(3) p32x_sh2_write16(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   const void **sh2_wmap = sh2->write16_tab; |   const void **sh2_wmap = sh2->write16_tab; | ||||||
|   sh2_write_handler *wh; |   sh2_write_handler *wh; | ||||||
| 
 | 
 | ||||||
|   wh = sh2_wmap[SH2MAP_ADDR2OFFS_W(a)]; |   wh = sh2_wmap[SH2MAP_ADDR2OFFS_W(a)]; | ||||||
|   return wh(a, d, sh2->is_slave); |   wh(a, d, sh2); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int REGPARM(3) p32x_sh2_write32(u32 a, u32 d, SH2 *sh2) | void REGPARM(3) p32x_sh2_write32(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   const void **sh2_wmap = sh2->write16_tab; |   const void **sh2_wmap = sh2->write16_tab; | ||||||
|   sh2_write_handler *handler; |   sh2_write_handler *wh; | ||||||
|   u32 offs; |   u32 offs; | ||||||
| 
 | 
 | ||||||
|   offs = SH2MAP_ADDR2OFFS_W(a); |   offs = SH2MAP_ADDR2OFFS_W(a); | ||||||
| 
 | 
 | ||||||
|   if (offs == SH2MAP_ADDR2OFFS_W(0xffffc000)) { |   if (offs == SH2MAP_ADDR2OFFS_W(0xffffc000)) { | ||||||
|     sh2_peripheral_write32(a, d, sh2->is_slave); |     sh2_peripheral_write32(a, d, sh2); | ||||||
|     return 0; |     return; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   handler = sh2_wmap[offs]; |   wh = sh2_wmap[offs]; | ||||||
|   handler(a, d >> 16, sh2->is_slave); |   wh(a, d >> 16, sh2); | ||||||
|   handler(a + 2, d, sh2->is_slave); |   wh(a + 2, d, sh2); | ||||||
|   return 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // -----------------------------------------------------------------
 | // -----------------------------------------------------------------
 | ||||||
|  | @ -1359,7 +1351,7 @@ static void get_bios(void) | ||||||
| #define MAP_MEMORY(m) ((uptr)(m) >> 1) | #define MAP_MEMORY(m) ((uptr)(m) >> 1) | ||||||
| #define MAP_HANDLER(h) ( ((uptr)(h) >> 1) | ((uptr)1 << (sizeof(uptr) * 8 - 1)) ) | #define MAP_HANDLER(h) ( ((uptr)(h) >> 1) | ((uptr)1 << (sizeof(uptr) * 8 - 1)) ) | ||||||
| 
 | 
 | ||||||
| static sh2_memmap sh2_read8_map[0x20], sh2_read16_map[0x20]; | static sh2_memmap sh2_read8_map[0x80], sh2_read16_map[0x80]; | ||||||
| // for writes we are using handlers only
 | // for writes we are using handlers only
 | ||||||
| static sh2_write_handler *sh2_write8_map[0x80], *sh2_write16_map[0x80]; | static sh2_write_handler *sh2_write8_map[0x80], *sh2_write16_map[0x80]; | ||||||
| 
 | 
 | ||||||
|  | @ -1375,8 +1367,8 @@ void Pico32xSwapDRAM(int b) | ||||||
|                  b ? m68k_write16_dram1_ow : m68k_write16_dram0_ow, 1); |                  b ? m68k_write16_dram1_ow : m68k_write16_dram0_ow, 1); | ||||||
| 
 | 
 | ||||||
|   // SH2
 |   // SH2
 | ||||||
|   sh2_read8_map[2].addr   = sh2_read8_map[6].addr   = |   sh2_read8_map[0x04/2].addr  = sh2_read8_map[0x24/2].addr  = | ||||||
|   sh2_read16_map[2].addr  = sh2_read16_map[6].addr  = MAP_MEMORY(Pico32xMem->dram[b]); |   sh2_read16_map[0x04/2].addr = sh2_read16_map[0x24/2].addr = MAP_MEMORY(Pico32xMem->dram[b]); | ||||||
| 
 | 
 | ||||||
|   sh2_write8_map[0x04/2]  = sh2_write8_map[0x24/2]  = b ? sh2_write8_dram1 : sh2_write8_dram0; |   sh2_write8_map[0x04/2]  = sh2_write8_map[0x24/2]  = b ? sh2_write8_dram1 : sh2_write8_dram0; | ||||||
|   sh2_write16_map[0x04/2] = sh2_write16_map[0x24/2] = b ? sh2_write16_dram1 : sh2_write16_dram0; |   sh2_write16_map[0x04/2] = sh2_write16_map[0x24/2] = b ? sh2_write16_dram1 : sh2_write16_dram0; | ||||||
|  | @ -1448,35 +1440,35 @@ void PicoMemSetup32x(void) | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   // CS0
 |   // CS0
 | ||||||
|   sh2_read8_map[0].addr   = sh2_read8_map[4].addr   = MAP_HANDLER(sh2_read8_cs0); |   sh2_read8_map[0x00/2].addr  = sh2_read8_map[0x20/2].addr  = MAP_HANDLER(sh2_read8_cs0); | ||||||
|   sh2_read16_map[0].addr  = sh2_read16_map[4].addr  = MAP_HANDLER(sh2_read16_cs0); |   sh2_read16_map[0x00/2].addr = sh2_read16_map[0x20/2].addr = MAP_HANDLER(sh2_read16_cs0); | ||||||
|   sh2_write8_map[0x00/2]  = sh2_write8_map[0x20/2]  = sh2_write8_cs0; |   sh2_write8_map[0x00/2]  = sh2_write8_map[0x20/2]  = sh2_write8_cs0; | ||||||
|   sh2_write16_map[0x00/2] = sh2_write16_map[0x20/2] = sh2_write16_cs0; |   sh2_write16_map[0x00/2] = sh2_write16_map[0x20/2] = sh2_write16_cs0; | ||||||
|   // CS1 - ROM
 |   // CS1 - ROM
 | ||||||
|   sh2_read8_map[1].addr   = sh2_read8_map[5].addr   = |   sh2_read8_map[0x02/2].addr  = sh2_read8_map[0x22/2].addr  = | ||||||
|   sh2_read16_map[1].addr  = sh2_read16_map[5].addr  = MAP_MEMORY(Pico.rom); |   sh2_read16_map[0x02/2].addr = sh2_read16_map[0x22/2].addr = MAP_MEMORY(Pico.rom); | ||||||
|   sh2_read8_map[1].mask   = sh2_read8_map[5].mask   = |   sh2_read8_map[0x02/2].mask  = sh2_read8_map[0x22/2].mask  = | ||||||
|   sh2_read16_map[1].mask  = sh2_read16_map[5].mask  = 0x3fffff; // FIXME
 |   sh2_read16_map[0x02/2].mask = sh2_read16_map[0x22/2].mask = 0x3fffff; // FIXME
 | ||||||
|   // CS2 - DRAM - done by Pico32xSwapDRAM()
 |   // CS2 - DRAM - done by Pico32xSwapDRAM()
 | ||||||
|   sh2_read8_map[2].mask   = sh2_read8_map[6].mask   = |   sh2_read8_map[0x04/2].mask  = sh2_read8_map[0x24/2].mask  = | ||||||
|   sh2_read16_map[2].mask  = sh2_read16_map[6].mask  = 0x01ffff; |   sh2_read16_map[0x04/2].mask = sh2_read16_map[0x24/2].mask = 0x01ffff; | ||||||
|   // CS3 - SDRAM
 |   // CS3 - SDRAM
 | ||||||
|   sh2_read8_map[3].addr   = sh2_read8_map[7].addr   = |   sh2_read8_map[0x06/2].addr   = sh2_read8_map[0x26/2].addr   = | ||||||
|   sh2_read16_map[3].addr  = sh2_read16_map[7].addr  = MAP_MEMORY(Pico32xMem->sdram); |   sh2_read16_map[0x06/2].addr  = sh2_read16_map[0x26/2].addr  = MAP_MEMORY(Pico32xMem->sdram); | ||||||
|   sh2_write8_map[0x06/2]  = sh2_write8_map[0x26/2]  = sh2_write8_sdram; |   sh2_write8_map[0x06/2]       = sh2_write8_map[0x26/2]       = sh2_write8_sdram; | ||||||
|   sh2_write16_map[0x06/2] = sh2_write16_map[0x26/2] = sh2_write16_sdram; |   sh2_write16_map[0x06/2]      = sh2_write16_map[0x26/2]      = sh2_write16_sdram; | ||||||
|   sh2_read8_map[3].mask   = sh2_read8_map[7].mask   = |   sh2_read8_map[0x06/2].mask   = sh2_read8_map[0x26/2].mask   = | ||||||
|   sh2_read16_map[3].mask  = sh2_read16_map[7].mask  = 0x03ffff; |   sh2_read16_map[0x06/2].mask  = sh2_read16_map[0x26/2].mask  = 0x03ffff; | ||||||
|   // SH2 data array
 |   // SH2 data array
 | ||||||
|   sh2_read8_map[0x18].addr   = MAP_HANDLER(sh2_read8_da); |   sh2_read8_map[0xc0/2].addr  = MAP_HANDLER(sh2_read8_da); | ||||||
|   sh2_read16_map[0x18].addr  = MAP_HANDLER(sh2_read16_da); |   sh2_read16_map[0xc0/2].addr = MAP_HANDLER(sh2_read16_da); | ||||||
|   sh2_write8_map[0xc0/2]     = sh2_write8_da; |   sh2_write8_map[0xc0/2]      = sh2_write8_da; | ||||||
|   sh2_write16_map[0xc0/2]    = sh2_write16_da; |   sh2_write16_map[0xc0/2]     = sh2_write16_da; | ||||||
|   // SH2 IO
 |   // SH2 IO
 | ||||||
|   sh2_read8_map[0x1f].addr   = MAP_HANDLER(sh2_peripheral_read8); |   sh2_read8_map[0xff/2].addr  = MAP_HANDLER(sh2_peripheral_read8); | ||||||
|   sh2_read16_map[0x1f].addr  = MAP_HANDLER(sh2_peripheral_read16); |   sh2_read16_map[0xff/2].addr = MAP_HANDLER(sh2_peripheral_read16); | ||||||
|   sh2_write8_map[0xff/2]     = sh2_peripheral_write8; |   sh2_write8_map[0xff/2]      = sh2_peripheral_write8; | ||||||
|   sh2_write16_map[0xff/2]    = sh2_peripheral_write16; |   sh2_write16_map[0xff/2]     = sh2_peripheral_write16; | ||||||
| 
 | 
 | ||||||
|   // map DRAM area, both 68k and SH2
 |   // map DRAM area, both 68k and SH2
 | ||||||
|   Pico32xSwapDRAM(1); |   Pico32xSwapDRAM(1); | ||||||
|  |  | ||||||
|  | @ -59,7 +59,7 @@ struct dmac { | ||||||
| 
 | 
 | ||||||
| static void dmac_te_irq(SH2 *sh2, struct dma_chan *chan) | static void dmac_te_irq(SH2 *sh2, struct dma_chan *chan) | ||||||
| { | { | ||||||
|   char *regs = (void *)Pico32xMem->sh2_peri_regs[sh2->is_slave]; |   char *regs = (void *)sh2->peri_regs; | ||||||
|   struct dmac *dmac = (void *)(regs + 0x180); |   struct dmac *dmac = (void *)(regs + 0x180); | ||||||
|   int level = PREG8(regs, 0xe2) & 0x0f; // IPRA
 |   int level = PREG8(regs, 0xe2) & 0x0f; // IPRA
 | ||||||
|   int vector = (chan == &dmac->chan[0]) ? |   int vector = (chan == &dmac->chan[0]) ? | ||||||
|  | @ -167,7 +167,7 @@ void p32x_timers_recalc(void) | ||||||
| 
 | 
 | ||||||
|   // SH2 timer step
 |   // SH2 timer step
 | ||||||
|   for (i = 0; i < 2; i++) { |   for (i = 0; i < 2; i++) { | ||||||
|     tmp = PREG8(Pico32xMem->sh2_peri_regs[i], 0x80) & 7; |     tmp = PREG8(sh2s[i].peri_regs, 0x80) & 7; | ||||||
|     // Sclk cycles per timer tick
 |     // Sclk cycles per timer tick
 | ||||||
|     if (tmp) |     if (tmp) | ||||||
|       cycles = 0x20 << tmp; |       cycles = 0x20 << tmp; | ||||||
|  | @ -186,7 +186,7 @@ void p32x_timers_do(unsigned int m68k_slice) | ||||||
| 
 | 
 | ||||||
|   // WDT timers
 |   // WDT timers
 | ||||||
|   for (i = 0; i < 2; i++) { |   for (i = 0; i < 2; i++) { | ||||||
|     void *pregs = Pico32xMem->sh2_peri_regs[i]; |     void *pregs = sh2s[i].peri_regs; | ||||||
|     if (PREG8(pregs, 0x80) & 0x20) { // TME
 |     if (PREG8(pregs, 0x80) & 0x20) { // TME
 | ||||||
|       timer_cycles[i] += cycles; |       timer_cycles[i] += cycles; | ||||||
|       cnt = PREG8(pregs, 0x81); |       cnt = PREG8(pregs, 0x81); | ||||||
|  | @ -211,44 +211,48 @@ void p32x_timers_do(unsigned int m68k_slice) | ||||||
| // SH2 internal peripheral memhandlers
 | // SH2 internal peripheral memhandlers
 | ||||||
| // we keep them in little endian format
 | // we keep them in little endian format
 | ||||||
| 
 | 
 | ||||||
| u32 sh2_peripheral_read8(u32 a, int id) | u32 sh2_peripheral_read8(u32 a, SH2 *sh2) | ||||||
| { | { | ||||||
|   u8 *r = (void *)Pico32xMem->sh2_peri_regs[id]; |   u8 *r = (void *)sh2->peri_regs; | ||||||
|   u32 d; |   u32 d; | ||||||
| 
 | 
 | ||||||
|   a &= 0x1ff; |   a &= 0x1ff; | ||||||
|   d = PREG8(r, a); |   d = PREG8(r, a); | ||||||
| 
 | 
 | ||||||
|   elprintf(EL_32XP, "%csh2 peri r8  [%08x]       %02x @%06x", id ? 's' : 'm', a | ~0x1ff, d, sh2_pc(id)); |   elprintf(EL_32XP, "%csh2 peri r8  [%08x]       %02x @%06x", | ||||||
|  |     sh2->is_slave ? 's' : 'm', a | ~0x1ff, d, sh2_pc(sh2)); | ||||||
|   return d; |   return d; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| u32 sh2_peripheral_read16(u32 a, int id) | u32 sh2_peripheral_read16(u32 a, SH2 *sh2) | ||||||
| { | { | ||||||
|   u16 *r = (void *)Pico32xMem->sh2_peri_regs[id]; |   u16 *r = (void *)sh2->peri_regs; | ||||||
|   u32 d; |   u32 d; | ||||||
| 
 | 
 | ||||||
|   a &= 0x1ff; |   a &= 0x1ff; | ||||||
|   d = r[(a / 2) ^ 1]; |   d = r[(a / 2) ^ 1]; | ||||||
| 
 | 
 | ||||||
|   elprintf(EL_32XP, "%csh2 peri r16 [%08x]     %04x @%06x", id ? 's' : 'm', a | ~0x1ff, d, sh2_pc(id)); |   elprintf(EL_32XP, "%csh2 peri r16 [%08x]     %04x @%06x", | ||||||
|  |     sh2->is_slave ? 's' : 'm', a | ~0x1ff, d, sh2_pc(sh2)); | ||||||
|   return d; |   return d; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| u32 sh2_peripheral_read32(u32 a, int id) | u32 sh2_peripheral_read32(u32 a, SH2 *sh2) | ||||||
| { | { | ||||||
|   u32 d; |   u32 d; | ||||||
|   a &= 0x1fc; |   a &= 0x1fc; | ||||||
|   d = Pico32xMem->sh2_peri_regs[id][a / 4]; |   d = sh2->peri_regs[a / 4]; | ||||||
| 
 | 
 | ||||||
|   elprintf(EL_32XP, "%csh2 peri r32 [%08x] %08x @%06x", id ? 's' : 'm', a | ~0x1ff, d, sh2_pc(id)); |   elprintf(EL_32XP, "%csh2 peri r32 [%08x] %08x @%06x", | ||||||
|  |     sh2->is_slave ? 's' : 'm', a | ~0x1ff, d, sh2_pc(sh2)); | ||||||
|   return d; |   return d; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int REGPARM(3) sh2_peripheral_write8(u32 a, u32 d, int id) | void REGPARM(3) sh2_peripheral_write8(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   u8 *r = (void *)Pico32xMem->sh2_peri_regs[id]; |   u8 *r = (void *)sh2->peri_regs; | ||||||
|   elprintf(EL_32XP, "%csh2 peri w8  [%08x]       %02x @%06x", id ? 's' : 'm', a, d, sh2_pc(id)); |   elprintf(EL_32XP, "%csh2 peri w8  [%08x]       %02x @%06x", | ||||||
|  |     sh2->is_slave ? 's' : 'm', a, d, sh2_pc(sh2)); | ||||||
| 
 | 
 | ||||||
|   a &= 0x1ff; |   a &= 0x1ff; | ||||||
|   PREG8(r, a) = d; |   PREG8(r, a) = d; | ||||||
|  | @ -256,22 +260,23 @@ int REGPARM(3) sh2_peripheral_write8(u32 a, u32 d, int id) | ||||||
|   // X-men SCI hack
 |   // X-men SCI hack
 | ||||||
|   if ((a == 2 &&  (d & 0x20)) || // transmiter enabled
 |   if ((a == 2 &&  (d & 0x20)) || // transmiter enabled
 | ||||||
|       (a == 4 && !(d & 0x80))) { // valid data in TDR
 |       (a == 4 && !(d & 0x80))) { // valid data in TDR
 | ||||||
|     void *oregs = Pico32xMem->sh2_peri_regs[id ^ 1]; |     void *oregs = sh2->other_sh2->peri_regs; | ||||||
|     if ((PREG8(oregs, 2) & 0x50) == 0x50) { // receiver + irq enabled
 |     if ((PREG8(oregs, 2) & 0x50) == 0x50) { // receiver + irq enabled
 | ||||||
|       int level = PREG8(oregs, 0x60) >> 4; |       int level = PREG8(oregs, 0x60) >> 4; | ||||||
|       int vector = PREG8(oregs, 0x63) & 0x7f; |       int vector = PREG8(oregs, 0x63) & 0x7f; | ||||||
|       elprintf(EL_32XP, "%csh2 SCI recv irq (%d, %d)", (id ^ 1) ? 's' : 'm', level, vector); |       elprintf(EL_32XP, "%csh2 SCI recv irq (%d, %d)", | ||||||
|       sh2_internal_irq(&sh2s[id ^ 1], level, vector); |         (sh2->is_slave ^ 1) ? 's' : 'm', level, vector); | ||||||
|       return 1; |       sh2_internal_irq(sh2->other_sh2, level, vector); | ||||||
|  |       return; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   return 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int REGPARM(3) sh2_peripheral_write16(u32 a, u32 d, int id) | void REGPARM(3) sh2_peripheral_write16(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   u16 *r = (void *)Pico32xMem->sh2_peri_regs[id]; |   u16 *r = (void *)sh2->peri_regs; | ||||||
|   elprintf(EL_32XP, "%csh2 peri w16 [%08x]     %04x @%06x", id ? 's' : 'm', a, d, sh2_pc(id)); |   elprintf(EL_32XP, "%csh2 peri w16 [%08x]     %04x @%06x", | ||||||
|  |     sh2->is_slave ? 's' : 'm', a, d, sh2_pc(sh2)); | ||||||
| 
 | 
 | ||||||
|   a &= 0x1ff; |   a &= 0x1ff; | ||||||
| 
 | 
 | ||||||
|  | @ -283,17 +288,17 @@ int REGPARM(3) sh2_peripheral_write16(u32 a, u32 d, int id) | ||||||
|     } |     } | ||||||
|     if ((d & 0xff00) == 0x5a00) // WTCNT
 |     if ((d & 0xff00) == 0x5a00) // WTCNT
 | ||||||
|       PREG8(r, 0x81) = d; |       PREG8(r, 0x81) = d; | ||||||
|     return 0; |     return; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   r[(a / 2) ^ 1] = d; |   r[(a / 2) ^ 1] = d; | ||||||
|   return 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void sh2_peripheral_write32(u32 a, u32 d, int id) | void sh2_peripheral_write32(u32 a, u32 d, SH2 *sh2) | ||||||
| { | { | ||||||
|   u32 *r = Pico32xMem->sh2_peri_regs[id]; |   u32 *r = sh2->peri_regs; | ||||||
|   elprintf(EL_32XP, "%csh2 peri w32 [%08x] %08x @%06x", id ? 's' : 'm', a, d, sh2_pc(id)); |   elprintf(EL_32XP, "%csh2 peri w32 [%08x] %08x @%06x", | ||||||
|  |     sh2->is_slave ? 's' : 'm', a, d, sh2_pc(sh2)); | ||||||
| 
 | 
 | ||||||
|   a &= 0x1fc; |   a &= 0x1fc; | ||||||
|   r[a / 4] = d; |   r[a / 4] = d; | ||||||
|  | @ -301,7 +306,8 @@ void sh2_peripheral_write32(u32 a, u32 d, int id) | ||||||
|   switch (a) { |   switch (a) { | ||||||
|     // division unit (TODO: verify):
 |     // division unit (TODO: verify):
 | ||||||
|     case 0x104: // DVDNT: divident L, starts divide
 |     case 0x104: // DVDNT: divident L, starts divide
 | ||||||
|       elprintf(EL_32XP, "%csh2 divide %08x / %08x", id ? 's' : 'm', d, r[0x100 / 4]); |       elprintf(EL_32XP, "%csh2 divide %08x / %08x", | ||||||
|  |         sh2->is_slave ? 's' : 'm', d, r[0x100 / 4]); | ||||||
|       if (r[0x100 / 4]) { |       if (r[0x100 / 4]) { | ||||||
|         signed int divisor = r[0x100 / 4]; |         signed int divisor = r[0x100 / 4]; | ||||||
|                        r[0x118 / 4] = r[0x110 / 4] = (signed int)d % divisor; |                        r[0x118 / 4] = r[0x110 / 4] = (signed int)d % divisor; | ||||||
|  | @ -312,7 +318,7 @@ void sh2_peripheral_write32(u32 a, u32 d, int id) | ||||||
|       break; |       break; | ||||||
|     case 0x114: |     case 0x114: | ||||||
|       elprintf(EL_32XP, "%csh2 divide %08x%08x / %08x @%08x", |       elprintf(EL_32XP, "%csh2 divide %08x%08x / %08x @%08x", | ||||||
|         id ? 's' : 'm', r[0x110 / 4], d, r[0x100 / 4], sh2_pc(id)); |         sh2->is_slave ? 's' : 'm', r[0x110 / 4], d, r[0x100 / 4], sh2_pc(sh2)); | ||||||
|       if (r[0x100 / 4]) { |       if (r[0x100 / 4]) { | ||||||
|         signed long long divident = (signed long long)r[0x110 / 4] << 32 | d; |         signed long long divident = (signed long long)r[0x110 / 4] << 32 | d; | ||||||
|         signed int divisor = r[0x100 / 4]; |         signed int divisor = r[0x100 / 4]; | ||||||
|  | @ -322,7 +328,8 @@ void sh2_peripheral_write32(u32 a, u32 d, int id) | ||||||
|         r[0x11c / 4] = r[0x114 / 4] = divident; |         r[0x11c / 4] = r[0x114 / 4] = divident; | ||||||
|         divident >>= 31; |         divident >>= 31; | ||||||
|         if ((unsigned long long)divident + 1 > 1) { |         if ((unsigned long long)divident + 1 > 1) { | ||||||
|           //elprintf(EL_32XP, "%csh2 divide overflow! @%08x", id ? 's' : 'm', sh2_pc(id));
 |           //elprintf(EL_32XP, "%csh2 divide overflow! @%08x",
 | ||||||
|  |           //  sh2->is_slave ? 's' : 'm', sh2_pc(sh2));
 | ||||||
|           r[0x11c / 4] = r[0x114 / 4] = divident > 0 ? 0x7fffffff : 0x80000000; // overflow
 |           r[0x11c / 4] = r[0x114 / 4] = divident > 0 ? 0x7fffffff : 0x80000000; // overflow
 | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|  | @ -333,14 +340,14 @@ void sh2_peripheral_write32(u32 a, u32 d, int id) | ||||||
| 
 | 
 | ||||||
|   // perhaps starting a DMA?
 |   // perhaps starting a DMA?
 | ||||||
|   if (a == 0x1b0 || a == 0x18c || a == 0x19c) { |   if (a == 0x1b0 || a == 0x18c || a == 0x19c) { | ||||||
|     struct dmac *dmac = (void *)&Pico32xMem->sh2_peri_regs[id][0x180 / 4]; |     struct dmac *dmac = (void *)&sh2->peri_regs[0x180 / 4]; | ||||||
|     if (!(dmac->dmaor & DMA_DME)) |     if (!(dmac->dmaor & DMA_DME)) | ||||||
|       return; |       return; | ||||||
| 
 | 
 | ||||||
|     if ((dmac->chan[0].chcr & (DMA_TE|DMA_DE)) == DMA_DE) |     if ((dmac->chan[0].chcr & (DMA_TE|DMA_DE)) == DMA_DE) | ||||||
|       dmac_trigger(&sh2s[id], &dmac->chan[0]); |       dmac_trigger(sh2, &dmac->chan[0]); | ||||||
|     if ((dmac->chan[1].chcr & (DMA_TE|DMA_DE)) == DMA_DE) |     if ((dmac->chan[1].chcr & (DMA_TE|DMA_DE)) == DMA_DE) | ||||||
|       dmac_trigger(&sh2s[id], &dmac->chan[1]); |       dmac_trigger(sh2, &dmac->chan[1]); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -401,8 +408,8 @@ static void dreq1_do(SH2 *sh2, struct dma_chan *chan) | ||||||
| 
 | 
 | ||||||
| void p32x_dreq0_trigger(void) | void p32x_dreq0_trigger(void) | ||||||
| { | { | ||||||
|   struct dmac *mdmac = (void *)&Pico32xMem->sh2_peri_regs[0][0x180 / 4]; |   struct dmac *mdmac = (void *)&msh2.peri_regs[0x180 / 4]; | ||||||
|   struct dmac *sdmac = (void *)&Pico32xMem->sh2_peri_regs[1][0x180 / 4]; |   struct dmac *sdmac = (void *)&ssh2.peri_regs[0x180 / 4]; | ||||||
| 
 | 
 | ||||||
|   elprintf(EL_32XP, "dreq0_trigger"); |   elprintf(EL_32XP, "dreq0_trigger"); | ||||||
|   if ((mdmac->dmaor & DMA_DME) && (mdmac->chan[0].chcr & 3) == DMA_DE) { |   if ((mdmac->dmaor & DMA_DME) && (mdmac->chan[0].chcr & 3) == DMA_DE) { | ||||||
|  | @ -415,8 +422,8 @@ void p32x_dreq0_trigger(void) | ||||||
| 
 | 
 | ||||||
| void p32x_dreq1_trigger(void) | void p32x_dreq1_trigger(void) | ||||||
| { | { | ||||||
|   struct dmac *mdmac = (void *)&Pico32xMem->sh2_peri_regs[0][0x180 / 4]; |   struct dmac *mdmac = (void *)&msh2.peri_regs[0x180 / 4]; | ||||||
|   struct dmac *sdmac = (void *)&Pico32xMem->sh2_peri_regs[1][0x180 / 4]; |   struct dmac *sdmac = (void *)&ssh2.peri_regs[0x180 / 4]; | ||||||
|   int hit = 0; |   int hit = 0; | ||||||
| 
 | 
 | ||||||
|   elprintf(EL_32XP, "dreq1_trigger"); |   elprintf(EL_32XP, "dreq1_trigger"); | ||||||
|  |  | ||||||
|  | @ -83,7 +83,7 @@ char *PDebug32x(void) | ||||||
|     i*2, r[i+0], r[i+1], r[i+2], r[i+3], r[i+4], r[i+5], r[i+6], r[i+7]); MVP; |     i*2, r[i+0], r[i+1], r[i+2], r[i+3], r[i+4], r[i+5], r[i+6], r[i+7]); MVP; | ||||||
| 
 | 
 | ||||||
|   sprintf(dstrp, "                   mSH2              sSH2\n"); MVP; |   sprintf(dstrp, "                   mSH2              sSH2\n"); MVP; | ||||||
|   sprintf(dstrp, "PC,SR %08x,     %03x %08x,     %03x\n", sh2_pc(0), sh2_sr(0), sh2_pc(1), sh2_sr(1)); MVP; |   sprintf(dstrp, "PC,SR %08x,     %03x %08x,     %03x\n", sh2_pc(&msh2), sh2_sr(0), sh2_pc(&ssh2), sh2_sr(1)); MVP; | ||||||
|   for (i = 0; i < 16/2; i++) { |   for (i = 0; i < 16/2; i++) { | ||||||
|     sprintf(dstrp, "R%d,%2d %08x,%08x %08x,%08x\n", i, i + 8, |     sprintf(dstrp, "R%d,%2d %08x,%08x %08x,%08x\n", i, i + 8, | ||||||
|       sh2_reg(0,i), sh2_reg(0,i+8), sh2_reg(1,i), sh2_reg(1,i+8)); MVP; |       sh2_reg(0,i), sh2_reg(0,i+8), sh2_reg(1,i), sh2_reg(1,i+8)); MVP; | ||||||
|  | @ -345,8 +345,8 @@ void PDebugDumpMem(void) | ||||||
|     dump_ram(Pico32xMem->dram[0], "dumps/dram0.bin"); |     dump_ram(Pico32xMem->dram[0], "dumps/dram0.bin"); | ||||||
|     dump_ram(Pico32xMem->dram[1], "dumps/dram1.bin"); |     dump_ram(Pico32xMem->dram[1], "dumps/dram1.bin"); | ||||||
|     dump_ram(Pico32xMem->pal, "dumps/pal32x.bin"); |     dump_ram(Pico32xMem->pal, "dumps/pal32x.bin"); | ||||||
|     dump_ram(Pico32xMem->data_array[0], "dumps/data_array0.bin"); |     dump_ram(msh2.data_array, "dumps/data_array0.bin"); | ||||||
|     dump_ram(Pico32xMem->data_array[1], "dumps/data_array1.bin"); |     dump_ram(ssh2.data_array, "dumps/data_array1.bin"); | ||||||
|   } |   } | ||||||
| #endif | #endif | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -245,7 +245,7 @@ extern SH2 sh2s[2]; | ||||||
|   } \ |   } \ | ||||||
| } while (0) | } while (0) | ||||||
| # define sh2_cycles_left(sh2) (sh2)->icount | # define sh2_cycles_left(sh2) (sh2)->icount | ||||||
| # define sh2_pc(c)    (c) ? ssh2.ppc : msh2.ppc | # define sh2_pc(sh2) (sh2)->ppc | ||||||
| #else | #else | ||||||
| # define sh2_end_run(sh2, after_) do { \ | # define sh2_end_run(sh2, after_) do { \ | ||||||
|   int left_ = (signed int)(sh2)->sr >> 12; \ |   int left_ = (signed int)(sh2)->sr >> 12; \ | ||||||
|  | @ -256,7 +256,7 @@ extern SH2 sh2s[2]; | ||||||
|   } \ |   } \ | ||||||
| } while (0) | } while (0) | ||||||
| # define sh2_cycles_left(sh2) ((signed int)(sh2)->sr >> 12) | # define sh2_cycles_left(sh2) ((signed int)(sh2)->sr >> 12) | ||||||
| # define sh2_pc(c)    (c) ? ssh2.pc : msh2.pc | # define sh2_pc(sh2) (sh2)->pc | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| #define sh2_cycles_done(sh2) ((int)(sh2)->cycles_timeslice - sh2_cycles_left(sh2)) | #define sh2_cycles_done(sh2) ((int)(sh2)->cycles_timeslice - sh2_cycles_left(sh2)) | ||||||
|  | @ -490,6 +490,7 @@ typedef struct | ||||||
| #define SH2_DRCBLK_RAM_SHIFT 1 | #define SH2_DRCBLK_RAM_SHIFT 1 | ||||||
| #define SH2_DRCBLK_DA_SHIFT  1 | #define SH2_DRCBLK_DA_SHIFT  1 | ||||||
| 
 | 
 | ||||||
|  | #define SH2_READ_SHIFT 25 | ||||||
| #define SH2_WRITE_SHIFT 25 | #define SH2_WRITE_SHIFT 25 | ||||||
| 
 | 
 | ||||||
| struct Pico32x | struct Pico32x | ||||||
|  | @ -526,7 +527,6 @@ struct Pico32xMem | ||||||
|     unsigned char  m68k_rom[0x100]; |     unsigned char  m68k_rom[0x100]; | ||||||
|     unsigned char  m68k_rom_bank[0x10000]; // M68K_BANK_SIZE
 |     unsigned char  m68k_rom_bank[0x10000]; // M68K_BANK_SIZE
 | ||||||
|   }; |   }; | ||||||
|   unsigned char  data_array[2][0x1000]; // cache in SH2s (can be used as RAM)
 |  | ||||||
| #ifdef DRC_SH2 | #ifdef DRC_SH2 | ||||||
|   unsigned short drcblk_da[2][1 << (12 - SH2_DRCBLK_DA_SHIFT)]; |   unsigned short drcblk_da[2][1 << (12 - SH2_DRCBLK_DA_SHIFT)]; | ||||||
| #endif | #endif | ||||||
|  | @ -534,7 +534,6 @@ struct Pico32xMem | ||||||
|   unsigned char  sh2_rom_s[0x400]; |   unsigned char  sh2_rom_s[0x400]; | ||||||
|   unsigned short pal[0x100]; |   unsigned short pal[0x100]; | ||||||
|   unsigned short pal_native[0x100];     // converted to native (for renderer)
 |   unsigned short pal_native[0x100];     // converted to native (for renderer)
 | ||||||
|   unsigned int   sh2_peri_regs[2][0x200/4]; // periphereal regs of SH2s
 |  | ||||||
|   signed short   pwm[2*PWM_BUFF_LEN];   // PWM buffer for current frame
 |   signed short   pwm[2*PWM_BUFF_LEN];   // PWM buffer for current frame
 | ||||||
|   signed short   pwm_fifo[2][4];        // [0] - current, others - fifo entries
 |   signed short   pwm_fifo[2][4];        // [0] - current, others - fifo entries
 | ||||||
| }; | }; | ||||||
|  | @ -794,12 +793,12 @@ void p32x_dreq0_trigger(void); | ||||||
| void p32x_dreq1_trigger(void); | void p32x_dreq1_trigger(void); | ||||||
| void p32x_timers_recalc(void); | void p32x_timers_recalc(void); | ||||||
| void p32x_timers_do(unsigned int m68k_slice); | void p32x_timers_do(unsigned int m68k_slice); | ||||||
| unsigned int sh2_peripheral_read8(unsigned int a, int id); | unsigned int sh2_peripheral_read8(unsigned int a, SH2 *sh2); | ||||||
| unsigned int sh2_peripheral_read16(unsigned int a, int id); | unsigned int sh2_peripheral_read16(unsigned int a, SH2 *sh2); | ||||||
| unsigned int sh2_peripheral_read32(unsigned int a, int id); | unsigned int sh2_peripheral_read32(unsigned int a, SH2 *sh2); | ||||||
| int sh2_peripheral_write8(unsigned int a, unsigned int d, int id); | void sh2_peripheral_write8(unsigned int a, unsigned int d, SH2 *sh2); | ||||||
| int sh2_peripheral_write16(unsigned int a, unsigned int d, int id); | void sh2_peripheral_write16(unsigned int a, unsigned int d, SH2 *sh2); | ||||||
| void sh2_peripheral_write32(unsigned int a, unsigned int d, int id); | void sh2_peripheral_write32(unsigned int a, unsigned int d, SH2 *sh2); | ||||||
| 
 | 
 | ||||||
| #else | #else | ||||||
| #define Pico32xInit() | #define Pico32xInit() | ||||||
|  |  | ||||||
							
								
								
									
										16
									
								
								pico/state.c
									
										
									
									
									
								
							
							
						
						
									
										16
									
								
								pico/state.c
									
										
									
									
									
								
							|  | @ -318,13 +318,13 @@ static int state_save(void *file) | ||||||
| 
 | 
 | ||||||
|     sh2_pack(&sh2s[0], cpubuff); |     sh2_pack(&sh2s[0], cpubuff); | ||||||
|     CHECKED_WRITE_BUFF(CHUNK_MSH2,      cpubuff); |     CHECKED_WRITE_BUFF(CHUNK_MSH2,      cpubuff); | ||||||
|     CHECKED_WRITE_BUFF(CHUNK_MSH2_DATA, Pico32xMem->data_array[0]); |     CHECKED_WRITE_BUFF(CHUNK_MSH2_DATA, sh2s[0].data_array); | ||||||
|     CHECKED_WRITE_BUFF(CHUNK_MSH2_PERI, Pico32xMem->sh2_peri_regs[0]); |     CHECKED_WRITE_BUFF(CHUNK_MSH2_PERI, sh2s[0].peri_regs); | ||||||
| 
 | 
 | ||||||
|     sh2_pack(&sh2s[1], cpubuff); |     sh2_pack(&sh2s[1], cpubuff); | ||||||
|     CHECKED_WRITE_BUFF(CHUNK_SSH2,      cpubuff); |     CHECKED_WRITE_BUFF(CHUNK_SSH2,      cpubuff); | ||||||
|     CHECKED_WRITE_BUFF(CHUNK_SSH2_DATA, Pico32xMem->data_array[1]); |     CHECKED_WRITE_BUFF(CHUNK_SSH2_DATA, sh2s[1].data_array); | ||||||
|     CHECKED_WRITE_BUFF(CHUNK_SSH2_PERI, Pico32xMem->sh2_peri_regs[1]); |     CHECKED_WRITE_BUFF(CHUNK_SSH2_PERI, sh2s[1].peri_regs); | ||||||
| 
 | 
 | ||||||
|     CHECKED_WRITE_BUFF(CHUNK_32XSYS,    Pico32x); |     CHECKED_WRITE_BUFF(CHUNK_32XSYS,    Pico32x); | ||||||
|     CHECKED_WRITE_BUFF(CHUNK_M68K_BIOS, Pico32xMem->m68k_rom); |     CHECKED_WRITE_BUFF(CHUNK_M68K_BIOS, Pico32xMem->m68k_rom); | ||||||
|  | @ -470,10 +470,10 @@ static int state_load(void *file) | ||||||
|         sh2_unpack(&sh2s[1], buff_sh2); |         sh2_unpack(&sh2s[1], buff_sh2); | ||||||
|         break; |         break; | ||||||
| 
 | 
 | ||||||
|       case CHUNK_MSH2_DATA:   CHECKED_READ_BUFF(Pico32xMem->data_array[0]); break; |       case CHUNK_MSH2_DATA:   CHECKED_READ_BUFF(sh2s[0].data_array); break; | ||||||
|       case CHUNK_MSH2_PERI:   CHECKED_READ_BUFF(Pico32xMem->sh2_peri_regs[0]); break; |       case CHUNK_MSH2_PERI:   CHECKED_READ_BUFF(sh2s[0].peri_regs); break; | ||||||
|       case CHUNK_SSH2_DATA:   CHECKED_READ_BUFF(Pico32xMem->data_array[1]); break; |       case CHUNK_SSH2_DATA:   CHECKED_READ_BUFF(sh2s[1].data_array); break; | ||||||
|       case CHUNK_SSH2_PERI:   CHECKED_READ_BUFF(Pico32xMem->sh2_peri_regs[1]); break; |       case CHUNK_SSH2_PERI:   CHECKED_READ_BUFF(sh2s[1].peri_regs); break; | ||||||
|       case CHUNK_32XSYS:      CHECKED_READ_BUFF(Pico32x); break; |       case CHUNK_32XSYS:      CHECKED_READ_BUFF(Pico32x); break; | ||||||
|       case CHUNK_M68K_BIOS:   CHECKED_READ_BUFF(Pico32xMem->m68k_rom); break; |       case CHUNK_M68K_BIOS:   CHECKED_READ_BUFF(Pico32xMem->m68k_rom); break; | ||||||
|       case CHUNK_MSH2_BIOS:   CHECKED_READ_BUFF(Pico32xMem->sh2_rom_m); break; |       case CHUNK_MSH2_BIOS:   CHECKED_READ_BUFF(Pico32xMem->sh2_rom_m); break; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 notaz
						notaz