#include "usf/usf.h"
#include "usf/usf_internal.h"
#include "usf/barray.h"
#include "rsp_lle/rsp_lle.h"
#include "rsp_core.h"
#include "main/main.h"
#include "memory/memory.h"
#include "r4300/cp0.h"
#include "r4300/r4300_core.h"
#include "r4300/interupt.h"
#include "rdp/rdp_core.h"
#include "ri/ri_controller.h"
#include <string.h>
void dma_sp_write(struct rsp_core* sp)
{
unsigned int i,j;
unsigned int l = sp->regs[SP_RD_LEN_REG];
unsigned int length = ((l & 0xfff) | 7) + 1;
unsigned int count = ((l >> 12) & 0xff) + 1;
unsigned int skip = ((l >> 20) & 0xfff);
unsigned int memaddr = sp->regs[SP_MEM_ADDR_REG] & 0xfff;
unsigned int dramaddr = sp->regs[SP_DRAM_ADDR_REG] & 0xffffff;
unsigned char *spmem = (unsigned char*)sp->mem + (sp->regs[SP_MEM_ADDR_REG] & 0x1000);
unsigned char *dram = (unsigned char*)sp->ri->rdram.dram;
if (sp->r4300->state->enable_trimming_mode) {
for(j=0; j<count; j++) {
for(i=0; i<length; i++) {
spmem[memaddr^S8] = dram[dramaddr^S8];
if (!bit_array_test(sp->r4300->state->barray_ram_written_first, dramaddr / 4))
bit_array_set(sp->r4300->state->barray_ram_read, dramaddr / 4);
memaddr++;
dramaddr++;
}
dramaddr+=skip;
}
} else {
for(j=0; j<count; j++) {
for(i=0; i<length; i++) {
spmem[memaddr^S8] = dram[dramaddr^S8];
memaddr++;
dramaddr++;
}
dramaddr+=skip;
}
}
}
void dma_sp_read(struct rsp_core* sp)
{
unsigned int i,j;
unsigned int l = sp->regs[SP_WR_LEN_REG];
unsigned int length = ((l & 0xfff) | 7) + 1;
unsigned int count = ((l >> 12) & 0xff) + 1;
unsigned int skip = ((l >> 20) & 0xfff);
unsigned int memaddr = sp->regs[SP_MEM_ADDR_REG] & 0xfff;
unsigned int dramaddr = sp->regs[SP_DRAM_ADDR_REG] & 0xffffff;
unsigned char *spmem = (unsigned char*)sp->mem + (sp->regs[SP_MEM_ADDR_REG] & 0x1000);
unsigned char *dram = (unsigned char*)sp->ri->rdram.dram;
if (sp->r4300->state->enable_trimming_mode) {
for(j=0; j<count; j++) {
for(i=0; i<length; i++) {
dram[dramaddr^S8] = spmem[memaddr^S8];
if (!bit_array_test(sp->r4300->state->barray_ram_read, dramaddr / 4))
bit_array_set(sp->r4300->state->barray_ram_written_first, dramaddr / 4);
memaddr++;
dramaddr++;
}
dramaddr+=skip;
}
} else {
for(j=0; j<count; j++) {
for(i=0; i<length; i++) {
dram[dramaddr^S8] = spmem[memaddr^S8];
memaddr++;
dramaddr++;
}
dramaddr+=skip;
}
}
}
static void update_sp_status(struct rsp_core* sp, uint32_t w)
{
if (w & 0x1) sp->regs[SP_STATUS_REG] &= ~0x1;
if (w & 0x2) sp->regs[SP_STATUS_REG] |= 0x1;
if (w & 0x4) sp->regs[SP_STATUS_REG] &= ~0x2;
if (w & 0x8)
{
clear_rcp_interrupt(sp->r4300, MI_INTR_SP);
}
if (w & 0x10)
{
signal_rcp_interrupt(sp->r4300, MI_INTR_SP);
}
if (w & 0x20) sp->regs[SP_STATUS_REG] &= ~0x20;
if (w & 0x40) sp->regs[SP_STATUS_REG] |= 0x20;
if (w & 0x80) sp->regs[SP_STATUS_REG] &= ~0x40;
if (w & 0x100) sp->regs[SP_STATUS_REG] |= 0x40;
if (w & 0x200) sp->regs[SP_STATUS_REG] &= ~0x80;
if (w & 0x400) sp->regs[SP_STATUS_REG] |= 0x80;
if (w & 0x800) sp->regs[SP_STATUS_REG] &= ~0x100;
if (w & 0x1000) sp->regs[SP_STATUS_REG] |= 0x100;
if (w & 0x2000) sp->regs[SP_STATUS_REG] &= ~0x200;
if (w & 0x4000) sp->regs[SP_STATUS_REG] |= 0x200;
if (w & 0x8000) sp->regs[SP_STATUS_REG] &= ~0x400;
if (w & 0x10000) sp->regs[SP_STATUS_REG] |= 0x400;
if (w & 0x20000) sp->regs[SP_STATUS_REG] &= ~0x800;
if (w & 0x40000) sp->regs[SP_STATUS_REG] |= 0x800;
if (w & 0x80000) sp->regs[SP_STATUS_REG] &= ~0x1000;
if (w & 0x100000) sp->regs[SP_STATUS_REG] |= 0x1000;
if (w & 0x200000) sp->regs[SP_STATUS_REG] &= ~0x2000;
if (w & 0x400000) sp->regs[SP_STATUS_REG] |= 0x2000;
if (w & 0x800000) sp->regs[SP_STATUS_REG] &= ~0x4000;
if (w & 0x1000000) sp->regs[SP_STATUS_REG] |= 0x4000;
if (!(w & 0x1) && !(w & 0x4))
return;
if (!(sp->regs[SP_STATUS_REG] & 0x3)) do_SP_Task(sp);
}
void connect_rsp(struct rsp_core* sp,
struct r4300_core* r4300,
struct rdp_core* dp,
struct ri_controller* ri)
{
sp->r4300 = r4300;
sp->dp = dp;
sp->ri = ri;
init_rsp_lle(r4300->state);
}
void init_rsp(struct rsp_core* sp)
{
memset(sp->mem, 0, SP_MEM_SIZE);
memset(sp->regs, 0, SP_REGS_COUNT*sizeof(uint32_t));
memset(sp->regs2, 0, SP_REGS2_COUNT*sizeof(uint32_t));
sp->regs[SP_STATUS_REG] = 1;
}
int read_rsp_mem(void* opaque, uint32_t address, uint32_t* value)
{
struct rsp_core* sp = (struct rsp_core*)opaque;
uint32_t addr = rsp_mem_address(address);
*value = sp->mem[addr];
return 0;
}
int write_rsp_mem(void* opaque, uint32_t address, uint32_t value, uint32_t mask)
{
struct rsp_core* sp = (struct rsp_core*)opaque;
uint32_t addr = rsp_mem_address(address);
masked_write(&sp->mem[addr], value, mask);
return 0;
}
int read_rsp_regs(void* opaque, uint32_t address, uint32_t* value)
{
struct rsp_core* sp = (struct rsp_core*)opaque;
uint32_t reg = rsp_reg(address);
*value = sp->regs[reg];
if (reg == SP_SEMAPHORE_REG)
{
sp->regs[SP_SEMAPHORE_REG] = 1;
}
return 0;
}
int write_rsp_regs(void* opaque, uint32_t address, uint32_t value, uint32_t mask)
{
struct rsp_core* sp = (struct rsp_core*)opaque;
uint32_t reg = rsp_reg(address);
switch(reg)
{
case SP_STATUS_REG:
update_sp_status(sp, value & mask);
case SP_DMA_FULL_REG:
case SP_DMA_BUSY_REG:
return 0;
}
masked_write(&sp->regs[reg], value, mask);
switch(reg)
{
case SP_RD_LEN_REG:
dma_sp_write(sp);
break;
case SP_WR_LEN_REG:
dma_sp_read(sp);
break;
case SP_SEMAPHORE_REG:
sp->regs[SP_SEMAPHORE_REG] = 0;
break;
}
return 0;
}
int read_rsp_regs2(void* opaque, uint32_t address, uint32_t* value)
{
struct rsp_core* sp = (struct rsp_core*)opaque;
uint32_t reg = rsp_reg2(address);
*value = sp->regs2[reg];
return 0;
}
int write_rsp_regs2(void* opaque, uint32_t address, uint32_t value, uint32_t mask)
{
struct rsp_core* sp = (struct rsp_core*)opaque;
uint32_t reg = rsp_reg2(address);
masked_write(&sp->regs2[reg], value, mask);
return 0;
}
void do_SP_Task(struct rsp_core* sp)
{
#ifdef DEBUG_INFO
fprintf(sp->r4300->state->debug_log, "RSP Task");
#endif
uint32_t save_pc = sp->regs2[SP_PC_REG] & ~0xfff;
if (sp->mem[0xfc0/4] == 1)
{
#ifdef DEBUG_INFO
fprintf(sp->r4300->state->debug_log, " - DList");
#endif
if (sp->dp->dpc_regs[DPC_STATUS_REG] & 0x2) {
#ifdef DEBUG_INFO
fprintf(sp->r4300->state->debug_log, " - frozen!\n");
#endif
return;
}
sp->regs2[SP_PC_REG] &= 0xfff;
real_run_rsp(sp->r4300->state, 0xffffffff);
sp->regs2[SP_PC_REG] |= save_pc;
update_count(sp->r4300->state);
if (sp->r4300->mi.regs[MI_INTR_REG] & MI_INTR_SP)
add_interupt_event(sp->r4300->state, SP_INT, sp->r4300->state->g_delay_sp ? 1000 : 0);
if (sp->r4300->mi.regs[MI_INTR_REG] & MI_INTR_DP)
add_interupt_event(sp->r4300->state, DP_INT, sp->r4300->state->g_delay_dp ? 1000 : 0);
#ifdef DEBUG_INFO
if (sp->r4300->mi.regs[MI_INTR_REG])
fprintf(sp->r4300->state->debug_log, " - interrupts fired %d", sp->r4300->mi.regs[MI_INTR_REG]);
#endif
sp->r4300->mi.regs[MI_INTR_REG] &= ~(MI_INTR_SP | MI_INTR_DP);
sp->regs[SP_STATUS_REG] &= ~0x303;
}
else if (sp->mem[0xfc0/4] == 2)
{
#ifdef DEBUG_INFO
fprintf(sp->r4300->state->debug_log, " - AList");
#endif
sp->regs2[SP_PC_REG] &= 0xfff;
real_run_rsp(sp->r4300->state, 0xffffffff);
sp->regs2[SP_PC_REG] |= save_pc;
update_count(sp->r4300->state);
if (sp->r4300->mi.regs[MI_INTR_REG] & MI_INTR_SP)
add_interupt_event(sp->r4300->state, SP_INT, sp->r4300->state->g_delay_sp ? 4000: 0);
#ifdef DEBUG_INFO
if (sp->r4300->mi.regs[MI_INTR_REG])
fprintf(sp->r4300->state->debug_log, " - interrupt fired %d", sp->r4300->mi.regs[MI_INTR_REG]);
#endif
sp->r4300->mi.regs[MI_INTR_REG] &= ~MI_INTR_SP;
sp->regs[SP_STATUS_REG] &= ~0x303;
}
else
{
#ifdef DEBUG_INFO
fprintf(sp->r4300->state->debug_log, " - Unknown task");
#endif
sp->regs2[SP_PC_REG] &= 0xfff;
real_run_rsp(sp->r4300->state, 0xffffffff);
sp->regs2[SP_PC_REG] |= save_pc;
update_count(sp->r4300->state);
if (sp->r4300->mi.regs[MI_INTR_REG] & MI_INTR_SP)
{
add_interupt_event(sp->r4300->state, SP_INT, 0);
}
#ifdef DEBUG_INFO
if (sp->r4300->mi.regs[MI_INTR_REG])
fprintf(sp->r4300->state->debug_log, " - interrupt fired %d", sp->r4300->mi.regs[MI_INTR_REG]);
#endif
sp->r4300->mi.regs[MI_INTR_REG] &= ~MI_INTR_SP;
sp->regs[SP_STATUS_REG] &= ~0x203;
}
#ifdef DEBUG_INFO
fprintf(sp->r4300->state->debug_log, "\n");
#endif
}
void rsp_interrupt_event(struct rsp_core* sp)
{
sp->regs[SP_STATUS_REG] |= 0x203;
if ((sp->regs[SP_STATUS_REG] & 0x40) != 0)
{
raise_rcp_interrupt(sp->r4300, MI_INTR_SP);
}
}