jak-project/game/overlord/jak2/iso_queue.cpp

607 lines
16 KiB
C++
Raw Permalink Normal View History

#include "iso_queue.h"
#include <cstring>
#include "common/log/log.h"
#include "common/util/Assert.h"
#include "game/overlord/common/iso.h"
#include "game/overlord/jak2/dma.h"
#include "game/overlord/jak2/iso.h"
#include "game/overlord/jak2/spustreams.h"
#include "game/overlord/jak2/vag.h"
#include "game/sce/iop.h"
#include "game/sound/sndshim.h"
using namespace iop;
namespace jak2 {
/// The data structure containing all memory pages.
PageList* SpMemoryBuffers = nullptr;
u8* ScratchPadMemory = nullptr;
constexpr int N_BUFFERS = 3;
constexpr int N_STR_BUFFERS = 8;
static Buffer sBuffer[N_BUFFERS];
static Buffer sStrBuffer[N_STR_BUFFERS];
static Buffer* sFreeBuffer = nullptr;
static Buffer* sFreeStrBuffer = nullptr;
u32 BuffersAlloc = 0;
u32 StrBuffersAlloc = 0;
u32 AllocdBuffersCount = 0;
u32 NextBuffer = 0;
u32 AllocdStrBuffersCount = 0;
u32 NextStrBuffer = 0;
int sSema = 0;
// note that these are different vag commands from the VagCmds array.
// These are used for the return values of GetVagCommand, and are used for queued commands.
// The VagCmds array is used for commands that are actually running.
u32 vag_cmd_cnt = 0;
VagCmd vag_cmds[32];
u32 vag_cmd_used = 0;
u32 max_vag_cmd_cnt = 0;
PriStackEntry gPriStack[N_PRIORITIES];
std::string gPriEntryNames[N_PRIORITIES][PRI_STACK_LENGTH]; // my addition for debug
void ReturnMessage(CmdHeader* param_1);
void FreeVAGCommand(VagCmd* param_1);
void FreeDataBuffer(u32* param_1, u32 param_2);
void iso_queue_init_globals() {
memset(sBuffer, 0, sizeof(sBuffer));
memset(gPriStack, 0, sizeof(gPriStack));
memset(vag_cmds, 0, sizeof(vag_cmds));
ScratchPadMemory = nullptr;
SpMemoryBuffers = nullptr;
sFreeBuffer = nullptr;
sFreeStrBuffer = nullptr;
BuffersAlloc = 0;
StrBuffersAlloc = 0;
AllocdBuffersCount = 0;
NextBuffer = 0;
sSema = 0;
vag_cmd_cnt = 0;
vag_cmd_used = 0;
max_vag_cmd_cnt = 0;
}
void InitBuffers() {
SpMemoryBuffers = (PageList*)ScratchPadMemory;
ScratchPadMemory += sizeof(PageList);
InitPagedMemory(SpMemoryBuffers, 0x12, 0x8000);
for (int i = 0; i < N_BUFFERS; i++) {
sBuffer[i].next = &sBuffer[i + 1];
sBuffer[i].decomp_buffer = nullptr;
sBuffer[i].decompressed_size = 0;
sBuffer[i].unk_12 = 0;
sBuffer[i].data_buffer_idx = -1;
sBuffer[i].use_mode = 1;
sBuffer[i].plist = SpMemoryBuffers;
sBuffer[i].num_pages = 1;
sBuffer[i].unk_32 = 0;
sBuffer[i].free_pages = 0;
sBuffer[i].page = nullptr;
sBuffer[i].unk_44 = 0;
};
sBuffer[N_BUFFERS - 1].next = nullptr;
sFreeBuffer = sBuffer;
BuffersAlloc = 0;
for (int i = 0; i < N_STR_BUFFERS; i++) {
sStrBuffer[i].next = &sStrBuffer[i + 1];
sStrBuffer[i].decomp_buffer = nullptr;
sStrBuffer[i].decompressed_size = 0;
sStrBuffer[i].unk_12 = 0;
sStrBuffer[i].data_buffer_idx = -1;
sStrBuffer[i].use_mode = 2;
sStrBuffer[i].plist = SpMemoryBuffers;
sStrBuffer[i].num_pages = 1;
sStrBuffer[i].unk_32 = 2;
sStrBuffer[i].free_pages = 0;
sStrBuffer[i].page = nullptr;
sStrBuffer[i].unk_44 = 0;
};
sStrBuffer[N_STR_BUFFERS - 1].next = nullptr;
sFreeStrBuffer = sStrBuffer;
StrBuffersAlloc = 0;
StreamSRAM[0] = 0x5040;
TrapSRAM[0] = 0x9040;
snd_SRAMMarkUsed(0x5040, 0x4040);
StreamSRAM[1] = 0x9080;
TrapSRAM[1] = 0xd080;
snd_SRAMMarkUsed(0x9080, 0x4040);
StreamSRAM[2] = 0xd0c0;
TrapSRAM[2] = 0x110c0;
snd_SRAMMarkUsed(0xd0c0, 0x4040);
StreamSRAM[3] = 0x11100;
TrapSRAM[3] = 0x15100;
snd_SRAMMarkUsed(0x11100, 0x4040);
for (int i = 0; i < 4; i++) {
if (DMA_SendToSPUAndSync(VAG_SilentLoop, 0x30, TrapSRAM[i], 0, 1)) {
ASSERT_NOT_REACHED();
break;
}
DelayThread(1000);
}
SemaParam param;
param.attr = 1;
param.init_count = 1;
param.max_count = 1;
param.option = 0;
sSema = CreateSema(&param);
if (sSema < 0) {
printf("IOP: ======================================================================\n");
printf("IOP: iso_queue InitBuffers: Can\'t create semaphore\n");
printf("IOP: ======================================================================\n");
ASSERT_NOT_REACHED();
}
}
u32 AllocDataBuffer(u32* param_1, u32 param_2) {
bool bVar1;
u32 uVar2;
u32 uVar3;
int iVar4;
u32 uVar5;
uVar5 = 0xffffffff;
bVar1 = false;
uVar3 = uVar5;
if (param_1 == &BuffersAlloc) {
if (AllocdBuffersCount < param_2) {
if ((BuffersAlloc & 1 << (NextBuffer & 0x1f)) == 0) {
bVar1 = true;
} else {
iVar4 = 0;
if (0 < (int)param_2) {
do {
NextBuffer = NextBuffer + 1;
if (param_2 <= NextBuffer) {
NextBuffer = 0;
}
iVar4 = iVar4 + 1;
if ((BuffersAlloc & 1 << (NextBuffer & 0x1f)) == 0) {
bVar1 = true;
iVar4 = param_2 + 1;
}
} while (iVar4 < (int)param_2);
}
}
uVar5 = NextBuffer;
if (bVar1) {
BuffersAlloc = BuffersAlloc | 1 << (NextBuffer & 0x1f);
AllocdBuffersCount = AllocdBuffersCount + 1;
NextBuffer = NextBuffer + 1;
uVar3 = uVar5;
if (param_2 <= NextBuffer) {
NextBuffer = 0;
}
}
}
} else {
uVar3 = 0xffffffff;
if ((param_1 == &StrBuffersAlloc) && (uVar3 = uVar5, AllocdStrBuffersCount < param_2)) {
if ((StrBuffersAlloc & 1 << (NextStrBuffer & 0x1f)) == 0) {
bVar1 = true;
} else {
iVar4 = 0;
if (0 < (int)param_2) {
do {
NextStrBuffer = NextStrBuffer + 1;
if (param_2 <= NextStrBuffer) {
NextStrBuffer = 0;
}
iVar4 = iVar4 + 1;
if ((StrBuffersAlloc & 1 << (NextStrBuffer & 0x1f)) == 0) {
bVar1 = true;
iVar4 = param_2 + 1;
}
} while (iVar4 < (int)param_2);
}
}
uVar2 = NextStrBuffer;
if (bVar1) {
StrBuffersAlloc = StrBuffersAlloc | 1 << (NextStrBuffer & 0x1f);
AllocdStrBuffersCount = AllocdStrBuffersCount + 1;
NextStrBuffer = NextStrBuffer + 1;
uVar3 = uVar2;
if (param_2 <= NextStrBuffer) {
NextStrBuffer = 0;
}
}
}
}
return uVar3;
}
Buffer* AllocateBuffer(int param_1, VagCmd* param_2, int /*param_3*/) {
PageList** ppPVar1;
int* piVar2;
int iVar3;
int iVar4;
int iVar5;
Buffer* pBVar6;
Buffer* pBVar7;
Page* pPVar8;
// if (param_3 == 1) {
// CpuSuspendIntr(local_28);
//}
pBVar6 = (Buffer*)0x0;
pPVar8 = (Page*)0x0;
if (param_1 == 1) {
if ((sFreeBuffer != (Buffer*)0x0) &&
(iVar3 = AllocDataBuffer(&BuffersAlloc, 3), pBVar7 = sFreeBuffer, -1 < iVar3)) {
ppPVar1 = &sFreeBuffer->plist;
piVar2 = &sFreeBuffer->num_pages;
sFreeBuffer->data_buffer_idx = iVar3;
sFreeBuffer->use_mode = 1;
sFreeBuffer = sFreeBuffer->next;
pPVar8 = (Page*)AllocPages(*ppPVar1, *piVar2);
if (pPVar8 != (Page*)0x0) {
pBVar7->page = pPVar8;
pBVar7->free_pages = pPVar8->free_pages;
pBVar7->decomp_buffer = (uint8_t*)pPVar8->buffer;
}
goto LAB_00006a0c;
}
LAB_00006a28:
pBVar7 = pBVar6;
if (pPVar8 != (Page*)0x0)
goto LAB_00006a44;
} else {
if (((param_1 != 2) || (sFreeStrBuffer == (Buffer*)0x0)) ||
(iVar3 = AllocDataBuffer(&StrBuffersAlloc, 8), pBVar7 = sFreeStrBuffer, iVar3 < 0))
goto LAB_00006a28;
sFreeStrBuffer->data_buffer_idx = iVar3;
sFreeStrBuffer->use_mode = 2;
pBVar6 = (param_2->header).callback_buffer;
pPVar8 = (Page*)0x0;
if (param_2->xfer_size == 0) {
iVar5 = sFreeStrBuffer->num_pages;
} else {
iVar4 = sFreeStrBuffer->plist->page_size;
iVar3 = param_2->xfer_size + iVar4 + -1;
if (iVar4 == 0) {
ASSERT_NOT_REACHED();
// trap(0x1c00);
}
if (pBVar6 == (Buffer*)0x0) {
iVar5 = 0;
} else {
iVar5 = -pBVar6->page->free_pages;
}
iVar5 = iVar3 / iVar4 + iVar5;
if (iVar5 < 0) {
iVar5 = 0;
}
}
pBVar6 = sFreeStrBuffer->next;
if (iVar5 != 0) {
if ((param_2->status_bytes[BYTE10] == '\0') || (param_2->unk_232 == '\0')) {
if (param_2->status_bytes[BYTE4] == '\0') {
iVar3 = sFreeStrBuffer->num_pages;
} else {
iVar3 = sFreeStrBuffer->unk_32;
}
if (iVar5 < iVar3) {
iVar3 = iVar5;
}
} else {
iVar3 = iVar5;
if (3 < iVar5) {
iVar3 = 3;
}
}
ppPVar1 = &sFreeStrBuffer->plist;
sFreeStrBuffer = sFreeStrBuffer->next;
pPVar8 = (Page*)AllocPages(*ppPVar1, iVar3);
pBVar6 = sFreeStrBuffer;
}
sFreeStrBuffer = pBVar6;
if (pPVar8 != (Page*)0x0) {
pBVar7->page = pPVar8;
pBVar7->free_pages = pPVar8->free_pages;
pBVar7->decomp_buffer = (uint8_t*)pPVar8->buffer;
}
LAB_00006a0c:
if (pPVar8 != (Page*)0x0) {
pBVar7->decompressed_size = 0;
pBVar7->next = (Buffer*)0x0;
pBVar7->unk_12 = pPVar8->buffer;
pBVar6 = pBVar7;
goto LAB_00006a28;
}
}
if (pBVar7) {
FreeBuffer(pBVar7, 0);
pBVar7 = (Buffer*)0x0;
}
LAB_00006a44:
// if (param_3 == 1) {
// CpuResumeIntr(local_28[0]);
//}
return pBVar7;
}
void FreeBuffer(Buffer* buf, int /*param_2*/) {
// if (param_2 == 1) {
// CpuSuspendIntr(local_18);
//}
if (buf->use_mode == 1) {
if ((BuffersAlloc & 1 << (buf->data_buffer_idx & 0x1fU)) != 0) {
buf->page = FreePagesList(buf->plist, buf->page);
FreeDataBuffer(&BuffersAlloc, buf->data_buffer_idx);
buf->next = sFreeBuffer;
buf->decompressed_size = 0;
buf->unk_12 = 0;
sFreeBuffer = buf;
buf->use_mode = 0;
buf->data_buffer_idx = -1;
}
} else if (buf->use_mode == 2) {
if ((StrBuffersAlloc & 1 << (buf->data_buffer_idx & 0x1fU)) != 0) {
buf->page = FreePagesList(buf->plist, buf->page);
FreeDataBuffer(&StrBuffersAlloc, buf->data_buffer_idx);
buf->next = sFreeStrBuffer;
buf->decompressed_size = 0;
buf->unk_12 = 0;
sFreeStrBuffer = buf;
buf->use_mode = 0;
buf->data_buffer_idx = -1;
}
}
// if (param_2 == 1) {
// CpuResumeIntr(local_18[0]);
//}
}
void ReleaseMessage(CmdHeader* cmd, int suspend_irq) {
while (cmd->callback_buffer != nullptr) {
auto cb_buf = cmd->callback_buffer;
cmd->callback_buffer = cb_buf->next;
FreeBuffer(cb_buf, suspend_irq);
}
if (cmd->lse != nullptr) {
isofs->close(cmd->lse);
}
UnqueueMessage(cmd, suspend_irq);
}
void DisplayQueue() {
for (int pri = 0; pri < N_PRIORITIES; pri++) {
for (int cmd = 0; cmd < (int)gPriStack[pri].count; cmd++) {
lg::debug(" PRI {} elt {} {} @ #x{:X}", pri, cmd, gPriEntryNames[pri][cmd],
(u64)gPriStack[pri].entries[cmd]);
}
}
}
int QueueMessage(CmdHeader* param_1, int param_2, const char* param_3, int param_4) {
int uVar1;
// undefined4 local_20[2];
if (param_4 == 1) {
// CpuSuspendIntr(local_20);
}
if (gPriStack[param_2].count == 8) {
param_1->status = 2;
// CpuResumeIntr(local_20[0]);
ReturnMessage(param_1);
uVar1 = 0;
} else {
gPriStack[param_2].entries[gPriStack[param_2].count] = param_1;
gPriEntryNames[param_2][gPriStack[param_2].count] = param_3;
gPriStack[param_2].count = gPriStack[param_2].count + 1;
if (param_4 == 1) {
// CpuResumeIntr(local_20[0]);
}
uVar1 = 1;
}
return uVar1;
}
void UnqueueMessage(CmdHeader* cmd, int suspend_irq) {
if (suspend_irq == 1) {
// CpuSuspendIntr(&stat);
}
for (int pri = 0; pri < N_PRIORITIES; pri++) {
auto pse = &gPriStack[pri];
for (int idx = 0; idx < pse->count; idx++) {
if (pse->entries[idx] == cmd) {
pse->count--;
while (idx < pse->count) {
pse->entries[idx] = pse->entries[idx + 1];
idx++;
}
goto exit;
}
}
}
exit:
if (suspend_irq == 1) {
// CpuResumeIntr(stat);
}
}
CmdHeader* GetMessage() {
for (int pri = (N_PRIORITIES - 1); pri >= 0; pri--) {
auto pse = &gPriStack[pri];
int idx = pse->count;
for (idx = idx - 1; idx >= 0; idx--) {
auto cmd = pse->entries[idx];
if (cmd->lse && (u32)cmd->status == CMD_STATUS_IN_PROGRESS && cmd->ready_for_data) {
if (cmd->callback_buffer == nullptr) {
return cmd;
}
if (cmd->callback_buffer->next == nullptr) {
return cmd;
}
}
}
}
return nullptr;
}
void ProcessMessageData() {
int iVar1;
CmdHeader* pCVar2;
Buffer* pBVar3;
int iVar4;
CmdHeader** ppCVar5;
PriStackEntry* iVar6;
int iVar7;
iVar7 = 2;
iVar6 = gPriStack + 2;
do {
iVar4 = iVar6->count + -1;
if (-1 < iVar4) {
ppCVar5 = iVar6->entries + iVar6->count + -1;
do {
pCVar2 = *ppCVar5;
if ((pCVar2 != (CmdHeader*)0x0) && (pCVar2->ready_for_data != 0)) {
iVar1 = pCVar2->status;
if (iVar1 == -1) {
pBVar3 = pCVar2->callback_buffer;
if (pBVar3 != (Buffer*)0x0) {
if (pCVar2->callback != ProcessVAGData) {
iVar1 = (pCVar2->callback)(pCVar2, pBVar3);
pCVar2->status = iVar1;
if (pBVar3->decompressed_size == 0) {
pCVar2->callback_buffer = pBVar3->next;
FreeBuffer(pBVar3, 1);
}
}
iVar1 = pCVar2->status;
}
if (iVar1 == -1)
goto LAB_00007308;
}
ReleaseMessage(pCVar2, 1);
ReturnMessage(pCVar2);
iVar6 = iVar6 + 1;
iVar7 = iVar7 + 1;
break;
}
LAB_00007308:
iVar4 = iVar4 + -1;
ppCVar5 = ppCVar5 + -1;
} while (-1 < iVar4);
}
iVar7 = iVar7 + -1;
iVar6 = iVar6 + -1;
if (iVar7 < 0) {
return;
}
} while (true);
}
void ReturnMessage(CmdHeader* param_1) {
if (param_1->mbx_to_reply == 0) {
if (param_1->thread_id == 0) {
FreeVAGCommand((VagCmd*)param_1);
} else {
WakeupThread(param_1->thread_id);
}
} else {
SendMbx(param_1->mbx_to_reply, param_1);
}
}
VagCmd* GetVAGCommand() {
while (true) {
while (vag_cmd_cnt == 31) {
DelayThread(100);
}
while (WaitSema(sSema))
;
for (int i = 0; i < 31; i++) {
// scan bits for free entry
if (((vag_cmd_used >> i) & 1) == 0) {
vag_cmd_used |= 1 << i;
vag_cmd_cnt++;
if (max_vag_cmd_cnt < vag_cmd_cnt) {
max_vag_cmd_cnt = vag_cmd_cnt;
}
SignalSema(sSema);
return &vag_cmds[i];
}
}
SignalSema(sSema);
}
}
void FreeVAGCommand(VagCmd* cmd) {
u32 vag_idx;
// get array index
vag_idx = (cmd - vag_cmds) / sizeof(VagCmd);
if ((vag_idx < 0x1f) && ((vag_cmd_used >> vag_idx) & 1U) != 0) {
while (WaitSema(sSema))
;
vag_cmd_used &= ~(1 << vag_idx);
--vag_cmd_cnt;
SignalSema(sSema);
}
}
u8* CheckForIsoPageBoundaryCrossing(Buffer* buf) {
Page* new_page;
u8* new_buffer;
u8* our_ptr;
u8* page_end;
our_ptr = buf->decomp_buffer;
page_end = buf->page->ptr;
if (page_end <= our_ptr) {
new_page = StepTopPage(buf->plist, buf->page);
buf->page = new_page;
if (new_page != nullptr) {
new_buffer = new_page->buffer;
buf->unk_12 = new_buffer;
buf->decomp_buffer = page_end + (new_buffer - (our_ptr + -1));
}
}
return buf->decomp_buffer;
}
void FreeDataBuffer(u32* param_1, u32 buffer_idx) {
if (param_1 == &BuffersAlloc) {
BuffersAlloc &= ~(1 << (buffer_idx & 0x1f));
--AllocdBuffersCount;
} else if (param_1 == &StrBuffersAlloc) {
StrBuffersAlloc &= ~(1 << (buffer_idx & 0x1f));
--AllocdStrBuffersCount;
}
}
} // namespace jak2