summaryrefslogtreecommitdiff
path: root/arm9/lib/src
diff options
context:
space:
mode:
authorred031000 <rubenru09@aol.com>2020-06-30 14:10:53 +0100
committerred031000 <rubenru09@aol.com>2020-06-30 14:12:05 +0100
commit1f9c3382c451382756934e0eaf7d35b51ce3b1d4 (patch)
treeddc326782b6c25a958fbcf6de83a9af8eb0a848a /arm9/lib/src
parent9396b082a208d4fc4375030c38a58ff589c9a72b (diff)
arm9 MI_dma_gxcommand, remove arm9_itcm.s
Diffstat (limited to 'arm9/lib/src')
-rw-r--r--arm9/lib/src/MI_dma_gxcommand.c153
1 files changed, 153 insertions, 0 deletions
diff --git a/arm9/lib/src/MI_dma_gxcommand.c b/arm9/lib/src/MI_dma_gxcommand.c
new file mode 100644
index 00000000..32f98f60
--- /dev/null
+++ b/arm9/lib/src/MI_dma_gxcommand.c
@@ -0,0 +1,153 @@
+#include "MI_dma_gxcommand.h"
+#include "function_target.h"
+#include "OS_interrupt.h"
+#include "OS_reset.h"
+#include "sections.h"
+
+static MIiGXDmaParams MIi_GXDmaParams = { FALSE };
+
+#pragma section ITCM begin
+ARM_FUNC void MI_SendGXCommand(u32 dmaNo, const void *src, u32 commandLength)
+{
+ vu32 *dmaCntp;
+ u32 leftLength = commandLength;
+ u32 currentSrc = (u32)src;
+ if (!leftLength)
+ {
+ return;
+ }
+
+ MIi_CheckDma0SourceAddress(dmaNo, (u32)src, commandLength, DMA_SRC_INC);
+
+ do
+ {
+ dmaCntp = &((vu32 *)REG_ADDR_DMA0SAD)[dmaNo * 3 + 2];
+ while (*dmaCntp & 0x80000000) {}
+ } while(0);
+
+ while (leftLength > 0)
+ {
+ u32 length = (leftLength > MIi_GX_LENGTH_ONCE) ? MIi_GX_LENGTH_ONCE : leftLength;
+ MIi_DmaSetParams(dmaNo, currentSrc, (u32)REG_GXFIFO_ADDR, MI_CNT_SEND32(length));
+ leftLength -= length;
+ currentSrc += length;
+ }
+
+ do
+ {
+ while (*dmaCntp & 0x80000000) {}
+ } while(0);
+}
+#pragma section ITCM end
+
+ARM_FUNC void MI_SendGXCommandAsync(u32 dmaNo, const void *src, u32 commandLength, MIDmaCallback callback, void *arg)
+{
+ vu32 *dmaCntp;
+ u32 leftLength = commandLength;
+ u32 currentSrc = (u32)src;
+
+ if (!commandLength)
+ {
+ MIi_CallCallback(callback, arg);
+ return;
+ }
+
+ while (MIi_GXDmaParams.isBusy) {}
+
+ while (!(G3X_GetCommandFifoStatus() & GX_FIFOSTAT_UNDERHALF)) {}
+
+ MIi_GXDmaParams.isBusy = TRUE;
+ MIi_GXDmaParams.dmaNo = dmaNo;
+ MIi_GXDmaParams.src = (u32)src;
+ MIi_GXDmaParams.length = commandLength;
+ MIi_GXDmaParams.callback = callback;
+ MIi_GXDmaParams.arg = arg;
+
+ MIi_CheckDma0SourceAddress(dmaNo, (u32)src, commandLength, DMA_SRC_INC);
+
+ MI_WaitDma(dmaNo);
+ {
+ OSIntrMode lastIntrMode = OS_DisableInterrupts();
+
+ MIi_GXDmaParams.fifoCond = (GXFifoIntrCond)((reg_G3X_GXSTAT & 0xc0000000) >> 30);
+ MIi_GXDmaParams.fifoFunc = OS_GetIrqFunction(OS_IE_GXFIFO);
+
+ G3X_SetFifoIntrCond(GX_FIFOINTR_COND_UNDERHALF);
+ OS_SetIrqFunction(OS_IE_GXFIFO, MIi_FIFOCallback);
+ (void)OS_EnableIrqMask(OS_IE_GXFIFO);
+
+ MIi_FIFOCallback();
+
+ (void)OS_RestoreInterrupts(lastIntrMode);
+ }
+}
+
+ARM_FUNC static void MIi_FIFOCallback(void)
+{
+ if (!MIi_GXDmaParams.length)
+ {
+ return;
+ }
+
+ u32 length = (MIi_GXDmaParams.length >= MIi_GX_LENGTH_ONCE) ? MIi_GX_LENGTH_ONCE : MIi_GXDmaParams.length;
+ u32 src = MIi_GXDmaParams.src;
+
+ MIi_GXDmaParams.length -= length;
+ MIi_GXDmaParams.src += length;
+
+ if (!MIi_GXDmaParams.length)
+ {
+ OSi_EnterDmaCallback(MIi_GXDmaParams.dmaNo, MIi_DMACallback, NULL);
+ MIi_DmaSetParams(MIi_GXDmaParams.dmaNo, src, (u32)REG_GXFIFO_ADDR, MI_CNT_SEND32_IF(length));
+ (void)OS_ResetRequestIrqMask(OS_IE_GXFIFO);
+ }
+ else
+ {
+ MIi_DmaSetParams(MIi_GXDmaParams.dmaNo, src, (u32)REG_GXFIFO_ADDR, MI_CNT_SEND32(length));
+ (void)OS_ResetRequestIrqMask(OS_IE_GXFIFO);
+ }
+}
+
+ARM_FUNC static void MIi_DMACallback(void *)
+{
+ (void)OS_DisableIrqMask(OS_IE_GXFIFO);
+
+ G3X_SetFifoIntrCond(MIi_GXDmaParams.fifoCond);
+ OS_SetIrqFunction(OS_IE_GXFIFO, MIi_GXDmaParams.fifoFunc);
+
+ MIi_GXDmaParams.isBusy = FALSE;
+
+ MIi_CallCallback(MIi_GXDmaParams.callback, MIi_GXDmaParams.arg);
+}
+
+ARM_FUNC void MI_SendGXCommandAsyncFast(u32 dmaNo, const void *src, u32 commandLength, MIDmaCallback callback, void *arg)
+{
+ if (!commandLength)
+ {
+ MIi_CallCallback(callback, arg);
+ return;
+ }
+
+ while (MIi_GXDmaParams.isBusy) {}
+
+ MIi_GXDmaParams.isBusy = TRUE;
+ MIi_GXDmaParams.dmaNo = dmaNo;
+ MIi_GXDmaParams.callback = callback;
+ MIi_GXDmaParams.arg = arg;
+
+ MIi_CheckAnotherAutoDMA(dmaNo, 0x38000000);
+
+ MIi_CheckDma0SourceAddress(dmaNo, (u32)src, commandLength, DMA_SRC_INC);
+
+ MI_WaitDma(dmaNo);
+
+ OSi_EnterDmaCallback(dmaNo, MIi_DMAFastCallback, NULL);
+ MIi_DmaSetParams(dmaNo, (u32)src, (u32)REG_GXFIFO_ADDR, MI_CNT_GXCOPY_IF(commandLength));
+}
+
+ARM_FUNC static void MIi_DMAFastCallback(void *)
+{
+ MIi_GXDmaParams.isBusy = FALSE;
+
+ MIi_CallCallback(MIi_GXDmaParams.callback, MIi_GXDmaParams.arg);
+}