#include "libultra_internal.h" #include "macros.h" #include "PR/os.h" #if defined(VERSION_EU) || defined(VERSION_SH) || defined(VERSION_CN) #include "new_func.h" #include "PR/rcp.h" #include "PR/ique.h" #define ALIGN16(val) (((val) + 0xF) & ~0xF) void __osDevMgrMain(void *args) { #ifdef VERSION_CN s32 loadedToTempBuffer = FALSE; #endif OSIoMesg *mb; OSMesg em; OSMesg dummy; s32 ret; OSMgrArgs *sp34; #ifdef VERSION_EU UNUSED u32 sp30; #endif u32 sp2c; __OSBlockInfo *sp28; __OSTranxInfo *sp24; #if defined(VERSION_SH) || defined(VERSION_CN) u32 tmp; #endif #ifdef VERSION_EU sp30 = 0; #endif sp2c = 0; mb = NULL; ret = 0; sp34 = (OSMgrArgs *) args; while (TRUE) { osRecvMesg(sp34->cmdQueue, (OSMesg) &mb, OS_MESG_BLOCK); if (mb->piHandle != NULL && mb->piHandle->type == 2 && (mb->piHandle->transferInfo.cmdType == 0 || mb->piHandle->transferInfo.cmdType == 1)) { sp24 = &mb->piHandle->transferInfo; sp28 = &sp24->block[sp24->blockNum]; sp24->sectorNum = -1; if (sp24->transferMode != 3) { sp28->dramAddr = (void *) ((u32) sp28->dramAddr - sp28->sectorSize); } if (sp24->transferMode == 2 && mb->piHandle->transferInfo.cmdType == 0) { sp2c = 1; } else { sp2c = 0; } osRecvMesg(sp34->accessQueue, &dummy, OS_MESG_BLOCK); __osResetGlobalIntMask(OS_IM_PI); osEPiRawWriteIo(mb->piHandle, 0x05000510, (sp24->bmCtlShadow | 0x80000000)); l1: osRecvMesg(sp34->eventQueue, &em, OS_MESG_BLOCK); #if defined(VERSION_SH) || defined(VERSION_CN) sp24 = &mb->piHandle->transferInfo; sp28 = &sp24->block[sp24->blockNum]; if (sp28->errStatus == 0x1D) { osEPiRawWriteIo(mb->piHandle, 0x5000510, sp24->bmCtlShadow | 0x10000000); osEPiRawWriteIo(mb->piHandle, 0x5000510, sp24->bmCtlShadow); osEPiRawReadIo(mb->piHandle, 0x5000508, &tmp); if ((tmp & 0x2000000) != 0) { osEPiRawWriteIo(mb->piHandle, 0x5000510, sp24->bmCtlShadow | 0x1000000); } sp28->errStatus = 4; IO_WRITE(PI_STATUS_REG, PI_STATUS_CLR_INTR); __osSetGlobalIntMask(0x100C01); } osSendMesg(mb->hdr.retQueue, mb, OS_MESG_NOBLOCK); if (sp2c == 1 && mb->piHandle->transferInfo.block[0].errStatus == 0) { sp2c = 0; goto l1; } #else sp30 = osSendMesg(mb->hdr.retQueue, mb, OS_MESG_NOBLOCK); if (sp2c == 1 && mb->piHandle->transferInfo.errStatus == 0) { sp2c = 0; goto l1; } #endif osSendMesg(sp34->accessQueue, NULL, OS_MESG_NOBLOCK); if (mb->piHandle->transferInfo.blockNum == 1) { osYieldThread(); } } else { switch (mb->hdr.type) { case 11: osRecvMesg(sp34->accessQueue, &dummy, OS_MESG_BLOCK); #ifdef VERSION_CN if (__osBbIsBb == 1 && ((uintptr_t) mb->dramAddr & 0x7f) >= 0x60) { loadedToTempBuffer = TRUE; ret = sp34->dma_func(OS_READ, mb->devAddr, (void *) 0x80600000, mb->size); } else #endif ret = sp34->dma_func(OS_READ, mb->devAddr, mb->dramAddr, mb->size); break; case 12: osRecvMesg(sp34->accessQueue, &dummy, OS_MESG_BLOCK); ret = sp34->dma_func(OS_WRITE, mb->devAddr, mb->dramAddr, mb->size); break; case 15: osRecvMesg(sp34->accessQueue, &dummy, OS_MESG_BLOCK); #ifdef VERSION_CN if (__osBbIsBb == 1 && ((uintptr_t) mb->dramAddr & 0x7f) >= 0x60) { loadedToTempBuffer = TRUE; ret = sp34->edma_func(mb->piHandle, OS_READ, mb->devAddr, (void *) 0x80600000, mb->size); } else #endif ret = sp34->edma_func(mb->piHandle, OS_READ, mb->devAddr, mb->dramAddr, mb->size); break; case 16: osRecvMesg(sp34->accessQueue, &dummy, OS_MESG_BLOCK); ret = sp34->edma_func(mb->piHandle, OS_WRITE, mb->devAddr, mb->dramAddr, mb->size); break; case 10: osSendMesg(mb->hdr.retQueue, mb, OS_MESG_NOBLOCK); ret = -1; break; break; default: ret = -1; break; } if (ret == 0) { osRecvMesg(sp34->eventQueue, &em, OS_MESG_BLOCK); #ifdef VERSION_CN if (__osBbIsBb == 1 && loadedToTempBuffer) { osInvalDCache((void *) 0x80600000, ALIGN16(mb->size)); bcopy((void *) 0x80600000, mb->dramAddr, mb->size); osWritebackDCache(mb->dramAddr, mb->size); loadedToTempBuffer = FALSE; } #endif #ifdef VERSION_EU sp30 = #endif osSendMesg(mb->hdr.retQueue, mb, OS_MESG_NOBLOCK); osSendMesg(sp34->accessQueue, NULL, OS_MESG_NOBLOCK); } } } } #else void __osDevMgrMain(void *args) { OSIoMesg *sp34; OSMesg sp30; OSMesg sp2c; s32 sp28; OSMgrArgs *sp24; sp34 = NULL; sp28 = 0; sp24 = (OSMgrArgs *) args; while (TRUE) { osRecvMesg(sp24->cmdQueue, (OSMesg) &sp34, OS_MESG_BLOCK); switch (sp34->hdr.type) { case 11: osRecvMesg(sp24->accessQueue, &sp2c, OS_MESG_BLOCK); sp28 = sp24->dma_func(OS_READ, sp34->devAddr, sp34->dramAddr, sp34->size); break; case 12: osRecvMesg(sp24->accessQueue, &sp2c, OS_MESG_BLOCK); sp28 = sp24->dma_func(OS_WRITE, sp34->devAddr, sp34->dramAddr, sp34->size); break; case 10: osSendMesg(sp34->hdr.retQueue, sp34, OS_MESG_NOBLOCK); sp28 = -1; break; default: sp28 = -1; break; } if (sp28 == 0) { osRecvMesg(sp24->eventQueue, &sp30, OS_MESG_BLOCK); osSendMesg(sp34->hdr.retQueue, sp34, OS_MESG_NOBLOCK); osSendMesg(sp24->accessQueue, NULL, OS_MESG_NOBLOCK); } } } #endif